hush: fix wait builtin

function                                             old     new   delta
builtin_wait                                         174     275    +101
sigwaitinfo                                            -      48     +48
__GI_sigwaitinfo                                       -      48     +48
check_and_run_traps                                  133     169     +36
checkjobs                                            349     380     +31
hush_main                                            971     991     +20
static.zero_timespec                                   -       8      +8
run_list                                            2010    2016      +6
file_get                                             254     260      +6
static.zero_ts                                         8       -      -8
------------------------------------------------------------------------------
(add/remove: 3/1 grow/shrink: 6/0 up/down: 304/-8)            Total: 296 bytes
This commit is contained in:
Denis Vlasenko 2009-03-31 17:24:49 +00:00
parent 70c6e40e47
commit 7566bae197

View File

@ -491,13 +491,14 @@ struct globals {
unsigned char charmap[256]; unsigned char charmap[256];
char user_input_buf[ENABLE_FEATURE_EDITING ? BUFSIZ : 2]; char user_input_buf[ENABLE_FEATURE_EDITING ? BUFSIZ : 2];
/* Signal and trap handling */ /* Signal and trap handling */
char **traps; /* char *traps[NSIG] */ // unsigned count_SIGCHLD;
// unsigned handled_SIGCHLD;
/* which signals have non-DFL handler (even with no traps set)? */ /* which signals have non-DFL handler (even with no traps set)? */
unsigned non_DFL_mask; unsigned non_DFL_mask;
char **traps; /* char *traps[NSIG] */
sigset_t blocked_set; sigset_t blocked_set;
sigset_t inherited_set; sigset_t inherited_set;
}; };
#define G (*ptr_to_globals) #define G (*ptr_to_globals)
/* Not #defining name to G.name - this quickly gets unwieldy /* Not #defining name to G.name - this quickly gets unwieldy
* (too many defines). Also, I actually prefer to see when a variable * (too many defines). Also, I actually prefer to see when a variable
@ -829,12 +830,18 @@ static void free_strings(char **strings)
* POSIX says pending signal mask is cleared in child - no need to clear it. * POSIX says pending signal mask is cleared in child - no need to clear it.
* Restore blocked signal set to one inherited by shell just prior to exec. * Restore blocked signal set to one inherited by shell just prior to exec.
* *
* Note: as a result, we do not use signal handlers much. The only use * Note: as a result, we do not use signal handlers much. The only uses
* is to restore terminal pgrp on exit. * are to count SIGCHLDs [disabled - bug somewhere, + bloat]
* and to restore tty pgrp on signal-induced exit.
* *
* TODO: check/fix wait builtin to be interruptible. * TODO: check/fix wait builtin to be interruptible.
*/ */
//static void SIGCHLD_handler(int sig UNUSED_PARAM)
//{
// G.count_SIGCHLD++;
//}
/* called once at shell init */ /* called once at shell init */
static void init_signal_mask(void) static void init_signal_mask(void)
{ {
@ -863,20 +870,24 @@ static void init_signal_mask(void)
mask >>= 1; mask >>= 1;
sig++; sig++;
} }
sigdelset(&G.blocked_set, SIGCHLD);
sigprocmask(SIG_SETMASK, &G.blocked_set, &G.inherited_set); sigprocmask(SIG_SETMASK, &G.blocked_set, &G.inherited_set);
} }
static void check_and_run_traps(void) static int check_and_run_traps(int sig)
{ {
static const struct timespec zero_ts = { 0, 0 }; static const struct timespec zero_timespec = { 0, 0 };
smalluint save_rcode; smalluint save_rcode;
int sig; int last_sig = 0;
if (sig)
goto jump_in;
while (1) { while (1) {
sig = sigtimedwait(&G.blocked_set, NULL, &zero_ts); sig = sigtimedwait(&G.blocked_set, NULL, &zero_timespec);
if (sig <= 0) if (sig <= 0)
break; break;
jump_in:
last_sig = sig;
if (G.traps && G.traps[sig]) { if (G.traps && G.traps[sig]) {
if (G.traps[sig][0]) { if (G.traps[sig][0]) {
/* We have user-defined handler */ /* We have user-defined handler */
@ -890,7 +901,11 @@ static void check_and_run_traps(void)
} }
/* not a trap: special action */ /* not a trap: special action */
switch (sig) { switch (sig) {
// case SIGCHLD:
// G.count_SIGCHLD++;
// break;
case SIGINT: case SIGINT:
bb_putchar('\n');
G.flag_SIGINT = 1; G.flag_SIGINT = 1;
break; break;
//TODO //TODO
@ -900,6 +915,7 @@ static void check_and_run_traps(void)
break; break;
} }
} }
return last_sig;
} }
#if ENABLE_HUSH_JOB #if ENABLE_HUSH_JOB
@ -1209,7 +1225,7 @@ static void get_user_input(struct in_str *i)
* only after <Enter>. (^C will work) */ * only after <Enter>. (^C will work) */
r = read_line_input(prompt_str, G.user_input_buf, BUFSIZ-1, G.line_input_state); r = read_line_input(prompt_str, G.user_input_buf, BUFSIZ-1, G.line_input_state);
/* catch *SIGINT* etc (^C is handled by read_line_input) */ /* catch *SIGINT* etc (^C is handled by read_line_input) */
check_and_run_traps(); check_and_run_traps(0);
} while (r == 0 || G.flag_SIGINT); /* repeat if ^C or SIGINT */ } while (r == 0 || G.flag_SIGINT); /* repeat if ^C or SIGINT */
i->eof_flag = (r < 0); i->eof_flag = (r < 0);
if (i->eof_flag) { /* EOF/error detected */ if (i->eof_flag) { /* EOF/error detected */
@ -1223,7 +1239,7 @@ static void get_user_input(struct in_str *i)
fflush(stdout); fflush(stdout);
G.user_input_buf[0] = r = fgetc(i->file); G.user_input_buf[0] = r = fgetc(i->file);
/*G.user_input_buf[1] = '\0'; - already is and never changed */ /*G.user_input_buf[1] = '\0'; - already is and never changed */
//do we need check_and_run_traps()? (maybe only if stdin) //do we need check_and_run_traps(0)? (maybe only if stdin)
} while (G.flag_SIGINT); } while (G.flag_SIGINT);
i->eof_flag = (r == EOF); i->eof_flag = (r == EOF);
#endif #endif
@ -2329,6 +2345,11 @@ static int checkjobs(struct pipe* fg_pipe)
debug_printf_jobs("checkjobs %p\n", fg_pipe); debug_printf_jobs("checkjobs %p\n", fg_pipe);
errno = 0;
// if (G.handled_SIGCHLD == G.count_SIGCHLD)
// /* avoid doing syscall, nothing there anyway */
// return rcode;
attributes = WUNTRACED; attributes = WUNTRACED;
if (fg_pipe == NULL) if (fg_pipe == NULL)
attributes |= WNOHANG; attributes |= WNOHANG;
@ -2346,10 +2367,21 @@ static int checkjobs(struct pipe* fg_pipe)
// + killall -STOP cat // + killall -STOP cat
wait_more: wait_more:
// TODO: safe_waitpid? while (1) {
while ((childpid = waitpid(-1, &status, attributes)) > 0) {
int i; int i;
const int dead = WIFEXITED(status) || WIFSIGNALED(status); int dead;
// i = G.count_SIGCHLD;
childpid = waitpid(-1, &status, attributes);
if (childpid <= 0) {
if (childpid && errno != ECHILD)
bb_perror_msg("waitpid");
// else /* Until next SIGCHLD, waitpid's are useless */
// G.handled_SIGCHLD = i;
break;
}
dead = WIFEXITED(status) || WIFSIGNALED(status);
#if DEBUG_JOBS #if DEBUG_JOBS
if (WIFSTOPPED(status)) if (WIFSTOPPED(status))
debug_printf_jobs("pid %d stopped by sig %d (exitcode %d)\n", debug_printf_jobs("pid %d stopped by sig %d (exitcode %d)\n",
@ -2428,10 +2460,6 @@ static int checkjobs(struct pipe* fg_pipe)
#endif #endif
} /* while (waitpid succeeds)... */ } /* while (waitpid succeeds)... */
/* wait found no children or failed */
if (childpid && errno != ECHILD)
bb_perror_msg("waitpid");
return rcode; return rcode;
} }
@ -3004,7 +3032,7 @@ static int run_list(struct pipe *pi)
if (r != -1) { if (r != -1) {
/* we only ran a builtin: rcode is already known /* we only ran a builtin: rcode is already known
* and we don't need to wait for anything. */ * and we don't need to wait for anything. */
check_and_run_traps(); check_and_run_traps(0);
#if ENABLE_HUSH_LOOPS #if ENABLE_HUSH_LOOPS
/* was it "break" or "continue"? */ /* was it "break" or "continue"? */
if (G.flag_break_continue) { if (G.flag_break_continue) {
@ -3029,7 +3057,7 @@ static int run_list(struct pipe *pi)
/* even bash 3.2 doesn't do that well with nested bg: /* even bash 3.2 doesn't do that well with nested bg:
* try "{ { sleep 10; echo DEEP; } & echo HERE; } &". * try "{ { sleep 10; echo DEEP; } & echo HERE; } &".
* I'm NOT treating inner &'s as jobs */ * I'm NOT treating inner &'s as jobs */
check_and_run_traps(); check_and_run_traps(0);
#if ENABLE_HUSH_JOB #if ENABLE_HUSH_JOB
if (G.run_list_level == 1) if (G.run_list_level == 1)
insert_bg_job(pi); insert_bg_job(pi);
@ -3040,13 +3068,13 @@ static int run_list(struct pipe *pi)
if (G.run_list_level == 1 && G.interactive_fd) { if (G.run_list_level == 1 && G.interactive_fd) {
/* waits for completion, then fg's main shell */ /* waits for completion, then fg's main shell */
rcode = checkjobs_and_fg_shell(pi); rcode = checkjobs_and_fg_shell(pi);
check_and_run_traps(); check_and_run_traps(0);
debug_printf_exec(": checkjobs_and_fg_shell returned %d\n", rcode); debug_printf_exec(": checkjobs_and_fg_shell returned %d\n", rcode);
} else } else
#endif #endif
{ /* this one just waits for completion */ { /* this one just waits for completion */
rcode = checkjobs(pi); rcode = checkjobs(pi);
check_and_run_traps(); check_and_run_traps(0);
debug_printf_exec(": checkjobs returned %d\n", rcode); debug_printf_exec(": checkjobs returned %d\n", rcode);
} }
} }
@ -3668,9 +3696,9 @@ static int process_command_subs(o_string *dest,
} }
debug_printf("done reading from pipe, pclose()ing\n"); debug_printf("done reading from pipe, pclose()ing\n");
/* This is the step that wait()s for the child. Should be pretty /* This is the step that waits for the child. Should be pretty
* safe, since we just read an EOF from its stdout. We could try * safe, since we just read an EOF from its stdout. We could try
* to do better, by using wait(), and keeping track of background jobs * to do better, by using waitpid, and keeping track of background jobs
* at the same time. That would be a lot of work, and contrary * at the same time. That would be a lot of work, and contrary
* to the KISS philosophy of this program. */ * to the KISS philosophy of this program. */
retcode = fclose(p); retcode = fclose(p);
@ -4586,7 +4614,7 @@ int hush_main(int argc, char **argv)
// to (inadvertently) close/redirect it // to (inadvertently) close/redirect it
} }
} }
init_signal_mask(); init_signal_mask(); /* note: ensures SIGCHLD is not masked */
debug_printf("G.interactive_fd=%d\n", G.interactive_fd); debug_printf("G.interactive_fd=%d\n", G.interactive_fd);
if (G.interactive_fd) { if (G.interactive_fd) {
fcntl(G.interactive_fd, F_SETFD, FD_CLOEXEC); fcntl(G.interactive_fd, F_SETFD, FD_CLOEXEC);
@ -4617,8 +4645,12 @@ int hush_main(int argc, char **argv)
fcntl(G.interactive_fd, F_SETFD, FD_CLOEXEC); fcntl(G.interactive_fd, F_SETFD, FD_CLOEXEC);
} }
} }
init_signal_mask(); init_signal_mask(); /* note: ensures SIGCHLD is not masked */
#endif #endif
/* POSIX allows shell to re-enable SIGCHLD
* even if it was SIG_IGN on entry */
// G.count_SIGCHLD++; /* ensure it is != G.handled_SIGCHLD */
signal(SIGCHLD, SIG_DFL); // SIGCHLD_handler);
#if ENABLE_HUSH_INTERACTIVE && !ENABLE_FEATURE_SH_EXTRA_QUIET #if ENABLE_HUSH_INTERACTIVE && !ENABLE_FEATURE_SH_EXTRA_QUIET
if (G.interactive_fd) { if (G.interactive_fd) {
@ -5164,12 +5196,48 @@ static int builtin_unset(char **argv)
static int builtin_wait(char **argv) static int builtin_wait(char **argv)
{ {
int ret = EXIT_SUCCESS; int ret = EXIT_SUCCESS;
int status; int status, sig;
if (*++argv == NULL) if (*++argv == NULL) {
/* don't care about exit status */ /* Don't care about wait results */
wait(NULL); /* Note 1: must wait until there are no more children */
/* Note 2: must be interruptible */
/* Examples:
* $ sleep 3 & sleep 6 & wait
* [1] 30934 sleep 3
* [2] 30935 sleep 6
* [1] Done sleep 3
* [2] Done sleep 6
* $ sleep 3 & sleep 6 & wait
* [1] 30936 sleep 3
* [2] 30937 sleep 6
* [1] Done sleep 3
* ^C <-- after ~4 sec from keyboard
* $
*/
sigaddset(&G.blocked_set, SIGCHLD);
sigprocmask(SIG_SETMASK, &G.blocked_set, NULL);
while (1) {
checkjobs(NULL);
if (errno == ECHILD)
break;
/* Wait for SIGCHLD or any other signal of interest */
/* sigtimedwait with infinite timeout: */
sig = sigwaitinfo(&G.blocked_set, NULL);
if (sig > 0) {
sig = check_and_run_traps(sig);
if (sig && sig != SIGCHLD) { /* see note 2 */
ret = 128 + sig;
break;
}
}
}
sigdelset(&G.blocked_set, SIGCHLD);
sigprocmask(SIG_SETMASK, &G.blocked_set, NULL);
return ret;
}
/* This is probably buggy wrt interruptible-ness */
while (*argv) { while (*argv) {
pid_t pid = bb_strtou(*argv, NULL, 10); pid_t pid = bb_strtou(*argv, NULL, 10);
if (errno) { if (errno) {