8322 nl: misleading-indentation
[unleashed/tickless.git] / usr / src / lib / libshell / common / sh / main.c
bloba1dc119bcbbae6c534cd6906dce4739766051c18
1 /***********************************************************************
2 * *
3 * This software is part of the ast package *
4 * Copyright (c) 1982-2010 AT&T Intellectual Property *
5 * and is licensed under the *
6 * Common Public License, Version 1.0 *
7 * by AT&T Intellectual Property *
8 * *
9 * A copy of the License is available at *
10 * http://www.opensource.org/licenses/cpl1.0.txt *
11 * (with md5 checksum 059e8cd6165cb4c31e351f2b69388fd9) *
12 * *
13 * Information and Software Systems Research *
14 * AT&T Research *
15 * Florham Park NJ *
16 * *
17 * David Korn <dgk@research.att.com> *
18 * *
19 ***********************************************************************/
20 #pragma prototyped
22 * UNIX shell
24 * S. R. Bourne
25 * Rewritten By David Korn
26 * AT&T Labs
30 #include <ast.h>
31 #include <sfio.h>
32 #include <stak.h>
33 #include <ls.h>
34 #include <fcin.h>
35 #include "defs.h"
36 #include "variables.h"
37 #include "path.h"
38 #include "io.h"
39 #include "jobs.h"
40 #include "shlex.h"
41 #include "shnodes.h"
42 #include "history.h"
43 #include "timeout.h"
44 #include "FEATURE/time"
45 #include "FEATURE/pstat"
46 #include "FEATURE/execargs"
47 #include "FEATURE/externs"
48 #ifdef _hdr_nc
49 # include <nc.h>
50 #endif /* _hdr_nc */
52 #define CMD_LENGTH 64
54 /* These routines are referenced by this module */
55 static void exfile(Shell_t*, Sfio_t*,int);
56 static void chkmail(Shell_t *shp, char*);
57 #if defined(_lib_fork) && !defined(_NEXT_SOURCE)
58 static void fixargs(char**,int);
59 #else
60 # define fixargs(a,b)
61 #endif
63 #ifndef environ
64 extern char **environ;
65 #endif
67 static struct stat lastmail;
68 static time_t mailtime;
69 static char beenhere = 0;
71 #ifdef _lib_sigvec
72 void clearsigmask(register int sig)
74 struct sigvec vec;
75 if(sigvec(sig,NIL(struct sigvec*),&vec)>=0 && vec.sv_mask)
77 vec.sv_mask = 0;
78 sigvec(sig,&vec,NIL(struct sigvec*));
81 #endif /* _lib_sigvec */
83 #ifdef _lib_fts_notify
84 # include <fts.h>
85 /* check for interrupts during tree walks */
86 static int fts_sigcheck(FTS* fp, FTSENT* ep, void* context)
88 Shell_t *shp = (Shell_t*)context;
89 NOT_USED(fp);
90 NOT_USED(ep);
91 if(shp->trapnote&SH_SIGSET)
93 errno = EINTR;
94 return(-1);
96 return(0);
98 #endif /* _lib_fts_notify */
100 #ifdef PATH_BFPATH
101 #define PATHCOMP NIL(Pathcomp_t*)
102 #else
103 #define PATHCOMP ""
104 #endif
107 * search for file and exfile() it if it exists
108 * 1 returned if file found, 0 otherwise
111 int sh_source(Shell_t *shp, Sfio_t *iop, const char *file)
113 char* oid;
114 char* nid;
115 int fd;
117 if (!file || !*file || (fd = path_open(file, PATHCOMP)) < 0)
119 REGRESS(source, "sh_source", ("%s:ENOENT", file));
120 return 0;
122 oid = error_info.id;
123 nid = error_info.id = strdup(file);
124 shp->st.filename = path_fullname(stakptr(PATH_OFFSET));
125 REGRESS(source, "sh_source", ("%s", file));
126 exfile(shp, iop, fd);
127 error_info.id = oid;
128 free(nid);
129 return 1;
132 #ifdef S_ISSOCK
133 #define REMOTE(m) (S_ISSOCK(m)||!(m))
134 #else
135 #define REMOTE(m) !(m)
136 #endif
138 int sh_main(int ac, char *av[], Shinit_f userinit)
140 register char *name;
141 register int fdin;
142 register Sfio_t *iop;
143 register Shell_t *shp;
144 struct stat statb;
145 int i, rshflag; /* set for restricted shell */
146 char *command;
147 #ifdef _lib_sigvec
148 /* This is to clear mask that may be left on by rlogin */
149 clearsigmask(SIGALRM);
150 clearsigmask(SIGHUP);
151 clearsigmask(SIGCHLD);
152 #endif /* _lib_sigvec */
153 #ifdef _hdr_nc
154 _NutConf(_NC_SET_SUFFIXED_SEARCHING, 1);
155 #endif /* _hdr_nc */
156 fixargs(av,0);
157 shp = sh_init(ac,av,userinit);
158 time(&mailtime);
159 if(rshflag=sh_isoption(SH_RESTRICTED))
160 sh_offoption(SH_RESTRICTED);
161 #ifdef _lib_fts_notify
162 fts_notify(fts_sigcheck,(void*)shp);
163 #endif /* _lib_fts_notify */
164 if(sigsetjmp(*((sigjmp_buf*)shp->jmpbuffer),0))
166 /* begin script execution here */
167 sh_reinit((char**)0);
169 shp->fn_depth = shp->dot_depth = 0;
170 command = error_info.id;
171 /* set pidname '$$' */
172 shp->pid = getpid();
173 srand(shp->pid&0x7fff);
174 shp->ppid = getppid();
175 if(nv_isnull(PS4NOD))
176 nv_putval(PS4NOD,e_traceprompt,NV_RDONLY);
177 path_pwd(1);
178 iop = (Sfio_t*)0;
179 #if SHOPT_BRACEPAT
180 sh_onoption(SH_BRACEEXPAND);
181 #endif
182 if((beenhere++)==0)
184 sh_onstate(SH_PROFILE);
185 ((Lex_t*)shp->lex_context)->nonstandard = 0;
186 if(shp->ppid==1)
187 shp->login_sh++;
188 if(shp->login_sh >= 2)
189 sh_onoption(SH_LOGIN_SHELL);
190 /* decide whether shell is interactive */
191 if(!sh_isoption(SH_INTERACTIVE) && !sh_isoption(SH_TFLAG) && !sh_isoption(SH_CFLAG) &&
192 sh_isoption(SH_SFLAG) && tty_check(0) && tty_check(ERRIO))
193 sh_onoption(SH_INTERACTIVE);
194 if(sh_isoption(SH_INTERACTIVE))
196 sh_onoption(SH_BGNICE);
197 sh_onoption(SH_RC);
199 if(!sh_isoption(SH_RC) && (sh_isoption(SH_BASH) && !sh_isoption(SH_POSIX)
200 #if SHOPT_REMOTE
201 || !fstat(0, &statb) && REMOTE(statb.st_mode)
202 #endif
204 sh_onoption(SH_RC);
205 for(i=0; i<elementsof(shp->offoptions.v); i++)
206 shp->options.v[i] &= ~shp->offoptions.v[i];
207 if(sh_isoption(SH_INTERACTIVE))
209 #ifdef SIGXCPU
210 signal(SIGXCPU,SIG_DFL);
211 #endif /* SIGXCPU */
212 #ifdef SIGXFSZ
213 signal(SIGXFSZ,SIG_DFL);
214 #endif /* SIGXFSZ */
215 sh_onoption(SH_MONITOR);
217 job_init(shp,sh_isoption(SH_LOGIN_SHELL));
218 if(sh_isoption(SH_LOGIN_SHELL))
220 /* system profile */
221 sh_source(shp, iop, e_sysprofile);
222 if(!sh_isoption(SH_NOUSRPROFILE) && !sh_isoption(SH_PRIVILEGED))
224 char **files = shp->login_files;
225 while ((name = *files++) && !sh_source(shp, iop, sh_mactry(shp,name)));
228 /* make sure PWD is set up correctly */
229 path_pwd(1);
230 if(!sh_isoption(SH_NOEXEC))
232 if(!sh_isoption(SH_NOUSRPROFILE) && !sh_isoption(SH_PRIVILEGED) && sh_isoption(SH_RC))
234 #if SHOPT_BASH
235 if(sh_isoption(SH_BASH) && !sh_isoption(SH_POSIX))
237 #if SHOPT_SYSRC
238 sh_source(shp, iop, e_bash_sysrc);
239 #endif
240 sh_source(shp, iop, shp->rcfile ? shp->rcfile : sh_mactry(shp,(char*)e_bash_rc));
242 else
243 #endif
245 if(name = sh_mactry(shp,nv_getval(ENVNOD)))
246 name = *name ? strdup(name) : (char*)0;
247 #if SHOPT_SYSRC
248 if(!strmatch(name, "?(.)/./*"))
249 sh_source(shp, iop, e_sysrc);
250 #endif
251 if(name)
253 sh_source(shp, iop, name);
254 free(name);
258 else if(sh_isoption(SH_INTERACTIVE) && sh_isoption(SH_PRIVILEGED))
259 sh_source(shp, iop, e_suidprofile);
261 shp->st.cmdname = error_info.id = command;
262 sh_offstate(SH_PROFILE);
263 if(rshflag)
264 sh_onoption(SH_RESTRICTED);
265 /* open input file if specified */
266 if(shp->comdiv)
268 shell_c:
269 iop = sfnew(NIL(Sfio_t*),shp->comdiv,strlen(shp->comdiv),0,SF_STRING|SF_READ);
271 else
273 name = error_info.id;
274 error_info.id = shp->shname;
275 if(sh_isoption(SH_SFLAG))
276 fdin = 0;
277 else
279 char *sp;
280 /* open stream should have been passed into shell */
281 if(strmatch(name,e_devfdNN))
283 #if !_WINIX
284 char *cp;
285 int type;
286 #endif
287 fdin = (int)strtol(name+8, (char**)0, 10);
288 if(fstat(fdin,&statb)<0)
289 errormsg(SH_DICT,ERROR_system(1),e_open,name);
290 #if !_WINIX
292 * try to undo effect of solaris 2.5+
293 * change for argv for setuid scripts
295 if(((type = sh_type(cp = av[0])) & SH_TYPE_SH) && (!(name = nv_getval(L_ARGNOD)) || !((type = sh_type(cp = name)) & SH_TYPE_SH)))
297 av[0] = (type & SH_TYPE_LOGIN) ? cp : path_basename(cp);
298 /* exec to change $0 for ps */
299 execv(pathshell(),av);
300 /* exec fails */
301 shp->st.dolv[0] = av[0];
302 fixargs(shp->st.dolv,1);
304 #endif
305 name = av[0];
306 sh_offoption(SH_VERBOSE);
307 sh_offoption(SH_XTRACE);
309 else
311 int isdir = 0;
312 if((fdin=sh_open(name,O_RDONLY,0))>=0 &&(fstat(fdin,&statb)<0 || S_ISDIR(statb.st_mode)))
314 close(fdin);
315 isdir = 1;
316 fdin = -1;
318 else
319 shp->st.filename = path_fullname(name);
320 sp = 0;
321 if(fdin < 0 && !strchr(name,'/'))
323 #ifdef PATH_BFPATH
324 if(path_absolute(name,NIL(Pathcomp_t*)))
325 sp = stakptr(PATH_OFFSET);
326 #else
327 sp = path_absolute(name,NIL(char*));
328 #endif
329 if(sp)
331 if((fdin=sh_open(sp,O_RDONLY,0))>=0)
332 shp->st.filename = path_fullname(sp);
335 if(fdin<0)
337 if(isdir)
338 errno = EISDIR;
339 error_info.id = av[0];
340 if(sp || errno!=ENOENT)
341 errormsg(SH_DICT,ERROR_system(ERROR_NOEXEC),e_open,name);
342 /* try sh -c 'name "$@"' */
343 sh_onoption(SH_CFLAG);
344 shp->comdiv = (char*)malloc(strlen(name)+7);
345 name = strcopy(shp->comdiv,name);
346 if(shp->st.dolc)
347 strcopy(name," \"$@\"");
348 goto shell_c;
350 if(fdin==0)
351 fdin = sh_iomovefd(fdin);
353 shp->readscript = shp->shname;
355 error_info.id = name;
356 shp->comdiv--;
357 #if SHOPT_ACCT
358 sh_accinit();
359 if(fdin != 0)
360 sh_accbegin(error_info.id);
361 #endif /* SHOPT_ACCT */
364 else
366 fdin = shp->infd;
367 fixargs(shp->st.dolv,1);
369 if(sh_isoption(SH_INTERACTIVE))
370 sh_onstate(SH_INTERACTIVE);
371 nv_putval(IFSNOD,(char*)e_sptbnl,NV_RDONLY);
372 exfile(shp,iop,fdin);
373 sh_done(shp,0);
374 /* NOTREACHED */
375 return(0);
379 * iop is not null when the input is a string
380 * fdin is the input file descriptor
383 static void exfile(register Shell_t *shp, register Sfio_t *iop,register int fno)
385 time_t curtime;
386 Shnode_t *t;
387 int maxtry=IOMAXTRY, tdone=0, execflags;
388 int states,jmpval;
389 struct checkpt buff;
390 sh_pushcontext(&buff,SH_JMPERREXIT);
391 /* open input stream */
392 nv_putval(SH_PATHNAMENOD, shp->st.filename ,NV_NOFREE);
393 if(!iop)
395 if(fno > 0)
397 int r;
398 if(fno < 10 && ((r=sh_fcntl(fno,F_DUPFD,10))>=10))
400 shp->fdstatus[r] = shp->fdstatus[fno];
401 sh_close(fno);
402 fno = r;
404 fcntl(fno,F_SETFD,FD_CLOEXEC);
405 shp->fdstatus[fno] |= IOCLEX;
406 iop = sh_iostream((void*)shp,fno);
408 else
409 iop = sfstdin;
411 else
412 fno = -1;
413 shp->infd = fno;
414 if(sh_isstate(SH_INTERACTIVE))
416 if(nv_isnull(PS1NOD))
417 nv_putval(PS1NOD,(shp->euserid?e_stdprompt:e_supprompt),NV_RDONLY);
418 sh_sigdone();
419 if(sh_histinit((void*)shp))
420 sh_onoption(SH_HISTORY);
422 else
424 if(!sh_isstate(SH_PROFILE))
426 buff.mode = SH_JMPEXIT;
427 sh_onoption(SH_TRACKALL);
428 sh_offoption(SH_MONITOR);
430 sh_offstate(SH_INTERACTIVE);
431 sh_offstate(SH_MONITOR);
432 sh_offstate(SH_HISTORY);
433 sh_offoption(SH_HISTORY);
435 states = sh_getstate();
436 jmpval = sigsetjmp(buff.buff,0);
437 if(jmpval)
439 Sfio_t *top;
440 sh_iorestore((void*)shp,0,jmpval);
441 hist_flush(shp->hist_ptr);
442 sfsync(shp->outpool);
443 shp->st.execbrk = shp->st.breakcnt = 0;
444 /* check for return from profile or env file */
445 if(sh_isstate(SH_PROFILE) && (jmpval==SH_JMPFUN || jmpval==SH_JMPEXIT))
447 sh_setstate(states);
448 goto done;
450 if(!sh_isoption(SH_INTERACTIVE) || sh_isstate(SH_FORKED) || (jmpval > SH_JMPERREXIT && job_close(shp) >=0))
452 sh_offstate(SH_INTERACTIVE);
453 sh_offstate(SH_MONITOR);
454 goto done;
456 /* skip over remaining input */
457 if(top = fcfile())
459 while(fcget()>0);
460 fcclose();
461 while(top=sfstack(iop,SF_POPSTACK))
462 sfclose(top);
464 /* make sure that we own the terminal */
465 #ifdef SIGTSTP
466 tcsetpgrp(job.fd,shp->pid);
467 #endif /* SIGTSTP */
469 /* error return here */
470 sfclrerr(iop);
471 sh_setstate(states);
472 shp->st.optindex = 1;
473 opt_info.offset = 0;
474 shp->st.loopcnt = 0;
475 shp->trapnote = 0;
476 shp->intrap = 0;
477 error_info.line = 1;
478 shp->inlineno = 1;
479 shp->binscript = 0;
480 if(sfeof(iop))
481 goto eof_or_error;
482 /* command loop */
483 while(1)
485 shp->nextprompt = 1;
486 sh_freeup(shp);
487 stakset(NIL(char*),0);
488 exitset();
489 sh_offstate(SH_STOPOK);
490 sh_offstate(SH_ERREXIT);
491 sh_offstate(SH_VERBOSE);
492 sh_offstate(SH_TIMING);
493 sh_offstate(SH_GRACE);
494 sh_offstate(SH_TTYWAIT);
495 if(sh_isoption(SH_VERBOSE))
496 sh_onstate(SH_VERBOSE);
497 sh_onstate(SH_ERREXIT);
498 /* -eim flags don't apply to profiles */
499 if(sh_isstate(SH_PROFILE))
501 sh_offstate(SH_INTERACTIVE);
502 sh_offstate(SH_ERREXIT);
503 sh_offstate(SH_MONITOR);
505 if(sh_isstate(SH_INTERACTIVE) && !tdone)
507 register char *mail;
508 #ifdef JOBS
509 sh_offstate(SH_MONITOR);
510 if(sh_isoption(SH_MONITOR))
511 sh_onstate(SH_MONITOR);
512 if(job.pwlist)
514 job_walk(sfstderr,job_list,JOB_NFLAG,(char**)0);
515 job_wait((pid_t)0);
517 #endif /* JOBS */
518 if((mail=nv_getval(MAILPNOD)) || (mail=nv_getval(MAILNOD)))
520 time(&curtime);
521 if ((curtime - mailtime) >= sh_mailchk)
523 chkmail(shp,mail);
524 mailtime = curtime;
527 if(shp->hist_ptr)
528 hist_eof(shp->hist_ptr);
529 /* sets timeout for command entry */
530 shp->timeout = shp->st.tmout;
531 #if SHOPT_TIMEOUT
532 if(shp->timeout <= 0 || shp->timeout > SHOPT_TIMEOUT)
533 shp->timeout = SHOPT_TIMEOUT;
534 #endif /* SHOPT_TIMEOUT */
535 shp->inlineno = 1;
536 error_info.line = 1;
537 shp->exitval = 0;
538 shp->trapnote = 0;
539 if(buff.mode == SH_JMPEXIT)
541 buff.mode = SH_JMPERREXIT;
542 #ifdef DEBUG
543 errormsg(SH_DICT,ERROR_warn(0),"%d: mode changed to JMP_EXIT",getpid());
544 #endif
547 errno = 0;
548 if(tdone || !sfreserve(iop,0,0))
550 eof_or_error:
551 if(sh_isstate(SH_INTERACTIVE) && !sferror(iop))
553 if(--maxtry>0 && sh_isoption(SH_IGNOREEOF) &&
554 !sferror(sfstderr) && (shp->fdstatus[fno]&IOTTY))
556 sfclrerr(iop);
557 errormsg(SH_DICT,0,e_logout);
558 continue;
560 else if(job_close(shp)<0)
561 continue;
563 if(errno==0 && sferror(iop) && --maxtry>0)
565 sfclrlock(iop);
566 sfclrerr(iop);
567 continue;
569 goto done;
571 maxtry = IOMAXTRY;
572 if(sh_isstate(SH_INTERACTIVE) && shp->hist_ptr)
574 job_wait((pid_t)0);
575 hist_eof(shp->hist_ptr);
576 sfsync(sfstderr);
578 if(sh_isoption(SH_HISTORY))
579 sh_onstate(SH_HISTORY);
580 job.waitall = job.curpgid = 0;
581 error_info.flags |= ERROR_INTERACTIVE;
582 t = (Shnode_t*)sh_parse(shp,iop,0);
583 if(!sh_isstate(SH_INTERACTIVE) && !sh_isstate(SH_CFLAG))
584 error_info.flags &= ~ERROR_INTERACTIVE;
585 shp->readscript = 0;
586 if(sh_isstate(SH_INTERACTIVE) && shp->hist_ptr)
587 hist_flush(shp->hist_ptr);
588 sh_offstate(SH_HISTORY);
589 if(t)
591 execflags = sh_state(SH_ERREXIT)|sh_state(SH_INTERACTIVE);
592 /* The last command may not have to fork */
593 if(!sh_isstate(SH_PROFILE) && !sh_isstate(SH_INTERACTIVE) &&
594 (fno<0 || !(shp->fdstatus[fno]&(IOTTY|IONOSEEK)))
595 && !sfreserve(iop,0,0))
597 execflags |= sh_state(SH_NOFORK);
599 shp->st.execbrk = 0;
600 sh_exec(t,execflags);
601 if(shp->forked)
603 sh_offstate(SH_INTERACTIVE);
604 goto done;
606 /* This is for sh -t */
607 if(sh_isoption(SH_TFLAG) && !sh_isstate(SH_PROFILE))
608 tdone++;
611 done:
612 sh_popcontext(&buff);
613 if(sh_isstate(SH_INTERACTIVE))
615 sfputc(sfstderr,'\n');
616 job_close(shp);
618 if(jmpval == SH_JMPSCRIPT)
619 siglongjmp(*shp->jmplist,jmpval);
620 else if(jmpval == SH_JMPEXIT)
621 sh_done(shp,0);
622 if(fno>0)
623 sh_close(fno);
624 if(shp->st.filename)
625 free((void*)shp->st.filename);
626 shp->st.filename = 0;
630 /* prints out messages if files in list have been modified since last call */
631 static void chkmail(Shell_t *shp, char *files)
633 register char *cp,*sp,*qp;
634 register char save;
635 struct argnod *arglist=0;
636 int offset = staktell();
637 char *savstak=stakptr(0);
638 struct stat statb;
639 if(*(cp=files) == 0)
640 return;
641 sp = cp;
644 /* skip to : or end of string saving first '?' */
645 for(qp=0;*sp && *sp != ':';sp++)
646 if((*sp == '?' || *sp=='%') && qp == 0)
647 qp = sp;
648 save = *sp;
649 *sp = 0;
650 /* change '?' to end-of-string */
651 if(qp)
652 *qp = 0;
655 /* see if time has been modified since last checked
656 * and the access time <= the modification time
658 if(stat(cp,&statb) >= 0 && statb.st_mtime >= mailtime
659 && statb.st_atime <= statb.st_mtime)
661 /* check for directory */
662 if(!arglist && S_ISDIR(statb.st_mode))
664 /* generate list of directory entries */
665 path_complete(cp,"/*",&arglist);
667 else
670 * If the file has shrunk,
671 * or if the size is zero
672 * then don't print anything
674 if(statb.st_size &&
675 ( statb.st_ino != lastmail.st_ino
676 || statb.st_dev != lastmail.st_dev
677 || statb.st_size > lastmail.st_size))
679 /* save and restore $_ */
680 char *save = shp->lastarg;
681 shp->lastarg = cp;
682 errormsg(SH_DICT,0,sh_mactry(shp,qp?qp+1:(char*)e_mailmsg));
683 shp->lastarg = save;
685 lastmail = statb;
686 break;
689 if(arglist)
691 cp = arglist->argval;
692 arglist = arglist->argchn.ap;
694 else
695 cp = 0;
697 while(cp);
698 if(qp)
699 *qp = '?';
700 *sp++ = save;
701 cp = sp;
703 while(save);
704 stakset(savstak,offset);
707 #undef EXECARGS
708 #undef PSTAT
709 #if defined(_hdr_execargs) && defined(pdp11)
710 # include <execargs.h>
711 # define EXECARGS 1
712 #endif
714 #if defined(_lib_pstat) && defined(_sys_pstat)
715 # include <sys/pstat.h>
716 # define PSTAT 1
717 #endif
719 #if defined(_lib_fork) && !defined(_NEXT_SOURCE)
721 * fix up command line for ps command
722 * mode is 0 for initialization
724 static void fixargs(char **argv, int mode)
726 #if EXECARGS
727 *execargs=(char *)argv;
728 #else
729 static char *buff;
730 static int command_len;
731 register char *cp;
732 int offset=0,size;
733 # ifdef PSTAT
734 union pstun un;
735 if(mode==0)
737 struct pst_static st;
738 un.pst_static = &st;
739 if(pstat(PSTAT_STATIC, un, sizeof(struct pst_static), 1, 0)<0)
740 return;
741 command_len = st.command_length;
742 return;
744 stakseek(command_len+2);
745 buff = stakseek(0);
746 # else
747 if(mode==0)
749 buff = argv[0];
750 while(cp = *argv++)
751 command_len += strlen(cp)+1;
752 if(environ && *environ==buff+command_len)
754 for(argv=environ; cp = *argv; cp++)
756 if(command_len > CMD_LENGTH)
758 command_len = CMD_LENGTH;
759 break;
761 *argv++ = strdup(cp);
762 command_len += strlen(cp)+1;
765 command_len -= 1;
766 return;
768 # endif /* PSTAT */
769 if(command_len==0)
770 return;
771 while((cp = *argv++) && offset < command_len)
773 if(offset + (size=strlen(cp)) >= command_len)
774 size = command_len - offset;
775 memcpy(buff+offset,cp,size);
776 offset += size;
777 buff[offset++] = ' ';
779 buff[offset-1] = 0;
780 # ifdef PSTAT
781 un.pst_command = stakptr(0);
782 pstat(PSTAT_SETCMD,un,0,0,0);
783 # endif /* PSTAT */
784 #endif /* EXECARGS */
786 #endif /* _lib_fork */