diff options
Diffstat (limited to 'toys/pending')
-rw-r--r-- | toys/pending/sh.c | 539 |
1 files changed, 376 insertions, 163 deletions
diff --git a/toys/pending/sh.c b/toys/pending/sh.c index 4e626c75..67b8459e 100644 --- a/toys/pending/sh.c +++ b/toys/pending/sh.c @@ -220,8 +220,7 @@ GLOBALS( // undo redirects, a=b at start, child PID, exit status, has !, job # int *urd, envlen, pid, exit, not, job; long long when; // when job backgrounded/suspended -// TODO struct sh_arg *raw; // for display - struct sh_arg arg; + struct sh_arg *raw, arg; } *pp; // currently running process struct sh_arg jobs, *arg; // job list, command line args for $* etc @@ -348,9 +347,15 @@ static struct sh_vars *addvar(char *s) } // TODO function to resolve a string into a number for $((1+2)) etc -long long do_math(char *s) +long long do_math(char **s) { - return atoll(s); + long long ll; + + while (isspace(**s)) ++*s; + ll = strtoll(*s, s, 0); + while (isspace(**s)) ++*s; + + return ll; } // Assign one variable from malloced key=val string, returns var struct @@ -376,10 +381,10 @@ static struct sh_vars *setvar(char *s) if (s[len] != '=') { error_msg("bad setvar %s\n", s); free(s); + return 0; } - if (len == 3 && !memcmp(s, "IFS", 3)) TT.ifs = s+4; - + if (!strncmp(s, "IFS=", 4)) TT.ifs = s+4; if (!(var = findvar(s))) return addvar(s); flags = var->flags; @@ -388,20 +393,26 @@ static struct sh_vars *setvar(char *s) free(s); return 0; - } else if (flags&VAR_MAGIC) { - if (*s == 'S') TT.SECONDS = millitime() - 1000*do_math(s+len-1); - else if (*s == 'R') srandom(do_math(s+len-1)); - } else if (flags&VAR_GLOBAL) xsetenv(var->str = s, 0); - else { - free(var->str); - var->str = s; } + // TODO if (flags&(VAR_TOUPPER|VAR_TOLOWER)) // unicode _is stupid enough for upper/lower case to be different utf8 byte // lengths. example: lowercase of U+0130 (C4 B0) is U+0069 (69) // TODO VAR_INT // TODO VAR_ARRAY VAR_DICT + if (flags&VAR_MAGIC) { + char *ss = s+len-1; + +// TODO: trailing garbage after do_math()? + if (*s == 'S') TT.SECONDS = millitime() - 1000*do_math(&ss); + else if (*s == 'R') srandom(do_math(&ss)); + } else if (flags&VAR_GLOBAL) xsetenv(var->str = s, 0); + else { + free(var->str); + var->str = s; + } + return var; } @@ -750,7 +761,7 @@ static int pipe_subshell(char *s, int len, int out) } // utf8 strchr: return wide char matched at wc from chrs, or 0 if not matched -// if len, save length of wc +// if len, save length of next wc (whether or not it's in list) static int utf8chr(char *wc, char *chrs, int *len) { wchar_t wc1, wc2; @@ -772,6 +783,54 @@ static int utf8chr(char *wc, char *chrs, int *len) return 0; } +// grab variable or special param (ala $$) up to len bytes. Return value. +// set *used to length consumed. Does not handle $* and $@ +char *getvar_special(char *str, int len, int *used, struct arg_list **delete) +{ + char *s = 0, *ss, cc = *str; + unsigned uu; + + *used = 1; + if (cc == '-') { + s = ss = xmalloc(8); + if (TT.options&OPT_I) *ss++ = 'i'; + if (TT.options&OPT_BRACE) *ss++ = 'B'; + if (TT.options&OPT_S) *ss++ = 's'; + if (TT.options&OPT_C) *ss++ = 'c'; + *ss = 0; + } else if (cc == '?') s = xmprintf("%d", toys.exitval); + else if (cc == '$') s = xmprintf("%d", TT.pid); + else if (cc == '#') s = xmprintf("%d", TT.arg->c?TT.arg->c-1:0); + else if (cc == '!') s = xmprintf("%d"+2*!TT.bangpid, TT.bangpid); + else { + delete = 0; + for (*used = uu = 0; *used<len && isdigit(str[*used]); ++*used) + uu = (10*uu)+str[*used]-'0'; + if (*used) { + if (uu) uu += TT.shift; + if (uu<TT.arg->c) s = TT.arg->v[uu]; + } else if ((*used = varend(str)-str)) return getvar(str); + } + if (s) push_arg(delete, s); + + return s; +} + +// Copy string until } including escaped } +char *slashcopy(char *s, char c) +{ + char *ss; + int ii, jj; + + for (ii = 0; s[ii] != c; ii++) if (s[ii] == '\\') ii++; + ss = xmalloc(ii+1); + for (ii = jj = 0; s[jj] != c; ii++) + if ('\\'==(s[ii] = s[jj++])) s[ii] = s[jj++]; + ss[ii] = 0; + + return ss; +} + #define NO_PATH (1<<0) // path expansion (wildcards) #define NO_SPLIT (1<<1) // word splitting #define NO_BRACE (1<<2) // {brace,expansion} @@ -789,8 +848,8 @@ static int utf8chr(char *wc, char *chrs, int *len) static int expand_arg_nobrace(struct sh_arg *arg, char *str, unsigned flags, struct arg_list **delete) { - char cc, qq = 0, *old = str, *new = str, *s, *ss, *ifs, **aa; - int ii = 0, dd, jj, kk, ll, oo = 0, nodel; + char cc, qq = 0, sep[6], *old = str, *new = str, *s, *ss, *ifs, *slice; + int ii = 0, oo = 0, xx, yy, dd, jj, kk, ll, mm; if (BUGBUG) dprintf(255, "expand %s\n", str); @@ -822,6 +881,7 @@ if (BUGBUG) dprintf(255, "expand %s\n", str); // parameter/variable expansion, and dequoting for (; (cc = str[ii++]); old!=new && (new[oo] = 0)) { + struct sh_arg aa = {0}; // skip literal chars if (!strchr("$'`\\\"", cc)) { @@ -834,11 +894,9 @@ if (BUGBUG) dprintf(255, "expand %s\n", str); new = xstrdup(new); new[oo = ii-1] = 0; } - ifs = 0; - aa = 0; - nodel = 0; + ifs = slice = 0; - // handle different types of escapes + // handle escapes and quoting if (cc == '\\') new[oo++] = str[ii] ? str[ii++] : cc; else if (cc == '"') qq++; else if (cc == '\'') { @@ -888,50 +946,188 @@ dprintf(2, "TODO: do math for %.*s\n", kk, s); for (kk = strlen(ifs); kk && ifs[kk-1]=='\n'; ifs[--kk] = 0); close(jj); } - } else if (cc == '$') { - // parse $ $'' ${} or $VAR + // $VARIABLE expansions - cc = str[ii++]; + } else if (cc == '$') { + cc = *(ss = str+ii++); if (cc=='\'') { for (s = str+ii; *s != '\''; oo += wcrtomb(new+oo, unescape2(&s, 0),0)); ii = s-str+1; continue; + } else if (cc=='"' && !(qq&1)) { + qq++; + + continue; } else if (cc == '{') { - cc = *(ss = str+ii); - if (!(jj = strchr(ss, '}')-ss)) ifs = (void *)1; - ii += jj+1; - - if (jj>1) { - // handle ${!x} and ${#x} - if (*ss == '!') { - if (!(ss = getvar(ss+1)) || !*ss) continue; - jj = varend(ss)-ss; - if (ss[jj]) ifs = (void *)1; - } else if (*ss == '#') { - if (jj == 2 && (*ss == '@' || *ss == '*')) jj--; - else ifs = xmprintf("%ld", (long)strlen(getvar(ss) ? : "")); + + // Skip escapes to find }, parse_word() guarantees ${} terminates + for (cc = *++ss; str[ii] != '}'; ii++) if (str[ii]=='\\') ii++; + ii++; + + if (cc == '}') ifs = (void *)1; + else if (strchr("#!", cc)) ss++; + jj = varend(ss)-ss; + if (!jj) while (isdigit(ss[jj])) jj++; + if (!jj && strchr("#$!_*", *ss)) jj++; + // parameter or operator? Maybe not a prefix: ${#-} vs ${#-x} + if (!jj && strchr("-?@", *ss)) if (ss[++jj]!='}' && ss[-1]!='{') ss--; + slice = ss+jj; // start of :operation + + if (!jj) { + ifs = (void *)1; + // literal ${#} or ${!} wasn't a prefix + if (strchr("#!", cc)) ifs = getvar_special(--ss, 1, &kk, delete); + } else if (ss[-1]=='{'); // not prefix, fall through + else if (cc == '#') { // TODO ${#x[@]} + dd = !!strchr("@*", *ss); // For ${#@} or ${#*} do normal ${#} + ifs = getvar_special(ss-dd, jj, &kk, delete) ? : ""; + if (!dd) push_arg(delete, ifs = xmprintf("%ld", strlen(ifs))); + // ${!@} ${!@Q} ${!x} ${!x@} ${!x@Q} ${!x#} ${!x[} ${!x[*]} + } else if (cc == '!') { // TODO: ${var[@]} array + + // special case: normal varname followed by @} or *} = prefix list + if (ss[jj] == '*' || (ss[jj] == '@' && !isalpha(ss[jj+1]))) { + for (slice++, kk = 0; kk<TT.varslen; kk++) { + if (!strncmp(s = TT.vars[kk].str, ss, jj)) { + arg_add(&aa, s = xstrndup(s, stridx(s, '='))); + push_arg(delete, s); + } + } + if (aa.c) push_arg(delete, (void *)aa.v); + + // else dereference to get new varname, discarding if none, check err + } else { + // First expansion + if (strchr("@*", *ss)) { // special case ${!*}/${!@} + expand_arg_nobrace(&aa, "\"$*\"", NO_PATH|NO_SPLIT, delete); + ifs = *aa.v; + free(aa.v); + memset(&aa, 0, sizeof(aa)); + jj = 1; + } else ifs = getvar_special(ss, jj, &jj, delete); + slice = ss+jj; + + // Second expansion + if (!jj) ifs = (void *)1; + else if (ifs && *(ss = ifs)) { + if (strchr("@*", cc)) { + aa.c = TT.arg->c-1; + aa.v = TT.arg->v+1; + jj = 1; + } else ifs = getvar_special(ifs, strlen(ifs), &jj, delete); + if (ss && ss[jj]) { + ifs = (void *)1; + slice = ss+strlen(ss); + } + } } } - } else { - ss = str+--ii; - if (!(jj = varend(ss)-ss)) jj++; - ii += jj; + + // Substitution error? + if (ifs == (void *)1) { +barf: + if (!(((unsigned long)ifs)>>1)) ifs = "bad substitution"; + error_msg("%.*s: %s", (int)(slice-ss), ss, ifs); + free(new); + + return 1; + } + } else jj = 1; + + // Resolve unprefixed variables + if (strchr("{$", ss[-1])) { + if (strchr("@*", cc)) { + aa.c = TT.arg->c-1; + aa.v = TT.arg->v+1; + } else { + ifs = getvar_special(ss, jj, &jj, delete); + if (!jj) { + if (ss[-1] == '{') goto barf; + new[oo++] = '$'; + ii--; + continue; + } else if (ss[-1] != '{') ii += jj-1; + } } + } + + // combine before/ifs/after sections & split words on $IFS in ifs + + // Fetch separator + *sep = 0; + if ((qq&1) && cc=='*') { + wchar_t wc; + + if (flags&SEMI_IFS) strcpy(sep, " "); + else if (0<(dd = utf8towc(&wc, TT.ifs, 4))) + sprintf(sep, "%.*s", dd, TT.ifs); + } + + // when aa proceed through entries until NULL, else process ifs once + mm = yy = 0; + do { + + // get next argument + if (aa.c) ifs = aa.v[mm++] ? : ""; + + // Are we performing surgery on this argument? + if (slice && *slice != '}') { + dd = slice[xx = (*slice == ':')]; + if (!ifs || (xx && !*ifs)) { + if (strchr("-?=", dd)) { // - use default = assign default ? error + push_arg(delete, ifs = slashcopy(slice+xx+1, '}')); + if (dd == '?' || (dd == '=' && + !(setvar(s = xmprintf("%.*s=%s", (int)(slice-ss), ss, ifs))))) + goto barf; + } + // use alternate value + } else if (dd == '+') push_arg(delete, ifs = slashcopy(slice+xx+1,'}')); + else if (xx) { // ${x::} + long long la, lb, lc; + +// TODO don't redo math in loop + ss = slice+1; + la = do_math(&s); + if (s && *s == ':') { + s++; + lb = do_math(&s); + } else lb = LLONG_MAX; + if (s && *s != '}') { + error_msg("%.*s: bad '%c'", (int)(slice-ss), ss, *s); + s = 0; + } + if (!s) { + free(new); + return 1; + } + + // This isn't quite what bash does, but close enough. + if (!(lc = aa.c)) lc = strlen(ifs); + else if (!la && !yy && strchr("@*", slice[1])) { + aa.v--; // ${*:0} shows $0 even though default is 1-indexed + aa.c++; + yy++; + } + if (la<0 && (la += lc)<0) continue; + if (lb<0) lb = lc+lb-la; + if (aa.c) { + if (mm<la || mm>=la+lb) continue; + } else if (la>=lc || lb<0) ifs = ""; + else if (la+lb>=lc) ifs += la; + else if (!*delete || ifs != (*delete)->arg) + push_arg(delete, ifs = xmprintf("%.*s", (int)lb, ifs+la)); + else { + for (dd = 0; dd<lb ; dd++) if (!(ifs[dd] = ifs[dd+la])) break; + ifs[dd] = 0; + } + } else { +// TODO test ${-abc} as error + ifs = slice; + goto barf; + } -// ${#nom} ${} ${x} -// ${x:-y} use default -// ${x:=y} assign default (error if positional) -// ${x:?y} err if null -// ${x:+y} alt value -// ${x:off} ${x:off:len} off<0 from end (must ": -"), len<0 also from end must -// 0-based indexing -// ${@:off:len} positional parameters, off -1 = len, -len is error -// 1-based indexing -// ${!x} deref (does bad substitution if name has : in it) -// ${!x*} ${!x@} names matching prefix -// note: expands something other than arg->c // ${x#y} remove shortest prefix ${x##y} remove longest prefix // x can be @ or * // ${x%y} ${x%%y} suffix @@ -943,130 +1139,60 @@ dprintf(2, "TODO: do math for %.*s\n", kk, s); // A=declare that recreates var a=attribute flags // x can be @* -// TODO: $_ is last arg of last command, and exported as path to exe run - if (ifs); - else if (cc == '-') { - s = ifs = xmalloc(8); - if (TT.options&OPT_I) *s++ = 'i'; - if (TT.options&OPT_BRACE) *s++ = 'B'; - if (TT.options&OPT_S) *s++ = 's'; - if (TT.options&OPT_C) *s++ = 'c'; - *s = 0; - } else if (cc == '?') ifs = xmprintf("%d", toys.exitval); - else if (cc == '$') ifs = xmprintf("%d", TT.pid); - else if (cc == '#') ifs = xmprintf("%d", TT.arg->c?TT.arg->c-1:0); - else if (cc == '!') ifs = xmprintf("%d"+2*!TT.bangpid, TT.bangpid); - else if (cc == '*' || cc == '@') aa = TT.arg->v+1; - else if (isdigit(cc)) { - for (kk = ll = 0; kk<jj && isdigit(ss[kk]); kk++) - ll = (10*ll)+ss[kk]-'0'; - if (ll) ll += TT.shift; - if (ll<TT.arg->c) ifs = TT.arg->v[ll]; - nodel = 1; - - // $VARIABLE - } else { - if (ss == varend(ss)) { - ii--; - if (ss[-1] == '$') new[oo++] = '$'; - else ifs = (void *)1; - } else ifs = getvar(ss); - nodel = 1; - } - } - // TODO: $((a=42)) can change var, affect lifetime // must replace ifs AND any previous output arg[] within pointer strlen() -// TODO ${blah} here - - if (ifs == (void *)1) { - error_msg("%.*s: bad substitution", (int)(s-(str+ii)+3), str+ii-2); - free(new); - - return 1; - } - // combine before/ifs/after sections, splitting words on $IFS in ifs - if (ifs || aa) { - char sep[8]; - - // If not gluing together, nothing to substitute, not quoted: do nothing - if (!aa && !*ifs && !qq) continue; - - // Fetch separator - *sep = 0; - if ((qq&1) && cc=='*') { - wchar_t wc; - - if (flags&SEMI_IFS) strcpy(sep, " "); - else if (0<(dd = utf8towc(&wc, TT.ifs, 4))) - sprintf(sep, "%.*s", dd, TT.ifs); } - // when aa proceed through entries until NULL, else process ifs once + // Nothing left to do? + if (!ifs) break; + if (!*ifs && !qq) continue; + + // loop within current ifs checking region to split words do { - // get next argument, is this last entry, find end of word - if (aa) { - ifs = *aa ? : ""; - if (*aa) aa++; - nodel = 1; - } + // find end of (split) word if (qq&1) ss = ifs+strlen(ifs); - else for (ss = ifs; *ss; ss += kk) - if ((ll = utf8chr(ss, TT.ifs, &kk))) break; - kk = !aa || !*aa; - - // loop within current ifs checking region to split words - do { - // fast path: use existing memory when no prefix, not splitting it, - // and either not last entry or no suffix - if (!oo && !*ss && (!kk || !str[ii]) && !((qq&1) && cc=='*')) { - if (!qq && ss==ifs) break; - arg_add_del(arg, ifs, nodel ? 0 : delete); - nodel = 1; - - continue; - } + else for (ss = ifs; *ss; ss += kk) if (utf8chr(ss, TT.ifs, &kk)) break; - // resize allocation and copy next chunk of IFS-free data - new = xrealloc(new, oo + (ss-ifs) + strlen(sep) + - ((jj = kk && !*ss) ? strlen(str+ii) : 0) + 1); - oo += sprintf(new + oo, "%.*s", (int)(ss-ifs), ifs); - if (!nodel) free(ifs); - nodel = 1; - if (jj) break; + // when no prefix, not splitting, no suffix: use existing memory + if (!oo && !*ss && !((mm==aa.c) ? str[ii] : ((qq&1) && cc=='*'))) { + if (qq || ss!=ifs) arg_add(arg, ifs); + continue; + } - // for single quoted "$*" append IFS - if ((qq&1) && cc == '*') oo += sprintf(new+oo, "%s", sep); + // resize allocation and copy next chunk of IFS-free data + new = xrealloc(new, oo + (ss-ifs) + strlen(sep) + + ((jj = (mm == aa.c) && !*ss) ? strlen(str+ii) : 0) + 1); + oo += sprintf(new + oo, "%.*s", (int)(ss-ifs), ifs); + if (jj) break; - // add argument if quoted, non-blank, or non-whitespace separator - else { - if (qq || *new || *ss) { - arg_add_del(arg, new, nodel ? 0 : delete); - nodel = 1; - } - qq &= 1; - new = xstrdup(str+ii); - oo = 0; - } + // for double quoted "$*" append first separator from IFS + if ((qq&1) && cc == '*') oo += sprintf(new+oo, "%s", sep); - // Skip trailing seperator (combining whitespace) - while ((jj = utf8chr(ss, TT.ifs, &ll)) && iswspace(jj)) ss += ll; + // finished arg: keep if quoted, non-blank, or non-whitespace separator + else { + if (qq || *new || *ss) arg_add(arg, xrealloc(new, strlen(new)+1)); + qq &= 1; + new = xstrdup(str+ii); + oo = 0; + } - } while (*(ifs = ss)); - } while (!kk); - } + // Skip trailing seperator (combining whitespace) + while ((jj = utf8chr(ss, TT.ifs, &ll))) { + ss += ll; + if (!iswspace(jj)) break; + } + } while (*(ifs = ss)); + } while (!(mm == aa.c)); } -// TODO globbing * ? [ +// TODO globbing * ? [] +() happens after variable resolution -// Word splitting completely eliminating argument when no non-$IFS data left +// TODO test word splitting completely eliminating argument when no non-$IFS data left // wordexp keeps pattern when no matches -// TODO NO_SPLIT cares about IFS, see also trailing \n - -// quote removal +// TODO test NO_SPLIT cares about IFS, see also trailing \n // Record result. if (*new || qq) @@ -1268,6 +1394,7 @@ static struct sh_process *expand_redir(struct sh_arg *arg, int envlen, int *urd) pp = xzalloc(sizeof(struct sh_process)); pp->urd = urd; + pp->raw = arg; // When we redirect, we copy each displaced filehandle to restore it later. @@ -1515,8 +1642,7 @@ if (BUGBUG) { int i; dprintf(255, "envlen=%d arg->c=%d run=", envlen, arg->c); f if (envlen == arg->c) { for (jj = 0; jj<envlen; jj++) { if (!(s = expand_one_arg(arg->v[jj], NO_PATH|NO_SPLIT, 0))) break; - if (s == arg->v[jj]) s = xstrdup(s); - setvar(s); + setvar((s == arg->v[jj]) ? xstrdup(s) : s); } goto out; } @@ -1524,7 +1650,7 @@ if (BUGBUG) { int i; dprintf(255, "envlen=%d arg->c=%d run=", envlen, arg->c); f // assign leading environment variables (if any) in temp environ copy jj = 0; env.v = 0; - if (envlen) { + if (!pp->exit && envlen) { for (env.c = 0; environ[env.c]; env.c++); memcpy(env.v = xmalloc(sizeof(char *)*(env.c+33)), environ, sizeof(char *)*(env.c+1)); @@ -2062,7 +2188,6 @@ TODO: a | b | c needs subshell for builtins? when to auto-exec? ps vs sh -c 'ps' vs sh -c '(ps)' */ - TT.hfd = 10; // iterate through pipeline segments @@ -2230,7 +2355,7 @@ dprintf(2, "TODO skipped running for((;;)), need math parser\n"); } // for && and || skip pipeline segment(s) based on return code - if (pl->type == 1 || pl->type == 3) + if (!pl->type || pl->type == 3) while (ctl && !strcmp(ctl, toys.exitval ? "&&" : "||")) ctl = (pl = pl->type ? pl->end : pl->next)?pl->arg->v[pl->arg->c]:0; @@ -2748,6 +2873,94 @@ void exec_main(void) environ = old; } +// Find + and - jobs. Returns index of plus, writes minus to *minus +int find_plus_minus(int *minus) +{ + long long when, then; + int i, plus; + + if (minus) *minus = 0; + for (then = i = plus = 0; i<TT.jobs.c; i++) { + if ((when = ((struct sh_process *)TT.jobs.v[i])->when) > then) { + then = when; + if (minus) *minus = plus; + plus = i; + } + } + + return plus; +} + +// Return T.jobs index or -1 from identifier +// Note, we don't return "ambiguous job spec", we return the first hit or -1. +// TODO %% %+ %- %?ab +int find_job(char *s) +{ + char *ss; + long ll = strtol(s, &ss, 10); + int i, j; + + if (!TT.jobs.c) return -1; + if (!*s || (!s[1] && strchr("%+-", *s))) { + int minus, plus = find_plus_minus(&minus); + + return (*s == '-') ? minus : plus; + } + + // Is this a %1 numeric jobspec? + if (s != ss && !*ss) + for (i = 0; i<TT.jobs.c; i++) + if (((struct sh_process *)TT.jobs.v[i])->job == ll) return i; + + // Match start of command or %?abc + for (i = 0; i<TT.jobs.c; i++) { + struct sh_process *pp = (void *)TT.jobs.v[i]; + + if (strstart(&s, *pp->arg.v)) return i; + if (*s != '?' || !s[1]) continue; + for (j = 0; j<pp->arg.c; j++) if (strstr(pp->arg.v[j], s+1)) return i; + } + + return -1; +} + +// We pass in dash to avoid looping over every job each time +void print_job(int i, char dash) +{ + struct sh_process *pp = (void *)TT.jobs.v[i]; + char *s = "Run"; + int j; + +// TODO Terminated (Exited) + if (pp->exit<0) s = "Stop"; + else if (pp->exit>126) s = "Kill"; + else if (pp->exit>0) s = "Done"; + printf("[%d]%c %-6s", pp->job, dash, s); + for (j = 0; j<pp->raw->c; j++) printf(" %s"+!j, pp->raw->v[j]); + printf("\n"); +} + +void jobs_main(void) +{ + int i, j, minus, plus = find_plus_minus(&minus); + char *s; + +// TODO -lnprs + + for (i = 0;;i++) { + if (toys.optc) { + if (!(s = toys.optargs[i])) break; + if ((j = find_job(s+('%' == *s))) == -1) { + perror_msg("%s: no such job", s); + + continue; + } + } else if ((j = i) >= TT.jobs.c) break; + + print_job(i, (i == plus) ? '+' : (i == minus) ? '-' : ' '); + } +} + void shift_main(void) { long long by = 1; |