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.
tdebase/konsole/konsole/TEPty.cpp

263 lines
6.3 KiB

/*
This file is part of Konsole, an X terminal.
Copyright (C) 1997,1998 by Lars Doelle <lars.doelle@on-line.de>
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 <config.h>
#endif
#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>
#include <errno.h>
#include <termios.h>
#include <kstandarddirs.h>
#include <klocale.h>
#include <kdebug.h>
#include <kpty.h>
#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("Uh oh.. can't get terminal attributes..");
return;
}
tios.c_cc[VERASE] = erase;
if(tcsetattr(fd, TCSANOW, &tios))
tqWarning("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(KProcess *, char *, int )),
this, TQT_SLOT(dataReceived(KProcess *,char *, int)));
connect(this, TQT_SIGNAL(processExited(KProcess *)),
this, TQT_SLOT(donePty()));
connect(this, TQT_SIGNAL(wroteStdin(KProcess *)),
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("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(KProcess *,char *buf, int len)
{
emit block_in(buf,len);
}
void TEPty::lockPty(bool lock)
{
if (lock)
suspend();
else
resume();
}
int TEPty::commSetupDoneC ()
{
int ok = KProcess::commSetupDoneC ();
if ( ok ) {
emit forkedChild();
}
return ok;
}
#include "TEPty.moc"