130fdc8d8SChris Lattner //===-- PseudoTerminal.cpp --------------------------------------*- C++ -*-===//
230fdc8d8SChris Lattner //
32946cd70SChandler Carruth // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
42946cd70SChandler Carruth // See https://llvm.org/LICENSE.txt for license information.
52946cd70SChandler Carruth // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
630fdc8d8SChris Lattner //
730fdc8d8SChris Lattner //===----------------------------------------------------------------------===//
830fdc8d8SChris Lattner //
930fdc8d8SChris Lattner // Created by Greg Clayton on 1/8/08.
1030fdc8d8SChris Lattner //
1130fdc8d8SChris Lattner //===----------------------------------------------------------------------===//
1230fdc8d8SChris Lattner
1330fdc8d8SChris Lattner #include "PseudoTerminal.h"
14*76e47d48SRaphael Isemann #include <cstdlib>
1530fdc8d8SChris Lattner #include <sys/ioctl.h>
16d34a329cSBenjamin Kramer #include <unistd.h>
1730fdc8d8SChris Lattner
1830fdc8d8SChris Lattner // PseudoTerminal constructor
PseudoTerminal()19b9c1b51eSKate Stone PseudoTerminal::PseudoTerminal()
200965b59bSJonas Devlieghere : m_primary_fd(invalid_fd), m_secondary_fd(invalid_fd) {}
2130fdc8d8SChris Lattner
2230fdc8d8SChris Lattner // Destructor
230965b59bSJonas Devlieghere // The primary and secondary file descriptors will get closed if they are
240965b59bSJonas Devlieghere // valid. Call the ReleasePrimaryFD()/ReleaseSecondaryFD() member functions
2530fdc8d8SChris Lattner // to release any file descriptors that are needed beyond the lifespan
2630fdc8d8SChris Lattner // of this object.
~PseudoTerminal()27b9c1b51eSKate Stone PseudoTerminal::~PseudoTerminal() {
280965b59bSJonas Devlieghere ClosePrimary();
290965b59bSJonas Devlieghere CloseSecondary();
3030fdc8d8SChris Lattner }
3130fdc8d8SChris Lattner
320965b59bSJonas Devlieghere // Close the primary file descriptor if it is valid.
ClosePrimary()330965b59bSJonas Devlieghere void PseudoTerminal::ClosePrimary() {
340965b59bSJonas Devlieghere if (m_primary_fd > 0) {
350965b59bSJonas Devlieghere ::close(m_primary_fd);
360965b59bSJonas Devlieghere m_primary_fd = invalid_fd;
3730fdc8d8SChris Lattner }
3830fdc8d8SChris Lattner }
3930fdc8d8SChris Lattner
400965b59bSJonas Devlieghere // Close the secondary file descriptor if it is valid.
CloseSecondary()410965b59bSJonas Devlieghere void PseudoTerminal::CloseSecondary() {
420965b59bSJonas Devlieghere if (m_secondary_fd > 0) {
430965b59bSJonas Devlieghere ::close(m_secondary_fd);
440965b59bSJonas Devlieghere m_secondary_fd = invalid_fd;
4530fdc8d8SChris Lattner }
4630fdc8d8SChris Lattner }
4730fdc8d8SChris Lattner
4830fdc8d8SChris Lattner // Open the first available pseudo terminal with OFLAG as the
490965b59bSJonas Devlieghere // permissions. The file descriptor is store in the m_primary_fd member
500965b59bSJonas Devlieghere // variable and can be accessed via the PrimaryFD() or ReleasePrimaryFD()
5130fdc8d8SChris Lattner // accessors.
5230fdc8d8SChris Lattner //
5330fdc8d8SChris Lattner // Suggested value for oflag is O_RDWR|O_NOCTTY
5430fdc8d8SChris Lattner //
5530fdc8d8SChris Lattner // RETURNS:
5630fdc8d8SChris Lattner // Zero when successful, non-zero indicating an error occurred.
OpenFirstAvailablePrimary(int oflag)570965b59bSJonas Devlieghere PseudoTerminal::Status PseudoTerminal::OpenFirstAvailablePrimary(int oflag) {
580965b59bSJonas Devlieghere // Open the primary side of a pseudo terminal
590965b59bSJonas Devlieghere m_primary_fd = ::posix_openpt(oflag);
600965b59bSJonas Devlieghere if (m_primary_fd < 0) {
6130fdc8d8SChris Lattner return err_posix_openpt_failed;
6230fdc8d8SChris Lattner }
6330fdc8d8SChris Lattner
640965b59bSJonas Devlieghere // Grant access to the secondary pseudo terminal
650965b59bSJonas Devlieghere if (::grantpt(m_primary_fd) < 0) {
660965b59bSJonas Devlieghere ClosePrimary();
6730fdc8d8SChris Lattner return err_grantpt_failed;
6830fdc8d8SChris Lattner }
6930fdc8d8SChris Lattner
700965b59bSJonas Devlieghere // Clear the lock flag on the secondary pseudo terminal
710965b59bSJonas Devlieghere if (::unlockpt(m_primary_fd) < 0) {
720965b59bSJonas Devlieghere ClosePrimary();
7330fdc8d8SChris Lattner return err_unlockpt_failed;
7430fdc8d8SChris Lattner }
7530fdc8d8SChris Lattner
7630fdc8d8SChris Lattner return success;
7730fdc8d8SChris Lattner }
7830fdc8d8SChris Lattner
790965b59bSJonas Devlieghere // Open the secondary pseudo terminal for the current primary pseudo
800965b59bSJonas Devlieghere // terminal. A primary pseudo terminal should already be valid prior to
810965b59bSJonas Devlieghere // calling this function (see PseudoTerminal::OpenFirstAvailablePrimary()).
820965b59bSJonas Devlieghere // The file descriptor is stored in the m_secondary_fd member variable and
830965b59bSJonas Devlieghere // can be accessed via the SecondaryFD() or ReleaseSecondaryFD() accessors.
8430fdc8d8SChris Lattner //
8530fdc8d8SChris Lattner // RETURNS:
8630fdc8d8SChris Lattner // Zero when successful, non-zero indicating an error occurred.
OpenSecondary(int oflag)870965b59bSJonas Devlieghere PseudoTerminal::Status PseudoTerminal::OpenSecondary(int oflag) {
880965b59bSJonas Devlieghere CloseSecondary();
8930fdc8d8SChris Lattner
900965b59bSJonas Devlieghere // Open the primary side of a pseudo terminal
910965b59bSJonas Devlieghere const char *secondary_name = SecondaryName();
9230fdc8d8SChris Lattner
930965b59bSJonas Devlieghere if (secondary_name == NULL)
9430fdc8d8SChris Lattner return err_ptsname_failed;
9530fdc8d8SChris Lattner
960965b59bSJonas Devlieghere m_secondary_fd = ::open(secondary_name, oflag);
9730fdc8d8SChris Lattner
980965b59bSJonas Devlieghere if (m_secondary_fd < 0)
990965b59bSJonas Devlieghere return err_open_secondary_failed;
10030fdc8d8SChris Lattner
10130fdc8d8SChris Lattner return success;
10230fdc8d8SChris Lattner }
10330fdc8d8SChris Lattner
1040965b59bSJonas Devlieghere // Get the name of the secondary pseudo terminal. A primary pseudo terminal
10530fdc8d8SChris Lattner // should already be valid prior to calling this function (see
1060965b59bSJonas Devlieghere // PseudoTerminal::OpenFirstAvailablePrimary()).
10730fdc8d8SChris Lattner //
10830fdc8d8SChris Lattner // RETURNS:
1090965b59bSJonas Devlieghere // NULL if no valid primary pseudo terminal or if ptsname() fails.
1100965b59bSJonas Devlieghere // The name of the secondary pseudo terminal as a NULL terminated C string
11130fdc8d8SChris Lattner // that comes from static memory, so a copy of the string should be
11230fdc8d8SChris Lattner // made as subsequent calls can change this value.
SecondaryName() const1130965b59bSJonas Devlieghere const char *PseudoTerminal::SecondaryName() const {
1140965b59bSJonas Devlieghere if (m_primary_fd < 0)
11530fdc8d8SChris Lattner return NULL;
1160965b59bSJonas Devlieghere return ::ptsname(m_primary_fd);
11730fdc8d8SChris Lattner }
11830fdc8d8SChris Lattner
11930fdc8d8SChris Lattner // Fork a child process that and have its stdio routed to a pseudo
12030fdc8d8SChris Lattner // terminal.
12130fdc8d8SChris Lattner //
1220965b59bSJonas Devlieghere // In the parent process when a valid pid is returned, the primary file
12330fdc8d8SChris Lattner // descriptor can be used as a read/write access to stdio of the
12430fdc8d8SChris Lattner // child process.
12530fdc8d8SChris Lattner //
12630fdc8d8SChris Lattner // In the child process the stdin/stdout/stderr will already be routed
1270965b59bSJonas Devlieghere // to the secondary pseudo terminal and the primary file descriptor will be
12830fdc8d8SChris Lattner // closed as it is no longer needed by the child process.
12930fdc8d8SChris Lattner //
1300965b59bSJonas Devlieghere // This class will close the file descriptors for the primary/secondary
1310965b59bSJonas Devlieghere // when the destructor is called, so be sure to call ReleasePrimaryFD()
1320965b59bSJonas Devlieghere // or ReleaseSecondaryFD() if any file descriptors are going to be used
13330fdc8d8SChris Lattner // past the lifespan of this object.
13430fdc8d8SChris Lattner //
13530fdc8d8SChris Lattner // RETURNS:
13630fdc8d8SChris Lattner // in the parent process: the pid of the child, or -1 if fork fails
13730fdc8d8SChris Lattner // in the child process: zero
13830fdc8d8SChris Lattner
Fork(PseudoTerminal::Status & error)13997206d57SZachary Turner pid_t PseudoTerminal::Fork(PseudoTerminal::Status &error) {
14030fdc8d8SChris Lattner pid_t pid = invalid_pid;
1410965b59bSJonas Devlieghere error = OpenFirstAvailablePrimary(O_RDWR | O_NOCTTY);
14230fdc8d8SChris Lattner
143b9c1b51eSKate Stone if (error == 0) {
1440965b59bSJonas Devlieghere // Successfully opened our primary pseudo terminal
14530fdc8d8SChris Lattner
14630fdc8d8SChris Lattner pid = ::fork();
147b9c1b51eSKate Stone if (pid < 0) {
14830fdc8d8SChris Lattner // Fork failed
14930fdc8d8SChris Lattner error = err_fork_failed;
150b9c1b51eSKate Stone } else if (pid == 0) {
15130fdc8d8SChris Lattner // Child Process
15230fdc8d8SChris Lattner ::setsid();
15330fdc8d8SChris Lattner
1540965b59bSJonas Devlieghere error = OpenSecondary(O_RDWR);
155b9c1b51eSKate Stone if (error == 0) {
1560965b59bSJonas Devlieghere // Successfully opened secondary
1570965b59bSJonas Devlieghere // We are done with the primary in the child process so lets close it
1580965b59bSJonas Devlieghere ClosePrimary();
15930fdc8d8SChris Lattner
16030fdc8d8SChris Lattner #if defined(TIOCSCTTY)
16130fdc8d8SChris Lattner // Acquire the controlling terminal
1620965b59bSJonas Devlieghere if (::ioctl(m_secondary_fd, TIOCSCTTY, (char *)0) < 0)
16330fdc8d8SChris Lattner error = err_failed_to_acquire_controlling_terminal;
16430fdc8d8SChris Lattner #endif
1650965b59bSJonas Devlieghere // Duplicate all stdio file descriptors to the secondary pseudo terminal
1660965b59bSJonas Devlieghere if (::dup2(m_secondary_fd, STDIN_FILENO) != STDIN_FILENO)
16730fdc8d8SChris Lattner error = error ? error : err_dup2_failed_on_stdin;
1680965b59bSJonas Devlieghere if (::dup2(m_secondary_fd, STDOUT_FILENO) != STDOUT_FILENO)
16930fdc8d8SChris Lattner error = error ? error : err_dup2_failed_on_stdout;
1700965b59bSJonas Devlieghere if (::dup2(m_secondary_fd, STDERR_FILENO) != STDERR_FILENO)
17130fdc8d8SChris Lattner error = error ? error : err_dup2_failed_on_stderr;
17230fdc8d8SChris Lattner }
173b9c1b51eSKate Stone } else {
17430fdc8d8SChris Lattner // Parent Process
17530fdc8d8SChris Lattner // Do nothing and let the pid get returned!
17630fdc8d8SChris Lattner }
17730fdc8d8SChris Lattner }
17830fdc8d8SChris Lattner return pid;
17930fdc8d8SChris Lattner }
180