830 lines
23 KiB
C++
830 lines
23 KiB
C++
/* fhandler_termios.cc
|
|
|
|
This file is part of Cygwin.
|
|
|
|
This software is a copyrighted work licensed under the terms of the
|
|
Cygwin license. Please consult the file "CYGWIN_LICENSE" for
|
|
details. */
|
|
|
|
#include "winsup.h"
|
|
#include <stdlib.h>
|
|
#include <ctype.h>
|
|
#include "cygerrno.h"
|
|
#include "path.h"
|
|
#include "fhandler.h"
|
|
#include "sigproc.h"
|
|
#include "pinfo.h"
|
|
#include "tty.h"
|
|
#include "cygtls.h"
|
|
#include "dtable.h"
|
|
#include "cygheap.h"
|
|
#include "child_info.h"
|
|
#include "ntdll.h"
|
|
|
|
/* Wait time for some treminal mutexes. This is set to 0 when the
|
|
process calls CreateProcess() with DEBUG_PROCESS flag, because
|
|
the debuggie may be suspended while it grabs the mutex. Without
|
|
this, GDB may cause deadlock in console or pty I/O. */
|
|
DWORD NO_COPY mutex_timeout = INFINITE;
|
|
|
|
/* Common functions shared by tty/console */
|
|
|
|
void
|
|
fhandler_termios::tcinit (bool is_pty_master)
|
|
{
|
|
/* Initial termios values */
|
|
|
|
if (is_pty_master || !tc ()->initialized ())
|
|
{
|
|
tc ()->ti.c_iflag = BRKINT | ICRNL | IXON | IUTF8;
|
|
tc ()->ti.c_oflag = OPOST | ONLCR;
|
|
tc ()->ti.c_cflag = B38400 | CS8 | CREAD;
|
|
tc ()->ti.c_lflag = ISIG | ICANON | ECHO | IEXTEN
|
|
| ECHOE | ECHOK | ECHOCTL | ECHOKE;
|
|
|
|
tc ()->ti.c_cc[VDISCARD] = CFLUSH;
|
|
tc ()->ti.c_cc[VEOL] = CEOL;
|
|
tc ()->ti.c_cc[VEOL2] = CEOL2;
|
|
tc ()->ti.c_cc[VEOF] = CEOF;
|
|
tc ()->ti.c_cc[VERASE] = CERASE;
|
|
tc ()->ti.c_cc[VINTR] = CINTR;
|
|
tc ()->ti.c_cc[VKILL] = CKILL;
|
|
tc ()->ti.c_cc[VLNEXT] = CLNEXT;
|
|
tc ()->ti.c_cc[VMIN] = CMIN;
|
|
tc ()->ti.c_cc[VQUIT] = CQUIT;
|
|
tc ()->ti.c_cc[VREPRINT] = CRPRNT;
|
|
tc ()->ti.c_cc[VSTART] = CSTART;
|
|
tc ()->ti.c_cc[VSTOP] = CSTOP;
|
|
tc ()->ti.c_cc[VSUSP] = CSUSP;
|
|
tc ()->ti.c_cc[VSWTC] = CSWTCH;
|
|
tc ()->ti.c_cc[VTIME] = CTIME;
|
|
tc ()->ti.c_cc[VWERASE] = CWERASE;
|
|
|
|
tc ()->ti.c_ispeed = tc ()->ti.c_ospeed = B38400;
|
|
tc ()->pgid = is_pty_master ? 0 : myself->pgid;
|
|
tc ()->initialized (true);
|
|
}
|
|
}
|
|
|
|
int
|
|
fhandler_termios::tcsetpgrp (const pid_t pgid)
|
|
{
|
|
termios_printf ("%s, pgid %d, sid %d, tsid %d", tc ()->ttyname (), pgid,
|
|
myself->sid, tc ()->getsid ());
|
|
if (myself->sid != tc ()->getsid ())
|
|
{
|
|
set_errno (EPERM);
|
|
return -1;
|
|
}
|
|
else if (pgid < 0)
|
|
{
|
|
set_errno (EINVAL);
|
|
return -1;
|
|
}
|
|
int res;
|
|
while (1)
|
|
{
|
|
res = bg_check (-SIGTTOU);
|
|
|
|
switch (res)
|
|
{
|
|
case bg_ok:
|
|
tc ()->setpgid (pgid);
|
|
if (tc ()->is_console && (strace.active () || !being_debugged ()))
|
|
tc ()->kill_pgrp (__SIGSETPGRP);
|
|
res = 0;
|
|
break;
|
|
case bg_signalled:
|
|
if (_my_tls.call_signal_handler ())
|
|
continue;
|
|
set_errno (EINTR);
|
|
fallthrough;
|
|
default:
|
|
res = -1;
|
|
break;
|
|
}
|
|
break;
|
|
}
|
|
return res;
|
|
}
|
|
|
|
int
|
|
fhandler_termios::tcgetpgrp ()
|
|
{
|
|
if (CTTY_IS_VALID (myself->ctty) && myself->ctty == tc ()->ntty)
|
|
return tc ()->pgid;
|
|
set_errno (ENOTTY);
|
|
return -1;
|
|
}
|
|
|
|
int
|
|
fhandler_pty_master::tcgetpgrp ()
|
|
{
|
|
return tc ()->pgid;
|
|
}
|
|
|
|
static inline bool
|
|
is_flush_sig (int sig)
|
|
{
|
|
return sig == SIGINT || sig == SIGQUIT || sig == SIGTSTP;
|
|
}
|
|
|
|
void
|
|
tty_min::kill_pgrp (int sig, pid_t target_pgid)
|
|
{
|
|
target_pgid = target_pgid ?: pgid;
|
|
bool killself = false;
|
|
if (is_flush_sig (sig) && cygheap->ctty)
|
|
cygheap->ctty->sigflush ();
|
|
winpids pids ((DWORD) PID_MAP_RW);
|
|
siginfo_t si = {0};
|
|
si.si_signo = sig;
|
|
si.si_code = SI_KERNEL;
|
|
if (sig > 0 && sig < _NSIG)
|
|
last_sig = sig;
|
|
|
|
for (unsigned i = 0; i < pids.npids; i++)
|
|
{
|
|
_pinfo *p = pids[i];
|
|
if (!p || !p->exists () || p->ctty != ntty || p->pgid != target_pgid)
|
|
continue;
|
|
if (p->process_state & PID_NOTCYGWIN)
|
|
continue; /* Do not send signal to non-cygwin process to prevent
|
|
cmd.exe from crash. */
|
|
if (p == myself)
|
|
killself = sig != __SIGSETPGRP && !exit_state;
|
|
else
|
|
sig_send (p, si);
|
|
}
|
|
if (killself)
|
|
sig_send (myself, si);
|
|
}
|
|
|
|
int
|
|
tty_min::is_orphaned_process_group (int pgid)
|
|
{
|
|
/* An orphaned process group is a process group in which the parent
|
|
of every member is either itself a member of the group or is not
|
|
a member of the group's session. */
|
|
termios_printf ("checking pgid %d, my sid %d, my parent %d", pgid, myself->sid, myself->ppid);
|
|
winpids pids ((DWORD) 0);
|
|
for (unsigned i = 0; i < pids.npids; i++)
|
|
{
|
|
_pinfo *p = pids[i];
|
|
termios_printf ("checking pid %d - has pgid %d\n", p->pid, p->pgid);
|
|
if (!p || !p->exists () || p->pgid != pgid)
|
|
continue;
|
|
pinfo ppid (p->ppid);
|
|
if (!ppid)
|
|
continue;
|
|
termios_printf ("ppid->pgid %d, ppid->sid %d", ppid->pgid, ppid->sid);
|
|
if (ppid->pgid != pgid && ppid->sid == myself->sid)
|
|
return 0;
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
/*
|
|
bg_check: check that this process is either in the foreground process group,
|
|
or if the terminal operation is allowed for processes which are in a
|
|
background process group.
|
|
|
|
If the operation is not permitted by the terminal configuration for processes
|
|
which are a member of a background process group, return an error or raise a
|
|
signal as appropriate.
|
|
|
|
This handles the following terminal operations:
|
|
|
|
write: sig = SIGTTOU
|
|
read: sig = SIGTTIN
|
|
change terminal settings: sig = -SIGTTOU
|
|
(tcsetattr, tcsetpgrp, etc.)
|
|
peek (poll, select): sig = SIGTTIN, dontsignal = TRUE
|
|
*/
|
|
bg_check_types
|
|
fhandler_termios::bg_check (int sig, bool dontsignal)
|
|
{
|
|
/* Ignore errors:
|
|
- this process isn't in a process group
|
|
- tty is invalid
|
|
|
|
Everything is ok if:
|
|
- this process is in the foreground process group, or
|
|
- this tty is not the controlling tty for this process (???), or
|
|
- writing, when TOSTOP TTY mode is not set on this tty
|
|
*/
|
|
if (!myself->pgid || !tc () || tc ()->getpgid () == myself->pgid ||
|
|
myself->ctty != tc ()->ntty ||
|
|
((sig == SIGTTOU) && !(tc ()->ti.c_lflag & TOSTOP)))
|
|
return bg_ok;
|
|
|
|
/* sig -SIGTTOU is used to indicate a change to terminal settings, where
|
|
TOSTOP TTY mode isn't considered when determining if we need to send a
|
|
signal. */
|
|
if (sig < 0)
|
|
sig = -sig;
|
|
|
|
termios_printf ("%s, bg I/O pgid %d, tpgid %d, myctty %s", tc ()->ttyname (),
|
|
myself->pgid, tc ()->getpgid (), myctty ());
|
|
|
|
if (tc ()->getsid () == 0)
|
|
{
|
|
/* The pty has been closed by the master. Return an EOF
|
|
indication. FIXME: There is nothing to stop somebody
|
|
from reallocating this pty. I think this is the case
|
|
which is handled by unlockpt on a Unix system. */
|
|
termios_printf ("closed by master");
|
|
return bg_eof;
|
|
}
|
|
|
|
threadlist_t *tl_entry;
|
|
tl_entry = cygheap->find_tls (_main_tls);
|
|
int sigs_ignored =
|
|
((void *) global_sigs[sig].sa_handler == (void *) SIG_IGN) ||
|
|
(_main_tls->sigmask & SIGTOMASK (sig));
|
|
cygheap->unlock_tls (tl_entry);
|
|
|
|
/* If the process is blocking or ignoring SIGTT*, then signals are not sent
|
|
and background IO is allowed */
|
|
if (sigs_ignored)
|
|
return bg_ok; /* Just allow the IO */
|
|
/* If the process group of the process is orphaned, return EIO */
|
|
else if (tc ()->is_orphaned_process_group (myself->pgid))
|
|
{
|
|
termios_printf ("process group is orphaned");
|
|
set_errno (EIO); /* This is an IO error */
|
|
return bg_error;
|
|
}
|
|
/* Otherwise, if signalling is desired, the signal is sent to all processes in
|
|
the process group */
|
|
else if (!dontsignal)
|
|
{
|
|
/* Don't raise a SIGTT* signal if we have already been
|
|
interrupted by another signal. */
|
|
if (cygwait ((DWORD) 0) != WAIT_SIGNALED)
|
|
{
|
|
siginfo_t si = {0};
|
|
si.si_signo = sig;
|
|
si.si_code = SI_KERNEL;
|
|
kill_pgrp (myself->pgid, si);
|
|
}
|
|
return bg_signalled;
|
|
}
|
|
return bg_ok;
|
|
}
|
|
|
|
#define set_input_done(x) input_done = input_done || (x)
|
|
|
|
int
|
|
fhandler_termios::eat_readahead (int n)
|
|
{
|
|
int oralen = ralen ();
|
|
if (n < 0)
|
|
n = ralen () - raixget ();
|
|
if (n > 0 && ralen () > raixget ())
|
|
{
|
|
if ((int) (ralen () -= n) < (int) raixget ())
|
|
ralen () = raixget ();
|
|
/* If IUTF8 is set, the terminal is in UTF-8 mode. If so, we erase
|
|
a complete UTF-8 multibyte sequence on VERASE/VWERASE. Otherwise,
|
|
if we only erase a single byte, invalid unicode chars are left in
|
|
the input. */
|
|
if (tc ()->ti.c_iflag & IUTF8)
|
|
while (ralen () > raixget () &&
|
|
((unsigned char) rabuf ()[ralen ()] & 0xc0) == 0x80)
|
|
--ralen ();
|
|
}
|
|
oralen = oralen - ralen ();
|
|
if (raixget () >= ralen ())
|
|
raixget () = raixput () = ralen () = 0;
|
|
else if (raixput () > ralen ())
|
|
raixput () = ralen ();
|
|
|
|
return oralen;
|
|
}
|
|
|
|
inline void
|
|
fhandler_termios::echo_erase (int force)
|
|
{
|
|
if (force || tc ()->ti.c_lflag & ECHO)
|
|
doecho ("\b \b", 3);
|
|
}
|
|
|
|
/* The basic policy is as follows:
|
|
- The signal generated by key press will be sent only to cygwin process.
|
|
- For non-cygwin process, CTRL_C_EVENT will be sent on Ctrl-C. */
|
|
/* The inferior of GDB is an exception. GDB does not support to hook signal
|
|
even if the inferior is a cygwin app. As a result, inferior cannot be
|
|
continued after interruption by Ctrl-C if SIGINT was sent. Therefore,
|
|
CTRL_C_EVENT rather than SIGINT is sent to the inferior of GDB. */
|
|
fhandler_termios::process_sig_state
|
|
fhandler_termios::process_sigs (char c, tty* ttyp, fhandler_termios *fh)
|
|
{
|
|
termios &ti = ttyp->ti;
|
|
pid_t pgid = ttyp->pgid;
|
|
|
|
/* The name *_nat stands for 'native' which means non-cygwin apps. */
|
|
bool ctrl_c_event_sent = false;
|
|
bool need_discard_input = false;
|
|
bool pg_with_nat = false; /* The process group has non-cygwin processes. */
|
|
bool need_send_sig = false; /* There is process which need the signal. */
|
|
bool nat_shell = false; /* The shell seems to be a non-cygwin process. */
|
|
bool cyg_reader = false; /* Cygwin process is reading the tty. */
|
|
bool with_debugger = false; /* GDB is debugging cygwin app. */
|
|
bool with_debugger_nat = false; /* GDB is debugging non-cygwin app. */
|
|
|
|
winpids pids ((DWORD) 0);
|
|
for (unsigned i = 0; i < pids.npids; i++)
|
|
{
|
|
_pinfo *p = pids[i];
|
|
/* PID_NOTCYGWIN: check this for non-cygwin process.
|
|
exec_dwProcessId == dwProcessId:
|
|
check this for GDB with non-cygwin inferior in pty
|
|
without pcon enabled. In this case, the inferior is not
|
|
cygwin process list. This condition is set true as
|
|
a marker for GDB with non-cygwin inferior in pty code.
|
|
!PID_CYGPARENT: check this for GDB with cygwin inferior or
|
|
cygwin apps started from non-cygwin shell. */
|
|
if (c == '\003' && p && p->ctty == ttyp->ntty && p->pgid == pgid
|
|
&& ((p->process_state & PID_NOTCYGWIN)
|
|
|| ((p->exec_dwProcessId == p->dwProcessId)
|
|
&& ttyp->pty_input_state_eq (tty::to_nat))
|
|
|| !(p->process_state & PID_CYGPARENT)))
|
|
{
|
|
/* Ctrl-C event will be sent only to the processes attaching
|
|
to the same console. Therefore, attach to the console to
|
|
which the target process is attaching before sending the
|
|
CTRL_C_EVENT. After sending the event, reattach to the
|
|
console to which the process was previously attached. */
|
|
DWORD resume_pid = 0;
|
|
if (fh && !fh->is_console ())
|
|
resume_pid =
|
|
fhandler_pty_common::attach_console_temporarily (p->dwProcessId);
|
|
if (fh && p == myself && being_debugged ())
|
|
{ /* Avoid deadlock in gdb on console. */
|
|
fh->tcflush(TCIFLUSH);
|
|
fh->release_input_mutex_if_necessary ();
|
|
}
|
|
/* CTRL_C_EVENT does not work for the process started with
|
|
CREATE_NEW_PROCESS_GROUP flag, so send CTRL_BREAK_EVENT
|
|
instead. */
|
|
if (p->process_state & PID_NEW_PG)
|
|
GenerateConsoleCtrlEvent (CTRL_BREAK_EVENT, p->dwProcessId);
|
|
else if ((!fh || fh->need_send_ctrl_c_event ()
|
|
|| p->exec_dwProcessId == p->dwProcessId)
|
|
&& !ctrl_c_event_sent)
|
|
{
|
|
GenerateConsoleCtrlEvent (CTRL_C_EVENT, 0);
|
|
ctrl_c_event_sent = true;
|
|
}
|
|
if (fh && !fh->is_console ())
|
|
{
|
|
/* If a process on pseudo console is killed by Ctrl-C,
|
|
this process may take over the ownership of the
|
|
pseudo console because this process attached to it
|
|
before sending CTRL_C_EVENT. In this case, closing
|
|
pseudo console is necessary. */
|
|
fhandler_pty_slave::release_ownership_of_nat_pipe (ttyp, fh);
|
|
fhandler_pty_common::resume_from_temporarily_attach (resume_pid);
|
|
}
|
|
need_discard_input = true;
|
|
}
|
|
if (p && p->ctty == ttyp->ntty && p->pgid == pgid)
|
|
{
|
|
if (p->process_state & PID_NOTCYGWIN)
|
|
pg_with_nat = true; /* The process group has non-cygwin process */
|
|
if (!(p->process_state & PID_NOTCYGWIN))
|
|
need_send_sig = true; /* Process which needs signal exists */
|
|
if (!p->cygstarted)
|
|
nat_shell = true; /* The shell seems to a non-cygwin shell */
|
|
if (p->process_state & PID_TTYIN)
|
|
cyg_reader = true; /* Theh process is reading the tty */
|
|
if (!p->cygstarted && !(p->process_state & PID_NOTCYGWIN)
|
|
&& (p->process_state & PID_DEBUGGED))
|
|
with_debugger = true; /* inferior is cygwin app */
|
|
if (!(p->process_state & PID_NOTCYGWIN)
|
|
&& (p->exec_dwProcessId == p->dwProcessId) /* Check marker */
|
|
&& ttyp->pty_input_state_eq (tty::to_nat)
|
|
&& p->pid == pgid)
|
|
with_debugger_nat = true; /* inferior is non-cygwin app */
|
|
}
|
|
}
|
|
if ((with_debugger || with_debugger_nat) && need_discard_input)
|
|
{
|
|
if (!(ti.c_lflag & NOFLSH) && fh)
|
|
{
|
|
fh->eat_readahead (-1);
|
|
fh->discard_input ();
|
|
}
|
|
ti.c_lflag &= ~FLUSHO;
|
|
return done_with_debugger;
|
|
}
|
|
if (with_debugger_nat)
|
|
return not_signalled; /* Do not process slgnal keys further.
|
|
The non-cygwin inferior of GDB cannot receive
|
|
the signals. */
|
|
/* Send SIGQUIT to non-cygwin process. Non-cygwin app will not be alerted
|
|
by kill_pgrp(), however, QUIT key should quit the non-cygwin app
|
|
if it is started along with cygwin process from cygwin shell. */
|
|
if ((ti.c_lflag & ISIG) && CCEQ (ti.c_cc[VQUIT], c)
|
|
&& pg_with_nat && need_send_sig && !nat_shell)
|
|
{
|
|
for (unsigned i = 0; i < pids.npids; i++)
|
|
{
|
|
_pinfo *p = pids[i];
|
|
if (p && p->ctty == ttyp->ntty && p->pgid == pgid
|
|
&& (p->process_state & PID_NOTCYGWIN))
|
|
sig_send (p, SIGQUIT);
|
|
}
|
|
}
|
|
if ((ti.c_lflag & ISIG) && need_send_sig)
|
|
{
|
|
int sig;
|
|
if (CCEQ (ti.c_cc[VINTR], c))
|
|
sig = SIGINT;
|
|
else if (CCEQ (ti.c_cc[VQUIT], c))
|
|
sig = SIGQUIT;
|
|
else if (pg_with_nat) /* If the process group has a non-cygwin process,
|
|
it cannot be suspended correctly. Therefore,
|
|
do not send SIGTSTP. */
|
|
goto not_a_sig;
|
|
else if (CCEQ (ti.c_cc[VSUSP], c))
|
|
sig = SIGTSTP;
|
|
else
|
|
goto not_a_sig;
|
|
|
|
termios_printf ("got interrupt %d, sending signal %d", c, sig);
|
|
if (!(ti.c_lflag & NOFLSH) && fh)
|
|
{
|
|
fh->eat_readahead (-1);
|
|
fh->discard_input ();
|
|
}
|
|
if (fh)
|
|
fh->release_input_mutex_if_necessary ();
|
|
ttyp->kill_pgrp (sig, pgid);
|
|
if (fh)
|
|
fh->acquire_input_mutex_if_necessary (mutex_timeout);
|
|
ti.c_lflag &= ~FLUSHO;
|
|
return signalled;
|
|
}
|
|
not_a_sig:
|
|
if ((ti.c_lflag & ISIG) && need_discard_input)
|
|
{
|
|
if (!(ti.c_lflag & NOFLSH) && fh)
|
|
{
|
|
fh->eat_readahead (-1);
|
|
fh->discard_input ();
|
|
}
|
|
ti.c_lflag &= ~FLUSHO;
|
|
return not_signalled_but_done;
|
|
}
|
|
bool to_nat = !cyg_reader && pg_with_nat;
|
|
return to_nat ? not_signalled_with_nat_reader : not_signalled;
|
|
}
|
|
|
|
bool
|
|
fhandler_termios::process_stop_start (char c, tty *ttyp)
|
|
{
|
|
termios &ti = ttyp->ti;
|
|
if (ti.c_iflag & IXON)
|
|
{
|
|
if (CCEQ (ti.c_cc[VSTOP], c))
|
|
{
|
|
ttyp->output_stopped = true;
|
|
return true;
|
|
}
|
|
else if (CCEQ (ti.c_cc[VSTART], c))
|
|
{
|
|
restart_output:
|
|
ttyp->output_stopped = false;
|
|
return true;
|
|
}
|
|
else if ((ti.c_iflag & IXANY) && ttyp->output_stopped)
|
|
goto restart_output;
|
|
}
|
|
if ((ti.c_lflag & ICANON) && (ti.c_lflag & IEXTEN)
|
|
&& CCEQ (ti.c_cc[VDISCARD], c))
|
|
{
|
|
ti.c_lflag ^= FLUSHO;
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
line_edit_status
|
|
fhandler_termios::line_edit (const char *rptr, size_t nread, termios& ti,
|
|
ssize_t *bytes_read)
|
|
{
|
|
line_edit_status ret = line_edit_ok;
|
|
char c;
|
|
int input_done = 0;
|
|
bool sawsig = false;
|
|
int iscanon = ti.c_lflag & ICANON;
|
|
size_t read_cnt = 0;
|
|
|
|
while (read_cnt < nread)
|
|
{
|
|
c = *rptr++;
|
|
read_cnt++;
|
|
|
|
paranoid_printf ("char %0o", c);
|
|
|
|
if (ti.c_iflag & ISTRIP)
|
|
c &= 0x7f;
|
|
bool disable_eof_key = false;
|
|
switch (process_sigs (c, get_ttyp (), this))
|
|
{
|
|
case signalled:
|
|
sawsig = true;
|
|
fallthrough;
|
|
case not_signalled_but_done:
|
|
case done_with_debugger:
|
|
get_ttyp ()->output_stopped = false;
|
|
continue;
|
|
case not_signalled_with_nat_reader:
|
|
disable_eof_key = true;
|
|
break;
|
|
default: /* Not signalled */
|
|
break;
|
|
}
|
|
if (process_stop_start (c, get_ttyp ()))
|
|
continue;
|
|
/* Check for special chars */
|
|
if (c == '\r')
|
|
{
|
|
if (ti.c_iflag & IGNCR)
|
|
continue;
|
|
if (ti.c_iflag & ICRNL)
|
|
{
|
|
c = '\n';
|
|
set_input_done (iscanon);
|
|
}
|
|
}
|
|
else if (c == '\n')
|
|
{
|
|
if (ti.c_iflag & INLCR)
|
|
c = '\r';
|
|
else
|
|
set_input_done (iscanon);
|
|
}
|
|
|
|
if (!iscanon)
|
|
/* nothing */;
|
|
else if (CCEQ (ti.c_cc[VERASE], c))
|
|
{
|
|
if (eat_readahead (1))
|
|
echo_erase ();
|
|
continue;
|
|
}
|
|
else if (CCEQ (ti.c_cc[VWERASE], c))
|
|
{
|
|
int ch;
|
|
do
|
|
if (!eat_readahead (1))
|
|
break;
|
|
else
|
|
echo_erase ();
|
|
while ((ch = peek_readahead (1)) >= 0 && !isspace (ch));
|
|
continue;
|
|
}
|
|
else if (CCEQ (ti.c_cc[VKILL], c))
|
|
{
|
|
int nchars = eat_readahead (-1);
|
|
if (ti.c_lflag & ECHO)
|
|
while (nchars--)
|
|
echo_erase (1);
|
|
continue;
|
|
}
|
|
else if (CCEQ (ti.c_cc[VREPRINT], c))
|
|
{
|
|
if (ti.c_lflag & ECHO)
|
|
{
|
|
doecho ("\n\r", 2);
|
|
doecho (rabuf (), ralen ());
|
|
}
|
|
continue;
|
|
}
|
|
else if (CCEQ (ti.c_cc[VEOF], c) && !disable_eof_key)
|
|
{
|
|
termios_printf ("EOF");
|
|
accept_input ();
|
|
ret = line_edit_input_done;
|
|
continue;
|
|
}
|
|
else if (CCEQ (ti.c_cc[VEOL], c) ||
|
|
CCEQ (ti.c_cc[VEOL2], c) ||
|
|
c == '\n')
|
|
{
|
|
set_input_done (1);
|
|
termios_printf ("EOL");
|
|
}
|
|
|
|
if (ti.c_iflag & IUCLC && isupper (c))
|
|
c = cyg_tolower (c);
|
|
|
|
put_readahead (c);
|
|
if (ti.c_lflag & ECHO)
|
|
doecho (&c, 1);
|
|
/* Write in chunks of 32 bytes to reduce the number of WriteFile calls
|
|
in non-canonical mode. */
|
|
if ((!iscanon && ralen () >= 32) || input_done)
|
|
{
|
|
int status = accept_input ();
|
|
if (status != 1)
|
|
{
|
|
ret = status ? line_edit_error : line_edit_pipe_full;
|
|
break;
|
|
}
|
|
ret = line_edit_input_done;
|
|
input_done = 0;
|
|
}
|
|
}
|
|
|
|
/* If we didn't write all bytes in non-canonical mode, write them now. */
|
|
if ((input_done || !iscanon) && ralen () > 0 && ret != line_edit_error)
|
|
{
|
|
int status;
|
|
int retry_count = 3;
|
|
while ((status = accept_input ()) != 1 &&
|
|
ralen () > 0 && --retry_count > 0)
|
|
cygwait ((DWORD) 10);
|
|
if (status != 1)
|
|
ret = status ? line_edit_error : line_edit_pipe_full;
|
|
else
|
|
ret = line_edit_input_done;
|
|
}
|
|
|
|
if (bytes_read)
|
|
*bytes_read = read_cnt;
|
|
|
|
if (sawsig)
|
|
ret = line_edit_signalled;
|
|
|
|
return ret;
|
|
}
|
|
|
|
off_t
|
|
fhandler_termios::lseek (off_t, int)
|
|
{
|
|
set_errno (ESPIPE);
|
|
return -1;
|
|
}
|
|
|
|
void
|
|
fhandler_termios::sigflush ()
|
|
{
|
|
/* FIXME: Checking get_ttyp() for NULL is not right since it should not
|
|
be NULL while this is alive. However, we can conceivably close a
|
|
ctty while exiting and that will zero this. */
|
|
if ((!have_execed || have_execed_cygwin) && tc ()
|
|
&& (tc ()->getpgid () == myself->pgid)
|
|
&& !(tc ()->ti.c_lflag & NOFLSH))
|
|
tcflush (TCIFLUSH);
|
|
}
|
|
|
|
pid_t
|
|
fhandler_termios::tcgetsid ()
|
|
{
|
|
if (CTTY_IS_VALID (myself->ctty) && myself->ctty == tc ()->ntty)
|
|
return tc ()->getsid ();
|
|
set_errno (ENOTTY);
|
|
return -1;
|
|
}
|
|
|
|
int
|
|
fhandler_termios::fstat (struct stat *buf)
|
|
{
|
|
fhandler_base::fstat (buf);
|
|
if (dev_referred_via > 0)
|
|
buf->st_rdev = dev_referred_via;
|
|
return 0;
|
|
}
|
|
|
|
static bool
|
|
is_console_app (const WCHAR *filename)
|
|
{
|
|
HANDLE h;
|
|
h = CreateFileW (filename, GENERIC_READ, FILE_SHARE_READ,
|
|
NULL, OPEN_EXISTING, 0, NULL);
|
|
char buf[1024];
|
|
DWORD n;
|
|
ReadFile (h, buf, sizeof (buf), &n, 0);
|
|
CloseHandle (h);
|
|
/* The offset of Subsystem is the same for both IMAGE_NT_HEADERS32 and
|
|
IMAGE_NT_HEADERS64, so only IMAGE_NT_HEADERS32 is used here. */
|
|
IMAGE_NT_HEADERS32 *p = (IMAGE_NT_HEADERS32 *) memmem (buf, n, "PE\0\0", 4);
|
|
if (p && (char *) &p->OptionalHeader.DllCharacteristics <= buf + n)
|
|
return p->OptionalHeader.Subsystem == IMAGE_SUBSYSTEM_WINDOWS_CUI;
|
|
wchar_t *e = wcsrchr (filename, L'.');
|
|
if (e && (wcscasecmp (e, L".bat") == 0 || wcscasecmp (e, L".cmd") == 0))
|
|
return true;
|
|
return false;
|
|
}
|
|
|
|
int
|
|
fhandler_termios::ioctl (int cmd, void *varg)
|
|
{
|
|
if (cmd != TIOCSCTTY)
|
|
return 1; /* Not handled by this function */
|
|
|
|
int arg = (int) (intptr_t) varg;
|
|
|
|
if (arg != 0 && arg != 1)
|
|
{
|
|
set_errno (EINVAL);
|
|
return -1;
|
|
}
|
|
|
|
termios_printf ("myself->ctty %d, myself->sid %d, myself->pid %d, arg %d, tc()->getsid () %d\n",
|
|
myself->ctty, myself->sid, myself->pid, arg, tc ()->getsid ());
|
|
if (CTTY_IS_VALID (myself->ctty) || myself->sid != myself->pid
|
|
|| (!arg && tc ()->getsid () > 0))
|
|
{
|
|
set_errno (EPERM);
|
|
return -1;
|
|
}
|
|
|
|
if (!myself->set_ctty (this, 0))
|
|
{
|
|
set_errno (EPERM);
|
|
return -1;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
void
|
|
fhandler_termios::spawn_worker::setup (bool iscygwin, HANDLE h_stdin,
|
|
const WCHAR *runpath, bool nopcon,
|
|
bool reset_sendsig,
|
|
const WCHAR *envblock)
|
|
{
|
|
fhandler_pty_slave *ptys_primary = NULL;
|
|
fhandler_console *cons_native = NULL;
|
|
|
|
for (int i = 0; i < 3; i ++)
|
|
{
|
|
const int chk_order[] = {1, 0, 2};
|
|
int fd = chk_order[i];
|
|
fhandler_base *fh = ::cygheap->fdtab[fd];
|
|
if (fh && fh->get_major () == DEV_PTYS_MAJOR && ptys_primary == NULL)
|
|
ptys_primary = (fhandler_pty_slave *) fh;
|
|
else if (fh && fh->get_major () == DEV_CONS_MAJOR
|
|
&& !iscygwin && cons_native == NULL)
|
|
cons_native = (fhandler_console *) fh;
|
|
}
|
|
if (cons_native)
|
|
{
|
|
cons_native->setup_for_non_cygwin_app ();
|
|
/* Console handles will be already closed by close_all_files()
|
|
when cleaning up, therefore, duplicate them here. */
|
|
cons_native->get_duplicated_handle_set (&cons_handle_set);
|
|
cons_need_cleanup = true;
|
|
}
|
|
if (!iscygwin)
|
|
{
|
|
int fd;
|
|
cygheap_fdenum cfd (false);
|
|
while ((fd = cfd.next ()) >= 0)
|
|
if (cfd->get_major () == DEV_PTYS_MAJOR)
|
|
{
|
|
fhandler_pty_slave *ptys
|
|
= (fhandler_pty_slave *)(fhandler_base *) cfd;
|
|
ptys->create_invisible_console ();
|
|
ptys->setup_locale ();
|
|
}
|
|
}
|
|
if (!iscygwin && ptys_primary && is_console_app (runpath))
|
|
{
|
|
if (h_stdin == ptys_primary->get_handle_nat ())
|
|
stdin_is_ptys = true;
|
|
if (reset_sendsig)
|
|
myself->sendsig = myself->exec_sendsig;
|
|
ptys_primary->setup_for_non_cygwin_app (nopcon, envblock, stdin_is_ptys);
|
|
if (reset_sendsig)
|
|
myself->sendsig = NULL;
|
|
ptys_primary->get_duplicated_handle_set (&ptys_handle_set);
|
|
ptys_ttyp = (tty *) ptys_primary->tc ();
|
|
ptys_need_cleanup = true;
|
|
}
|
|
}
|
|
|
|
void
|
|
fhandler_termios::spawn_worker::cleanup ()
|
|
{
|
|
if (ptys_need_cleanup)
|
|
fhandler_pty_slave::cleanup_for_non_cygwin_app (&ptys_handle_set,
|
|
ptys_ttyp, stdin_is_ptys);
|
|
if (cons_need_cleanup)
|
|
fhandler_console::cleanup_for_non_cygwin_app (&cons_handle_set);
|
|
close_handle_set ();
|
|
}
|
|
|
|
void
|
|
fhandler_termios::spawn_worker::close_handle_set ()
|
|
{
|
|
if (ptys_need_cleanup)
|
|
fhandler_pty_slave::close_handle_set (&ptys_handle_set);
|
|
if (cons_need_cleanup)
|
|
fhandler_console::close_handle_set (&cons_handle_set);
|
|
}
|