|  | #define NOPLAN9DEFINES | 
|  | #include	"mk.h" | 
|  | #include	<sys/wait.h> | 
|  | #include	<signal.h> | 
|  | #include	<sys/stat.h> | 
|  | #include	<sys/time.h> | 
|  |  | 
|  | char	*shell = "/bin/sh"; | 
|  | char	*shellname = "sh"; | 
|  |  | 
|  | extern char **environ; | 
|  |  | 
|  | static void | 
|  | mkperror(char *s) | 
|  | { | 
|  | fprint(2, "%s: %r\n", s); | 
|  | } | 
|  |  | 
|  | void | 
|  | readenv(void) | 
|  | { | 
|  | char **p, *s; | 
|  | Word *w; | 
|  |  | 
|  | for(p = environ; *p; p++){ | 
|  | /* rsc 5/5/2004 -- This misparses fn#cd={whatever} | 
|  | s = shname(*p); | 
|  | if(*s == '=') { | 
|  | *s = 0; | 
|  | w = newword(s+1); | 
|  | } else | 
|  | w = newword(""); | 
|  | */ | 
|  | s = strchr(*p, '='); | 
|  | if(s){ | 
|  | *s = 0; | 
|  | w = newword(s+1); | 
|  | } else | 
|  | w = newword(""); | 
|  | if (symlook(*p, S_INTERNAL, 0)) | 
|  | continue; | 
|  | s = strdup(*p); | 
|  | setvar(s, (void *)w); | 
|  | symlook(s, S_EXPORTED, (void*)"")->u.ptr = ""; | 
|  | } | 
|  | } | 
|  |  | 
|  | /* | 
|  | *	done on child side of fork, so parent's env is not affected | 
|  | *	and we don't care about freeing memory because we're going | 
|  | *	to exec immediately after this. | 
|  | */ | 
|  | void | 
|  | exportenv(Envy *e, Shell *sh) | 
|  | { | 
|  | int i; | 
|  | char **p; | 
|  | static char buf[16384]; | 
|  |  | 
|  | p = 0; | 
|  | for(i = 0; e->name; e++, i++) { | 
|  | p = (char**) Realloc(p, (i+2)*sizeof(char*)); | 
|  | if(e->values) | 
|  | snprint(buf, sizeof buf, "%s=%s", e->name,  wtos(e->values, sh->iws)); | 
|  | else | 
|  | snprint(buf, sizeof buf, "%s=", e->name); | 
|  | p[i] = strdup(buf); | 
|  | } | 
|  | p[i] = 0; | 
|  | environ = p; | 
|  | } | 
|  |  | 
|  | int | 
|  | waitfor(char *msg) | 
|  | { | 
|  | int status; | 
|  | int pid; | 
|  |  | 
|  | *msg = 0; | 
|  | pid = wait(&status); | 
|  | if(pid > 0) { | 
|  | if(status&0x7f) { | 
|  | if(status&0x80) | 
|  | snprint(msg, ERRMAX, "signal %d, core dumped", status&0x7f); | 
|  | else | 
|  | snprint(msg, ERRMAX, "signal %d", status&0x7f); | 
|  | } else if(status&0xff00) | 
|  | snprint(msg, ERRMAX, "exit(%d)", (status>>8)&0xff); | 
|  | } | 
|  | return pid; | 
|  | } | 
|  |  | 
|  | void | 
|  | expunge(int pid, char *msg) | 
|  | { | 
|  | if(strcmp(msg, "interrupt")) | 
|  | kill(pid, SIGINT); | 
|  | else | 
|  | kill(pid, SIGHUP); | 
|  | } | 
|  |  | 
|  | int mypid; | 
|  |  | 
|  | int | 
|  | shargv(Word *cmd, int extra, char ***pargv) | 
|  | { | 
|  | char **argv; | 
|  | int i, n; | 
|  | Word *w; | 
|  |  | 
|  | n = 0; | 
|  | for(w=cmd; w; w=w->next) | 
|  | n++; | 
|  |  | 
|  | argv = Malloc((n+extra+1)*sizeof(argv[0])); | 
|  | i = 0; | 
|  | for(w=cmd; w; w=w->next) | 
|  | argv[i++] = w->s; | 
|  | argv[n] = 0; | 
|  | *pargv = argv; | 
|  | return n; | 
|  | } | 
|  |  | 
|  | int | 
|  | execsh(char *args, char *cmd, Bufblock *buf, Envy *e, Shell *sh, Word *shellcmd) | 
|  | { | 
|  | char *p, **argv; | 
|  | int tot, n, pid, in[2], out[2]; | 
|  |  | 
|  | if(buf && pipe(out) < 0){ | 
|  | mkperror("pipe"); | 
|  | Exit(); | 
|  | } | 
|  | pid = fork(); | 
|  | mypid = getpid(); | 
|  | if(pid < 0){ | 
|  | mkperror("mk fork"); | 
|  | Exit(); | 
|  | } | 
|  | if(pid == 0){ | 
|  | if(buf) | 
|  | close(out[0]); | 
|  | if(pipe(in) < 0){ | 
|  | mkperror("pipe"); | 
|  | Exit(); | 
|  | } | 
|  | pid = fork(); | 
|  | if(pid < 0){ | 
|  | mkperror("mk fork"); | 
|  | Exit(); | 
|  | } | 
|  | if(pid != 0){ | 
|  | dup2(in[0], 0); | 
|  | if(buf){ | 
|  | dup2(out[1], 1); | 
|  | close(out[1]); | 
|  | } | 
|  | close(in[0]); | 
|  | close(in[1]); | 
|  | if (e) | 
|  | exportenv(e, sh); | 
|  | n = shargv(shellcmd, 1, &argv); | 
|  | argv[n++] = args; | 
|  | argv[n] = 0; | 
|  | execvp(argv[0], argv); | 
|  | mkperror(shell); | 
|  | _exit(1); | 
|  | } | 
|  | close(out[1]); | 
|  | close(in[0]); | 
|  | if(DEBUG(D_EXEC)) | 
|  | fprint(1, "starting: %s\n", cmd); | 
|  | p = cmd+strlen(cmd); | 
|  | while(cmd < p){ | 
|  | n = write(in[1], cmd, p-cmd); | 
|  | if(n < 0) | 
|  | break; | 
|  | cmd += n; | 
|  | } | 
|  | close(in[1]); | 
|  | _exit(0); | 
|  | } | 
|  | if(buf){ | 
|  | close(out[1]); | 
|  | tot = 0; | 
|  | for(;;){ | 
|  | if (buf->current >= buf->end) | 
|  | growbuf(buf); | 
|  | n = read(out[0], buf->current, buf->end-buf->current); | 
|  | if(n <= 0) | 
|  | break; | 
|  | buf->current += n; | 
|  | tot += n; | 
|  | } | 
|  | if (tot && buf->current[-1] == '\n') | 
|  | buf->current--; | 
|  | close(out[0]); | 
|  | } | 
|  | return pid; | 
|  | } | 
|  |  | 
|  | int | 
|  | pipecmd(char *cmd, Envy *e, int *fd, Shell *sh, Word *shellcmd) | 
|  | { | 
|  | int pid, pfd[2]; | 
|  | int n; | 
|  | char **argv; | 
|  |  | 
|  | if(DEBUG(D_EXEC)) | 
|  | fprint(1, "pipecmd='%s'\n", cmd);/**/ | 
|  |  | 
|  | if(fd && pipe(pfd) < 0){ | 
|  | mkperror("pipe"); | 
|  | Exit(); | 
|  | } | 
|  | pid = fork(); | 
|  | if(pid < 0){ | 
|  | mkperror("mk fork"); | 
|  | Exit(); | 
|  | } | 
|  | if(pid == 0){ | 
|  | if(fd){ | 
|  | close(pfd[0]); | 
|  | dup2(pfd[1], 1); | 
|  | close(pfd[1]); | 
|  | } | 
|  | if(e) | 
|  | exportenv(e, sh); | 
|  | n = shargv(shellcmd, 2, &argv); | 
|  | argv[n++] = "-c"; | 
|  | argv[n++] = cmd; | 
|  | argv[n] = 0; | 
|  | execvp(argv[0], argv); | 
|  | mkperror(shell); | 
|  | _exit(1); | 
|  | } | 
|  | if(fd){ | 
|  | close(pfd[1]); | 
|  | *fd = pfd[0]; | 
|  | } | 
|  | return pid; | 
|  | } | 
|  |  | 
|  | void | 
|  | Exit(void) | 
|  | { | 
|  | while(wait(0) >= 0) | 
|  | ; | 
|  | exits("error"); | 
|  | } | 
|  |  | 
|  | static	struct | 
|  | { | 
|  | int	sig; | 
|  | char	*msg; | 
|  | }	sigmsgs[] = | 
|  | { | 
|  | SIGALRM,	"alarm", | 
|  | SIGFPE,		"sys: fp: fptrap", | 
|  | SIGPIPE,	"sys: write on closed pipe", | 
|  | SIGILL,		"sys: trap: illegal instruction", | 
|  | /*	SIGSEGV,	"sys: segmentation violation", */ | 
|  | 0,		0 | 
|  | }; | 
|  |  | 
|  | static void | 
|  | notifyf(int sig) | 
|  | { | 
|  | int i; | 
|  |  | 
|  | for(i = 0; sigmsgs[i].msg; i++) | 
|  | if(sigmsgs[i].sig == sig) | 
|  | killchildren(sigmsgs[i].msg); | 
|  |  | 
|  | /* should never happen */ | 
|  | signal(sig, SIG_DFL); | 
|  | kill(getpid(), sig); | 
|  | } | 
|  |  | 
|  | void | 
|  | catchnotes(void) | 
|  | { | 
|  | int i; | 
|  |  | 
|  | for(i = 0; sigmsgs[i].msg; i++) | 
|  | signal(sigmsgs[i].sig, notifyf); | 
|  | } | 
|  |  | 
|  | char* | 
|  | maketmp(int *pfd) | 
|  | { | 
|  | static char temp[] = "/tmp/mkargXXXXXX"; | 
|  | static char buf[100]; | 
|  | int fd; | 
|  |  | 
|  | strcpy(buf, temp); | 
|  | fd = mkstemp(buf); | 
|  | if(fd < 0) | 
|  | return 0; | 
|  | *pfd = fd; | 
|  | return buf; | 
|  | } | 
|  |  | 
|  | int | 
|  | chgtime(char *name) | 
|  | { | 
|  | if(access(name, 0) >= 0) | 
|  | return utimes(name, 0); | 
|  | return close(creat(name, 0666)); | 
|  | } | 
|  |  | 
|  | void | 
|  | rcopy(char **to, Resub *match, int n) | 
|  | { | 
|  | int c; | 
|  | char *p; | 
|  |  | 
|  | *to = match->s.sp;		/* stem0 matches complete target */ | 
|  | for(to++, match++; --n > 0; to++, match++){ | 
|  | if(match->s.sp && match->e.ep){ | 
|  | p = match->e.ep; | 
|  | c = *p; | 
|  | *p = 0; | 
|  | *to = strdup(match->s.sp); | 
|  | *p = c; | 
|  | } | 
|  | else | 
|  | *to = 0; | 
|  | } | 
|  | } | 
|  |  | 
|  | unsigned long | 
|  | mkmtime(char *name) | 
|  | { | 
|  | struct stat st; | 
|  |  | 
|  | if(stat(name, &st) < 0) | 
|  | return 0; | 
|  |  | 
|  | return st.st_mtime; | 
|  | } |