1 /* Determine name of the slave side of a pseudo-terminal.
2 Copyright (C) 1998, 2002, 2010-2025 Free Software Foundation, Inc.
4 This file is free software: you can redistribute it and/or modify
5 it under the terms of the GNU Lesser General Public License as
6 published by the Free Software Foundation; either version 2.1 of the
7 License, or (at your option) any later version.
9 This file is distributed in the hope that it will be useful,
10 but WITHOUT ANY WARRANTY; without even the implied warranty of
11 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 GNU Lesser General Public License for more details.
14 You should have received a copy of the GNU Lesser General Public License
15 along with this program. If not, see <https://www.gnu.org/licenses/>. */
29 # define _PATH_TTY "/dev/tty"
32 # define _PATH_DEV "/dev/"
35 /* Get the major, minor macros. */
37 # include <sys/mkdev.h>
39 #if MAJOR_IN_SYSMACROS
40 # include <sys/sysmacros.h>
44 /* Get ioctl() and 'struct strioctl'. */
47 # include <sys/stream.h>
48 # include <sys/ptms.h>
52 #if defined _AIX || defined __osf__
53 /* Get ioctl(), ISPTM. */
54 # include <sys/ioctl.h>
58 #if defined __DragonFly__
59 /* Get fdevname_r(). */
63 /* Store at most BUFLEN characters of the pathname of the slave pseudo
64 terminal associated with the master FD is open on in BUF.
65 Return 0 on success, otherwise an error number. */
67 ptsname_r (int fd
, char *buf
, size_t buflen
)
70 #if HAVE_ESSENTIALLY_WORKING_PTSNAME_R
71 # if defined __NetBSD__ || defined __sun
75 if (buflen
>= sizeof (tmpbuf
))
76 /* ERANGE should not happen in this case. */
77 return ptsname_r (fd
, buf
, buflen
);
80 int ret
= ptsname_r (fd
, tmpbuf
, sizeof (tmpbuf
));
85 size_t len
= strlen (tmpbuf
);
88 memcpy (buf
, tmpbuf
, len
+ 1);
93 int ret
= ptsname_r (fd
, buf
, buflen
);
99 #elif defined __DragonFly__
100 int saved_errno
= errno
;
101 char tmpbuf
[5 + 4 + 10 + 1];
109 /* The result of fdevname_r is typically of the form ptm/N. */
110 ret
= fdevname_r (fd
, tmpbuf
+ 5, sizeof (tmpbuf
) - 5);
111 if (ret
< 0 || strncmp (tmpbuf
+ 5, "ptm/", 4) != 0)
116 /* Turn it into /dev/pts/N. */
117 memcpy (tmpbuf
, "/dev/pts/", 5 + 4);
124 memcpy (buf
, tmpbuf
, n
+ 1);
125 /* Don't do a final stat(), since the file name /dev/pts/N does not actually
130 int saved_errno
= errno
;
139 # if defined __sun /* Solaris */
140 if (fstat (fd
, &st
) < 0)
142 if (!S_ISCHR (st
.st_mode
))
148 /* Master ptys can be recognized through a STREAMS ioctl. See
149 "STREAMS-based Pseudo-Terminal Subsystem"
150 <https://docs.oracle.com/cd/E18752_01/html/816-4855/termsub15-44781.html>
151 and "STREAMS ioctl commands"
152 <https://docs.oracle.com/cd/E18752_01/html/816-5177/streamio-7i.html>
154 struct strioctl ioctl_arg
;
155 ioctl_arg
.ic_cmd
= ISPTM
;
156 ioctl_arg
.ic_timout
= 0;
157 ioctl_arg
.ic_len
= 0;
158 ioctl_arg
.ic_dp
= NULL
;
160 if (ioctl (fd
, I_STR
, &ioctl_arg
) < 0)
167 char tmpbuf
[9 + 10 + 1];
168 int n
= sprintf (tmpbuf
, "/dev/pts/%u", minor (st
.st_rdev
));
174 memcpy (buf
, tmpbuf
, n
+ 1);
176 # elif defined _AIX || defined __osf__ /* AIX, OSF/1 */
177 /* This implementation returns /dev/pts/N, like ptsname() does.
178 Whereas the generic implementation below returns /dev/ttypN.
179 Both are correct, but let's be consistent with ptsname(). */
180 if (fstat (fd
, &st
) < 0)
182 if (!S_ISCHR (st
.st_mode
))
190 char tmpbuf
[9 + 10 + 1];
193 ret
= ioctl (fd
, ISPTM
, &dev
);
196 ret
= ioctl (fd
, ISPTM
, NULL
);
204 n
= sprintf (tmpbuf
, "/dev/pts/%u", minor (dev
));
210 memcpy (buf
, tmpbuf
, n
+ 1);
215 # if ISATTY_FAILS_WITHOUT_SETTING_ERRNO && defined F_GETFL /* IRIX, Solaris */
217 if (fcntl (fd
, F_GETFL
) != -1)
220 /* We rely on isatty to set errno properly (i.e. EBADF or ENOTTY). */
225 if (buflen
< strlen (_PATH_TTY
) + 3)
231 int err
= ttyname_r (fd
, buf
, buflen
);
238 if (strncmp(buf
, "/dev/pts/", strlen("/dev/pts/")) != 0)
239 buf
[sizeof (_PATH_DEV
) - 1] = 't';
242 if (stat (buf
, &st
) < 0 && errno
!= EOVERFLOW
)