#include "defs.h"
static struct dolnod *copyargs();
static void freedolh(void);
extern struct dolnod *freeargs();
static struct dolnod *dolh;
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
};
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] == '-')
{
cp = argp[1];
if (cp[1] == '-')
{
argp[1] = argp[0];
argc--;
return(argc);
}
if (cp[1] == '\0')
flags &= ~(execpr|readpr);
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] == '+')
{
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++;
if (!any(wc, "sicrp") && wc == *flagc) {
flags &= ~(flagval[flagc-flagchar]);
if (wc == 'e')
eflag = 0;
}
cp += len;
}
argp[1] = argp[0];
argc--;
}
flagp = flagadr;
if (flags)
{
flagc = flagchar;
while (*flagc)
{
if (flags & flagval[flagc-flagchar])
*flagp++ = *flagc;
flagc++;
}
}
*flagp = 0;
return(argc);
}
void
setargs(unsigned char *argi[])
{
unsigned char **argp = argi;
int argn = 0;
while (*argp++ != (unsigned char *)ENDARGS)
argn++;
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;
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)
{
if(globdolv)
dolv = globdolv;
if(globdolc)
dolc = globdolc;
if(globdolh)
dolh = globdolh;
globdolv = 0;
globdolc = 0;
globdolh = 0;
while (argfor = clean_args(argfor))
;
while (pop())
;
if(savpipe != -1) {
close(savpipe);
savpipe = -1;
}
while (poptemp())
;
}
struct dolnod *savargs(funcnt)
int funcnt;
{
if (!funcnt) {
globdolh = dolh;
globdolv = dolv;
globdolc = dolc;
}
useargs();
return(dolh);
}
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++;
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);
}