2 * Please note: this implementation of openpty() is far from complete.
3 * it is just enough for portable OpenSSH's needs.
7 * Copyright (c) 2004 Damien Miller <djm@mindrot.org>
9 * Permission to use, copy, modify, and distribute this software for any
10 * purpose with or without fee is hereby granted, provided that the above
11 * copyright notice and this permission notice appear in all copies.
13 * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
14 * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
15 * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
16 * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
17 * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
18 * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
19 * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
23 * Author: Tatu Ylonen <ylo@cs.hut.fi>
24 * Copyright (c) 1995 Tatu Ylonen <ylo@cs.hut.fi>, Espoo, Finland
26 * Allocating a pseudo-terminal, and making it the controlling tty.
28 * As far as I am concerned, the code I have written for this software
29 * can be used freely for any purpose. Any derived versions of this
30 * software must be clearly marked as such, and if the derived work is
31 * incompatible with the protocol description in the RFC file, it must be
32 * called by a name other than "ssh" or "Secure Shell".
36 #if !defined(HAVE_OPENPTY)
38 #include <sys/types.h>
40 #ifdef HAVE_SYS_STAT_H
41 # include <sys/stat.h>
50 #endif /* HAVE_UTIL_H */
55 #if defined(HAVE_DEV_PTMX) && defined(HAVE_SYS_STROPTS_H)
56 # include <sys/stropts.h>
66 openpty(int *amaster, int *aslave, char *name, struct termios *termp,
69 #if defined(HAVE__GETPTY)
71 * _getpty(3) exists in SGI Irix 4.x, 5.x & 6.x -- it generates more
72 * pty's automagically when needed
76 if ((slave = _getpty(amaster, O_RDWR, 0622, 0)) == NULL)
79 /* Open the slave side. */
80 if ((*aslave = open(slave, O_RDWR | O_NOCTTY)) == -1) {
86 #elif defined(HAVE_DEV_PTMX)
88 * This code is used e.g. on Solaris 2.x. (Note that Solaris 2.3
89 * also has bsd-style ptys, but they simply do not work.)
95 if ((ptm = open("/dev/ptmx", O_RDWR | O_NOCTTY)) == -1)
98 /* XXX: need to close ptm on error? */
99 old_signal = signal(SIGCHLD, SIG_DFL);
100 if (grantpt(ptm) < 0)
102 signal(SIGCHLD, old_signal);
104 if (unlockpt(ptm) < 0)
107 if ((pts = ptsname(ptm)) == NULL)
111 /* Open the slave side. */
112 if ((*aslave = open(pts, O_RDWR | O_NOCTTY)) == -1) {
118 * Try to push the appropriate streams modules, as described
121 ioctl(*aslave, I_PUSH, "ptem");
122 ioctl(*aslave, I_PUSH, "ldterm");
124 ioctl(*aslave, I_PUSH, "ttcompat");
129 #elif defined(HAVE_DEV_PTS_AND_PTC)
130 /* AIX-style pty code. */
133 if ((*amaster = open("/dev/ptc", O_RDWR | O_NOCTTY)) == -1)
135 if ((ttname = ttyname(*amaster)) == NULL)
137 if ((*aslave = open(ttname, O_RDWR | O_NOCTTY)) == -1) {
143 #elif defined(_UNICOS)
144 char ptbuf[64], ttbuf[64];
150 if ((highpty = sysconf(_SC_CRAY_NPTY)) == -1)
152 #endif /* _SC_CRAY_NPTY */
154 for (i = 0; i < highpty; i++) {
155 snprintf(ptbuf, sizeof(ptbuf), "/dev/pty/%03d", i);
156 snprintf(ttbuf, sizeof(ttbuf), "/dev/ttyp%03d", i);
157 if ((*amaster = open(ptbuf, O_RDWR|O_NOCTTY)) == -1)
159 /* Open the slave side. */
160 if ((*aslave = open(ttbuf, O_RDWR|O_NOCTTY)) == -1) {
169 /* BSD-style pty code. */
170 char ptbuf[64], ttbuf[64];
172 const char *ptymajors = "pqrstuvwxyzabcdefghijklmno"
173 "ABCDEFGHIJKLMNOPQRSTUVWXYZ";
174 const char *ptyminors = "0123456789abcdef";
175 int num_minors = strlen(ptyminors);
176 int num_ptys = strlen(ptymajors) * num_minors;
179 for (i = 0; i < num_ptys; i++) {
180 snprintf(ptbuf, sizeof(ptbuf), "/dev/pty%c%c",
181 ptymajors[i / num_minors], ptyminors[i % num_minors]);
182 snprintf(ttbuf, sizeof(ttbuf), "/dev/tty%c%c",
183 ptymajors[i / num_minors], ptyminors[i % num_minors]);
185 if ((*amaster = open(ptbuf, O_RDWR | O_NOCTTY)) == -1) {
186 /* Try SCO style naming */
187 snprintf(ptbuf, sizeof(ptbuf), "/dev/ptyp%d", i);
188 snprintf(ttbuf, sizeof(ttbuf), "/dev/ttyp%d", i);
189 if ((*amaster = open(ptbuf, O_RDWR | O_NOCTTY)) == -1)
193 /* Open the slave side. */
194 if ((*aslave = open(ttbuf, O_RDWR | O_NOCTTY)) == -1) {
198 /* set tty modes to a sane state for broken clients */
199 if (tcgetattr(*amaster, &tio) != -1) {
200 tio.c_lflag |= (ECHO | ISIG | ICANON);
201 tio.c_oflag |= (OPOST | ONLCR);
202 tio.c_iflag |= ICRNL;
203 tcsetattr(*amaster, TCSANOW, &tio);
212 #endif /* !defined(HAVE_OPENPTY) */