7aa4a7e25f
children happens to exit while we are waiting for another child to exit. This can happen with code like sh -c ' sleep 5 & exec sh -c "sleep 10 & wait !$" ' when the inner "sh" is waiting for the 10 second sleep to be done, the 5 second sleep started earlier terminates. It is a child of our process, as the inner shell is the same process as the outer one, but not a known child (the inner shell has no idea what the outer one did before it started). This was observed in the wild by Martijn Dekker (where the outer shell was bash but that's irrelevant). XXX pullup -9
1830 lines
40 KiB
C
1830 lines
40 KiB
C
/* $NetBSD: jobs.c,v 1.107 2020/02/07 02:06:12 kre Exp $ */
|
|
|
|
/*-
|
|
* Copyright (c) 1991, 1993
|
|
* The Regents of the University of California. All rights reserved.
|
|
*
|
|
* This code is derived from software contributed to Berkeley by
|
|
* Kenneth Almquist.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions
|
|
* are met:
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
* documentation and/or other materials provided with the distribution.
|
|
* 3. Neither the name of the University nor the names of its contributors
|
|
* may be used to endorse or promote products derived from this software
|
|
* without specific prior written permission.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
|
|
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
* ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
|
|
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
|
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
|
|
* OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
|
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
|
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
|
|
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
|
* SUCH DAMAGE.
|
|
*/
|
|
|
|
#include <sys/cdefs.h>
|
|
#ifndef lint
|
|
#if 0
|
|
static char sccsid[] = "@(#)jobs.c 8.5 (Berkeley) 5/4/95";
|
|
#else
|
|
__RCSID("$NetBSD: jobs.c,v 1.107 2020/02/07 02:06:12 kre Exp $");
|
|
#endif
|
|
#endif /* not lint */
|
|
|
|
#include <stdio.h>
|
|
#include <fcntl.h>
|
|
#include <signal.h>
|
|
#include <errno.h>
|
|
#include <unistd.h>
|
|
#include <stdlib.h>
|
|
#include <paths.h>
|
|
#include <sys/types.h>
|
|
#include <sys/param.h>
|
|
#ifdef BSD
|
|
#include <sys/wait.h>
|
|
#include <sys/time.h>
|
|
#include <sys/resource.h>
|
|
#endif
|
|
#include <sys/ioctl.h>
|
|
|
|
#include "shell.h"
|
|
#if JOBS
|
|
#if OLD_TTY_DRIVER
|
|
#include "sgtty.h"
|
|
#else
|
|
#include <termios.h>
|
|
#endif
|
|
#undef CEOF /* syntax.h redefines this */
|
|
#endif
|
|
#include "redir.h"
|
|
#include "show.h"
|
|
#include "main.h"
|
|
#include "parser.h"
|
|
#include "nodes.h"
|
|
#include "jobs.h"
|
|
#include "var.h"
|
|
#include "options.h"
|
|
#include "builtins.h"
|
|
#include "trap.h"
|
|
#include "syntax.h"
|
|
#include "input.h"
|
|
#include "output.h"
|
|
#include "memalloc.h"
|
|
#include "error.h"
|
|
#include "mystring.h"
|
|
|
|
|
|
#ifndef WCONTINUED
|
|
#define WCONTINUED 0 /* So we can compile on old systems */
|
|
#endif
|
|
#ifndef WIFCONTINUED
|
|
#define WIFCONTINUED(x) (0) /* ditto */
|
|
#endif
|
|
|
|
|
|
static struct job *jobtab; /* array of jobs */
|
|
static int njobs; /* size of array */
|
|
static int jobs_invalid; /* set in child */
|
|
MKINIT pid_t backgndpid = -1; /* pid of last background process */
|
|
#if JOBS
|
|
int initialpgrp; /* pgrp of shell on invocation */
|
|
static int curjob = -1; /* current job */
|
|
#endif
|
|
static int ttyfd = -1;
|
|
|
|
STATIC void restartjob(struct job *);
|
|
STATIC void freejob(struct job *);
|
|
STATIC struct job *getjob(const char *, int);
|
|
STATIC int dowait(int, struct job *, struct job **);
|
|
#define WBLOCK 1
|
|
#define WNOFREE 2
|
|
#define WSILENT 4
|
|
STATIC int jobstatus(const struct job *, int);
|
|
STATIC int waitproc(int, struct job *, int *);
|
|
STATIC void cmdtxt(union node *);
|
|
STATIC void cmdlist(union node *, int);
|
|
STATIC void cmdputs(const char *);
|
|
inline static void cmdputi(int);
|
|
|
|
#ifdef SYSV
|
|
STATIC int onsigchild(void);
|
|
#endif
|
|
|
|
#ifdef OLD_TTY_DRIVER
|
|
static pid_t tcgetpgrp(int fd);
|
|
static int tcsetpgrp(int fd, pid_t pgrp);
|
|
|
|
static pid_t
|
|
tcgetpgrp(int fd)
|
|
{
|
|
pid_t pgrp;
|
|
if (ioctl(fd, TIOCGPGRP, (char *)&pgrp) == -1)
|
|
return -1;
|
|
else
|
|
return pgrp;
|
|
}
|
|
|
|
static int
|
|
tcsetpgrp(int fd, pid_tpgrp)
|
|
{
|
|
return ioctl(fd, TIOCSPGRP, (char *)&pgrp);
|
|
}
|
|
#endif
|
|
|
|
static void
|
|
ttyfd_change(int from, int to)
|
|
{
|
|
if (ttyfd == from)
|
|
ttyfd = to;
|
|
}
|
|
|
|
/*
|
|
* Turn job control on and off.
|
|
*
|
|
* Note: This code assumes that the third arg to ioctl is a character
|
|
* pointer, which is true on Berkeley systems but not System V. Since
|
|
* System V doesn't have job control yet, this isn't a problem now.
|
|
*/
|
|
|
|
MKINIT int jobctl;
|
|
|
|
void
|
|
setjobctl(int on)
|
|
{
|
|
#ifdef OLD_TTY_DRIVER
|
|
int ldisc;
|
|
#endif
|
|
|
|
if (on == jobctl || rootshell == 0)
|
|
return;
|
|
if (on) {
|
|
#if defined(FIOCLEX) || defined(FD_CLOEXEC)
|
|
int i;
|
|
|
|
if (ttyfd != -1)
|
|
sh_close(ttyfd);
|
|
if ((ttyfd = open("/dev/tty", O_RDWR)) == -1) {
|
|
for (i = 0; i < 3; i++) {
|
|
if (isatty(i) && (ttyfd = dup(i)) != -1)
|
|
break;
|
|
}
|
|
if (i == 3)
|
|
goto out;
|
|
}
|
|
ttyfd = to_upper_fd(ttyfd); /* Move to a high fd */
|
|
register_sh_fd(ttyfd, ttyfd_change);
|
|
#else
|
|
out2str("sh: Need FIOCLEX or FD_CLOEXEC to support job control");
|
|
goto out;
|
|
#endif
|
|
do { /* while we are in the background */
|
|
if ((initialpgrp = tcgetpgrp(ttyfd)) < 0) {
|
|
out:
|
|
out2str("sh: can't access tty; job control turned off\n");
|
|
mflag = 0;
|
|
return;
|
|
}
|
|
if (initialpgrp == -1)
|
|
initialpgrp = getpgrp();
|
|
else if (initialpgrp != getpgrp()) {
|
|
killpg(0, SIGTTIN);
|
|
continue;
|
|
}
|
|
} while (0);
|
|
|
|
#ifdef OLD_TTY_DRIVER
|
|
if (ioctl(ttyfd, TIOCGETD, (char *)&ldisc) < 0
|
|
|| ldisc != NTTYDISC) {
|
|
out2str("sh: need new tty driver to run job control; job control turned off\n");
|
|
mflag = 0;
|
|
return;
|
|
}
|
|
#endif
|
|
setsignal(SIGTSTP, 0);
|
|
setsignal(SIGTTOU, 0);
|
|
setsignal(SIGTTIN, 0);
|
|
if (getpgrp() != rootpid && setpgid(0, rootpid) == -1)
|
|
error("Cannot set process group (%s) at %d",
|
|
strerror(errno), __LINE__);
|
|
if (tcsetpgrp(ttyfd, rootpid) == -1)
|
|
error("Cannot set tty process group (%s) at %d",
|
|
strerror(errno), __LINE__);
|
|
} else { /* turning job control off */
|
|
if (getpgrp() != initialpgrp && setpgid(0, initialpgrp) == -1)
|
|
error("Cannot set process group (%s) at %d",
|
|
strerror(errno), __LINE__);
|
|
if (tcsetpgrp(ttyfd, initialpgrp) == -1)
|
|
error("Cannot set tty process group (%s) at %d",
|
|
strerror(errno), __LINE__);
|
|
sh_close(ttyfd);
|
|
ttyfd = -1;
|
|
setsignal(SIGTSTP, 0);
|
|
setsignal(SIGTTOU, 0);
|
|
setsignal(SIGTTIN, 0);
|
|
}
|
|
jobctl = on;
|
|
}
|
|
|
|
|
|
#ifdef mkinit
|
|
INCLUDE <stdlib.h>
|
|
|
|
SHELLPROC {
|
|
backgndpid = -1;
|
|
#if JOBS
|
|
jobctl = 0;
|
|
#endif
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if JOBS
|
|
static int
|
|
do_fgcmd(const char *arg_ptr)
|
|
{
|
|
struct job *jp;
|
|
int i;
|
|
int status;
|
|
|
|
if (jobs_invalid)
|
|
error("No current jobs");
|
|
jp = getjob(arg_ptr, 0);
|
|
if (jp->jobctl == 0)
|
|
error("job not created under job control");
|
|
out1fmt("%s", jp->ps[0].cmd);
|
|
for (i = 1; i < jp->nprocs; i++)
|
|
out1fmt(" | %s", jp->ps[i].cmd );
|
|
out1c('\n');
|
|
flushall();
|
|
|
|
for (i = 0; i < jp->nprocs; i++)
|
|
if (tcsetpgrp(ttyfd, jp->ps[i].pid) != -1)
|
|
break;
|
|
|
|
if (i >= jp->nprocs) {
|
|
error("Cannot set tty process group (%s) at %d",
|
|
strerror(errno), __LINE__);
|
|
}
|
|
INTOFF;
|
|
restartjob(jp);
|
|
status = waitforjob(jp);
|
|
INTON;
|
|
return status;
|
|
}
|
|
|
|
int
|
|
fgcmd(int argc, char **argv)
|
|
{
|
|
nextopt("");
|
|
return do_fgcmd(*argptr);
|
|
}
|
|
|
|
int
|
|
fgcmd_percent(int argc, char **argv)
|
|
{
|
|
nextopt("");
|
|
return do_fgcmd(*argv);
|
|
}
|
|
|
|
static void
|
|
set_curjob(struct job *jp, int mode)
|
|
{
|
|
struct job *jp1, *jp2;
|
|
int i, ji;
|
|
|
|
ji = jp - jobtab;
|
|
|
|
/* first remove from list */
|
|
if (ji == curjob)
|
|
curjob = jp->prev_job;
|
|
else {
|
|
for (i = 0; i < njobs; i++) {
|
|
if (jobtab[i].prev_job != ji)
|
|
continue;
|
|
jobtab[i].prev_job = jp->prev_job;
|
|
break;
|
|
}
|
|
}
|
|
|
|
/* Then re-insert in correct position */
|
|
switch (mode) {
|
|
case 0: /* job being deleted */
|
|
jp->prev_job = -1;
|
|
break;
|
|
case 1: /* newly created job or backgrounded job,
|
|
put after all stopped jobs. */
|
|
if (curjob != -1 && jobtab[curjob].state == JOBSTOPPED) {
|
|
for (jp1 = jobtab + curjob; ; jp1 = jp2) {
|
|
if (jp1->prev_job == -1)
|
|
break;
|
|
jp2 = jobtab + jp1->prev_job;
|
|
if (jp2->state != JOBSTOPPED)
|
|
break;
|
|
}
|
|
jp->prev_job = jp1->prev_job;
|
|
jp1->prev_job = ji;
|
|
break;
|
|
}
|
|
/* FALLTHROUGH */
|
|
case 2: /* newly stopped job - becomes curjob */
|
|
jp->prev_job = curjob;
|
|
curjob = ji;
|
|
break;
|
|
}
|
|
}
|
|
|
|
int
|
|
bgcmd(int argc, char **argv)
|
|
{
|
|
struct job *jp;
|
|
int i;
|
|
|
|
nextopt("");
|
|
if (jobs_invalid)
|
|
error("No current jobs");
|
|
do {
|
|
jp = getjob(*argptr, 0);
|
|
if (jp->jobctl == 0)
|
|
error("job not created under job control");
|
|
set_curjob(jp, 1);
|
|
out1fmt("[%ld] %s", (long)(jp - jobtab + 1), jp->ps[0].cmd);
|
|
for (i = 1; i < jp->nprocs; i++)
|
|
out1fmt(" | %s", jp->ps[i].cmd );
|
|
out1c('\n');
|
|
flushall();
|
|
restartjob(jp);
|
|
} while (*argptr && *++argptr);
|
|
return 0;
|
|
}
|
|
|
|
|
|
STATIC void
|
|
restartjob(struct job *jp)
|
|
{
|
|
struct procstat *ps;
|
|
int i, e;
|
|
|
|
if (jp->state == JOBDONE)
|
|
return;
|
|
INTOFF;
|
|
for (e = i = 0; i < jp->nprocs; i++) {
|
|
if (killpg(jp->ps[i].pid, SIGCONT) != -1)
|
|
break;
|
|
if (e == 0 && errno != ESRCH)
|
|
e = errno;
|
|
}
|
|
if (i >= jp->nprocs)
|
|
error("Cannot continue job (%s)", strerror(e ? e : ESRCH));
|
|
for (ps = jp->ps, i = jp->nprocs ; --i >= 0 ; ps++) {
|
|
if (WIFSTOPPED(ps->status)) {
|
|
VTRACE(DBG_JOBS, (
|
|
"restartjob: [%zu] pid %d status change"
|
|
" from %#x (stopped) to -1 (running)\n",
|
|
(size_t)(jp-jobtab+1), ps->pid, ps->status));
|
|
ps->status = -1;
|
|
jp->state = JOBRUNNING;
|
|
}
|
|
}
|
|
INTON;
|
|
}
|
|
#endif
|
|
|
|
inline static void
|
|
cmdputi(int n)
|
|
{
|
|
char str[20];
|
|
|
|
fmtstr(str, sizeof str, "%d", n);
|
|
cmdputs(str);
|
|
}
|
|
|
|
static void
|
|
showjob(struct output *out, struct job *jp, int mode)
|
|
{
|
|
int procno;
|
|
int st;
|
|
struct procstat *ps;
|
|
int col;
|
|
char s[64];
|
|
|
|
#if JOBS
|
|
if (mode & SHOW_PGID) {
|
|
/* just output process (group) id of pipeline */
|
|
outfmt(out, "%ld\n", (long)jp->ps->pid);
|
|
return;
|
|
}
|
|
#endif
|
|
|
|
procno = jp->nprocs;
|
|
if (!procno)
|
|
return;
|
|
|
|
if (mode & SHOW_PID)
|
|
mode |= SHOW_MULTILINE;
|
|
|
|
if ((procno > 1 && !(mode & SHOW_MULTILINE))
|
|
|| (mode & SHOW_SIGNALLED)) {
|
|
/* See if we have more than one status to report */
|
|
ps = jp->ps;
|
|
st = ps->status;
|
|
do {
|
|
int st1 = ps->status;
|
|
if (st1 != st)
|
|
/* yes - need multi-line output */
|
|
mode |= SHOW_MULTILINE;
|
|
if (st1 == -1 || !(mode & SHOW_SIGNALLED) || WIFEXITED(st1))
|
|
continue;
|
|
if (WIFSTOPPED(st1) || ((st1 = WTERMSIG(st1) & 0x7f)
|
|
&& st1 != SIGINT && st1 != SIGPIPE))
|
|
mode |= SHOW_ISSIG;
|
|
|
|
} while (ps++, --procno);
|
|
procno = jp->nprocs;
|
|
}
|
|
|
|
if (mode & SHOW_SIGNALLED && !(mode & SHOW_ISSIG)) {
|
|
if (jp->state == JOBDONE && !(mode & SHOW_NO_FREE)) {
|
|
VTRACE(DBG_JOBS, ("showjob: freeing job %d\n",
|
|
jp - jobtab + 1));
|
|
freejob(jp);
|
|
}
|
|
return;
|
|
}
|
|
|
|
for (ps = jp->ps; --procno >= 0; ps++) { /* for each process */
|
|
if (ps == jp->ps)
|
|
fmtstr(s, 16, "[%ld] %c ",
|
|
(long)(jp - jobtab + 1),
|
|
#if JOBS
|
|
jp - jobtab == curjob ?
|
|
'+' :
|
|
curjob != -1 &&
|
|
jp - jobtab == jobtab[curjob].prev_job ?
|
|
'-' :
|
|
#endif
|
|
' ');
|
|
else
|
|
fmtstr(s, 16, " " );
|
|
col = strlen(s);
|
|
if (mode & SHOW_PID) {
|
|
fmtstr(s + col, 16, "%ld ", (long)ps->pid);
|
|
col += strlen(s + col);
|
|
}
|
|
if (ps->status == -1) {
|
|
scopy("Running", s + col);
|
|
} else if (WIFEXITED(ps->status)) {
|
|
st = WEXITSTATUS(ps->status);
|
|
if (st)
|
|
fmtstr(s + col, 16, "Done(%d)", st);
|
|
else
|
|
fmtstr(s + col, 16, "Done");
|
|
} else {
|
|
#if JOBS
|
|
if (WIFSTOPPED(ps->status))
|
|
st = WSTOPSIG(ps->status);
|
|
else /* WIFSIGNALED(ps->status) */
|
|
#endif
|
|
st = WTERMSIG(ps->status);
|
|
scopyn(strsignal(st), s + col, 32);
|
|
if (WCOREDUMP(ps->status)) {
|
|
col += strlen(s + col);
|
|
scopyn(" (core dumped)", s + col, 64 - col);
|
|
}
|
|
}
|
|
col += strlen(s + col);
|
|
outstr(s, out);
|
|
do {
|
|
outc(' ', out);
|
|
col++;
|
|
} while (col < 30);
|
|
outstr(ps->cmd, out);
|
|
if (mode & SHOW_MULTILINE) {
|
|
if (procno > 0) {
|
|
outc(' ', out);
|
|
outc('|', out);
|
|
}
|
|
} else {
|
|
while (--procno >= 0)
|
|
outfmt(out, " | %s", (++ps)->cmd );
|
|
}
|
|
outc('\n', out);
|
|
}
|
|
flushout(out);
|
|
jp->flags &= ~JOBCHANGED;
|
|
if (jp->state == JOBDONE && !(mode & SHOW_NO_FREE))
|
|
freejob(jp);
|
|
}
|
|
|
|
int
|
|
jobscmd(int argc, char **argv)
|
|
{
|
|
int mode, m;
|
|
|
|
mode = 0;
|
|
while ((m = nextopt("lp")))
|
|
if (m == 'l')
|
|
mode = SHOW_PID;
|
|
else
|
|
mode = SHOW_PGID;
|
|
|
|
if (!iflag && !posix)
|
|
mode |= SHOW_NO_FREE;
|
|
|
|
if (*argptr) {
|
|
do
|
|
showjob(out1, getjob(*argptr,0), mode);
|
|
while (*++argptr);
|
|
} else
|
|
showjobs(out1, mode);
|
|
return 0;
|
|
}
|
|
|
|
|
|
/*
|
|
* Print a list of jobs. If "change" is nonzero, only print jobs whose
|
|
* statuses have changed since the last call to showjobs.
|
|
*
|
|
* If the shell is interrupted in the process of creating a job, the
|
|
* result may be a job structure containing zero processes. Such structures
|
|
* will be freed here.
|
|
*/
|
|
|
|
void
|
|
showjobs(struct output *out, int mode)
|
|
{
|
|
int jobno;
|
|
struct job *jp;
|
|
int silent = 0, gotpid;
|
|
|
|
CTRACE(DBG_JOBS, ("showjobs(%x) called\n", mode));
|
|
|
|
/* If not even one one job changed, there is nothing to do */
|
|
gotpid = dowait(WSILENT, NULL, NULL);
|
|
while (dowait(WSILENT, NULL, NULL) > 0)
|
|
continue;
|
|
#ifdef JOBS
|
|
/*
|
|
* Check if we are not in our foreground group, and if not
|
|
* put us in it.
|
|
*/
|
|
if (mflag && gotpid != -1 && tcgetpgrp(ttyfd) != getpid()) {
|
|
if (tcsetpgrp(ttyfd, getpid()) == -1)
|
|
error("Cannot set tty process group (%s) at %d",
|
|
strerror(errno), __LINE__);
|
|
VTRACE(DBG_JOBS|DBG_INPUT, ("repaired tty process group\n"));
|
|
silent = 1;
|
|
}
|
|
#endif
|
|
|
|
for (jobno = 1, jp = jobtab ; jobno <= njobs ; jobno++, jp++) {
|
|
if (!jp->used)
|
|
continue;
|
|
if (jp->nprocs == 0) {
|
|
if (!jobs_invalid)
|
|
freejob(jp);
|
|
continue;
|
|
}
|
|
if ((mode & SHOW_CHANGED) && !(jp->flags & JOBCHANGED))
|
|
continue;
|
|
if (silent && (jp->flags & JOBCHANGED)) {
|
|
jp->flags &= ~JOBCHANGED;
|
|
continue;
|
|
}
|
|
showjob(out, jp, mode);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Mark a job structure as unused.
|
|
*/
|
|
|
|
STATIC void
|
|
freejob(struct job *jp)
|
|
{
|
|
INTOFF;
|
|
if (jp->ps != &jp->ps0) {
|
|
ckfree(jp->ps);
|
|
jp->ps = &jp->ps0;
|
|
}
|
|
jp->nprocs = 0;
|
|
jp->used = 0;
|
|
#if JOBS
|
|
set_curjob(jp, 0);
|
|
#endif
|
|
INTON;
|
|
}
|
|
|
|
/*
|
|
* Extract the status of a completed job (for $?)
|
|
*/
|
|
STATIC int
|
|
jobstatus(const struct job *jp, int raw)
|
|
{
|
|
int status = 0;
|
|
int retval;
|
|
|
|
if ((jp->flags & JPIPEFAIL) && jp->nprocs) {
|
|
int i;
|
|
|
|
for (i = 0; i < jp->nprocs; i++)
|
|
if (jp->ps[i].status != 0)
|
|
status = jp->ps[i].status;
|
|
} else
|
|
status = jp->ps[jp->nprocs ? jp->nprocs - 1 : 0].status;
|
|
|
|
if (raw)
|
|
return status;
|
|
|
|
if (WIFEXITED(status))
|
|
retval = WEXITSTATUS(status);
|
|
#if JOBS
|
|
else if (WIFSTOPPED(status))
|
|
retval = WSTOPSIG(status) + 128;
|
|
#endif
|
|
else {
|
|
/* XXX: limits number of signals */
|
|
retval = WTERMSIG(status) + 128;
|
|
}
|
|
|
|
return retval;
|
|
}
|
|
|
|
|
|
|
|
int
|
|
waitcmd(int argc, char **argv)
|
|
{
|
|
struct job *job, *last;
|
|
int retval;
|
|
struct job *jp;
|
|
int i;
|
|
int any = 0;
|
|
int found;
|
|
char *pid = NULL, *fpid;
|
|
char **arg;
|
|
char idstring[20];
|
|
|
|
while ((i = nextopt("np:")) != '\0') {
|
|
switch (i) {
|
|
case 'n':
|
|
any = 1;
|
|
break;
|
|
case 'p':
|
|
if (pid)
|
|
error("more than one -p unsupported");
|
|
pid = optionarg;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (pid != NULL) {
|
|
if (!validname(pid, '\0', NULL))
|
|
error("invalid name: -p '%s'", pid);
|
|
if (unsetvar(pid, 0))
|
|
error("%s readonly", pid);
|
|
}
|
|
|
|
/*
|
|
* If we have forked, and not yet created any new jobs, then
|
|
* we have no children, whatever jobtab claims,
|
|
* so simply return in that case.
|
|
*
|
|
* The return code is 127 if we had any pid args (none are found)
|
|
* or if we had -n (nothing exited), but 0 for plain old "wait".
|
|
*/
|
|
if (jobs_invalid) {
|
|
CTRACE(DBG_WAIT, ("builtin wait%s%s in child, invalid jobtab\n",
|
|
any ? " -n" : "", *argptr ? " pid..." : ""));
|
|
return (any || *argptr) ? 127 : 0;
|
|
}
|
|
|
|
/*
|
|
* clear stray flags left from previous waitcmd
|
|
* or set them instead if anything will do ("wait -n")
|
|
*/
|
|
for (jp = jobtab, i = njobs ; --i >= 0 ; jp++) {
|
|
if (any && *argptr == NULL)
|
|
jp->flags |= JOBWANTED;
|
|
else
|
|
jp->flags &= ~JOBWANTED;
|
|
jp->ref = NULL;
|
|
}
|
|
|
|
CTRACE(DBG_WAIT,
|
|
("builtin wait%s%s\n", any ? " -n" : "", *argptr ? " pid..." : ""));
|
|
|
|
/*
|
|
* First, validate the jobnum args, count how many refer to
|
|
* (different) running jobs, and if we had -n, and found that one has
|
|
* already finished, we return that one. Otherwise remember
|
|
* which ones we are looking for (JOBWANTED).
|
|
*/
|
|
found = 0;
|
|
last = NULL;
|
|
for (arg = argptr; *arg; arg++) {
|
|
last = jp = getjob(*arg, 1);
|
|
if (!jp)
|
|
continue;
|
|
if (jp->ref == NULL)
|
|
jp->ref = *arg;
|
|
if (any && jp->state == JOBDONE) {
|
|
/*
|
|
* We just want any of them, and this one is
|
|
* ready for consumption, bon apetit ...
|
|
*/
|
|
retval = jobstatus(jp, 0);
|
|
if (pid)
|
|
setvar(pid, *arg, 0);
|
|
if (!iflag)
|
|
freejob(jp);
|
|
CTRACE(DBG_WAIT, ("wait -n found %s already done: %d\n", *arg, retval));
|
|
return retval;
|
|
}
|
|
if (!(jp->flags & JOBWANTED)) {
|
|
/*
|
|
* It is possible to list the same job several
|
|
* times - the obvious "wait 1 1 1" or
|
|
* "wait %% %2 102" where job 2 is current and pid 102
|
|
* However many times it is requested, it is found once.
|
|
*/
|
|
found++;
|
|
jp->flags |= JOBWANTED;
|
|
}
|
|
job = jp;
|
|
}
|
|
|
|
VTRACE(DBG_WAIT, ("wait %s%s%sfound %d candidates (last %s)\n",
|
|
any ? "-n " : "", *argptr ? *argptr : "",
|
|
argptr[0] && argptr[1] ? "... " : " ", found,
|
|
job ? (job->ref ? job->ref : "<no-arg>") : "none"));
|
|
|
|
/*
|
|
* If we were given a list of jobnums:
|
|
* and none of those exist, then we're done.
|
|
*/
|
|
if (*argptr && found == 0)
|
|
return 127;
|
|
|
|
/*
|
|
* Otherwise we need to wait for something to complete
|
|
* When it does, we check and see if it is one of the
|
|
* jobs we're waiting on, and if so, we clean it up.
|
|
* If we had -n, then we're done, otherwise we do it all again
|
|
* until all we had listed are done, of if there were no
|
|
* jobnum args, all are done.
|
|
*/
|
|
|
|
retval = any || *argptr ? 127 : 0;
|
|
fpid = NULL;
|
|
for (;;) {
|
|
VTRACE(DBG_WAIT, ("wait waiting (%d remain): ", found));
|
|
job = NULL;
|
|
for (jp = jobtab, i = njobs; --i >= 0; jp++) {
|
|
if (jp->used && jp->flags & JOBWANTED &&
|
|
jp->state == JOBDONE) {
|
|
job = jp;
|
|
break;
|
|
}
|
|
if (jp->used && jp->state == JOBRUNNING)
|
|
job = jp;
|
|
}
|
|
if (i < 0 && job == NULL) {
|
|
CTRACE(DBG_WAIT, ("nothing running (ret: %d) fpid %s\n",
|
|
retval, fpid ? fpid : "unset"));
|
|
if (pid && fpid)
|
|
setvar(pid, fpid, 0);
|
|
return retval;
|
|
}
|
|
jp = job;
|
|
VTRACE(DBG_WAIT, ("found @%d/%d state: %d\n", njobs-i, njobs,
|
|
jp->state));
|
|
|
|
/*
|
|
* There is at least 1 job running, so we can
|
|
* safely wait() for something to exit.
|
|
*/
|
|
if (jp->state == JOBRUNNING) {
|
|
job = NULL;
|
|
if ((i = dowait(WBLOCK|WNOFREE, NULL, &job)) == -1)
|
|
return 128 + lastsig();
|
|
|
|
if (job == NULL) /* an interloper */
|
|
continue;
|
|
|
|
/*
|
|
* one of the job's processes exited,
|
|
* but there are more
|
|
*/
|
|
if (job->state == JOBRUNNING)
|
|
continue;
|
|
} else
|
|
job = jp; /* we want this, and it is done */
|
|
|
|
if (job->flags & JOBWANTED) {
|
|
int rv;
|
|
|
|
job->flags &= ~JOBWANTED; /* got it */
|
|
rv = jobstatus(job, 0);
|
|
VTRACE(DBG_WAIT, (
|
|
"wanted %d (%s) done: st=%d", i,
|
|
job->ref ? job->ref : "", rv));
|
|
if (any || job == last) {
|
|
retval = rv;
|
|
fpid = job->ref;
|
|
|
|
VTRACE(DBG_WAIT, (" save"));
|
|
if (pid) {
|
|
/*
|
|
* don't need fpid unless we are going
|
|
* to return it.
|
|
*/
|
|
if (fpid == NULL) {
|
|
/*
|
|
* this only happens with "wait -n"
|
|
* (that is, no pid args)
|
|
*/
|
|
snprintf(idstring, sizeof idstring,
|
|
"%d", job->ps[ job->nprocs ?
|
|
job->nprocs-1 :
|
|
0 ].pid);
|
|
fpid = idstring;
|
|
}
|
|
VTRACE(DBG_WAIT, (" (for %s)", fpid));
|
|
}
|
|
}
|
|
|
|
if (job->state == JOBDONE) {
|
|
VTRACE(DBG_WAIT, (" free"));
|
|
freejob(job);
|
|
}
|
|
|
|
if (any || (found > 0 && --found == 0)) {
|
|
if (pid && fpid)
|
|
setvar(pid, fpid, 0);
|
|
VTRACE(DBG_WAIT, (" return %d\n", retval));
|
|
return retval;
|
|
}
|
|
VTRACE(DBG_WAIT, ("\n"));
|
|
continue;
|
|
}
|
|
|
|
/* this is to handle "wait" (no args) */
|
|
if (found == 0 && job->state == JOBDONE) {
|
|
VTRACE(DBG_JOBS|DBG_WAIT, ("Cleanup: %d\n", i));
|
|
freejob(job);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
int
|
|
jobidcmd(int argc, char **argv)
|
|
{
|
|
struct job *jp;
|
|
int i;
|
|
int pg = 0, onep = 0, job = 0;
|
|
|
|
while ((i = nextopt("gjp"))) {
|
|
switch (i) {
|
|
case 'g': pg = 1; break;
|
|
case 'j': job = 1; break;
|
|
case 'p': onep = 1; break;
|
|
}
|
|
}
|
|
CTRACE(DBG_JOBS, ("jobidcmd%s%s%s%s %s\n", pg ? " -g" : "",
|
|
onep ? " -p" : "", job ? " -j" : "", jobs_invalid ? " [inv]" : "",
|
|
*argptr ? *argptr : "<implicit %%>"));
|
|
if (pg + onep + job > 1)
|
|
error("-g -j and -p options cannot be combined");
|
|
|
|
if (argptr[0] && argptr[1])
|
|
error("usage: jobid [-g|-p|-r] jobid");
|
|
|
|
jp = getjob(*argptr, 0);
|
|
if (job) {
|
|
out1fmt("%%%zu\n", (size_t)(jp - jobtab + 1));
|
|
return 0;
|
|
}
|
|
if (pg) {
|
|
if (jp->pgrp != 0) {
|
|
out1fmt("%ld\n", (long)jp->pgrp);
|
|
return 0;
|
|
}
|
|
return 1;
|
|
}
|
|
if (onep) {
|
|
i = jp->nprocs - 1;
|
|
if (i < 0)
|
|
return 1;
|
|
out1fmt("%ld\n", (long)jp->ps[i].pid);
|
|
return 0;
|
|
}
|
|
for (i = 0 ; i < jp->nprocs ; ) {
|
|
out1fmt("%ld", (long)jp->ps[i].pid);
|
|
out1c(++i < jp->nprocs ? ' ' : '\n');
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
getjobpgrp(const char *name)
|
|
{
|
|
struct job *jp;
|
|
|
|
if (jobs_invalid)
|
|
error("No such job: %s", name);
|
|
jp = getjob(name, 1);
|
|
if (jp == 0)
|
|
return 0;
|
|
return -jp->ps[0].pid;
|
|
}
|
|
|
|
/*
|
|
* Convert a job name to a job structure.
|
|
*/
|
|
|
|
STATIC struct job *
|
|
getjob(const char *name, int noerror)
|
|
{
|
|
int jobno = -1;
|
|
struct job *jp;
|
|
int pid;
|
|
int i;
|
|
const char *err_msg = "No such job: %s";
|
|
|
|
if (name == NULL) {
|
|
#if JOBS
|
|
jobno = curjob;
|
|
#endif
|
|
err_msg = "No current job";
|
|
} else if (name[0] == '%') {
|
|
if (is_number(name + 1)) {
|
|
jobno = number(name + 1) - 1;
|
|
} else if (!name[1] || !name[2]) {
|
|
switch (name[1]) {
|
|
#if JOBS
|
|
case 0:
|
|
case '+':
|
|
case '%':
|
|
jobno = curjob;
|
|
err_msg = "No current job";
|
|
break;
|
|
case '-':
|
|
jobno = curjob;
|
|
if (jobno != -1)
|
|
jobno = jobtab[jobno].prev_job;
|
|
err_msg = "No previous job";
|
|
break;
|
|
#endif
|
|
default:
|
|
goto check_pattern;
|
|
}
|
|
} else {
|
|
struct job *found;
|
|
check_pattern:
|
|
found = NULL;
|
|
for (jp = jobtab, i = njobs ; --i >= 0 ; jp++) {
|
|
if (!jp->used || jp->nprocs <= 0)
|
|
continue;
|
|
if ((name[1] == '?'
|
|
&& strstr(jp->ps[0].cmd, name + 2))
|
|
|| prefix(name + 1, jp->ps[0].cmd)) {
|
|
if (found) {
|
|
err_msg = "%s: ambiguous";
|
|
found = 0;
|
|
break;
|
|
}
|
|
found = jp;
|
|
}
|
|
}
|
|
if (found)
|
|
return found;
|
|
}
|
|
|
|
} else if (is_number(name)) {
|
|
pid = number(name);
|
|
for (jp = jobtab, i = njobs ; --i >= 0 ; jp++) {
|
|
if (jp->used && jp->nprocs > 0
|
|
&& jp->ps[jp->nprocs - 1].pid == pid)
|
|
return jp;
|
|
}
|
|
}
|
|
|
|
if (jobno >= 0 && jobno < njobs) {
|
|
jp = jobtab + jobno;
|
|
if (jp->used)
|
|
return jp;
|
|
}
|
|
if (!noerror)
|
|
error(err_msg, name);
|
|
return 0;
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
* Return a new job structure,
|
|
*/
|
|
|
|
struct job *
|
|
makejob(union node *node, int nprocs)
|
|
{
|
|
int i;
|
|
struct job *jp;
|
|
|
|
if (jobs_invalid) {
|
|
for (i = njobs, jp = jobtab ; --i >= 0 ; jp++) {
|
|
if (jp->used)
|
|
freejob(jp);
|
|
}
|
|
jobs_invalid = 0;
|
|
}
|
|
|
|
for (i = njobs, jp = jobtab ; ; jp++) {
|
|
if (--i < 0) {
|
|
INTOFF;
|
|
if (njobs == 0) {
|
|
jobtab = ckmalloc(4 * sizeof jobtab[0]);
|
|
} else {
|
|
jp = ckmalloc((njobs + 4) * sizeof jobtab[0]);
|
|
memcpy(jp, jobtab, njobs * sizeof jp[0]);
|
|
/* Relocate `ps' pointers */
|
|
for (i = 0; i < njobs; i++)
|
|
if (jp[i].ps == &jobtab[i].ps0)
|
|
jp[i].ps = &jp[i].ps0;
|
|
ckfree(jobtab);
|
|
jobtab = jp;
|
|
}
|
|
jp = jobtab + njobs;
|
|
for (i = 4 ; --i >= 0 ; njobs++) {
|
|
jobtab[njobs].used = 0;
|
|
jobtab[njobs].prev_job = -1;
|
|
}
|
|
INTON;
|
|
break;
|
|
}
|
|
if (jp->used == 0)
|
|
break;
|
|
}
|
|
INTOFF;
|
|
jp->state = JOBRUNNING;
|
|
jp->used = 1;
|
|
jp->flags = pipefail ? JPIPEFAIL : 0;
|
|
jp->nprocs = 0;
|
|
jp->pgrp = 0;
|
|
#if JOBS
|
|
jp->jobctl = jobctl;
|
|
set_curjob(jp, 1);
|
|
#endif
|
|
if (nprocs > 1) {
|
|
jp->ps = ckmalloc(nprocs * sizeof (struct procstat));
|
|
} else {
|
|
jp->ps = &jp->ps0;
|
|
}
|
|
INTON;
|
|
VTRACE(DBG_JOBS, ("makejob(%p, %d)%s returns %%%d\n", (void *)node,
|
|
nprocs, (jp->flags&JPIPEFAIL)?" PF":"", jp - jobtab + 1));
|
|
return jp;
|
|
}
|
|
|
|
|
|
/*
|
|
* Fork off a subshell. If we are doing job control, give the subshell its
|
|
* own process group. Jp is a job structure that the job is to be added to.
|
|
* N is the command that will be evaluated by the child. Both jp and n may
|
|
* be NULL. The mode parameter can be one of the following:
|
|
* FORK_FG - Fork off a foreground process.
|
|
* FORK_BG - Fork off a background process.
|
|
* FORK_NOJOB - Like FORK_FG, but don't give the process its own
|
|
* process group even if job control is on.
|
|
*
|
|
* When job control is turned off, background processes have their standard
|
|
* input redirected to /dev/null (except for the second and later processes
|
|
* in a pipeline).
|
|
*/
|
|
|
|
int
|
|
forkshell(struct job *jp, union node *n, int mode)
|
|
{
|
|
pid_t pid;
|
|
int serrno;
|
|
|
|
CTRACE(DBG_JOBS, ("forkshell(%%%d, %p, %d) called\n",
|
|
jp - jobtab, n, mode));
|
|
|
|
switch ((pid = fork())) {
|
|
case -1:
|
|
serrno = errno;
|
|
VTRACE(DBG_JOBS, ("Fork failed, errno=%d\n", serrno));
|
|
error("Cannot fork (%s)", strerror(serrno));
|
|
break;
|
|
case 0:
|
|
SHELL_FORKED();
|
|
forkchild(jp, n, mode, 0);
|
|
return 0;
|
|
default:
|
|
return forkparent(jp, n, mode, pid);
|
|
}
|
|
}
|
|
|
|
int
|
|
forkparent(struct job *jp, union node *n, int mode, pid_t pid)
|
|
{
|
|
int pgrp;
|
|
|
|
if (rootshell && mode != FORK_NOJOB && mflag) {
|
|
if (jp == NULL || jp->nprocs == 0)
|
|
pgrp = pid;
|
|
else
|
|
pgrp = jp->ps[0].pid;
|
|
jp->pgrp = pgrp;
|
|
/* This can fail because we are doing it in the child also */
|
|
(void)setpgid(pid, pgrp);
|
|
}
|
|
if (mode == FORK_BG)
|
|
backgndpid = pid; /* set $! */
|
|
if (jp) {
|
|
struct procstat *ps = &jp->ps[jp->nprocs++];
|
|
ps->pid = pid;
|
|
ps->status = -1;
|
|
ps->cmd[0] = 0;
|
|
if (/* iflag && rootshell && */ n)
|
|
commandtext(ps, n);
|
|
}
|
|
CTRACE(DBG_JOBS, ("In parent shell: child = %d (mode %d)\n",pid,mode));
|
|
return pid;
|
|
}
|
|
|
|
void
|
|
forkchild(struct job *jp, union node *n, int mode, int vforked)
|
|
{
|
|
int wasroot;
|
|
int pgrp;
|
|
const char *devnull = _PATH_DEVNULL;
|
|
const char *nullerr = "Can't open %s";
|
|
|
|
wasroot = rootshell;
|
|
CTRACE(DBG_JOBS, ("Child shell %d %sforked from %d (mode %d)\n",
|
|
getpid(), vforked?"v":"", getppid(), mode));
|
|
|
|
if (!vforked) {
|
|
rootshell = 0;
|
|
handler = &main_handler;
|
|
}
|
|
|
|
closescript(vforked);
|
|
clear_traps(vforked);
|
|
#if JOBS
|
|
if (!vforked)
|
|
jobctl = 0; /* do job control only in root shell */
|
|
if (wasroot && mode != FORK_NOJOB && mflag) {
|
|
if (jp == NULL || jp->nprocs == 0)
|
|
pgrp = getpid();
|
|
else
|
|
pgrp = jp->ps[0].pid;
|
|
/* This can fail because we are doing it in the parent also */
|
|
(void)setpgid(0, pgrp);
|
|
if (mode == FORK_FG) {
|
|
if (tcsetpgrp(ttyfd, pgrp) == -1)
|
|
error("Cannot set tty process group (%s) at %d",
|
|
strerror(errno), __LINE__);
|
|
}
|
|
setsignal(SIGTSTP, vforked);
|
|
setsignal(SIGTTOU, vforked);
|
|
} else if (mode == FORK_BG) {
|
|
ignoresig(SIGINT, vforked);
|
|
ignoresig(SIGQUIT, vforked);
|
|
if ((jp == NULL || jp->nprocs == 0) &&
|
|
! fd0_redirected_p ()) {
|
|
close(0);
|
|
if (open(devnull, O_RDONLY) != 0)
|
|
error(nullerr, devnull);
|
|
}
|
|
}
|
|
#else
|
|
if (mode == FORK_BG) {
|
|
ignoresig(SIGINT, vforked);
|
|
ignoresig(SIGQUIT, vforked);
|
|
if ((jp == NULL || jp->nprocs == 0) &&
|
|
! fd0_redirected_p ()) {
|
|
close(0);
|
|
if (open(devnull, O_RDONLY) != 0)
|
|
error(nullerr, devnull);
|
|
}
|
|
}
|
|
#endif
|
|
if (wasroot && iflag) {
|
|
setsignal(SIGINT, vforked);
|
|
setsignal(SIGQUIT, vforked);
|
|
setsignal(SIGTERM, vforked);
|
|
}
|
|
|
|
if (!vforked)
|
|
jobs_invalid = 1;
|
|
}
|
|
|
|
/*
|
|
* Wait for job to finish.
|
|
*
|
|
* Under job control we have the problem that while a child process is
|
|
* running interrupts generated by the user are sent to the child but not
|
|
* to the shell. This means that an infinite loop started by an inter-
|
|
* active user may be hard to kill. With job control turned off, an
|
|
* interactive user may place an interactive program inside a loop. If
|
|
* the interactive program catches interrupts, the user doesn't want
|
|
* these interrupts to also abort the loop. The approach we take here
|
|
* is to have the shell ignore interrupt signals while waiting for a
|
|
* forground process to terminate, and then send itself an interrupt
|
|
* signal if the child process was terminated by an interrupt signal.
|
|
* Unfortunately, some programs want to do a bit of cleanup and then
|
|
* exit on interrupt; unless these processes terminate themselves by
|
|
* sending a signal to themselves (instead of calling exit) they will
|
|
* confuse this approach.
|
|
*/
|
|
|
|
int
|
|
waitforjob(struct job *jp)
|
|
{
|
|
#if JOBS
|
|
int mypgrp = getpgrp();
|
|
#endif
|
|
int status;
|
|
int st;
|
|
|
|
INTOFF;
|
|
VTRACE(DBG_JOBS, ("waitforjob(%%%d) called\n", jp - jobtab + 1));
|
|
while (jp->state == JOBRUNNING) {
|
|
dowait(WBLOCK, jp, NULL);
|
|
}
|
|
#if JOBS
|
|
if (jp->jobctl) {
|
|
if (tcsetpgrp(ttyfd, mypgrp) == -1)
|
|
error("Cannot set tty process group (%s) at %d",
|
|
strerror(errno), __LINE__);
|
|
}
|
|
if (jp->state == JOBSTOPPED && curjob != jp - jobtab)
|
|
set_curjob(jp, 2);
|
|
#endif
|
|
status = jobstatus(jp, 1);
|
|
|
|
/* convert to 8 bits */
|
|
if (WIFEXITED(status))
|
|
st = WEXITSTATUS(status);
|
|
#if JOBS
|
|
else if (WIFSTOPPED(status))
|
|
st = WSTOPSIG(status) + 128;
|
|
#endif
|
|
else
|
|
st = WTERMSIG(status) + 128;
|
|
|
|
VTRACE(DBG_JOBS, ("waitforjob: job %d, nproc %d, status %d, st %x\n",
|
|
jp - jobtab + 1, jp->nprocs, status, st));
|
|
#if JOBS
|
|
if (jp->jobctl) {
|
|
/*
|
|
* This is truly gross.
|
|
* If we're doing job control, then we did a TIOCSPGRP which
|
|
* caused us (the shell) to no longer be in the controlling
|
|
* session -- so we wouldn't have seen any ^C/SIGINT. So, we
|
|
* intuit from the subprocess exit status whether a SIGINT
|
|
* occurred, and if so interrupt ourselves. Yuck. - mycroft
|
|
*/
|
|
if (WIFSIGNALED(status) && WTERMSIG(status) == SIGINT)
|
|
raise(SIGINT);
|
|
}
|
|
#endif
|
|
if (! JOBS || jp->state == JOBDONE)
|
|
freejob(jp);
|
|
INTON;
|
|
return st;
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
* Wait for a process to terminate.
|
|
*/
|
|
|
|
STATIC int
|
|
dowait(int flags, struct job *job, struct job **changed)
|
|
{
|
|
int pid;
|
|
int status;
|
|
struct procstat *sp;
|
|
struct job *jp;
|
|
struct job *thisjob;
|
|
int done;
|
|
int stopped;
|
|
|
|
VTRACE(DBG_JOBS|DBG_PROCS, ("dowait(%x) called\n", flags));
|
|
|
|
if (changed != NULL)
|
|
*changed = NULL;
|
|
|
|
do {
|
|
pid = waitproc(flags & WBLOCK, job, &status);
|
|
VTRACE(DBG_JOBS|DBG_PROCS, ("wait returns pid %d, status %#x\n",
|
|
pid, status));
|
|
} while (pid == -1 && errno == EINTR && pendingsigs == 0);
|
|
if (pid <= 0)
|
|
return pid;
|
|
INTOFF;
|
|
thisjob = NULL;
|
|
for (jp = jobtab ; jp < jobtab + njobs ; jp++) {
|
|
if (jp->used) {
|
|
done = 1;
|
|
stopped = 1;
|
|
for (sp = jp->ps ; sp < jp->ps + jp->nprocs ; sp++) {
|
|
if (sp->pid == -1)
|
|
continue;
|
|
if (sp->pid == pid &&
|
|
(sp->status==-1 || WIFSTOPPED(sp->status))) {
|
|
VTRACE(DBG_JOBS | DBG_PROCS,
|
|
("Job %d: changing status of proc %d from %#x to %#x\n",
|
|
jp - jobtab + 1, pid,
|
|
sp->status, status));
|
|
if (WIFCONTINUED(status)) {
|
|
if (sp->status != -1)
|
|
jp->flags |= JOBCHANGED;
|
|
sp->status = -1;
|
|
jp->state = 0;
|
|
} else
|
|
sp->status = status;
|
|
thisjob = jp;
|
|
if (changed != NULL)
|
|
*changed = jp;
|
|
}
|
|
if (sp->status == -1)
|
|
stopped = 0;
|
|
else if (WIFSTOPPED(sp->status))
|
|
done = 0;
|
|
}
|
|
if (stopped) { /* stopped or done */
|
|
int state = done ? JOBDONE : JOBSTOPPED;
|
|
|
|
if (jp->state != state) {
|
|
VTRACE(DBG_JOBS,
|
|
("Job %d: changing state from %d to %d\n",
|
|
jp - jobtab + 1, jp->state, state));
|
|
jp->state = state;
|
|
#if JOBS
|
|
if (done)
|
|
set_curjob(jp, 0);
|
|
#endif
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (thisjob &&
|
|
(thisjob->state != JOBRUNNING || thisjob->flags & JOBCHANGED)) {
|
|
int mode = 0;
|
|
|
|
if (!rootshell || !iflag)
|
|
mode = SHOW_SIGNALLED;
|
|
if ((job == thisjob && (flags & WNOFREE) == 0) ||
|
|
job != thisjob)
|
|
mode = SHOW_SIGNALLED | SHOW_NO_FREE;
|
|
if (mode && (flags & WSILENT) == 0)
|
|
showjob(out2, thisjob, mode);
|
|
else {
|
|
VTRACE(DBG_JOBS,
|
|
("Not printing status, rootshell=%d, job=%p\n",
|
|
rootshell, job));
|
|
thisjob->flags |= JOBCHANGED;
|
|
}
|
|
}
|
|
|
|
INTON;
|
|
return pid;
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
* Do a wait system call. If job control is compiled in, we accept
|
|
* stopped processes. If block is zero, we return a value of zero
|
|
* rather than blocking.
|
|
*
|
|
* System V doesn't have a non-blocking wait system call. It does
|
|
* have a SIGCLD signal that is sent to a process when one of its
|
|
* children dies. The obvious way to use SIGCLD would be to install
|
|
* a handler for SIGCLD which simply bumped a counter when a SIGCLD
|
|
* was received, and have waitproc bump another counter when it got
|
|
* the status of a process. Waitproc would then know that a wait
|
|
* system call would not block if the two counters were different.
|
|
* This approach doesn't work because if a process has children that
|
|
* have not been waited for, System V will send it a SIGCLD when it
|
|
* installs a signal handler for SIGCLD. What this means is that when
|
|
* a child exits, the shell will be sent SIGCLD signals continuously
|
|
* until is runs out of stack space, unless it does a wait call before
|
|
* restoring the signal handler. The code below takes advantage of
|
|
* this (mis)feature by installing a signal handler for SIGCLD and
|
|
* then checking to see whether it was called. If there are any
|
|
* children to be waited for, it will be.
|
|
*
|
|
* If neither SYSV nor BSD is defined, we don't implement nonblocking
|
|
* waits at all. In this case, the user will not be informed when
|
|
* a background process until the next time she runs a real program
|
|
* (as opposed to running a builtin command or just typing return),
|
|
* and the jobs command may give out of date information.
|
|
*/
|
|
|
|
#ifdef SYSV
|
|
STATIC int gotsigchild;
|
|
|
|
STATIC int onsigchild() {
|
|
gotsigchild = 1;
|
|
}
|
|
#endif
|
|
|
|
|
|
STATIC int
|
|
waitproc(int block, struct job *jp, int *status)
|
|
{
|
|
#ifdef BSD
|
|
int flags = 0;
|
|
|
|
#if JOBS
|
|
if (mflag || (jp != NULL && jp->jobctl))
|
|
flags |= WUNTRACED | WCONTINUED;
|
|
#endif
|
|
if (block == 0)
|
|
flags |= WNOHANG;
|
|
VTRACE(DBG_WAIT, ("waitproc: doing waitpid(flags=%#x)\n", flags));
|
|
return waitpid(-1, status, flags);
|
|
#else
|
|
#ifdef SYSV
|
|
int (*save)();
|
|
|
|
if (block == 0) {
|
|
gotsigchild = 0;
|
|
save = signal(SIGCLD, onsigchild);
|
|
signal(SIGCLD, save);
|
|
if (gotsigchild == 0)
|
|
return 0;
|
|
}
|
|
return wait(status);
|
|
#else
|
|
if (block == 0)
|
|
return 0;
|
|
return wait(status);
|
|
#endif
|
|
#endif
|
|
}
|
|
|
|
/*
|
|
* return 1 if there are stopped jobs, otherwise 0
|
|
*/
|
|
int job_warning = 0;
|
|
int
|
|
stoppedjobs(void)
|
|
{
|
|
int jobno;
|
|
struct job *jp;
|
|
|
|
if (job_warning || jobs_invalid)
|
|
return (0);
|
|
for (jobno = 1, jp = jobtab; jobno <= njobs; jobno++, jp++) {
|
|
if (jp->used == 0)
|
|
continue;
|
|
if (jp->state == JOBSTOPPED) {
|
|
out2str("You have stopped jobs.\n");
|
|
job_warning = 2;
|
|
return (1);
|
|
}
|
|
}
|
|
|
|
return (0);
|
|
}
|
|
|
|
/*
|
|
* Return a string identifying a command (to be printed by the
|
|
* jobs command).
|
|
*/
|
|
|
|
STATIC char *cmdnextc;
|
|
STATIC int cmdnleft;
|
|
|
|
void
|
|
commandtext(struct procstat *ps, union node *n)
|
|
{
|
|
int len;
|
|
|
|
cmdnextc = ps->cmd;
|
|
if (iflag || mflag || sizeof(ps->cmd) <= 60)
|
|
len = sizeof(ps->cmd);
|
|
else if (sizeof ps->cmd <= 400)
|
|
len = 50;
|
|
else if (sizeof ps->cmd <= 800)
|
|
len = 80;
|
|
else
|
|
len = sizeof(ps->cmd) / 10;
|
|
cmdnleft = len;
|
|
cmdtxt(n);
|
|
if (cmdnleft <= 0) {
|
|
char *p = ps->cmd + len - 4;
|
|
p[0] = '.';
|
|
p[1] = '.';
|
|
p[2] = '.';
|
|
p[3] = 0;
|
|
} else
|
|
*cmdnextc = '\0';
|
|
|
|
VTRACE(DBG_JOBS,
|
|
("commandtext: ps->cmd %p, end %p, left %d\n\t\"%s\"\n",
|
|
ps->cmd, cmdnextc, cmdnleft, ps->cmd));
|
|
}
|
|
|
|
|
|
STATIC void
|
|
cmdtxt(union node *n)
|
|
{
|
|
union node *np;
|
|
struct nodelist *lp;
|
|
const char *p;
|
|
int i;
|
|
|
|
if (n == NULL || cmdnleft <= 0)
|
|
return;
|
|
switch (n->type) {
|
|
case NSEMI:
|
|
cmdtxt(n->nbinary.ch1);
|
|
cmdputs("; ");
|
|
cmdtxt(n->nbinary.ch2);
|
|
break;
|
|
case NAND:
|
|
cmdtxt(n->nbinary.ch1);
|
|
cmdputs(" && ");
|
|
cmdtxt(n->nbinary.ch2);
|
|
break;
|
|
case NOR:
|
|
cmdtxt(n->nbinary.ch1);
|
|
cmdputs(" || ");
|
|
cmdtxt(n->nbinary.ch2);
|
|
break;
|
|
case NDNOT:
|
|
cmdputs("! ");
|
|
/* FALLTHROUGH */
|
|
case NNOT:
|
|
cmdputs("! ");
|
|
cmdtxt(n->nnot.com);
|
|
break;
|
|
case NPIPE:
|
|
for (lp = n->npipe.cmdlist ; lp ; lp = lp->next) {
|
|
cmdtxt(lp->n);
|
|
if (lp->next)
|
|
cmdputs(" | ");
|
|
}
|
|
if (n->npipe.backgnd)
|
|
cmdputs(" &");
|
|
break;
|
|
case NSUBSHELL:
|
|
cmdputs("(");
|
|
cmdtxt(n->nredir.n);
|
|
cmdputs(")");
|
|
break;
|
|
case NREDIR:
|
|
case NBACKGND:
|
|
cmdtxt(n->nredir.n);
|
|
break;
|
|
case NIF:
|
|
cmdputs("if ");
|
|
cmdtxt(n->nif.test);
|
|
cmdputs("; then ");
|
|
cmdtxt(n->nif.ifpart);
|
|
if (n->nif.elsepart) {
|
|
cmdputs("; else ");
|
|
cmdtxt(n->nif.elsepart);
|
|
}
|
|
cmdputs("; fi");
|
|
break;
|
|
case NWHILE:
|
|
cmdputs("while ");
|
|
goto until;
|
|
case NUNTIL:
|
|
cmdputs("until ");
|
|
until:
|
|
cmdtxt(n->nbinary.ch1);
|
|
cmdputs("; do ");
|
|
cmdtxt(n->nbinary.ch2);
|
|
cmdputs("; done");
|
|
break;
|
|
case NFOR:
|
|
cmdputs("for ");
|
|
cmdputs(n->nfor.var);
|
|
cmdputs(" in ");
|
|
cmdlist(n->nfor.args, 1);
|
|
cmdputs("; do ");
|
|
cmdtxt(n->nfor.body);
|
|
cmdputs("; done");
|
|
break;
|
|
case NCASE:
|
|
cmdputs("case ");
|
|
cmdputs(n->ncase.expr->narg.text);
|
|
cmdputs(" in ");
|
|
for (np = n->ncase.cases; np; np = np->nclist.next) {
|
|
cmdtxt(np->nclist.pattern);
|
|
cmdputs(") ");
|
|
cmdtxt(np->nclist.body);
|
|
switch (n->type) { /* switch (not if) for later */
|
|
case NCLISTCONT:
|
|
cmdputs(";& ");
|
|
break;
|
|
default:
|
|
cmdputs(";; ");
|
|
break;
|
|
}
|
|
}
|
|
cmdputs("esac");
|
|
break;
|
|
case NDEFUN:
|
|
cmdputs(n->narg.text);
|
|
cmdputs("() { ... }");
|
|
break;
|
|
case NCMD:
|
|
cmdlist(n->ncmd.args, 1);
|
|
cmdlist(n->ncmd.redirect, 0);
|
|
if (n->ncmd.backgnd)
|
|
cmdputs(" &");
|
|
break;
|
|
case NARG:
|
|
cmdputs(n->narg.text);
|
|
break;
|
|
case NTO:
|
|
p = ">"; i = 1; goto redir;
|
|
case NCLOBBER:
|
|
p = ">|"; i = 1; goto redir;
|
|
case NAPPEND:
|
|
p = ">>"; i = 1; goto redir;
|
|
case NTOFD:
|
|
p = ">&"; i = 1; goto redir;
|
|
case NFROM:
|
|
p = "<"; i = 0; goto redir;
|
|
case NFROMFD:
|
|
p = "<&"; i = 0; goto redir;
|
|
case NFROMTO:
|
|
p = "<>"; i = 0; goto redir;
|
|
redir:
|
|
if (n->nfile.fd != i)
|
|
cmdputi(n->nfile.fd);
|
|
cmdputs(p);
|
|
if (n->type == NTOFD || n->type == NFROMFD) {
|
|
if (n->ndup.dupfd < 0)
|
|
cmdputs("-");
|
|
else
|
|
cmdputi(n->ndup.dupfd);
|
|
} else {
|
|
cmdtxt(n->nfile.fname);
|
|
}
|
|
break;
|
|
case NHERE:
|
|
case NXHERE:
|
|
cmdputs("<<...");
|
|
break;
|
|
default:
|
|
cmdputs("???");
|
|
break;
|
|
}
|
|
}
|
|
|
|
STATIC void
|
|
cmdlist(union node *np, int sep)
|
|
{
|
|
for (; np; np = np->narg.next) {
|
|
if (!sep)
|
|
cmdputs(" ");
|
|
cmdtxt(np);
|
|
if (sep && np->narg.next)
|
|
cmdputs(" ");
|
|
}
|
|
}
|
|
|
|
|
|
STATIC void
|
|
cmdputs(const char *s)
|
|
{
|
|
const char *p, *str = 0;
|
|
char c, cc[2] = " ";
|
|
char *nextc;
|
|
int nleft;
|
|
int subtype = 0;
|
|
int quoted = 0;
|
|
static char vstype[16][4] = { "", "}", "-", "+", "?", "=",
|
|
"#", "##", "%", "%%", "}" };
|
|
|
|
p = s;
|
|
nextc = cmdnextc;
|
|
nleft = cmdnleft;
|
|
while (nleft > 0 && (c = *p++) != 0) {
|
|
switch (c) {
|
|
case CTLNONL:
|
|
c = '\0';
|
|
break;
|
|
case CTLESC:
|
|
c = *p++;
|
|
break;
|
|
case CTLVAR:
|
|
subtype = *p++;
|
|
if (subtype & VSLINENO) { /* undo LINENO hack */
|
|
if ((subtype & VSTYPE) == VSLENGTH)
|
|
str = "${#LINENO"; /*}*/
|
|
else
|
|
str = "${LINENO"; /*}*/
|
|
while (is_digit(*p))
|
|
p++;
|
|
} else if ((subtype & VSTYPE) == VSLENGTH)
|
|
str = "${#"; /*}*/
|
|
else
|
|
str = "${"; /*}*/
|
|
if (!(subtype & VSQUOTE) != !(quoted & 1)) {
|
|
quoted ^= 1;
|
|
c = '"';
|
|
} else {
|
|
c = *str++;
|
|
}
|
|
break;
|
|
case CTLENDVAR: /*{*/
|
|
c = '}';
|
|
if (quoted & 1)
|
|
str = "\"";
|
|
quoted >>= 1;
|
|
subtype = 0;
|
|
break;
|
|
case CTLBACKQ:
|
|
c = '$';
|
|
str = "(...)";
|
|
break;
|
|
case CTLBACKQ+CTLQUOTE:
|
|
c = '"';
|
|
str = "$(...)\"";
|
|
break;
|
|
case CTLARI:
|
|
c = '$';
|
|
if (*p == ' ')
|
|
p++;
|
|
str = "(("; /*))*/
|
|
break;
|
|
case CTLENDARI: /*((*/
|
|
c = ')';
|
|
str = ")";
|
|
break;
|
|
case CTLQUOTEMARK:
|
|
quoted ^= 1;
|
|
c = '"';
|
|
break;
|
|
case CTLQUOTEEND:
|
|
quoted >>= 1;
|
|
c = '"';
|
|
break;
|
|
case '=':
|
|
if (subtype == 0)
|
|
break;
|
|
str = vstype[subtype & VSTYPE];
|
|
if (subtype & VSNUL)
|
|
c = ':';
|
|
else
|
|
c = *str++; /*{*/
|
|
if (c != '}')
|
|
quoted <<= 1;
|
|
else if (*p == CTLENDVAR)
|
|
c = *str++;
|
|
subtype = 0;
|
|
break;
|
|
case '\'':
|
|
case '\\':
|
|
case '"':
|
|
case '$':
|
|
/* These can only happen inside quotes */
|
|
cc[0] = c;
|
|
str = cc;
|
|
c = '\\';
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
if (c != '\0') do { /* c == 0 implies nothing in str */
|
|
*nextc++ = c;
|
|
} while (--nleft > 0 && str && (c = *str++));
|
|
str = 0;
|
|
}
|
|
if ((quoted & 1) && nleft) {
|
|
*nextc++ = '"';
|
|
nleft--;
|
|
}
|
|
cmdnleft = nleft;
|
|
cmdnextc = nextc;
|
|
}
|