/* * CDDL HEADER START * * The contents of this file are subject to the terms of the * Common Development and Distribution License, Version 1.0 only * (the "License"). You may not use this file except in compliance * with the License. * * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE * or http://www.opensolaris.org/os/licensing. * See the License for the specific language governing permissions * and limitations under the License. * * When distributing Covered Code, include this CDDL HEADER in each * file and include the License file at usr/src/OPENSOLARIS.LICENSE. * If applicable, add the following below this CDDL HEADER, with the * fields enclosed by brackets "[]" replaced with your own identifying * information: Portions Copyright [yyyy] [name of copyright owner] * * CDDL HEADER END */ /* * Copyright 1995 Sun Microsystems, Inc. All rights reserved. * Use is subject to license terms. */ /* Copyright (c) 1984, 1986, 1987, 1988, 1989 AT&T */ /* All Rights Reserved */ /* * UNIX shell */ #include "defs.h" static struct dolnod *copyargs(); static void freedolh(void); extern struct dolnod *freeargs(); static struct dolnod *dolh; /* Used to save outermost positional parameters */ static struct dolnod *globdolh; static unsigned char **globdolv; static int globdolc; unsigned char flagadr[16]; unsigned char flagchar[] = { 'x', 'n', 'v', 't', STDFLG, 'i', 'e', 'r', 'k', 'u', 'h', 'f', 'a', 'm', 'p', 0 }; long flagval[] = { execpr, noexec, readpr, oneflg, stdflg, intflg, errflg, rshflg, keyflg, setflg, hashflg, nofngflg, exportflg, monitorflg, privflg, 0 }; /* ======== option handling ======== */ int options(int argc, unsigned char **argv) { unsigned char *cp; unsigned char **argp = argv; unsigned char *flagc; unsigned char *flagp; int len; wchar_t wc; if (argc > 1 && *argp[1] == '-') { /* * if first argument is "--" then options are not * to be changed. Fix for problems getting * $1 starting with a "-" */ cp = argp[1]; if (cp[1] == '-') { argp[1] = argp[0]; argc--; return(argc); } if (cp[1] == '\0') flags &= ~(execpr|readpr); /* * Step along 'flagchar[]' looking for matches. * 'sicrp' are not legal with 'set' command. */ cp++; while (*cp) { if ((len = mbtowc(&wc, (char *)cp, MB_LEN_MAX)) <= 0) { len = 1; wc = (unsigned char)*cp; failed(argv[1],badopt); } cp += len; flagc = flagchar; while (*flagc && wc != *flagc) flagc++; if (wc == *flagc) { if (eq(argv[0], "set") && any(wc, "sicrp")) failed(argv[1], badopt); else { flags |= flagval[flagc-flagchar]; if (flags & errflg) eflag = errflg; } } else if (wc == 'c' && argc > 2 && comdiv == 0) { comdiv = argp[2]; argp[1] = argp[0]; argp++; argc--; } else failed(argv[1],badopt); } argp[1] = argp[0]; argc--; } else if (argc > 1 && *argp[1] == '+') /* unset flags x, k, t, n, v, e, u */ { cp = argp[1]; cp++; while (*cp) { if ((len = mbtowc(&wc, (char *)cp, MB_LEN_MAX)) <= 0) { cp++; continue; } flagc = flagchar; while (*flagc && wc != *flagc) flagc++; /* * step through flags */ if (!any(wc, "sicrp") && wc == *flagc) { flags &= ~(flagval[flagc-flagchar]); if (wc == 'e') eflag = 0; } cp += len; } argp[1] = argp[0]; argc--; } /* * set up $- */ flagp = flagadr; if (flags) { flagc = flagchar; while (*flagc) { if (flags & flagval[flagc-flagchar]) *flagp++ = *flagc; flagc++; } } *flagp = 0; return(argc); } /* * sets up positional parameters */ void setargs(unsigned char *argi[]) { unsigned char **argp = argi; /* count args */ int argn = 0; while (*argp++ != (unsigned char *)ENDARGS) argn++; /* * free old ones unless on for loop chain */ freedolh(); dolh = copyargs(argi, argn); dolc = argn - 1; } static void freedolh(void) { unsigned char **argp; struct dolnod *argblk; if (argblk = dolh) { if ((--argblk->doluse) == 0) { for (argp = argblk->dolarg; *argp != (unsigned char *)ENDARGS; argp++) free(*argp); free(argblk->dolarg); free(argblk); } } } struct dolnod * freeargs(blk) struct dolnod *blk; { unsigned char **argp; struct dolnod *argr = 0; struct dolnod *argblk; int cnt; if (argblk = blk) { argr = argblk->dolnxt; cnt = --argblk->doluse; if (argblk == dolh) { if (cnt == 1) return(argr); else return(argblk); } else { if (cnt == 0) { for (argp = argblk->dolarg; *argp != (unsigned char *)ENDARGS; argp++) free(*argp); free(argblk->dolarg); free(argblk); } } } return(argr); } static struct dolnod * copyargs(unsigned char *from[], int n) { struct dolnod *np = (struct dolnod *)alloc(sizeof (struct dolnod)); unsigned char **fp = from; unsigned char **pp; np -> dolnxt = 0; np->doluse = 1; /* use count */ pp = np->dolarg = (unsigned char **)alloc((n+1)*sizeof(char *)); dolv = pp; while (n--) *pp++ = make(*fp++); *pp++ = ENDARGS; return(np); } struct dolnod * clean_args(struct dolnod *blk) { unsigned char **argp; struct dolnod *argr = 0; struct dolnod *argblk; if (argblk = blk) { argr = argblk->dolnxt; if (argblk == dolh) argblk->doluse = 1; else { for (argp = argblk->dolarg; *argp != (unsigned char *)ENDARGS; argp++) free(*argp); free(argblk->dolarg); free(argblk); } } return(argr); } void clearup(void) { /* * force `for' $* lists to go away */ if(globdolv) dolv = globdolv; if(globdolc) dolc = globdolc; if(globdolh) dolh = globdolh; globdolv = 0; globdolc = 0; globdolh = 0; while (argfor = clean_args(argfor)) ; /* * clean up io files */ while (pop()) ; /* * Clean up pipe file descriptor * from command substitution */ if(savpipe != -1) { close(savpipe); savpipe = -1; } /* * clean up tmp files */ while (poptemp()) ; } /* * Save positiional parameters before outermost function invocation * in case we are interrupted. * Increment use count for current positional parameters so that they aren't thrown * away. */ struct dolnod *savargs(funcnt) int funcnt; { if (!funcnt) { globdolh = dolh; globdolv = dolv; globdolc = dolc; } useargs(); return(dolh); } /* After function invocation, free positional parameters, * restore old positional parameters, and restore * use count. */ void restorargs(struct dolnod *olddolh, int funcnt) { if(argfor != olddolh) while ((argfor = clean_args(argfor)) != olddolh && argfor); if(!argfor) return; freedolh(); dolh = olddolh; if(dolh) dolh -> doluse++; /* increment use count so arguments aren't freed */ argfor = freeargs(dolh); if(funcnt == 1) { globdolh = 0; globdolv = 0; globdolc = 0; } } struct dolnod * useargs(void) { if (dolh) { if (dolh->doluse++ == 1) { dolh->dolnxt = argfor; argfor = dolh; } } return(dolh); }