summaryrefslogtreecommitdiffstats
path: root/src/kash/jobs.c
diff options
context:
space:
mode:
Diffstat (limited to '')
-rw-r--r--src/kash/jobs.c1521
1 files changed, 1521 insertions, 0 deletions
diff --git a/src/kash/jobs.c b/src/kash/jobs.c
new file mode 100644
index 0000000..43f359b
--- /dev/null
+++ b/src/kash/jobs.c
@@ -0,0 +1,1521 @@
+/* $NetBSD: jobs.c,v 1.63 2005/06/01 15:41:19 lukem 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.
+ */
+
+#if 0
+#ifndef lint
+static char sccsid[] = "@(#)jobs.c 8.5 (Berkeley) 5/4/95";
+#else
+__RCSID("$NetBSD: jobs.c,v 1.63 2005/06/01 15:41:19 lukem Exp $");
+#endif /* not lint */
+#endif
+
+#include <fcntl.h>
+#include <errno.h>
+#include <stdlib.h>
+#include <sys/types.h>
+
+#include "shell.h"
+#if JOBS && !defined(_MSC_VER)
+# include <termios.h>
+#endif
+#include "redir.h"
+#include "show.h"
+#include "main.h"
+#include "parser.h"
+#include "nodes.h"
+#include "jobs.h"
+#include "options.h"
+#include "trap.h"
+#include "syntax.h"
+#include "input.h"
+#include "output.h"
+#include "memalloc.h"
+#include "error.h"
+#include "mystring.h"
+#include "init.h"
+#include "shinstance.h"
+
+//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(shinstance *, struct job *);
+STATIC void freejob(shinstance *, struct job *);
+STATIC struct job *getjob(shinstance *, const char *, int);
+STATIC shpid dowait(shinstance *, int, struct job *);
+STATIC shpid waitproc(shinstance *, int, struct job *, int *);
+STATIC void cmdtxt(shinstance *, union node *);
+STATIC void cmdlist(shinstance *, union node *, int);
+STATIC void cmdredirlist(shinstance *, union node *, int);
+STATIC void cmdputs(shinstance *, const char *);
+STATIC shpid forkparent(shinstance *psh, struct job *jp, union node *n, int mode, shpid pid);
+STATIC void forkchild(shinstance *psh, shpid pgrp, union node *n, int mode);
+#ifdef KASH_USE_FORKSHELL2
+# ifndef SH_FORKED_MODE
+struct forkshell2args
+{
+ shinstance *psh;
+ int mode;
+ shpid pgrp; /**< The forkchild() pgrp argument (-1 if not in group). */
+ union node *n;
+ void *argp; /**< Points to child callback data following this structure. */
+ int (* child)(shinstance *, union node *, void *);
+ struct stackmark smark; /* do we need this? */
+};
+static int forkshell2_thread(shinstance *psh, void *argp);
+# endif
+#endif
+
+
+/*
+ * 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(shinstance *psh, int on)
+{
+ if (on == psh->jobctl || psh->rootshell == 0)
+ return;
+ if (on) {
+ int err;
+ int i;
+ if (psh->ttyfd != -1)
+ shfile_close(&psh->fdtab, psh->ttyfd);
+ if ((psh->ttyfd = shfile_open(&psh->fdtab, "/dev/tty", O_RDWR, 0)) == -1) {
+ for (i = 0; i < 3; i++) {
+ if (shfile_isatty(&psh->fdtab, i)
+ && (psh->ttyfd = shfile_dup(&psh->fdtab, i)) != -1)
+ break;
+ }
+ if (i == 3)
+ goto out;
+ }
+ /* Move to a high fd */
+ for (i = 10; i > 2; i--) {
+ if ((err = shfile_fcntl(&psh->fdtab, psh->ttyfd, F_DUPFD, (1 << i) - 1)) != -1)
+ break;
+ }
+ if (err != -1) {
+ shfile_close(&psh->fdtab, psh->ttyfd);
+ psh->ttyfd = err;
+ }
+ err = shfile_cloexec(&psh->fdtab, psh->ttyfd, 1);
+ if (err == -1) {
+ shfile_close(&psh->fdtab, psh->ttyfd);
+ psh->ttyfd = -1;
+ goto out;
+ }
+ do { /* while we are in the background */
+ if ((psh->initialpgrp = sh_tcgetpgrp(psh, psh->ttyfd)) < 0) {
+out:
+ out2str(psh, "sh: can't access tty; job control turned off\n");
+ mflag(psh) = 0;
+ return;
+ }
+ if (psh->initialpgrp == -1)
+ psh->initialpgrp = sh_getpgrp(psh);
+ else if (psh->initialpgrp != sh_getpgrp(psh)) {
+ sh_killpg(psh, 0, SIGTTIN);
+ continue;
+ }
+ } while (0);
+
+ setsignal(psh, SIGTSTP);
+ setsignal(psh, SIGTTOU);
+ setsignal(psh, SIGTTIN);
+ if (sh_getpgid(psh, 0) != psh->rootpid && sh_setpgid(psh, 0, psh->rootpid) == -1)
+ error(psh, "Cannot set process group (%s) at %d",
+ sh_strerror(psh, errno), __LINE__);
+ if (sh_tcsetpgrp(psh, psh->ttyfd, psh->rootpid) == -1)
+ error(psh, "Cannot set tty process group (%s) at %d",
+ sh_strerror(psh, errno), __LINE__);
+ } else { /* turning job control off */
+ if (sh_getpgid(psh, 0) != psh->initialpgrp && sh_setpgid(psh, 0, psh->initialpgrp) == -1)
+ error(psh, "Cannot set process group (%s) at %d",
+ sh_strerror(psh, errno), __LINE__);
+ if (sh_tcsetpgrp(psh, psh->ttyfd, psh->initialpgrp) == -1)
+ error(psh, "Cannot set tty process group (%s) at %d",
+ sh_strerror(psh, errno), __LINE__);
+ shfile_close(&psh->fdtab, psh->ttyfd);
+ psh->ttyfd = -1;
+ setsignal(psh, SIGTSTP);
+ setsignal(psh, SIGTTOU);
+ setsignal(psh, SIGTTIN);
+ }
+ psh->jobctl = on;
+}
+
+
+#ifdef mkinit
+INCLUDE <stdlib.h>
+
+SHELLPROC {
+ psh->backgndpid = -1;
+#if JOBS
+ psh->jobctl = 0;
+#endif
+}
+
+#endif
+
+
+
+#if JOBS
+int
+fgcmd(shinstance *psh, int argc, char **argv)
+{
+ struct job *jp;
+ int i;
+ int status;
+
+ nextopt(psh, "");
+ jp = getjob(psh, *psh->argptr, 0);
+ if (jp->jobctl == 0)
+ error(psh, "job not created under job control");
+ out1fmt(psh, "%s", jp->ps[0].cmd);
+ for (i = 1; i < jp->nprocs; i++)
+ out1fmt(psh, " | %s", jp->ps[i].cmd );
+ out1c(psh, '\n');
+ output_flushall(psh);
+
+ for (i = 0; i < jp->nprocs; i++)
+ if (sh_tcsetpgrp(psh, psh->ttyfd, jp->ps[i].pid) != -1)
+ break;
+
+ if (i >= jp->nprocs) {
+ error(psh, "Cannot set tty process group (%s) at %d",
+ sh_strerror(psh, errno), __LINE__);
+ }
+ restartjob(psh, jp);
+ INTOFF;
+ status = waitforjob(psh, jp);
+ INTON;
+ return status;
+}
+
+static void
+set_curjob(shinstance *psh, struct job *jp, int mode)
+{
+ struct job *jp1, *jp2;
+ int i, ji;
+
+ ji = (int)(jp - psh->jobtab);
+
+ /* first remove from list */
+ if (ji == psh->curjob)
+ psh->curjob = jp->prev_job;
+ else {
+ for (i = 0; i < psh->njobs; i++) {
+ if (psh->jobtab[i].prev_job != ji)
+ continue;
+ psh->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 (psh->curjob != -1 && psh->jobtab[psh->curjob].state == JOBSTOPPED) {
+ for (jp1 = psh->jobtab + psh->curjob; ; jp1 = jp2) {
+ if (jp1->prev_job == -1)
+ break;
+ jp2 = psh->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 psh->curjob */
+ jp->prev_job = psh->curjob;
+ psh->curjob = ji;
+ break;
+ }
+}
+
+int
+bgcmd(shinstance *psh, int argc, char **argv)
+{
+ struct job *jp;
+ int i;
+
+ nextopt(psh, "");
+ do {
+ jp = getjob(psh, *psh->argptr, 0);
+ if (jp->jobctl == 0)
+ error(psh, "job not created under job control");
+ set_curjob(psh, jp, 1);
+ out1fmt(psh, "[%ld] %s", (long)(jp - psh->jobtab + 1), jp->ps[0].cmd);
+ for (i = 1; i < jp->nprocs; i++)
+ out1fmt(psh, " | %s", jp->ps[i].cmd );
+ out1c(psh, '\n');
+ output_flushall(psh);
+ restartjob(psh, jp);
+ } while (*psh->argptr && *++psh->argptr);
+ return 0;
+}
+
+
+STATIC void
+restartjob(shinstance *psh, struct job *jp)
+{
+ struct procstat *ps;
+ int i;
+
+ if (jp->state == JOBDONE)
+ return;
+ INTOFF;
+ for (i = 0; i < jp->nprocs; i++)
+ if (sh_killpg(psh, jp->ps[i].pid, SIGCONT) != -1)
+ break;
+ if (i >= jp->nprocs)
+ error(psh, "Cannot continue job (%s)", sh_strerror(psh, errno));
+ for (ps = jp->ps, i = jp->nprocs ; --i >= 0 ; ps++) {
+ if (WIFSTOPPED(ps->status)) {
+ ps->status = -1;
+ jp->state = JOBRUNNING;
+ }
+ }
+ INTON;
+}
+#endif
+
+static void
+showjob(shinstance *psh, struct output *out, struct job *jp, int mode)
+{
+ int procno;
+ int st;
+ struct procstat *ps;
+ size_t col;
+ char s[64];
+
+#if JOBS
+ if (mode & SHOW_PGID) {
+ /* just output process (group) id of pipeline */
+ outfmt(out, "%" SHPID_PRI "\n", 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)) {
+ TRACE((psh, "showjob: freeing job %d\n", jp - psh->jobtab + 1));
+ freejob(psh, jp);
+ }
+ return;
+ }
+
+ for (ps = jp->ps; --procno >= 0; ps++) { /* for each process */
+ if (ps == jp->ps)
+ fmtstr(s, 16, "[%ld] %c ",
+ (long)(jp - psh->jobtab + 1),
+#if JOBS
+ jp == psh->jobtab + psh->curjob ? '+' :
+ psh->curjob != -1 && jp == psh->jobtab +
+ psh->jobtab[psh->curjob].prev_job ? '-' :
+#endif
+ ' ');
+ else
+ fmtstr(s, 16, " " );
+ col = strlen(s);
+ if (mode & SHOW_PID) {
+ fmtstr(s + col, 16, "%" SHPID_PRI " ", 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 {
+ const char *pszSigNm;
+#if JOBS
+ if (WIFSTOPPED(ps->status))
+ st = WSTOPSIG(ps->status);
+ else /* WIFSIGNALED(ps->status) */
+#endif
+ st = WTERMSIG(ps->status);
+ st &= 0x7f;
+ pszSigNm = st < NSIG ? strsignal(st) : NULL;
+ if (pszSigNm)
+ scopyn(pszSigNm, s + col, 32);
+ else
+ fmtstr(s + col, 16, "Signal %d", st);
+ 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->changed = 0;
+ if (jp->state == JOBDONE && !(mode & SHOW_NO_FREE))
+ freejob(psh, jp);
+}
+
+
+int
+jobscmd(shinstance *psh, int argc, char **argv)
+{
+ int mode, m;
+ int sv = psh->jobs_invalid;
+
+ psh->jobs_invalid = 0;
+ mode = 0;
+ while ((m = nextopt(psh, "lp")))
+ if (m == 'l')
+ mode = SHOW_PID;
+ else
+ mode = SHOW_PGID;
+ if (*psh->argptr)
+ do
+ showjob(psh, psh->out1, getjob(psh, *psh->argptr,0), mode);
+ while (*++psh->argptr);
+ else
+ showjobs(psh, psh->out1, mode);
+ psh->jobs_invalid = sv;
+ 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(shinstance *psh, struct output *out, int mode)
+{
+ int jobno;
+ struct job *jp;
+ int silent = 0;
+ shpid gotpid;
+
+ TRACE((psh, "showjobs(%x) called\n", mode));
+
+ /* If not even one one job changed, there is nothing to do */
+ gotpid = dowait(psh, 0, NULL);
+ while (dowait(psh, 0, NULL) > 0)
+ continue;
+#ifdef JOBS
+ /*
+ * Check if we are not in our foreground group, and if not
+ * put us in it.
+ */
+ if (mflag(psh) && gotpid != -1 && sh_tcgetpgrp(psh, psh->ttyfd) != sh_getpid(psh)) {
+ if (sh_tcsetpgrp(psh, psh->ttyfd, sh_getpid(psh)) == -1)
+ error(psh, "Cannot set tty process group (%s) at %d",
+ sh_strerror(psh, errno), __LINE__);
+ TRACE((psh, "repaired tty process group\n"));
+ silent = 1;
+ }
+#endif
+ if (psh->jobs_invalid)
+ return;
+
+ for (jobno = 1, jp = psh->jobtab ; jobno <= psh->njobs ; jobno++, jp++) {
+ if (!jp->used)
+ continue;
+ if (jp->nprocs == 0) {
+ freejob(psh, jp);
+ continue;
+ }
+ if ((mode & SHOW_CHANGED) && !jp->changed)
+ continue;
+ if (silent && jp->changed) {
+ jp->changed = 0;
+ continue;
+ }
+ showjob(psh, out, jp, mode);
+ }
+}
+
+/*
+ * Mark a job structure as unused.
+ */
+
+STATIC void
+freejob(shinstance *psh, struct job *jp)
+{
+ INTOFF;
+ if (jp->ps != &jp->ps0) {
+ ckfree(psh, jp->ps);
+ jp->ps = &jp->ps0;
+ }
+ jp->nprocs = 0;
+ jp->used = 0;
+#if JOBS
+ set_curjob(psh, jp, 0);
+#endif
+ INTON;
+}
+
+
+
+int
+waitcmd(shinstance *psh, int argc, char **argv)
+{
+ struct job *job;
+ int status, retval;
+ struct job *jp;
+
+ nextopt(psh, "");
+
+ if (!*psh->argptr) {
+ /* wait for all jobs */
+ jp = psh->jobtab;
+ if (psh->jobs_invalid)
+ return 0;
+ for (;;) {
+ if (jp >= psh->jobtab + psh->njobs) {
+ /* no running procs */
+ return 0;
+ }
+ if (!jp->used || jp->state != JOBRUNNING) {
+ jp++;
+ continue;
+ }
+ if (dowait(psh, 1, (struct job *)NULL) == -1)
+ return 128 + SIGINT;
+ jp = psh->jobtab;
+ }
+ }
+
+ retval = 127; /* XXXGCC: -Wuninitialized */
+ for (; *psh->argptr; psh->argptr++) {
+ job = getjob(psh, *psh->argptr, 1);
+ if (!job) {
+ retval = 127;
+ continue;
+ }
+ /* loop until process terminated or stopped */
+ while (job->state == JOBRUNNING) {
+ if (dowait(psh, 1, (struct job *)NULL) == -1)
+ return 128 + SIGINT;
+ }
+ status = job->ps[job->nprocs].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;
+ }
+ if (!iflag(psh))
+ freejob(psh, job);
+ }
+ return retval;
+}
+
+
+
+int
+jobidcmd(shinstance *psh, int argc, char **argv)
+{
+ struct job *jp;
+ int i;
+
+ nextopt(psh, "");
+ jp = getjob(psh, *psh->argptr, 0);
+ for (i = 0 ; i < jp->nprocs ; ) {
+ out1fmt(psh, "%" SHPID_PRI, jp->ps[i].pid);
+ out1c(psh, ++i < jp->nprocs ? ' ' : '\n');
+ }
+ return 0;
+}
+
+shpid
+getjobpgrp(shinstance *psh, const char *name)
+{
+ struct job *jp;
+
+ jp = getjob(psh, name, 1);
+ if (jp == 0)
+ return 0;
+ return -jp->ps[0].pid;
+}
+
+/*
+ * Convert a job name to a job structure.
+ */
+
+STATIC struct job *
+getjob(shinstance *psh, 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 = psh->curjob;
+#endif
+ err_msg = "No current job";
+ } else if (name[0] == '%') {
+ if (is_number(name + 1)) {
+ jobno = number(psh, name + 1) - 1;
+ } else if (!name[2]) {
+ switch (name[1]) {
+#if JOBS
+ case 0:
+ case '+':
+ case '%':
+ jobno = psh->curjob;
+ err_msg = "No current job";
+ break;
+ case '-':
+ jobno = psh->curjob;
+ if (jobno != -1)
+ jobno = psh->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 = psh->jobtab, i = psh->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(psh, name);
+ for (jp = psh->jobtab, i = psh->njobs ; --i >= 0 ; jp++) {
+ if (jp->used && jp->nprocs > 0
+ && jp->ps[jp->nprocs - 1].pid == pid)
+ return jp;
+ }
+ }
+
+ if (!psh->jobs_invalid && jobno >= 0 && jobno < psh->njobs) {
+ jp = psh->jobtab + jobno;
+ if (jp->used)
+ return jp;
+ }
+ if (!noerror)
+ error(psh, err_msg, name);
+ return 0;
+}
+
+
+
+/*
+ * Return a new job structure,
+ */
+
+struct job *
+makejob(shinstance *psh, union node *node, int nprocs)
+{
+ int i;
+ struct job *jp;
+
+ if (psh->jobs_invalid) {
+ for (i = psh->njobs, jp = psh->jobtab ; --i >= 0 ; jp++) {
+ if (jp->used)
+ freejob(psh, jp);
+ }
+ psh->jobs_invalid = 0;
+ }
+
+ for (i = psh->njobs, jp = psh->jobtab ; ; jp++) {
+ if (--i < 0) {
+ INTOFF;
+ if (psh->njobs == 0) {
+ psh->jobtab = ckmalloc(psh, 4 * sizeof psh->jobtab[0]);
+ } else {
+ jp = ckmalloc(psh, (psh->njobs + 4) * sizeof psh->jobtab[0]);
+ memcpy(jp, psh->jobtab, psh->njobs * sizeof jp[0]);
+ /* Relocate `ps' pointers */
+ for (i = 0; i < psh->njobs; i++)
+ if (jp[i].ps == &psh->jobtab[i].ps0)
+ jp[i].ps = &jp[i].ps0;
+ ckfree(psh, psh->jobtab);
+ psh->jobtab = jp;
+ }
+ jp = psh->jobtab + psh->njobs;
+ for (i = 4 ; --i >= 0 ; psh->jobtab[psh->njobs++].used = 0) { /*empty*/ }
+ INTON;
+ break;
+ }
+ if (jp->used == 0)
+ break;
+ }
+ INTOFF;
+ jp->state = JOBRUNNING;
+ jp->used = 1;
+ jp->changed = 0;
+ jp->nprocs = 0;
+#if JOBS
+ jp->jobctl = psh->jobctl;
+ set_curjob(psh, jp, 1);
+#endif
+ if (nprocs > 1) {
+ jp->ps = ckmalloc(psh, nprocs * sizeof (struct procstat));
+ } else {
+ jp->ps = &jp->ps0;
+ }
+ INTON;
+ TRACE((psh, "makejob(0x%lx, %d) returns %%%d\n", (long)node, nprocs,
+ jp - psh->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).
+ */
+
+#ifndef KASH_USE_FORKSHELL2
+shpid
+forkshell(shinstance *psh, struct job *jp, union node *n, int mode)
+{
+ int pid;
+
+ TRACE((psh, "forkshell(%%%d, %p, %d) called\n", jp - psh->jobtab, n, mode));
+ switch ((pid = sh_fork(psh))) {
+ case -1:
+ TRACE((psh, "Fork failed, errno=%d\n", errno));
+ INTON;
+ error(psh, "Cannot fork");
+ return -1; /* won't get here */
+ case 0:
+ forkchild(psh, jp == NULL || jp->nprocs == 0 ? -1 : jp->ps[0].pid, n, mode);
+ return 0;
+ default:
+ return forkparent(psh, jp, n, mode, pid);
+ }
+}
+#else /* KASH_USE_FORKSHELL2 */
+shpid
+forkshell2(shinstance *psh, struct job *jp, union node *n, int mode,
+ int (*child)(struct shinstance *, void *, union node *),
+ union node *nchild, void *argp, size_t arglen,
+ void (*setupchild)(struct shinstance *, struct shinstance *, void *))
+{
+ shpid pid;
+
+# ifdef SH_FORKED_MODE
+ /*
+ * fork variant.
+ */
+ pid = sh_fork(psh);
+ if (pid == 0)
+ {
+ /* child */
+ forkchild(psh, jp == NULL || jp->nprocs == 0 ? -1 : jp->ps[0].pid, n, mode);
+ sh__exit(psh, child(psh, nchild, argp));
+ return 0;
+ }
+
+ /* parent */
+ if (pid != -1)
+ return forkparent(psh, jp, n, mode, pid);
+ TRACE((psh, "Fork failed, errno=%d\n", errno));
+ INTON;
+ (void)arglen;
+ (void)setupchild;
+ error(psh, "Cannot fork");
+ return -1; /* won't get here */
+
+# else
+ /*
+ * Clone the shell and start a thread to service the subshell.
+ */
+ struct shinstance *pshchild;
+
+ TRACE((psh, "forkshell2(%%%d, %p, %d, %p, %p, %p, %d) called\n",
+ jp - psh->jobtab, n, mode, child, nchild, argp, (int)arglen));
+
+ pshchild = sh_create_child_shell(psh);
+ if (pshchild) {
+ /* pack arguments */
+ struct forkshell2args *args = (struct forkshell2args *)sh_calloc(pshchild, sizeof(*args) + arglen, 1);
+ args->psh = pshchild;
+ args->argp = memcpy(args + 1, argp, arglen);
+ args->child = child;
+ args->mode = mode;
+ args->pgrp = jp == NULL || jp->nprocs == 0 ? -1 : jp->ps[0].pid;
+ setstackmark(pshchild, &args->smark);
+ args->n = copyparsetree(pshchild, n);
+ if (setupchild)
+ setupchild(pshchild, psh, args->argp);
+
+ /* start the thread */
+ pid = sh_thread_start(psh, pshchild, forkshell2_thread, args);
+ if (pid >= 0)
+ return forkparent(psh, jp, n, mode, pid);
+ error(psh, "sh_start_child_thread failed (%d)!", (int)pid);
+ }
+ else
+ error(psh, "sh_create_child_shell failed!");
+ return -1;
+# endif
+}
+#endif
+
+STATIC shpid
+forkparent(shinstance *psh, struct job *jp, union node *n, int mode, shpid pid)
+{
+ shpid pgrp;
+
+ if (psh->rootshell && mode != FORK_NOJOB && mflag(psh)) {
+ if (jp == NULL || jp->nprocs == 0)
+ pgrp = pid;
+ else
+ pgrp = jp->ps[0].pid;
+ /* This can fail because we are doing it in the child also */
+ (void)sh_setpgid(psh, pid, pgrp);
+ }
+ if (mode == FORK_BG)
+ psh->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(psh, ps, n);
+ }
+ TRACE((psh, "In parent shell: child = %" SHPID_PRI "\n", pid));
+ return pid;
+}
+
+STATIC void
+forkchild(shinstance *psh, shpid pgrp, union node *n, int mode)
+{
+ int wasroot;
+ const char *devnull = _PATH_DEVNULL;
+ const char *nullerr = "Can't open %s";
+
+ wasroot = psh->rootshell;
+ TRACE((psh, "Child shell %" SHPID_PRI "\n", sh_getpid(psh)));
+ psh->rootshell = 0;
+
+ closescript(psh);
+ clear_traps(psh);
+#if JOBS
+ psh->jobctl = 0; /* do job control only in root shell */
+ if (wasroot && mode != FORK_NOJOB && mflag(psh)) {
+ if (pgrp == -1)
+ pgrp = sh_getpid(psh);
+ /* This can fail because we are doing it in the parent also.
+ And we must ignore SIGTTOU at this point or we'll be stopped! */
+ (void)sh_setpgid(psh, 0, pgrp);
+ if (mode == FORK_FG) {
+ if (sh_tcsetpgrp(psh, psh->ttyfd, pgrp) == -1)
+ error(psh, "Cannot set tty process group (%s) at %d",
+ sh_strerror(psh, errno), __LINE__);
+ }
+ setsignal(psh, SIGTSTP);
+ setsignal(psh, SIGTTOU);
+ } else
+#endif
+ if (mode == FORK_BG) {
+ ignoresig(psh, SIGINT);
+ ignoresig(psh, SIGQUIT);
+ if (pgrp == -1 && ! fd0_redirected_p(psh)) {
+ shfile_close(&psh->fdtab, 0);
+ if (shfile_open(&psh->fdtab, devnull, O_RDONLY, 0) != 0)
+ error(psh, nullerr, devnull);
+ }
+ }
+ if (wasroot && iflag(psh)) {
+ setsignal(psh, SIGINT);
+ setsignal(psh, SIGQUIT);
+ setsignal(psh, SIGTERM);
+ }
+
+ psh->jobs_invalid = 1;
+}
+
+#if defined(KASH_USE_FORKSHELL2) && !defined(SH_FORKED_MODE)
+/** thread procedure */
+static int forkshell2_thread(shinstance *psh, void *argp)
+{
+ struct forkshell2args * volatile volargs = (struct forkshell2args *)argp;
+ struct jmploc jmp;
+ TRACE2((psh, "forkshell2_thread:\n"));
+
+ if (setjmp(jmp.loc) == 0) {
+ struct forkshell2args * const args = volargs;
+
+ forkchild(psh, args->pgrp, args->n, args->mode);
+
+ psh->handler = &jmp;
+ return args->child(psh, args->n, args->argp);
+ }
+
+ /*
+ * (We longjmp'ed here.)
+ *
+ * This is copied from main() and simplified:
+ */
+ for (;;) {
+ psh = volargs->psh; /* longjmp paranoia */
+
+ if (psh->exception != EXSHELLPROC) {
+ if (psh->exception == EXEXEC)
+ psh->exitstatus = psh->exerrno;
+ else if (psh->exception == EXERROR)
+ psh->exitstatus = 2;
+ TRACE2((psh, "forkshell2_thread: exception=%d -> exitshell2(,%d)\n", psh->exception, psh->exitstatus));
+ return exitshell2(psh, psh->exitstatus);
+ }
+
+ /* EXSHELLPROC - tryexec gets us here and it wants to run a program
+ hoping (?) it's a shell script. We must reset the shell state and
+ turn ourselves into a root shell before doing so. */
+ TRACE2((psh, "forkshell2_thread: exception=EXSHELLPROC\n"));
+ psh->rootpid = /*getpid()*/ psh->pid;
+ psh->rootshell = 1;
+ psh->minusc = NULL;
+
+ reset(psh);
+ popstackmark(psh, &volargs->smark);
+
+ FORCEINTON; /* enable interrupts */
+
+ /* state3: */
+ if (sflag(psh) == 0) {
+# ifdef SIGTSTP
+ static int sigs[] = { SIGINT, SIGQUIT, SIGHUP, SIGPIPE, SIGTSTP };
+# else
+ static int sigs[] = { SIGINT, SIGQUIT, SIGHUP, SIGPIPE };
+# endif
+ unsigned i;
+ for (i = 0; i < K_ELEMENTS(sigs); i++)
+ setsignal(psh, sigs[i]);
+ }
+
+ if (setjmp(jmp.loc) == 0) {
+ psh->handler = &jmp;
+ cmdloop(psh, 1);
+ TRACE2((psh, "forkshell2_thread: cmdloop returned -> exitshell2(,%d)\n", psh->exitstatus));
+ return exitshell2(psh, psh->exitstatus);
+ }
+ }
+}
+#endif
+
+
+/*
+ * 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(shinstance *psh, struct job *jp)
+{
+#if JOBS
+ shpid mypgrp = sh_getpgrp(psh);
+#endif
+ int status;
+ int st;
+
+ INTOFF;
+ TRACE((psh, "waitforjob(%%%d) called\n", jp - psh->jobtab + 1));
+ while (jp->state == JOBRUNNING) {
+ dowait(psh, 1, jp);
+ }
+#if JOBS
+ if (jp->jobctl) {
+ if (sh_tcsetpgrp(psh, psh->ttyfd, mypgrp) == -1)
+ error(psh, "Cannot set tty process group (%s) at %d",
+ sh_strerror(psh, errno), __LINE__);
+ }
+ if (jp->state == JOBSTOPPED && psh->curjob != jp - psh->jobtab)
+ set_curjob(psh, jp, 2);
+#endif
+ status = jp->ps[jp->nprocs - 1].status;
+ /* 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;
+ TRACE((psh, "waitforjob: job %d, nproc %d, status %x, st %x\n",
+ jp - psh->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)
+ sh_raise_sigint(psh);/*raise(SIGINT);*/
+ }
+#endif
+ if (! JOBS || jp->state == JOBDONE)
+ freejob(psh, jp);
+ INTON;
+ return st;
+}
+
+
+
+/*
+ * Wait for a process to terminate.
+ */
+
+STATIC shpid
+dowait(shinstance *psh, int block, struct job *job)
+{
+ shpid pid;
+ int status;
+ struct procstat *sp;
+ struct job *jp;
+ struct job *thisjob;
+ int done;
+ int stopped;
+
+ TRACE((psh, "dowait(%d) called\n", block));
+ do {
+ pid = waitproc(psh, block, job, &status);
+ TRACE((psh, "wait returns pid %" SHPID_PRI ", status %d\n", pid, status));
+ } while (pid == -1 && errno == EINTR && psh->gotsig[SIGINT - 1] == 0);
+ if (pid <= 0)
+ return pid;
+ INTOFF;
+ thisjob = NULL;
+ for (jp = psh->jobtab ; jp < psh->jobtab + psh->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) {
+ TRACE((psh, "Job %d: changing status of proc %" SHPID_PRI " from 0x%x to 0x%x\n",
+ jp - psh->jobtab + 1, pid, sp->status, status));
+ sp->status = status;
+ thisjob = 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) {
+ TRACE((psh, "Job %d: changing state from %d to %d\n", jp - psh->jobtab + 1, jp->state, state));
+ jp->state = state;
+#if JOBS
+ if (done)
+ set_curjob(psh, jp, 0);
+#endif
+ }
+ }
+ }
+ }
+
+ if (thisjob && thisjob->state != JOBRUNNING) {
+ int mode = 0;
+ if (!psh->rootshell || !iflag(psh))
+ mode = SHOW_SIGNALLED;
+ if (job == thisjob)
+ mode = SHOW_SIGNALLED | SHOW_NO_FREE;
+ if (mode)
+ showjob(psh, psh->out2, thisjob, mode);
+ else {
+ TRACE((psh, "Not printing status, rootshell=%d, job=%p\n",
+ psh->rootshell, job));
+ thisjob->changed = 1;
+ }
+ }
+
+ 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.
+ */
+STATIC shpid
+waitproc(shinstance *psh, int block, struct job *jp, int *status)
+{
+ int flags = 0;
+
+#if JOBS
+ if (jp != NULL && jp->jobctl)
+ flags |= WUNTRACED;
+#endif
+ if (block == 0)
+ flags |= WNOHANG;
+ return sh_waitpid(psh, -1, status, flags);
+}
+
+/*
+ * return 1 if there are stopped jobs, otherwise 0
+ */
+//int job_warning = 0;
+int
+stoppedjobs(shinstance *psh)
+{
+ int jobno;
+ struct job *jp;
+
+ if (psh->job_warning || psh->jobs_invalid)
+ return (0);
+ for (jobno = 1, jp = psh->jobtab; jobno <= psh->njobs; jobno++, jp++) {
+ if (jp->used == 0)
+ continue;
+ if (jp->state == JOBSTOPPED) {
+ out2str(psh, "You have stopped jobs.\n");
+ psh->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(shinstance *psh, struct procstat *ps, union node *n)
+{
+ int len;
+
+ psh->cmdnextc = ps->cmd;
+ if (iflag(psh) || mflag(psh) || sizeof(ps->cmd) < 100)
+ len = sizeof(ps->cmd);
+ else
+ len = sizeof(ps->cmd) / 10;
+ psh->cmdnleft = len;
+ cmdtxt(psh, n);
+ if (psh->cmdnleft <= 0) {
+ char *p = ps->cmd + len - 4;
+ p[0] = '.';
+ p[1] = '.';
+ p[2] = '.';
+ p[3] = 0;
+ } else
+ *psh->cmdnextc = '\0';
+ TRACE((psh, "commandtext: ps->cmd %x, end %x, left %d\n\t\"%s\"\n",
+ ps->cmd, psh->cmdnextc, psh->cmdnleft, ps->cmd));
+}
+
+
+STATIC void
+cmdtxt(shinstance *psh, union node *n)
+{
+ union node *np;
+ struct nodelist *lp;
+ const char *p;
+ int i;
+ char s[2];
+
+ if (n == NULL || psh->cmdnleft <= 0)
+ return;
+ switch (n->type) {
+ case NSEMI:
+ cmdtxt(psh, n->nbinary.ch1);
+ cmdputs(psh, "; ");
+ cmdtxt(psh, n->nbinary.ch2);
+ break;
+ case NAND:
+ cmdtxt(psh, n->nbinary.ch1);
+ cmdputs(psh, " && ");
+ cmdtxt(psh, n->nbinary.ch2);
+ break;
+ case NOR:
+ cmdtxt(psh, n->nbinary.ch1);
+ cmdputs(psh, " || ");
+ cmdtxt(psh, n->nbinary.ch2);
+ break;
+ case NPIPE:
+ for (lp = n->npipe.cmdlist ; lp ; lp = lp->next) {
+ cmdtxt(psh, lp->n);
+ if (lp->next)
+ cmdputs(psh, " | ");
+ }
+ break;
+ case NSUBSHELL:
+ cmdputs(psh, "(");
+ cmdtxt(psh, n->nredir.n);
+ cmdputs(psh, ")");
+ break;
+ case NREDIR:
+ case NBACKGND:
+ cmdtxt(psh, n->nredir.n);
+ break;
+ case NIF:
+ cmdputs(psh, "if ");
+ cmdtxt(psh, n->nif.test);
+ cmdputs(psh, "; then ");
+ cmdtxt(psh, n->nif.ifpart);
+ if (n->nif.elsepart) {
+ cmdputs(psh, "; else ");
+ cmdtxt(psh, n->nif.elsepart);
+ }
+ cmdputs(psh, "; fi");
+ break;
+ case NWHILE:
+ cmdputs(psh, "while ");
+ goto until;
+ case NUNTIL:
+ cmdputs(psh, "until ");
+until:
+ cmdtxt(psh, n->nbinary.ch1);
+ cmdputs(psh, "; do ");
+ cmdtxt(psh, n->nbinary.ch2);
+ cmdputs(psh, "; done");
+ break;
+ case NFOR:
+ cmdputs(psh, "for ");
+ cmdputs(psh, n->nfor.var);
+ cmdputs(psh, " in ");
+ cmdlist(psh, n->nfor.args, 1);
+ cmdputs(psh, "; do ");
+ cmdtxt(psh, n->nfor.body);
+ cmdputs(psh, "; done");
+ break;
+ case NCASE:
+ cmdputs(psh, "case ");
+ cmdputs(psh, n->ncase.expr->narg.text);
+ cmdputs(psh, " in ");
+ for (np = n->ncase.cases; np; np = np->nclist.next) {
+ cmdtxt(psh, np->nclist.pattern);
+ cmdputs(psh, ") ");
+ cmdtxt(psh, np->nclist.body);
+ cmdputs(psh, ";; ");
+ }
+ cmdputs(psh, "esac");
+ break;
+ case NDEFUN:
+ cmdputs(psh, n->narg.text);
+ cmdputs(psh, "() { ... }");
+ break;
+ case NCMD:
+ cmdlist(psh, n->ncmd.args, 1);
+ cmdredirlist(psh, n->ncmd.redirect, 0);
+ break;
+ case NARG:
+ cmdputs(psh, 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) {
+ s[0] = n->nfile.fd + '0';
+ s[1] = '\0';
+ cmdputs(psh, s);
+ }
+ cmdputs(psh, p);
+ if (n->type == NTOFD || n->type == NFROMFD) {
+ s[0] = n->ndup.dupfd + '0';
+ s[1] = '\0';
+ cmdputs(psh, s);
+ } else {
+ cmdtxt(psh, n->nfile.fname);
+ }
+ break;
+ case NHERE:
+ case NXHERE:
+ cmdputs(psh, "<<...");
+ break;
+ default:
+ cmdputs(psh, "???");
+ break;
+ }
+}
+
+STATIC void
+cmdlist(shinstance *psh, union node *np, int sep)
+{
+ for (; np; np = np->narg.next) {
+ if (!sep)
+ cmdputs(psh, " ");
+ cmdtxt(psh, np);
+ if (sep && np->narg.next)
+ cmdputs(psh, " ");
+ }
+}
+
+STATIC void
+cmdredirlist(shinstance *psh, union node *np, int sep)
+{
+ for (; np; np = np->nfile.next) {
+ if (!sep)
+ cmdputs(psh, " ");
+ cmdtxt(psh, np);
+ if (sep && np->nfile.next)
+ cmdputs(psh, " ");
+ }
+}
+
+
+STATIC void
+cmdputs(shinstance *psh, 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 = psh->cmdnextc;
+ nleft = psh->cmdnleft;
+ while (nleft > 0 && (c = *p++) != 0) {
+ switch (c) {
+ case CTLESC:
+ c = *p++;
+ break;
+ case CTLVAR:
+ subtype = *p++;
+ if ((subtype & VSTYPE) == VSLENGTH)
+ str = "${#";
+ else
+ str = "${";
+ if (!(subtype & VSQUOTE) != !(quoted & 1)) {
+ quoted ^= 1;
+ c = '"';
+ } else
+ c = *str++;
+ break;
+ case CTLENDVAR:
+ if (quoted & 1) {
+ c = '"';
+ str = "}";
+ } else
+ c = '}';
+ quoted >>= 1;
+ subtype = 0;
+ break;
+ case CTLBACKQ:
+ c = '$';
+ str = "(...)";
+ break;
+ case CTLBACKQ+CTLQUOTE:
+ c = '"';
+ str = "$(...)\"";
+ break;
+ case CTLARI:
+ c = '$';
+ str = "((";
+ break;
+ case CTLENDARI:
+ c = ')';
+ str = ")";
+ break;
+ case CTLQUOTEMARK:
+ quoted ^= 1;
+ c = '"';
+ break;
+ case '=':
+ if (subtype == 0)
+ break;
+ str = vstype[subtype & VSTYPE];
+ if (subtype & VSNUL)
+ c = ':';
+ else
+ c = *str++;
+ if (c != '}')
+ quoted <<= 1;
+ break;
+ case '\'':
+ case '\\':
+ case '"':
+ case '$':
+ /* These can only happen inside quotes */
+ cc[0] = c;
+ str = cc;
+ c = '\\';
+ break;
+ default:
+ break;
+ }
+ do {
+ *nextc++ = c;
+ } while (--nleft > 0 && str && (c = *str++));
+ str = 0;
+ }
+ if ((quoted & 1) && nleft) {
+ *nextc++ = '"';
+ nleft--;
+ }
+ psh->cmdnleft = nleft;
+ psh->cmdnextc = nextc;
+}