blob: 599cbd2cebdaea6a5de6e0c6bc32e71bc24b4b93 [file] [log] [blame]
Elliott Hughes49167062014-07-25 17:24:00 -07001/*
2 * Copyright (C) 2008 The Android Open Source Project
3 * All rights reserved.
4 *
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions
7 * are met:
8 * * Redistributions of source code must retain the above copyright
9 * notice, this list of conditions and the following disclaimer.
10 * * Redistributions in binary form must reproduce the above copyright
11 * notice, this list of conditions and the following disclaimer in
12 * the documentation and/or other materials provided with the
13 * distribution.
14 *
15 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
16 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
17 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
18 * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
19 * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
20 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
21 * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
22 * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
23 * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
24 * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
25 * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
26 * SUCH DAMAGE.
27 */
28
29#include <errno.h>
30#include <fcntl.h>
Elliott Hughes65f0df72014-12-03 14:39:20 -080031#include <pty.h>
Elliott Hughes49167062014-07-25 17:24:00 -070032#include <stdio.h>
33#include <stdlib.h>
34#include <sys/ioctl.h>
35#include <termios.h>
36#include <unistd.h>
Elliott Hughes65f0df72014-12-03 14:39:20 -080037#include <utmp.h>
Elliott Hughes49167062014-07-25 17:24:00 -070038
Josh Gao5e2285d2017-02-22 12:19:05 -080039#include "bionic/pthread_internal.h"
Elliott Hughes4d215aa2017-10-18 15:54:56 -070040#include "private/FdPath.h"
Elliott Hughesa381fe82014-12-09 20:30:23 -080041
Elliott Hughes65f0df72014-12-03 14:39:20 -080042int getpt() {
Elliott Hughes49167062014-07-25 17:24:00 -070043 return posix_openpt(O_RDWR|O_NOCTTY);
44}
45
46int grantpt(int) {
47 return 0;
48}
49
50int posix_openpt(int flags) {
51 return open("/dev/ptmx", flags);
52}
53
54char* ptsname(int fd) {
Josh Gao5e2285d2017-02-22 12:19:05 -080055 bionic_tls& tls = __get_bionic_tls();
56 char* buf = tls.ptsname_buf;
57 int error = ptsname_r(fd, buf, sizeof(tls.ptsname_buf));
Elliott Hughes61706932015-03-31 10:56:58 -070058 return (error == 0) ? buf : NULL;
Elliott Hughes49167062014-07-25 17:24:00 -070059}
60
61int ptsname_r(int fd, char* buf, size_t len) {
62 if (buf == NULL) {
63 errno = EINVAL;
64 return errno;
65 }
66
67 unsigned int pty_num;
68 if (ioctl(fd, TIOCGPTN, &pty_num) != 0) {
69 errno = ENOTTY;
70 return errno;
71 }
72
73 if (snprintf(buf, len, "/dev/pts/%u", pty_num) >= static_cast<int>(len)) {
74 errno = ERANGE;
75 return errno;
76 }
77
78 return 0;
79}
80
81char* ttyname(int fd) {
Josh Gao5e2285d2017-02-22 12:19:05 -080082 bionic_tls& tls = __get_bionic_tls();
83 char* buf = tls.ttyname_buf;
84 int error = ttyname_r(fd, buf, sizeof(tls.ttyname_buf));
Elliott Hughes61706932015-03-31 10:56:58 -070085 return (error == 0) ? buf : NULL;
Elliott Hughes49167062014-07-25 17:24:00 -070086}
87
88int ttyname_r(int fd, char* buf, size_t len) {
89 if (buf == NULL) {
90 errno = EINVAL;
91 return errno;
92 }
93
94 if (!isatty(fd)) {
95 return errno;
96 }
97
Elliott Hughes4d215aa2017-10-18 15:54:56 -070098 ssize_t count = readlink(FdPath(fd).c_str(), buf, len);
Elliott Hughes49167062014-07-25 17:24:00 -070099 if (count == -1) {
100 return errno;
101 }
102 if (static_cast<size_t>(count) == len) {
103 errno = ERANGE;
104 return errno;
105 }
106 buf[count] = '\0';
107 return 0;
108}
109
110int unlockpt(int fd) {
111 int unlock = 0;
112 return ioctl(fd, TIOCSPTLCK, &unlock);
113}
Elliott Hughes65f0df72014-12-03 14:39:20 -0800114
115int openpty(int* master, int* slave, char* name, const termios* t, const winsize* ws) {
116 *master = getpt();
117 if (*master == -1) {
118 return -1;
119 }
120
121 if (grantpt(*master) == -1 || unlockpt(*master) == -1) {
122 close(*master);
123 return -1;
124 }
125
126 char buf[32];
127 if (name == NULL) {
128 name = buf;
129 }
130 if (ptsname_r(*master, name, sizeof(buf)) != 0) {
131 close(*master);
132 return -1;
133 }
134
135 *slave = open(name, O_RDWR|O_NOCTTY);
136 if (*slave == -1) {
137 close(*master);
138 return -1;
139 }
140
141 if (t != NULL) {
142 tcsetattr(*slave, TCSAFLUSH, t);
143 }
144 if (ws != NULL) {
145 ioctl(*slave, TIOCSWINSZ, ws);
146 }
147
148 return 0;
149}
150
Josh Gao6d7c1ee2016-03-04 18:04:41 -0800151int forkpty(int* amaster, char* name, const termios* t, const winsize* ws) {
152 int master;
Elliott Hughes65f0df72014-12-03 14:39:20 -0800153 int slave;
Josh Gao6d7c1ee2016-03-04 18:04:41 -0800154 if (openpty(&master, &slave, name, t, ws) == -1) {
Elliott Hughes65f0df72014-12-03 14:39:20 -0800155 return -1;
156 }
157
158 pid_t pid = fork();
159 if (pid == -1) {
Josh Gao6d7c1ee2016-03-04 18:04:41 -0800160 close(master);
Elliott Hughes65f0df72014-12-03 14:39:20 -0800161 close(slave);
162 return -1;
163 }
164
165 if (pid == 0) {
166 // Child.
Josh Gao6d7c1ee2016-03-04 18:04:41 -0800167 *amaster = -1;
168 close(master);
Elliott Hughes65f0df72014-12-03 14:39:20 -0800169 if (login_tty(slave) == -1) {
170 _exit(1);
171 }
172 return 0;
173 }
174
175 // Parent.
Josh Gao6d7c1ee2016-03-04 18:04:41 -0800176 *amaster = master;
Elliott Hughes65f0df72014-12-03 14:39:20 -0800177 close(slave);
178 return pid;
179}
180
181int login_tty(int fd) {
182 setsid();
183
184 if (ioctl(fd, TIOCSCTTY, NULL) == -1) {
185 return -1;
186 }
187
188 dup2(fd, STDIN_FILENO);
189 dup2(fd, STDOUT_FILENO);
190 dup2(fd, STDERR_FILENO);
191 if (fd > STDERR_FILENO) {
192 close(fd);
193 }
194
195 return 0;
196}