forked from abligh/pty
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathptyfork.c
81 lines (74 loc) · 2.11 KB
/
ptyfork.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
#include "apue.h"
#include <termios.h>
pid_t
pty_fork (int *ptrfdm, char *slave_name, int slave_namesz,
const struct termios *slave_termios,
const struct winsize *slave_winsize)
{
int fdm, fds;
pid_t pid;
char pts_name[20];
if ((fdm = ptym_open (pts_name, sizeof (pts_name))) < 0)
err_sys ("can't open master pty: %s, error %d", pts_name, fdm);
if (slave_name != NULL)
{
/*
* Return name of slave. Null terminate to handle case
* where strlen(pts_name) > slave_namesz.
*/
strncpy (slave_name, pts_name, slave_namesz);
slave_name[slave_namesz - 1] = '\0';
}
if ((pid = fork ()) < 0)
{
return (-1);
}
else if (pid == 0)
{ /* child */
if (setsid () < 0)
err_sys ("setsid error");
/*
* System V acquires controlling terminal on open().
*/
if ((fds = ptys_open (pts_name)) < 0)
err_sys ("can't open slave pty");
close (fdm); /* all done with master in child */
#if defined(BSD)
/*
* TIOCSCTTY is the BSD way to acquire a controlling terminal.
*/
if (ioctl (fds, TIOCSCTTY, (char *) 0) < 0)
err_sys ("TIOCSCTTY error");
#endif
/*
* Set slave's termios and window size.
*/
if (slave_termios != NULL)
{
if (tcsetattr (fds, TCSANOW, slave_termios) < 0)
err_sys ("tcsetattr error on slave pty");
}
if (slave_winsize != NULL)
{
if (ioctl (fds, TIOCSWINSZ, slave_winsize) < 0)
err_sys ("TIOCSWINSZ error on slave pty");
}
/*
* Slave becomes stdin/stdout/stderr of child.
*/
if (dup2 (fds, STDIN_FILENO) != STDIN_FILENO)
err_sys ("dup2 error to stdin");
if (dup2 (fds, STDOUT_FILENO) != STDOUT_FILENO)
err_sys ("dup2 error to stdout");
if (dup2 (fds, STDERR_FILENO) != STDERR_FILENO)
err_sys ("dup2 error to stderr");
if (fds != STDIN_FILENO && fds != STDOUT_FILENO && fds != STDERR_FILENO)
close (fds);
return (0); /* child returns 0 just like fork() */
}
else
{ /* parent */
*ptrfdm = fdm; /* return fd of master */
return (pid); /* parent returns pid of child */
}
}