hush: add support for "set -o pipefail"

function                                             old     new   delta
checkjobs                                            467     517     +50
builtin_set                                          259     286     +27
o_opt_strings                                          -      10     +10
hush_main                                           1011    1013      +2
------------------------------------------------------------------------------
(add/remove: 1/0 grow/shrink: 3/0 up/down: 89/0)               Total: 89 bytes

Signed-off-by: Denys Vlasenko <vda.linux@googlemail.com>
This commit is contained in:
Denys Vlasenko 2010-11-14 02:01:50 +01:00
parent c08c3f5d26
commit 6696eac274
3 changed files with 186 additions and 23 deletions

View File

@ -507,6 +507,7 @@ struct command {
# define CMD_FUNCDEF 3 # define CMD_FUNCDEF 3
#endif #endif
smalluint cmd_exitcode;
/* if non-NULL, this "command" is { list }, ( list ), or a compound statement */ /* if non-NULL, this "command" is { list }, ( list ), or a compound statement */
struct pipe *group; struct pipe *group;
#if !BB_MMU #if !BB_MMU
@ -637,6 +638,43 @@ struct function {
#endif #endif
/* set -/+o OPT support. (TODO: make it optional)
* bash supports the following opts:
* allexport off
* braceexpand on
* emacs on
* errexit off
* errtrace off
* functrace off
* hashall on
* histexpand off
* history on
* ignoreeof off
* interactive-comments on
* keyword off
* monitor on
* noclobber off
* noexec off
* noglob off
* nolog off
* notify off
* nounset off
* onecmd off
* physical off
* pipefail off
* posix off
* privileged off
* verbose off
* vi off
* xtrace off
*/
static const char o_opt_strings[] ALIGN1 = "pipefail\0";
enum {
OPT_O_PIPEFAIL,
NUM_OPT_O
};
/* "Globals" within this file */ /* "Globals" within this file */
/* Sorted roughly by size (smaller offsets == smaller code) */ /* Sorted roughly by size (smaller offsets == smaller code) */
struct globals { struct globals {
@ -675,6 +713,7 @@ struct globals {
int last_jobid; int last_jobid;
pid_t saved_tty_pgrp; pid_t saved_tty_pgrp;
struct pipe *job_list; struct pipe *job_list;
char o_opt[NUM_OPT_O];
# define G_saved_tty_pgrp (G.saved_tty_pgrp) # define G_saved_tty_pgrp (G.saved_tty_pgrp)
#else #else
# define G_saved_tty_pgrp 0 # define G_saved_tty_pgrp 0
@ -6315,24 +6354,23 @@ static int checkjobs(struct pipe *fg_pipe)
if (fg_pipe->cmds[i].pid != childpid) if (fg_pipe->cmds[i].pid != childpid)
continue; continue;
if (dead) { if (dead) {
int ex;
fg_pipe->cmds[i].pid = 0; fg_pipe->cmds[i].pid = 0;
fg_pipe->alive_cmds--; fg_pipe->alive_cmds--;
if (i == fg_pipe->num_cmds - 1) { ex = WEXITSTATUS(status);
/* last process gives overall exitstatus */ /* bash prints killer signal's name for *last*
rcode = WEXITSTATUS(status); * process in pipe (prints just newline for SIGINT).
/* bash prints killer signal's name for *last* * Mimic this. Example: "sleep 5" + (^\ or kill -QUIT)
* process in pipe (prints just newline for SIGINT). */
* Mimic this. Example: "sleep 5" + (^\ or kill -QUIT) if (WIFSIGNALED(status)) {
*/ int sig = WTERMSIG(status);
if (WIFSIGNALED(status)) { if (i == fg_pipe->num_cmds-1)
int sig = WTERMSIG(status);
printf("%s\n", sig == SIGINT ? "" : get_signame(sig)); printf("%s\n", sig == SIGINT ? "" : get_signame(sig));
/* TODO: MIPS has 128 sigs (1..128), what if sig==128 here? /* TODO: MIPS has 128 sigs (1..128), what if sig==128 here?
* Maybe we need to use sig | 128? */ * Maybe we need to use sig | 128? */
rcode = sig + 128; ex = sig + 128;
}
IF_HAS_KEYWORDS(if (fg_pipe->pi_inverted) rcode = !rcode;)
} }
fg_pipe->cmds[i].cmd_exitcode = ex;
} else { } else {
fg_pipe->cmds[i].is_stopped = 1; fg_pipe->cmds[i].is_stopped = 1;
fg_pipe->stopped_cmds++; fg_pipe->stopped_cmds++;
@ -6341,6 +6379,15 @@ static int checkjobs(struct pipe *fg_pipe)
fg_pipe->alive_cmds, fg_pipe->stopped_cmds); fg_pipe->alive_cmds, fg_pipe->stopped_cmds);
if (fg_pipe->alive_cmds == fg_pipe->stopped_cmds) { if (fg_pipe->alive_cmds == fg_pipe->stopped_cmds) {
/* All processes in fg pipe have exited or stopped */ /* All processes in fg pipe have exited or stopped */
i = fg_pipe->num_cmds;
while (--i >= 0) {
rcode = fg_pipe->cmds[i].cmd_exitcode;
/* usually last process gives overall exitstatus,
* but with "set -o pipefail", last *failed* process does */
if (G.o_opt[OPT_O_PIPEFAIL] == 0 || rcode != 0)
break;
}
IF_HAS_KEYWORDS(if (fg_pipe->pi_inverted) rcode = !rcode;)
/* Note: *non-interactive* bash does not continue if all processes in fg pipe /* Note: *non-interactive* bash does not continue if all processes in fg pipe
* are stopped. Testcase: "cat | cat" in a script (not on command line!) * are stopped. Testcase: "cat | cat" in a script (not on command line!)
* and "killall -STOP cat" */ * and "killall -STOP cat" */
@ -7340,13 +7387,41 @@ static void set_fatal_handlers(void)
} }
#endif #endif
static int set_mode(const char cstate, const char mode) static int set_mode(int state, char mode, const char *o_opt)
{ {
int state = (cstate == '-' ? 1 : 0); int idx;
switch (mode) { switch (mode) {
case 'n': G.n_mode = state; break; case 'n':
case 'x': IF_HUSH_MODE_X(G_x_mode = state;) break; G.n_mode = state;
default: return EXIT_FAILURE; break;
case 'x':
IF_HUSH_MODE_X(G_x_mode = state;)
break;
case 'o':
if (!o_opt) {
/* "set -+o" without parameter.
* in bash, set -o produces this output:
* pipefail off
* and set +o:
* set +o pipefail
* We always use the second form.
*/
const char *p = o_opt_strings;
idx = 0;
while (*p) {
printf("set %co %s\n", (G.o_opt[idx] ? '-' : '+'), p);
idx++;
p += strlen(p) + 1;
}
break;
}
idx = index_in_strings(o_opt_strings, o_opt);
if (idx >= 0) {
G.o_opt[idx] = state;
break;
}
default:
return EXIT_FAILURE;
} }
return EXIT_SUCCESS; return EXIT_SUCCESS;
} }
@ -7586,7 +7661,7 @@ int hush_main(int argc, char **argv)
#endif #endif
case 'n': case 'n':
case 'x': case 'x':
if (set_mode('-', opt) == 0) /* no error */ if (set_mode(1, opt, NULL) == 0) /* no error */
break; break;
default: default:
#ifndef BB_VER #ifndef BB_VER
@ -8376,15 +8451,18 @@ static int FAST_FUNC builtin_set(char **argv)
} }
do { do {
if (!strcmp(arg, "--")) { if (strcmp(arg, "--") == 0) {
++argv; ++argv;
goto set_argv; goto set_argv;
} }
if (arg[0] != '+' && arg[0] != '-') if (arg[0] != '+' && arg[0] != '-')
break; break;
for (n = 1; arg[n]; ++n) for (n = 1; arg[n]; ++n) {
if (set_mode(arg[0], arg[n])) if (set_mode((arg[0] == '-'), arg[n], argv[1]))
goto error; goto error;
if (arg[n] == 'o' && argv[1])
argv++;
}
} while ((arg = *++argv) != NULL); } while ((arg = *++argv) != NULL);
/* Now argv[0] is 1st argument */ /* Now argv[0] is 1st argument */

View File

@ -0,0 +1,40 @@
Default:
true | true:
0
1
true | false:
1
0
false | true:
0
1
exit 2 | exit 3 | exit 4:
4
0
Pipefail on:
true | true:
0
1
true | false:
1
0
false | true:
1
0
exit 2 | exit 3 | exit 4:
4
0
Pipefail off:
true | true:
0
1
true | false:
1
0
false | true:
0
1
exit 2 | exit 3 | exit 4:
4
0
Done

View File

@ -0,0 +1,45 @@
echo Default:
echo "true | true:"
true | true; echo $?
! true | true; echo $?
echo "true | false:"
true | false; echo $?
! true | false; echo $?
echo "false | true:"
false | true; echo $?
! false | true; echo $?
echo "exit 2 | exit 3 | exit 4:"
exit 2 | exit 3 | exit 4; echo $?
! exit 2 | exit 3 | exit 4; echo $?
echo Pipefail on:
set -o pipefail
echo "true | true:"
true | true; echo $?
! true | true; echo $?
echo "true | false:"
true | false; echo $?
! true | false; echo $?
echo "false | true:"
false | true; echo $?
! false | true; echo $?
echo "exit 2 | exit 3 | exit 4:"
exit 2 | exit 3 | exit 4; echo $?
! exit 2 | exit 3 | exit 4; echo $?
echo Pipefail off:
set +o pipefail
echo "true | true:"
true | true; echo $?
! true | true; echo $?
echo "true | false:"
true | false; echo $?
! true | false; echo $?
echo "false | true:"
false | true; echo $?
! false | true; echo $?
echo "exit 2 | exit 3 | exit 4:"
exit 2 | exit 3 | exit 4; echo $?
! exit 2 | exit 3 | exit 4; echo $?
echo Done