diff options
author | Rob Landley <rob@landley.net> | 2020-04-08 23:08:57 -0500 |
---|---|---|
committer | Rob Landley <rob@landley.net> | 2020-04-08 23:08:57 -0500 |
commit | 41dfa883e178c239bed0ec02fa38af09739d55b0 (patch) | |
tree | 1f3f6488f05d00130d485dee0e07c28d5c260c7f | |
parent | 4f1e357b5d2064b433006bd707a2e7307d79542a (diff) | |
download | toybox-41dfa883e178c239bed0ec02fa38af09739d55b0.tar.gz |
Implement ! and $$ and fix quoted "$(subshell)"
-rw-r--r-- | toys/pending/sh.c | 30 |
1 files changed, 18 insertions, 12 deletions
diff --git a/toys/pending/sh.c b/toys/pending/sh.c index 55c30a19..183d5fd1 100644 --- a/toys/pending/sh.c +++ b/toys/pending/sh.c @@ -485,7 +485,7 @@ static int redir_prefix(char *word) } // parse next word from command line. Returns end, or 0 if need continuation -// caller eats leading spaces. If early, stop at first unquoted char. +// caller eats leading spaces. early = skip one quote block (or return start) static char *parse_word(char *start, int early) { int i, quote = 0, q, qc = 0; @@ -497,8 +497,7 @@ static char *parse_word(char *start, int early) // Redirections. 123<<file- parses as 2 args: "123<<" "file-". s = end + redir_prefix(end); - if ((i = anystart(s, (void *)redirectors))) s += i; - if (s != end) return (end == start) ? s : end; + if ((i = anystart(s, (void *)redirectors))) return s+i; // (( is a special quote at the start of a word if (strstart(&end, "((")) toybuf[quote++] = 254; @@ -535,7 +534,8 @@ static char *parse_word(char *start, int early) else if (q == '\'') end++; else i++; - // loop if we already handled a symbol + // loop if we already handled a symbol and aren't stopping early + if (early && !quote) return end; if (!i) continue; } else { // Things that only matter when unquoted @@ -565,10 +565,9 @@ static char *parse_word(char *start, int early) toybuf[quote++] = ")}]"[i]; end++; } - } else { - if (early && !quote) break; - end++; } + if (early && !quote) return end; + end++; } return quote ? 0 : end; @@ -871,7 +870,7 @@ if (BUGBUG) dprintf(255, "expand %s\n", str); off_t pp = 0; s = str+ii-1; - kk = parse_word(str+ii-1, 1)-s; + kk = parse_word(s, 1)-s; if (*toybuf == 255) { s += 3; kk -= 5; @@ -913,6 +912,7 @@ dprintf(2, "TODO: do math for %.*s\n", kk, s); new[oo++] = cc; break; } else if (cc == '?') ifs = del = xmprintf("%d", toys.exitval); + else if (cc == '$') ifs = del = xmprintf("%d", TT.pid); else if (cc == '#') ifs = del = xmprintf("%d", TT.arg->c?TT.arg->c-1:0); else if (cc == '*' || cc == '@') { // If not doing word split, handle here @@ -1025,7 +1025,7 @@ static void expand_arg(struct sh_arg *arg, char *old, unsigned flags, { struct brace { struct brace *next, *prev, *stack; - int active, cnt, idx, commas[]; + int active, cnt, idx, dots[2], commas[]; } *bb = 0, *blist = 0, *bstk, *bnext; int i, j; char *s, *ss; @@ -1214,6 +1214,12 @@ static struct sh_process *expand_redir(struct sh_arg *arg, int envlen, int *urd) s = arg->v[j]; + if (!strcmp(s, "!")) { + pp->not ^= 1; + + continue; + } + // Handle <() >() redirectionss if ((*s == '<' || *s == '>') && s[1] == '(') { int new = pipe_subshell(s+2, strlen(s+2)-1, *s == '>'); @@ -1884,7 +1890,7 @@ static int wait_pipeline(struct sh_process *pp) pp->pid = 0; } // TODO handle set -o pipefail here - rc = pp->exit; + rc = pp->not ? !pp->exit : pp->exit; } return rc; @@ -2640,9 +2646,9 @@ void export_main(void) void eval_main(void) { int len = 1; - char *s = merge_args("", toys.optc+1, toys.argv, " ", &len, ""); + char *s; - sh_run(s); + sh_run(s = merge_args("", toys.optc+1, toys.argv, " ", &len, "")); free(s); } |