You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
494 lines
10 KiB
494 lines
10 KiB
/* vi: ts=8 sts=4 sw=4
|
|
*
|
|
*
|
|
* This file is part of the KDE project, module tdesu.
|
|
* Copyright (C) 1999,2000 Geert Jansen <jansen@kde.org>
|
|
*
|
|
* This file contains code from TEShell.C of the KDE konsole.
|
|
* Copyright (c) 1997,1998 by Lars Doelle <lars.doelle@on-line.de>
|
|
*
|
|
* This is free software; you can use this library under the GNU Library
|
|
* General Public License, version 2. See the file "COPYING.LIB" for the
|
|
* exact licensing terms.
|
|
*
|
|
* process.cpp: Functionality to build a front end to password asking
|
|
* terminal programs.
|
|
*/
|
|
|
|
#include <config.h>
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <unistd.h>
|
|
#include <fcntl.h>
|
|
#include <signal.h>
|
|
#include <errno.h>
|
|
#include <string.h>
|
|
#include <termios.h>
|
|
#include <signal.h>
|
|
|
|
#include <sys/types.h>
|
|
#include <sys/wait.h>
|
|
#include <sys/stat.h>
|
|
#include <sys/time.h>
|
|
#include <sys/resource.h>
|
|
#include <sys/socket.h>
|
|
|
|
#if defined(__SVR4) && defined(sun)
|
|
#include <stropts.h>
|
|
#include <sys/stream.h>
|
|
#endif
|
|
|
|
#ifdef HAVE_SYS_SELECT_H
|
|
#include <sys/select.h> // Needed on some systems.
|
|
#endif
|
|
|
|
#include <tqglobal.h>
|
|
#include <tqcstring.h>
|
|
#include <tqfile.h>
|
|
|
|
#include <kdebug.h>
|
|
#include <kstandarddirs.h>
|
|
|
|
#include "process.h"
|
|
#include <tdesu/tdesu_pty.h>
|
|
#include <tdesu/kcookie.h>
|
|
|
|
|
|
MyPtyProcess::MyPtyProcess()
|
|
{
|
|
m_bTerminal = false;
|
|
m_bErase = false;
|
|
m_pPTY = 0L;
|
|
m_Pid = -1;
|
|
m_Fd = -1;
|
|
}
|
|
|
|
|
|
int MyPtyProcess::init()
|
|
{
|
|
delete m_pPTY;
|
|
m_pPTY = new PTY();
|
|
m_Fd = m_pPTY->getpt();
|
|
if (m_Fd < 0)
|
|
return -1;
|
|
if ((m_pPTY->grantpt() < 0) || (m_pPTY->unlockpt() < 0))
|
|
{
|
|
kdError(PTYPROC) << k_lineinfo << "Master setup failed.\n" << endl;
|
|
m_Fd = -1;
|
|
return -1;
|
|
}
|
|
m_TTY = m_pPTY->ptsname();
|
|
m_stdoutBuf.resize(0);
|
|
m_stderrBuf.resize(0);
|
|
m_ptyBuf.resize(0);
|
|
return 0;
|
|
}
|
|
|
|
|
|
MyPtyProcess::~MyPtyProcess()
|
|
{
|
|
delete m_pPTY;
|
|
}
|
|
|
|
|
|
/*
|
|
* Read one line of input. The terminal is in canonical mode, so you always
|
|
* read a line at at time, but it's possible to receive multiple lines in
|
|
* one time.
|
|
*/
|
|
|
|
|
|
TQCString MyPtyProcess::readLineFrom(int fd, TQCString& inbuf, bool block)
|
|
{
|
|
int pos;
|
|
TQCString ret;
|
|
|
|
if (!inbuf.isEmpty())
|
|
{
|
|
|
|
pos = inbuf.find('\n');
|
|
|
|
if (pos == -1)
|
|
{
|
|
ret = inbuf;
|
|
inbuf.resize(0);
|
|
} else
|
|
{
|
|
ret = inbuf.left(pos);
|
|
inbuf = inbuf.mid(pos+1);
|
|
}
|
|
return ret;
|
|
|
|
}
|
|
|
|
int flags = fcntl(fd, F_GETFL);
|
|
if (flags < 0)
|
|
{
|
|
kdError(PTYPROC) << k_lineinfo << "fcntl(F_GETFL): " << perror << "\n";
|
|
return ret;
|
|
}
|
|
if (block)
|
|
flags &= ~O_NONBLOCK;
|
|
else
|
|
flags |= O_NONBLOCK;
|
|
if (fcntl(fd, F_SETFL, flags) < 0)
|
|
{
|
|
kdError(PTYPROC) << k_lineinfo << "fcntl(F_SETFL): " << perror << "\n";
|
|
return ret;
|
|
}
|
|
|
|
int nbytes;
|
|
char buf[256];
|
|
while (1)
|
|
{
|
|
nbytes = read(fd, buf, 255);
|
|
if (nbytes == -1)
|
|
{
|
|
if (errno == EINTR)
|
|
continue;
|
|
else break;
|
|
}
|
|
if (nbytes == 0)
|
|
break; // eof
|
|
|
|
buf[nbytes] = '\000';
|
|
inbuf += buf;
|
|
|
|
pos = inbuf.find('\n');
|
|
if (pos == -1)
|
|
{
|
|
ret = inbuf;
|
|
inbuf.resize(0);
|
|
} else
|
|
{
|
|
ret = inbuf.left(pos);
|
|
inbuf = inbuf.mid(pos+1);
|
|
}
|
|
break;
|
|
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
void MyPtyProcess::writeLine(TQCString line, bool addnl)
|
|
{
|
|
if (!line.isEmpty())
|
|
write(m_Fd, line, line.length());
|
|
if (addnl)
|
|
write(m_Fd, "\n", 1);
|
|
}
|
|
|
|
void MyPtyProcess::unreadLineFrom(TQCString inbuf, TQCString line, bool addnl)
|
|
{
|
|
if (addnl)
|
|
line += '\n';
|
|
if (!line.isEmpty())
|
|
inbuf.prepend(line);
|
|
}
|
|
|
|
|
|
/*
|
|
* Fork and execute the command. This returns in the parent.
|
|
*/
|
|
|
|
int MyPtyProcess::exec(TQCString command, QCStringList args)
|
|
{
|
|
kdDebug(PTYPROC) << "MyPtyProcess::exec(): " << command << endl;// << ", args = " << args << endl;
|
|
|
|
if (init() < 0)
|
|
return -1;
|
|
|
|
// Open the pty slave before forking. See SetupTTY()
|
|
int slave = open(m_TTY, O_RDWR);
|
|
if (slave < 0)
|
|
{
|
|
kdError(PTYPROC) << k_lineinfo << "Could not open slave pty.\n";
|
|
return -1;
|
|
}
|
|
|
|
// Also create a socket pair to connect to standard in/out.
|
|
// This will allow use to bypass the terminal.
|
|
int inout[2];
|
|
int err[2];
|
|
int ok = 1;
|
|
ok &= socketpair(AF_UNIX, SOCK_STREAM, 0, inout) >= 0;
|
|
ok &= socketpair(AF_UNIX, SOCK_STREAM, 0, err ) >= 0;
|
|
if( !ok ) {
|
|
kdDebug(PTYPROC) << "Could not create socket" << endl;
|
|
return -1;
|
|
}
|
|
m_stdinout = inout[0];
|
|
m_err = err[0];
|
|
|
|
if ((m_Pid = fork()) == -1)
|
|
{
|
|
kdError(PTYPROC) << k_lineinfo << "fork(): " << perror << "\n";
|
|
return -1;
|
|
}
|
|
|
|
// Parent
|
|
if (m_Pid)
|
|
{
|
|
close(slave);
|
|
close(inout[1]);
|
|
close(err[1]);
|
|
return 0;
|
|
}
|
|
|
|
// Child
|
|
|
|
ok = 1;
|
|
ok &= dup2(inout[1], STDIN_FILENO) >= 0;
|
|
ok &= dup2(inout[1], STDOUT_FILENO) >= 0;
|
|
ok &= dup2(err[1], STDERR_FILENO) >= 0;
|
|
|
|
if( !ok )
|
|
{
|
|
kdError(PTYPROC) << "dup of socket descriptor failed" << endl;
|
|
_exit(1);
|
|
}
|
|
|
|
close(inout[1]);
|
|
close(inout[0]);
|
|
close(err[1]);
|
|
close(err[0]);
|
|
|
|
if (SetupTTY(slave) < 0)
|
|
_exit(1);
|
|
|
|
// From now on, terminal output goes through the tty.
|
|
TQCString path;
|
|
if (command.contains('/'))
|
|
path = command;
|
|
else
|
|
{
|
|
TQString file = TDEStandardDirs::findExe(command);
|
|
if (file.isEmpty())
|
|
{
|
|
kdError(PTYPROC) << k_lineinfo << command << " not found\n";
|
|
_exit(1);
|
|
}
|
|
path = TQFile::encodeName(file);
|
|
}
|
|
|
|
int i;
|
|
const char * argp[32];
|
|
argp[0] = path;
|
|
QCStringList::Iterator it;
|
|
for (i=1, it=args.begin(); it!=args.end() && i<31; it++) {
|
|
argp[i++] = *it;
|
|
kdDebug(PTYPROC) << *it << endl;
|
|
}
|
|
argp[i] = 0L;
|
|
execv(path, (char * const *)argp);
|
|
kdError(PTYPROC) << k_lineinfo << "execv(\"" << path << "\"): " << perror << "\n";
|
|
_exit(1);
|
|
return -1; // Shut up compiler. Never reached.
|
|
}
|
|
|
|
/*
|
|
* Wait until the terminal is set into no echo mode. At least one su
|
|
* (RH6 w/ Linux-PAM patches) sets noecho mode AFTER writing the Password:
|
|
* prompt, using TCSAFLUSH. This flushes the terminal I/O queues, possibly
|
|
* taking the password with it. So we wait until no echo mode is set
|
|
* before writing the password.
|
|
* Note that this is done on the slave fd. While Linux allows tcgetattr() on
|
|
* the master side, Solaris doesn't.
|
|
*/
|
|
|
|
int MyPtyProcess::WaitSlave()
|
|
{
|
|
int slave = open(m_TTY, O_RDWR);
|
|
if (slave < 0)
|
|
{
|
|
kdError(PTYPROC) << k_lineinfo << "Could not open slave tty.\n";
|
|
return -1;
|
|
}
|
|
|
|
struct termios tio;
|
|
struct timeval tv;
|
|
while (1)
|
|
{
|
|
if (tcgetattr(slave, &tio) < 0)
|
|
{
|
|
kdError(PTYPROC) << k_lineinfo << "tcgetattr(): " << perror << "\n";
|
|
close(slave);
|
|
return -1;
|
|
}
|
|
if (tio.c_lflag & ECHO)
|
|
{
|
|
kdDebug(PTYPROC) << k_lineinfo << "Echo mode still on." << endl;
|
|
// sleep 1/10 sec
|
|
tv.tv_sec = 0; tv.tv_usec = 100000;
|
|
select(slave, 0L, 0L, 0L, &tv);
|
|
continue;
|
|
}
|
|
break;
|
|
}
|
|
close(slave);
|
|
return 0;
|
|
}
|
|
|
|
|
|
int MyPtyProcess::enableLocalEcho(bool enable)
|
|
{
|
|
int slave = open(m_TTY, O_RDWR);
|
|
if (slave < 0)
|
|
{
|
|
kdError(PTYPROC) << k_lineinfo << "Could not open slave tty.\n";
|
|
return -1;
|
|
}
|
|
struct termios tio;
|
|
if (tcgetattr(slave, &tio) < 0)
|
|
{
|
|
kdError(PTYPROC) << k_lineinfo << "tcgetattr(): " << perror << "\n";
|
|
close(slave); return -1;
|
|
}
|
|
if (enable)
|
|
tio.c_lflag |= ECHO;
|
|
else
|
|
tio.c_lflag &= ~ECHO;
|
|
if (tcsetattr(slave, TCSANOW, &tio) < 0)
|
|
{
|
|
kdError(PTYPROC) << k_lineinfo << "tcsetattr(): " << perror << "\n";
|
|
close(slave); return -1;
|
|
}
|
|
close(slave);
|
|
return 0;
|
|
}
|
|
|
|
|
|
/*
|
|
* Copy output to stdout until the child process exists, or a line of output
|
|
* matches `m_Exit'.
|
|
* We have to use waitpid() to test for exit. Merely waiting for EOF on the
|
|
* pty does not work, because the target process may have children still
|
|
* attached to the terminal.
|
|
*/
|
|
|
|
int MyPtyProcess::waitForChild()
|
|
{
|
|
int ret, state, retval = 1;
|
|
struct timeval tv;
|
|
|
|
fd_set fds;
|
|
FD_ZERO(&fds);
|
|
|
|
while (1)
|
|
{
|
|
tv.tv_sec = 1; tv.tv_usec = 0;
|
|
FD_SET(m_Fd, &fds);
|
|
ret = select(m_Fd+1, &fds, 0L, 0L, &tv);
|
|
if (ret == -1)
|
|
{
|
|
if (errno == EINTR) continue;
|
|
else
|
|
{
|
|
kdError(PTYPROC) << k_lineinfo << "select(): " << perror << "\n";
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
if (ret)
|
|
{
|
|
TQCString line = readLine(false);
|
|
while (!line.isNull())
|
|
{
|
|
if (!m_Exit.isEmpty() && !tqstrnicmp(line, m_Exit, m_Exit.length()))
|
|
kill(m_Pid, SIGTERM);
|
|
if (m_bTerminal)
|
|
{
|
|
fputs(line, stdout);
|
|
fputc('\n', stdout);
|
|
}
|
|
line = readLine(false);
|
|
}
|
|
}
|
|
|
|
// Check if the process is still alive
|
|
ret = waitpid(m_Pid, &state, WNOHANG);
|
|
if (ret < 0)
|
|
{
|
|
if (errno == ECHILD)
|
|
retval = 0;
|
|
else
|
|
kdError(PTYPROC) << k_lineinfo << "waitpid(): " << perror << "\n";
|
|
break;
|
|
}
|
|
if (ret == m_Pid)
|
|
{
|
|
if (WIFEXITED(state))
|
|
retval = WEXITSTATUS(state);
|
|
break;
|
|
}
|
|
}
|
|
|
|
return -retval;
|
|
}
|
|
|
|
/*
|
|
* SetupTTY: Creates a new session. The filedescriptor "fd" should be
|
|
* connected to the tty. It is closed after the tty is reopened to make it
|
|
* our controlling terminal. This way the tty is always opened at least once
|
|
* so we'll never get EIO when reading from it.
|
|
*/
|
|
|
|
int MyPtyProcess::SetupTTY(int fd)
|
|
{
|
|
// Reset signal handlers
|
|
for (int sig = 1; sig < NSIG; sig++)
|
|
signal(sig, SIG_DFL);
|
|
signal(SIGHUP, SIG_IGN);
|
|
|
|
// Close all file handles
|
|
// struct rlimit rlp;
|
|
// getrlimit(RLIMIT_NOFILE, &rlp);
|
|
// for (int i = 0; i < (int)rlp.rlim_cur; i++)
|
|
// if (i != fd) close(i);
|
|
|
|
// Create a new session.
|
|
setsid();
|
|
|
|
// Open slave. This will make it our controlling terminal
|
|
int slave = open(m_TTY, O_RDWR);
|
|
if (slave < 0)
|
|
{
|
|
kdError(PTYPROC) << k_lineinfo << "Could not open slave side: " << perror << "\n";
|
|
return -1;
|
|
}
|
|
close(fd);
|
|
|
|
#if defined(__SVR4) && defined(sun)
|
|
|
|
// Solaris STREAMS environment.
|
|
// Push these modules to make the stream look like a terminal.
|
|
ioctl(slave, I_PUSH, "ptem");
|
|
ioctl(slave, I_PUSH, "ldterm");
|
|
|
|
#endif
|
|
|
|
// Connect stdin, stdout and stderr
|
|
// dup2(slave, 0); dup2(slave, 1); dup2(slave, 2);
|
|
// if (slave > 2)
|
|
// close(slave);
|
|
|
|
// Disable OPOST processing. Otherwise, '\n' are (on Linux at least)
|
|
// translated to '\r\n'.
|
|
struct termios tio;
|
|
if (tcgetattr(slave, &tio) < 0)
|
|
{
|
|
kdError(PTYPROC) << k_lineinfo << "tcgetattr(): " << perror << "\n";
|
|
return -1;
|
|
}
|
|
tio.c_oflag &= ~OPOST;
|
|
if (tcsetattr(slave, TCSANOW, &tio) < 0)
|
|
{
|
|
kdError(PTYPROC) << k_lineinfo << "tcsetattr(): " << perror << "\n";
|
|
return -1;
|
|
}
|
|
|
|
return 0;
|
|
}
|