/* This file is part of Konsole, an X terminal. Copyright (C) 1997,1998 by Lars Doelle This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. */ /* If you're compiling konsole on non-Linux platforms and find problems that you can track down to this file, please have a look into ../README.ports, too. */ /*! \file */ /*! \class TEPty \brief Ptys provide a pseudo terminal connection to a program. Although closely related to pipes, these pseudo terminal connections have some ability, that makes it nessesary to uses them. Most importent, they know about changing screen sizes and UNIX job control. Within the terminal emulation framework, this class represents the host side of the terminal together with the connecting serial line. One can create many instances of this class within a program. As a side effect of using this class, a signal(2) handler is installed on SIGCHLD. \par FIXME [NOTE: much of the technical stuff below will be replaced by forkpty.] publish the SIGCHLD signal if not related to an instance. clearify TEPty::done vs. TEPty::~TEPty semantics. check if pty is restartable via run after done. \par Pseudo terminals Pseudo terminals are a unique feature of UNIX, and always come in form of pairs of devices (/dev/ptyXX and /dev/ttyXX), which are connected to each other by the operating system. One may think of them as two serial devices linked by a null-modem cable. Being based on devices the number of simultanous instances of this class is (globally) limited by the number of those device pairs, which is 256. Another technic are UNIX 98 PTY's. These are supported also, and preferred over the (obsolete) predecessor. There's a sinister ioctl(2), signal(2) and job control stuff nessesary to make everything work as it should. Much of the stuff can be simplified by using openpty from glibc2. Compatibility issues with obsolete installations and other unixes my prevent this. */ #ifdef HAVE_CONFIG_H #include #endif #include #include #include #include #include #include #include #include #include #include "TEPty.h" void TEPty::donePty() { emit done(exitStatus()); } void TEPty::setSize(int lines, int cols) { pty()->setWinSize(lines, cols); } void TEPty::setXonXoff(bool on) { pty()->setXonXoff(on); } void TEPty::useUtf8(bool on) { pty()->setUtf8Mode(on); } void TEPty::setErase(char erase) { struct termios tios; int fd = pty()->slaveFd(); if(tcgetattr(fd, &tios)) { tqWarning("[Konsole-TEPty] Uh oh.. can't get terminal attributes.."); return; } tios.c_cc[VERASE] = erase; if(tcsetattr(fd, TCSANOW, &tios)) tqWarning("[Konsole-TEPty] Uh oh.. can't set terminal attributes.."); } /*! start the client program. */ int TEPty::run(const char* _pgm, TQStrList & _args, const char* _term, ulong winid, bool _addutmp, const char* _konsole_dcop, const char* _konsole_dcop_session) { clearArguments(); setBinaryExecutable(_pgm); TQStrListIterator it( _args ); for (; it.current(); ++it ) arguments.append(it.current()); if (_term && _term[0]) setEnvironment("TERM",_term); if (_konsole_dcop && _konsole_dcop[0]) setEnvironment("KONSOLE_DCOP",_konsole_dcop); if (_konsole_dcop_session && _konsole_dcop_session[0]) setEnvironment("KONSOLE_DCOP_SESSION", _konsole_dcop_session); setEnvironment("WINDOWID", TQString::number(winid)); setUsePty(All, _addutmp); if ( start(NotifyOnExit, (Communication) (Stdin | Stdout)) == false ) return -1; resume(); // Start... return 0; } void TEPty::setWriteable(bool writeable) { struct stat sbuf; stat(pty()->ttyName(), &sbuf); if (writeable) chmod(pty()->ttyName(), sbuf.st_mode | S_IWGRP); else chmod(pty()->ttyName(), sbuf.st_mode & ~(S_IWGRP|S_IWOTH)); } /*! Create an instance. */ TEPty::TEPty() { m_bufferFull = false; connect(this, TQT_SIGNAL(receivedStdout(TDEProcess *, char *, int )), this, TQT_SLOT(dataReceived(TDEProcess *,char *, int))); connect(this, TQT_SIGNAL(processExited(TDEProcess *)), this, TQT_SLOT(donePty())); connect(this, TQT_SIGNAL(wroteStdin(TDEProcess *)), this, TQT_SLOT(writeReady())); setUsePty(All, false); // utmp will be overridden later } /*! Destructor. */ TEPty::~TEPty() { } /*! sends a character through the line */ void TEPty::send_byte(char c) { send_bytes(&c,1); } /*! sends a 0 terminated string through the line */ void TEPty::send_string(const char* s) { send_bytes(s,strlen(s)); } void TEPty::writeReady() { pendingSendJobs.remove(pendingSendJobs.begin()); m_bufferFull = false; doSendJobs(); } void TEPty::doSendJobs() { if(pendingSendJobs.isEmpty()) { emit buffer_empty(); return; } SendJob& job = pendingSendJobs.first(); if (!writeStdin(job.buffer.data(), job.length)) { tqWarning("[Konsole-TEPty] Uh oh.. can't write data.."); return; } m_bufferFull = true; } void TEPty::appendSendJob(const char* s, int len) { pendingSendJobs.append(SendJob(s,len)); } /*! sends len bytes through the line */ void TEPty::send_bytes(const char* s, int len) { appendSendJob(s,len); if (!m_bufferFull) doSendJobs(); } /*! indicates that a block of data is received */ void TEPty::dataReceived(TDEProcess *,char *buf, int len) { emit block_in(buf,len); } void TEPty::lockPty(bool lock) { if (lock) suspend(); else resume(); } int TEPty::commSetupDoneC () { int ok = TDEProcess::commSetupDoneC (); if ( ok ) { emit forkedChild(); } return ok; } #include "TEPty.moc"