📄 var.c
字号:
#include "sh.h"#include "ksh_time.h"#include "ksh_limval.h"#include "ksh_stat.h"#include <ctype.h>/* * Variables * * WARNING: unreadable code, needs a rewrite * * if (flag&INTEGER), val.i contains integer value, and type contains base. * otherwise, (val.s + type) contains string value. * if (flag&EXPORT), val.s contains "name=value" for E-Z exporting. */static struct tbl vtemp;static struct table specials;static char *formatstr ARGS((struct tbl *vp, const char *s));static void export ARGS((struct tbl *vp, const char *val));static int special ARGS((const char *name));static void unspecial ARGS((const char *name));static void getspec ARGS((struct tbl *vp));static void setspec ARGS((struct tbl *vp));static void unsetspec ARGS((struct tbl *vp));static struct tbl *arraysearch ARGS((struct tbl *, int));/* * create a new block for function calls and simple commands * assume caller has allocated and set up e->loc */voidnewblock(){ register struct block *l; static char *const empty[] = {null}; l = (struct block *) alloc(sizeof(struct block), ATEMP); l->flags = 0; ainit(&l->area); /* todo: could use e->area (l->area => l->areap) */ if (!e->loc) { l->argc = 0; l->argv = (char **) empty; } else { l->argc = e->loc->argc; l->argv = e->loc->argv; } l->exit = l->error = NULL; tinit(&l->vars, &l->area, 0); tinit(&l->funs, &l->area, 0); l->next = e->loc; e->loc = l;}/* * pop a block handling special variables */voidpopblock(){ register struct block *l = e->loc; register struct tbl *vp, **vpp = l->vars.tbls, *vq; register int i; e->loc = l->next; /* pop block */ for (i = l->vars.size; --i >= 0; ) if ((vp = *vpp++) != NULL && (vp->flag&SPECIAL)) if ((vq = global(vp->name))->flag & ISSET) setspec(vq); else unsetspec(vq); if (l->flags & BF_DOGETOPTS) user_opt = l->getopts_state; afreeall(&l->area); afree(l, ATEMP);}/* called by main() to initialize variable data structures */voidinitvar(){ static const struct { const char *name; int v; } names[] = { { "COLUMNS", V_COLUMNS }, { "IFS", V_IFS }, { "OPTIND", V_OPTIND }, { "PATH", V_PATH }, { "POSIXLY_CORRECT", V_POSIXLY_CORRECT }, { "TMPDIR", V_TMPDIR },#ifdef HISTORY { "HISTFILE", V_HISTFILE }, { "HISTSIZE", V_HISTSIZE },#endif /* HISTORY */#ifdef EDIT { "EDITOR", V_EDITOR }, { "VISUAL", V_VISUAL },#endif /* EDIT */#ifdef KSH { "MAIL", V_MAIL }, { "MAILCHECK", V_MAILCHECK }, { "MAILPATH", V_MAILPATH }, { "RANDOM", V_RANDOM }, { "SECONDS", V_SECONDS }, { "TMOUT", V_TMOUT },#endif /* KSH */ { "LINENO", V_LINENO }, { (char *) 0, 0 } }; int i; struct tbl *tp; tinit(&specials, APERM, 32); /* must be 2^n (currently 17 specials) */ for (i = 0; names[i].name; i++) { tp = tenter(&specials, names[i].name, hash(names[i].name)); tp->flag = DEFINED|ISSET; tp->type = names[i].v; }}/* Used to calculate an array index for global()/local(). Sets *arrayp to * non-zero if this is an array, sets *valp to the array index, returns * the basename of the array. */const char *array_index_calc(const char *n, bool_t *arrayp, int *valp){ const char *p; int len; *arrayp = FALSE; p = skip_varname(n, FALSE); if (p != n && *p == '[' && (len = array_ref_len(p))) { char *sub, *tmp; long rval; /* Calculate the value of the subscript */ *arrayp = TRUE; tmp = str_nsave(p+1, len-2, ATEMP); sub = substitute(tmp, 0); afree(tmp, ATEMP); n = str_nsave(n, p - n, ATEMP); evaluate(sub, &rval, KSH_UNWIND_ERROR); if (rval < 0 || rval > ARRAYMAX) errorf("%s: subscript out of range", n); *valp = rval; afree(sub, ATEMP); } return n;}/* * Search for variable, if not found create globally. */struct tbl *global(n) register const char *n;{ register struct block *l = e->loc; register struct tbl *vp; register int c; unsigned h; bool_t array; int val; /* Check to see if this is an array */ n = array_index_calc(n, &array, &val); h = hash(n); c = n[0]; if (!letter(c)) { if (array) errorf("bad substitution"); vp = &vtemp; vp->flag = DEFINED; vp->type = 0; vp->areap = ATEMP; *vp->name = c; if (digit(c)) { for (c = 0; digit(*n); n++) c = c*10 + *n-'0'; if (c <= l->argc) /* setstr can't fail here */ setstr(vp, l->argv[c], KSH_RETURN_ERROR); vp->flag |= RDONLY; return vp; } vp->flag |= RDONLY; if (n[1] != '\0') return vp; vp->flag |= ISSET|INTEGER; switch (c) { case '$': vp->val.i = kshpid; break; case '!': /* If no job, expand to nothing */ if ((vp->val.i = j_async()) == 0) vp->flag &= ~(ISSET|INTEGER); break; case '?': vp->val.i = exstat; break; case '#': vp->val.i = l->argc; break; case '-': vp->flag &= ~INTEGER; vp->val.s = getoptions(); break; default: vp->flag &= ~(ISSET|INTEGER); } return vp; } for (l = e->loc; ; l = l->next) { vp = tsearch(&l->vars, n, h); if (vp != NULL) if (array) return arraysearch(vp, val); else return vp; if (l->next == NULL) break; } vp = tenter(&l->vars, n, h); if (array) vp = arraysearch(vp, val); vp->flag |= DEFINED; if (special(n)) vp->flag |= SPECIAL; return vp;}/* * Search for local variable, if not found create locally. */struct tbl *local(n, copy) register const char *n; bool_t copy;{ register struct block *l = e->loc; register struct tbl *vp; unsigned h; bool_t array; int val; /* Check to see if this is an array */ n = array_index_calc(n, &array, &val); h = hash(n); if (!letter(*n)) { vp = &vtemp; vp->flag = DEFINED|RDONLY; vp->type = 0; vp->areap = ATEMP; return vp; } vp = tenter(&l->vars, n, h); if (copy && !(vp->flag & DEFINED)) { struct block *ll = l; struct tbl *vq = (struct tbl *) 0; while ((ll = ll->next) && !(vq = tsearch(&ll->vars, n, h))) ; if (vq) { vp->flag |= vq->flag & (EXPORT|INTEGER|RDONLY |LJUST|RJUST|ZEROFIL |LCASEV|UCASEV_AL|INT_U|INT_L); if (vq->flag & INTEGER) vp->type = vq->type; vp->u2.field = vq->u2.field; } } if (array) vp = arraysearch(vp, val); vp->flag |= DEFINED; if (special(n)) vp->flag |= SPECIAL; return vp;}/* get variable string value */char *str_val(vp) register struct tbl *vp;{ char *s; if ((vp->flag&SPECIAL)) getspec(vp); if (!(vp->flag&ISSET)) s = null; /* special to dollar() */ else if (!(vp->flag&INTEGER)) /* string source */ s = vp->val.s + vp->type; else { /* integer source */ /* worst case number length is when base=2, so use BITS(long) */ /* minus base # number null */ static char strbuf[1 + 2 + 1 + BITS(long) + 1]; const char *digits = (vp->flag & UCASEV_AL) ? "0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZ" : "0123456789abcdefghijklmnopqrstuvwxyz"; register unsigned long n; register int base; s = strbuf + sizeof(strbuf); if (vp->flag & INT_U) n = (unsigned long) vp->val.i; else n = (vp->val.i < 0) ? -vp->val.i : vp->val.i; base = (vp->type == 0) ? 10 : vp->type; *--s = '\0'; do { *--s = digits[n % base]; n /= base; } while (n != 0); if (base != 10) { *--s = '#'; *--s = digits[base % 10]; if (base >= 10) *--s = digits[base / 10]; } if (!(vp->flag & INT_U) && vp->val.i < 0) *--s = '-'; if (vp->flag & (RJUST|LJUST)) /* case already dealt with */ s = formatstr(vp, s); } return s;}/* get variable integer value, with error checking */longintval(vp) register struct tbl *vp;{ long num; int base; base = getint(vp, &num); if (base == -1) /* XXX check calls - is error here ok by POSIX? */ errorf("%s: bad number", str_val(vp)); return num;}/* set variable to string value */intsetstr(vq, s, error_ok) register struct tbl *vq; const char *s; int error_ok;{ if (vq->flag & RDONLY) { warningf(TRUE, "%s: is read only", vq->name); if (!error_ok) errorf(null); return 0; } if (!(vq->flag&INTEGER)) { /* string dest */ if ((vq->flag&ALLOC)) { /* debugging */ if (s >= vq->val.s && s <= vq->val.s + strlen(vq->val.s)) internal_errorf(TRUE, "setstr: %s=%s: assigning to self", vq->name, s); afree((void*)vq->val.s, vq->areap); } vq->flag &= ~(ISSET|ALLOC); vq->type = 0; if (s && (vq->flag & (UCASEV_AL|LCASEV|LJUST|RJUST))) s = formatstr(vq, s); if ((vq->flag&EXPORT)) export(vq, s); else { vq->val.s = str_save(s, vq->areap); if (vq->val.s) /* <sjg> don't lie */ vq->flag |= ALLOC; } } else /* integer dest */ if (!v_evaluate(vq, s, error_ok)) return 0; vq->flag |= ISSET; if ((vq->flag&SPECIAL)) setspec(vq); return 1;}/* set variable to integer */voidsetint(vq, n) register struct tbl *vq; long n;{ if (!(vq->flag&INTEGER)) { register struct tbl *vp = &vtemp; vp->flag = (ISSET|INTEGER); vp->type = 0; vp->areap = ATEMP; vp->val.i = n; /* setstr can't fail here */ setstr(vq, str_val(vp), KSH_RETURN_ERROR); } else vq->val.i = n; vq->flag |= ISSET; if ((vq->flag&SPECIAL)) setspec(vq);}intgetint(vp, nump) struct tbl *vp; long *nump;{ register char *s; register int c; int base, neg; int have_base = 0; long num; if (vp->flag&SPECIAL) getspec(vp); /* XXX is it possible for ISSET to be set and val.s to be 0? */ if (!(vp->flag&ISSET) || (!(vp->flag&INTEGER) && vp->val.s == NULL)) return -1; if (vp->flag&INTEGER) { *nump = vp->val.i; return vp->type; } s = vp->val.s + vp->type; if (s == NULL) /* redundent given initial test */ s = null; base = 10; num = 0; neg = 0; for (c = *s++; c ; c = *s++) { if (c == '-') { neg++; } else if (c == '#') { base = (int) num; if (have_base || base < 2 || base > 36) return -1; num = 0; have_base = 1; } else if (letnum(c)) { if (isdigit(c)) c -= '0'; else if (islower(c)) c -= 'a' - 10; /* todo: assumes ascii */ else if (isupper(c)) c -= 'A' - 10; /* todo: assumes ascii */ else c = -1; /* _: force error */ if (c < 0 || c >= base) return -1; num = num * base + c; } else return -1; } if (neg) num = -num; *nump = num; return base;}/* convert variable vq to integer variable, setting its value from vp * (vq and vp may be the same) */struct tbl *setint_v(vq, vp) register struct tbl *vq, *vp;{ int base; long num; if ((base = getint(vp, &num)) == -1) return NULL; if (!(vq->flag & INTEGER) && (vq->flag & ALLOC)) { vq->flag &= ~ALLOC; afree(vq->val.s, vq->areap); } vq->val.i = num; if (vq->type == 0) /* default base */ vq->type = base; vq->flag |= ISSET|INTEGER; if (vq->flag&SPECIAL) setspec(vq); return vq;}static char *formatstr(vp, s) struct tbl *vp; const char *s;{ int olen, nlen; char *p, *q; olen = strlen(s); if (vp->flag & (RJUST|LJUST)) { if (!vp->u2.field) /* default field width */ vp->u2.field = olen; nlen = vp->u2.field; } else nlen = olen; p = (char *) alloc(nlen + 1, ATEMP); if (vp->flag & (RJUST|LJUST)) { int slen; if (vp->flag & RJUST) { const char *q = s + olen; /* strip trailing spaces (at&t ksh uses q[-1] == ' ') */ while (q > s && isspace(q[-1])) --q; slen = q - s; if (slen > vp->u2.field) { s += slen - vp->u2.field; slen = vp->u2.field; } shf_snprintf(p, nlen + 1, ((vp->flag & ZEROFIL) && digit(*s)) ? "%0*s%.*s" : "%*s%.*s", vp->u2.field - slen, null, slen, s); } else { /* strip leading spaces/zeros */ while (isspace(*s)) s++; if (vp->flag & ZEROFIL) while (*s == '0') s++; shf_snprintf(p, nlen + 1, "%-*.*s", vp->u2.field, vp->u2.field, s); } } else memcpy(p, s, olen + 1); if (vp->flag & UCASEV_AL) { for (q = p; *q; q++) if (islower(*q)) *q = toupper(*q); } else if (vp->flag & LCASEV) { for (q = p; *q; q++) if (isupper(*q)) *q = tolower(*q); } return p;}/* * make vp->val.s be "name=value" for quick exporting. */static voidexport(vp, val) register struct tbl *vp; const char *val;{ register char *xp; char *op = (vp->flag&ALLOC) ? vp->val.s : NULL; int namelen = strlen(vp->name); int vallen = strlen(val) + 1; vp->flag |= ALLOC; xp = (char*)alloc(namelen + 1 + vallen, vp->areap); memcpy(vp->val.s = xp, vp->name, namelen); xp += namelen; *xp++ = '='; vp->type = xp - vp->val.s; /* offset to value */ memcpy(xp, val, vallen); if (op != NULL) afree((void*)op, vp->areap);}/* * lookup variable (according to (set&LOCAL)), * set its attributes (INTEGER, RDONLY, EXPORT, TRACE, LJUST, RJUST, ZEROFIL, * LCASEV, UCASEV_AL), and optionally set its value if an assignment. */struct tbl *typeset(var, set, clr, field, base) register const char *var; Tflag clr, set; int field, base;{ register struct tbl *vp; struct tbl *vpbase, *t; char *tvar; const char *val; /* check for valid variable name, search for value */ val = skip_varname(var, FALSE); if (val == var) return NULL; if (*val == '[') { int len; len = array_ref_len(val); if (len == 0) return NULL; /* IMPORT is only used when the shell starts up and is * setting up its environment. Allow only simple array * references at this time since parameter/command substitution * is preformed on the [expression], which would be a major * security hole. */ if (set & IMPORT) { int i; for (i = 1; i < len - 1; i++) if (!digit(val[i])) return NULL; } val += len;
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -