Source-Changes-HG archive
[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]
[src/trunk]: src/bin/sh Fixes to shell expand (that is, $ stuff) from FreeBSD...
details: https://anonhg.NetBSD.org/src/rev/cef901085725
branches: trunk
changeset: 354072:cef901085725
user: kre <kre%NetBSD.org@localhost>
date: Sat Jun 03 10:31:16 2017 +0000
description:
Fixes to shell expand (that is, $ stuff) from FreeBSD (implemented
differently...)
In particular ${01} is now $1 not $0 (for ${0any-digits})
${4294967297} is most probably now ""
(unless you have a very large number of params)
it is no longer an alias for $1 (4294967297 & 0xFFFFFFFF) == 1
$(( expr $(( more )) stuff )) is no longer the same as
$(( expr (( more )) stuff )) which was sometimes OK, as in:
$(( 3 + $(( 2 - 1 )) * 3 ))
but not always as in:
$(( 1$((1 + 1))1 ))
which should be 121, but was an arith syntax error as
1((1 + 1))1
is meaningless.
Probably some more. This also sprinkles a little const, splits a big
func that had 2 (kind of unrelated) purposes into two simpler ones,
and avoids some (semi-dubious) modifications (and restores) in the input
string to insert \0's when they were needed.
diffstat:
bin/sh/expand.c | 374 ++++++++++++++++++++++++++++++++++-------------------
bin/sh/expand.h | 12 +-
bin/sh/memalloc.h | 3 +-
bin/sh/parser.c | 12 +-
4 files changed, 257 insertions(+), 144 deletions(-)
diffs (truncated from 830 to 300 lines):
diff -r f28359f5130d -r cef901085725 bin/sh/expand.c
--- a/bin/sh/expand.c Sat Jun 03 10:27:05 2017 +0000
+++ b/bin/sh/expand.c Sat Jun 03 10:31:16 2017 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: expand.c,v 1.106 2017/05/28 00:38:01 kre Exp $ */
+/* $NetBSD: expand.c,v 1.107 2017/06/03 10:31:16 kre Exp $ */
/*-
* Copyright (c) 1991, 1993
@@ -37,7 +37,7 @@
#if 0
static char sccsid[] = "@(#)expand.c 8.5 (Berkeley) 5/15/95";
#else
-__RCSID("$NetBSD: expand.c,v 1.106 2017/05/28 00:38:01 kre Exp $");
+__RCSID("$NetBSD: expand.c,v 1.107 2017/06/03 10:31:16 kre Exp $");
#endif
#endif /* not lint */
@@ -97,13 +97,15 @@
struct ifsregion *ifslastp; /* last struct in list */
struct arglist exparg; /* holds expanded arg list */
-STATIC void argstr(char *, int);
-STATIC char *exptilde(char *, int);
+STATIC const char *argstr(const char *, int);
+STATIC const char *exptilde(const char *, int);
STATIC void expbackq(union node *, int, int);
-STATIC int subevalvar(char *, char *, int, int, int, int, int);
-STATIC char *evalvar(char *, int);
-STATIC int varisset(char *, int);
-STATIC void varvalue(char *, int, int, int);
+STATIC const char *expari(const char *, int);
+STATIC int subevalvar(const char *, const char *, int, int, int);
+STATIC int subevalvar_trim(const char *, int, int, int, int, int);
+STATIC const char *evalvar(const char *, int);
+STATIC int varisset(const char *, int);
+STATIC void varvalue(const char *, int, int, int);
STATIC void recordregion(int, int, int);
STATIC void removerecordregions(int);
STATIC void ifsbreakup(char *, struct arglist *);
@@ -116,6 +118,7 @@
STATIC int patmatch(const char *, const char *, int);
STATIC char *cvtnum(int, char *);
static int collate_range_cmp(wchar_t, wchar_t);
+STATIC void add_args(struct strlist *);
/*
* Expand shell variables and backquotes inside a here document.
@@ -156,12 +159,16 @@
struct strlist *sp;
char *p;
+ if (fflag) /* no filename expandsion */
+ flag &= ~EXP_GLOB;
+
argbackq = arg->narg.backquote;
STARTSTACKSTR(expdest);
ifsfirst.next = NULL;
ifslastp = NULL;
argstr(arg->narg.text, flag);
if (arglist == NULL) {
+ STACKSTRNUL(expdest);
return; /* here document expanded */
}
STPUTC('\0', expdest);
@@ -170,11 +177,14 @@
/*
* TODO - EXP_REDIR
*/
- if (flag & EXP_FULL) {
+ if (flag & EXP_SPLIT) {
ifsbreakup(p, &exparg);
*exparg.lastp = NULL;
exparg.lastp = &exparg.list;
- expandmeta(exparg.list, flag);
+ if (flag & EXP_GLOB)
+ expandmeta(exparg.list, flag);
+ else
+ add_args(exparg.list);
} else {
if (flag & EXP_REDIR) /*XXX - for now, just remove escapes */
rmescapes(p);
@@ -195,35 +205,50 @@
/*
* Perform variable and command substitution.
- * If EXP_FULL is set, output CTLESC characters to allow for further processing.
+ * If EXP_GLOB is set, output CTLESC characters to allow for further processing.
+ * If EXP_SPLIT is set, remember location of result for later,
* Otherwise treat $@ like $* since no splitting will be performed.
*/
-STATIC void
-argstr(char *p, int flag)
+STATIC const char *
+argstr(const char *p, int flag)
{
char c;
- int quotes = flag & (EXP_FULL | EXP_CASE | EXP_REDIR); /* do CTLESC */
+ int quotes = flag & (EXP_GLOB | EXP_CASE | EXP_REDIR); /* do CTLESC */
int firsteq = 1;
const char *ifs = NULL;
int ifs_split = EXP_IFS_SPLIT;
+#ifdef DEBUG
+ const char *ed = expdest;
+#endif
if (flag & EXP_IFS_SPLIT)
ifs = ifsval();
+ VTRACE(DBG_EXPAND, ("argstr(\"%s\", %#x) quotes=%#x\n", p,flag,quotes));
+
if (*p == '~' && (flag & (EXP_TILDE | EXP_VARTILDE)))
p = exptilde(p, flag);
for (;;) {
switch (c = *p++) {
case '\0':
+ VTRACE(DBG_EXPAND, ("argstr returning at \"\" "
+ "added \"%.*s\" to expdest (followed by: %2.2x)\n",
+ expdest-ed, ed, *expdest));
+ return p - 1;
case CTLENDVAR: /* end of expanding yyy in ${xxx-yyy} */
- return;
+ case CTLENDARI: /* end of a $(( )) string */
+ VTRACE(DBG_EXPAND, ("argstr returning at \"%.6s\"..."
+ " after %2.2x; "
+ "added \"%.*s\" to expdest (followed by: %2.2x)\n",
+ p, p[-1]&0xff, expdest-ed, ed, *expdest));
+ return p;
case CTLQUOTEMARK:
/* "$@" syntax adherence hack */
if (p[0] == CTLVAR && p[1] & VSQUOTE &&
p[2] == '@' && p[3] == '=')
break;
- if ((flag & EXP_FULL) != 0)
+ if ((flag & EXP_SPLIT) != 0)
STPUTC(c, expdest);
ifs_split = 0;
break;
@@ -238,14 +263,23 @@
break;
case CTLVAR:
p = evalvar(p, (flag & ~EXP_IFS_SPLIT) | (flag & ifs_split));
+ VTRACE(DBG_EXPAND, ("argstr evalvar "
+ "added \"%.*s\" to expdest (followed by: %2.2x)\n",
+ expdest-ed, ed, *expdest));
break;
case CTLBACKQ:
case CTLBACKQ|CTLQUOTE:
expbackq(argbackq->n, c & CTLQUOTE, flag);
argbackq = argbackq->next;
+ VTRACE(DBG_EXPAND, ("argstr expbackq "
+ "added \"%.*s\" to expdest (followed by: %2.2x)\n",
+ expdest-ed, ed, *expdest));
break;
- case CTLENDARI:
- expari(flag);
+ case CTLARI:
+ p = expari(p, flag);
+ VTRACE(DBG_EXPAND, ("argstr expari "
+ "+ \"%.*s\" to dest (fwd by: %2.2x) p=\"%.5s...\"\n",
+ expdest-ed, ed, *expdest, p));
break;
case ':':
case '=':
@@ -276,18 +310,25 @@
}
}
-STATIC char *
-exptilde(char *p, int flag)
+STATIC const char *
+exptilde(const char *p, int flag)
{
- char c, *startp = p;
+ char c;
+ const char *startp = p;
struct passwd *pw;
const char *home;
- int quotes = flag & (EXP_FULL | EXP_CASE);
+ int quotes = flag & (EXP_GLOB | EXP_CASE);
+ char *user;
+ user = expdest; /* we will just borrow top of stack */
while ((c = *p) != '\0') {
switch(c) {
case CTLESC:
- return (startp);
+ case CTLVAR:
+ case CTLBACKQ:
+ case CTLBACKQ | CTLQUOTE:
+ case CTLARI:
+ case CTLENDARI:
case CTLQUOTEMARK:
return (startp);
case ':':
@@ -297,30 +338,27 @@
case '/':
goto done;
}
+ STPUTC(c, user);
p++;
}
-done:
- *p = '\0';
- if (*(startp+1) == '\0') {
- if ((home = lookupvar("HOME")) == NULL)
- goto lose;
- } else {
- if ((pw = getpwnam(startp+1)) == NULL)
- goto lose;
+ done:
+ STACKSTRNUL(user);
+ if (*expdest == '\0')
+ home = lookupvar("HOME");
+ else if ((pw = getpwnam(expdest)) == NULL)
+ home = NULL;
+ else
home = pw->pw_dir;
- }
- if (*home == '\0')
- goto lose;
- *p = c;
+ STUNSTR(user - expdest); /* return borrowed string space */
+
+ if (home == NULL || *home == '\0')
+ return startp;
while ((c = *home++) != '\0') {
if (quotes && SQSYNTAX[(int)c] == CCTL)
STPUTC(CTLESC, expdest);
STPUTC(c, expdest);
}
return (p);
-lose:
- *p = c;
- return (startp);
}
@@ -368,15 +406,18 @@
* Expand arithmetic expression. Backup to start of expression,
* evaluate, place result in (backed up) result, adjust string position.
*/
-void
-expari(int flag)
+STATIC const char *
+expari(const char *p, int flag)
{
- char *p, *q, *start;
+ char *q, *start;
intmax_t result;
int adjustment;
int begoff;
- int quotes = flag & (EXP_FULL | EXP_CASE);
+ int quotes = flag & (EXP_GLOB | EXP_CASE);
int quoted;
+#ifdef DEBUG
+ const char *ed = expdest;
+#endif
/* ifsfree(); */
@@ -400,46 +441,54 @@
* generate decimal, which takes (often) less digits (never more)
* so this is safe, if occasionally slightly wasteful.
*/
-#define SPACE_NEEDED ((sizeof(intmax_t) * CHAR_BIT + 2) / 3 + 2)
+#define SPACE_NEEDED ((int)((sizeof(intmax_t) * CHAR_BIT + 2) / 3 + 2))
/*
* Only need check for SPACE_NEEDED-2 as we have already
* consumed (but will overwrite) 2 chars, the CTLARI, and the
* flags (quoting) byte (if those had not already been seen,
* we would not be here.)
*/
- CHECKSTRSPACE((int)(SPACE_NEEDED - 2), expdest);
- USTPUTC('\0', expdest);
- start = stackblock();
- p = expdest - 1;
- while (*p != CTLARI && p >= start)
- --p;
- if (*p != CTLARI)
- error("missing CTLARI (shouldn't happen)");
- if (p > start && *(p-1) == CTLESC)
- for (p = start; *p != CTLARI; p++)
- if (*p == CTLESC)
- p++;
-
- if (p[1] == '"')
- quoted=1;
- else
- quoted=0;
- begoff = p - start;
+ quoted = *p++ == '"';
+ begoff = expdest - stackblock();
+ VTRACE(DBG_EXPAND, ("expari: '%c' \"%s\" begoff %d\n", p[-1],p,begoff));
+ p = argstr(p, 0); /* expand $(( )) string */
+ STPUTC('\0', expdest);
+ start = stackblock() + begoff;
+ VTRACE(DBG_EXPAND, ("expari: argstr added: '%s' start: \"%.8s\"\n",
+ ed, start));
Home |
Main Index |
Thread Index |
Old Index