summaryrefslogtreecommitdiff
path: root/shell/ash.c
diff options
context:
space:
mode:
authorDenys Vlasenko2016-07-25 03:56:00 +0200
committerDenys Vlasenko2016-07-25 03:56:00 +0200
commitf8ddbe1ccce9eceaaac28b4b1aa71631fcc56db6 (patch)
tree0782f9ae0ab8bfb74ad8c3fca455f2dcd4de6b33 /shell/ash.c
parent0fb0045aa9261be1dda49dfdfb95cbc585402a8b (diff)
downloadbusybox-f8ddbe1ccce9eceaaac28b4b1aa71631fcc56db6.zip
busybox-f8ddbe1ccce9eceaaac28b4b1aa71631fcc56db6.tar.gz
ash: fix handling of ${VAR: -2}
Signed-off-by: Denys Vlasenko <vda.linux@googlemail.com>
Diffstat (limited to 'shell/ash.c')
-rw-r--r--shell/ash.c12
1 files changed, 8 insertions, 4 deletions
diff --git a/shell/ash.c b/shell/ash.c
index 4f6376f..496167f 100644
--- a/shell/ash.c
+++ b/shell/ash.c
@@ -6323,6 +6323,8 @@ subevalvar(char *p, char *varname, int strloc, int subtype,
#if ENABLE_ASH_BASH_COMPAT
case VSSUBSTR:
+//TODO: support more general format ${v:EXPR:EXPR},
+// where EXPR follows $(()) rules
loc = str = stackblock() + strloc;
/* Read POS in ${var:POS:LEN} */
pos = atoi(loc); /* number(loc) errors out on "1:4" */
@@ -11577,15 +11579,18 @@ parsesub: {
STPUTC('=', out);
flags = 0;
if (subtype == 0) {
+ static const char types[] ALIGN1 = "}-+?=";
/* ${VAR...} but not $VAR or ${#VAR} */
/* c == first char after VAR */
switch (c) {
case ':':
c = pgetc();
#if ENABLE_ASH_BASH_COMPAT
- if (c == ':' || c == '$' || isdigit(c)) {
-//TODO: support more general format ${v:EXPR:EXPR},
-// where EXPR follows $(()) rules
+ /* This check is only needed to not misinterpret
+ * ${VAR:-WORD}, ${VAR:+WORD}, ${VAR:=WORD}, ${VAR:?WORD}
+ * constructs.
+ */
+ if (!strchr(types, c)) {
subtype = VSSUBSTR;
pungetc();
break; /* "goto do_pungetc" is bigger (!) */
@@ -11594,7 +11599,6 @@ parsesub: {
flags = VSNUL;
/*FALLTHROUGH*/
default: {
- static const char types[] ALIGN1 = "}-+?=";
const char *p = strchr(types, c);
if (p == NULL)
goto badsub;