X-Git-Url: https://git.librecmc.org/?a=blobdiff_plain;f=runit%2Frunsv.c;h=48b83960df3a99afaa6ddfe07804feefe8c655f1;hb=8cd04d1cb658227d3c6eeb2a377314ca9cdf249d;hp=bd0f3dcc2583bd20534e0163af543a81b77e9bd5;hpb=a60f84ebf07863e390b72a2b6150e461a1ec18e9;p=oweals%2Fbusybox.git diff --git a/runit/runsv.c b/runit/runsv.c index bd0f3dcc2..48b83960d 100644 --- a/runit/runsv.c +++ b/runit/runsv.c @@ -78,12 +78,13 @@ struct svdir { int pid; smallint state; smallint ctrl; - smallint want; + smallint sd_want; smallint islog; struct timespec start; int fdlock; int fdcontrol; int fdcontrolwrite; + int wstat; }; struct globals { @@ -138,7 +139,8 @@ static void s_term(int sig_no UNUSED_PARAM) write(selfpipe.wr, "", 1); /* XXX */ } -static char *add_str(char *p, const char *to_add) +/* libbb candidate */ +static char *bb_stpcpy(char *p, const char *to_add) { while ((*p = *to_add) != '\0') { p++; @@ -189,24 +191,26 @@ static void update_status(struct svdir *s) char *p = stat_buf; switch (s->state) { case S_DOWN: - p = add_str(p, "down"); + p = bb_stpcpy(p, "down"); break; case S_RUN: - p = add_str(p, "run"); + p = bb_stpcpy(p, "run"); break; case S_FINISH: - p = add_str(p, "finish"); + p = bb_stpcpy(p, "finish"); break; } - if (s->ctrl & C_PAUSE) p = add_str(p, ", paused"); - if (s->ctrl & C_TERM) p = add_str(p, ", got TERM"); + if (s->ctrl & C_PAUSE) + p = bb_stpcpy(p, ", paused"); + if (s->ctrl & C_TERM) + p = bb_stpcpy(p, ", got TERM"); if (s->state != S_DOWN) - switch (s->want) { + switch (s->sd_want) { case W_DOWN: - p = add_str(p, ", want down"); + p = bb_stpcpy(p, ", want down"); break; case W_EXIT: - p = add_str(p, ", want exit"); + p = bb_stpcpy(p, ", want exit"); break; } *p++ = '\n'; @@ -224,7 +228,7 @@ static void update_status(struct svdir *s) status.pid_le32 = SWAP_LE32(s->pid); if (s->ctrl & C_PAUSE) status.paused = 1; - if (s->want == W_UP) + if (s->sd_want == W_UP) status.want = 'u'; else status.want = 'd'; @@ -247,13 +251,13 @@ static void update_status(struct svdir *s) static unsigned custom(struct svdir *s, char c) { - int pid; + pid_t pid; int w; char a[10]; struct stat st; - char *prog[2]; - if (s->islog) return 0; + if (s->islog) + return 0; strcpy(a, "control/?"); a[8] = c; /* replace '?' */ if (stat(a, &st) == 0) { @@ -263,21 +267,19 @@ static unsigned custom(struct svdir *s, char c) warn_cannot("vfork for control/?"); return 0; } - if (!pid) { + if (pid == 0) { /* child */ if (haslog && dup2(logpipe.wr, 1) == -1) warn_cannot("setup stdout for control/?"); - prog[0] = a; - prog[1] = NULL; - execv(a, prog); + execl(a, a, (char *) NULL); fatal_cannot("run control/?"); } /* parent */ - while (safe_waitpid(pid, &w, 0) == -1) { + if (safe_waitpid(pid, &w, 0) == -1) { warn_cannot("wait for child control/?"); return 0; } - return !wait_exitcode(w); + return WEXITSTATUS(w) == 0; } } else { if (errno != ENOENT) @@ -293,12 +295,12 @@ static void stopservice(struct svdir *s) s->ctrl |= C_TERM; update_status(s); } - if (s->want == W_DOWN) { + if (s->sd_want == W_DOWN) { kill(s->pid, SIGCONT); custom(s, 'd'); return; } - if (s->want == W_EXIT) { + if (s->sd_want == W_EXIT) { kill(s->pid, SIGCONT); custom(s, 'x'); } @@ -307,15 +309,33 @@ static void stopservice(struct svdir *s) static void startservice(struct svdir *s) { int p; - char *run[2]; - - if (s->state == S_FINISH) - run[0] = (char*)"./finish"; - else { - run[0] = (char*)"./run"; + const char *arg[4]; + char exitcode[sizeof(int)*3 + 2]; + + if (s->state == S_FINISH) { +/* Two arguments are given to ./finish. The first one is ./run exit code, + * or -1 if ./run didnt exit normally. The second one is + * the least significant byte of the exit status as determined by waitpid; + * for instance it is 0 if ./run exited normally, and the signal number + * if ./run was terminated by a signal. If runsv cannot start ./run + * for some reason, the exit code is 111 and the status is 0. + */ + arg[0] = "./finish"; + arg[1] = "-1"; + if (WIFEXITED(s->wstat)) { + *utoa_to_buf(WEXITSTATUS(s->wstat), exitcode, sizeof(exitcode)) = '\0'; + arg[1] = exitcode; + } + //arg[2] = "0"; + //if (WIFSIGNALED(s->wstat)) { + arg[2] = utoa(WTERMSIG(s->wstat)); + //} + arg[3] = NULL; + } else { + arg[0] = "./run"; + arg[1] = NULL; custom(s, 'u'); } - run[1] = NULL; if (s->pid != 0) stopservice(s); /* should never happen */ @@ -336,14 +356,15 @@ static void startservice(struct svdir *s) xdup2(logpipe.wr, 1); } } - bb_signals(0 + /* Non-ignored signals revert to SIG_DFL on exec anyway */ + /*bb_signals(0 + (1 << SIGCHLD) + (1 << SIGTERM) - , SIG_DFL); + , SIG_DFL);*/ sig_unblock(SIGCHLD); sig_unblock(SIGTERM); - execvp(*run, run); - fatal2_cannot(s->islog ? "start log/" : "start ", *run); + execv(arg[0], (char**) arg); + fatal2_cannot(s->islog ? "start log/" : "start ", arg[0]); } /* parent */ if (s->state != S_FINISH) { @@ -362,13 +383,13 @@ static int ctrl(struct svdir *s, char c) switch (c) { case 'd': /* down */ - s->want = W_DOWN; + s->sd_want = W_DOWN; update_status(s); if (s->pid && s->state != S_FINISH) stopservice(s); break; case 'u': /* up */ - s->want = W_UP; + s->sd_want = W_UP; update_status(s); if (s->pid == 0) startservice(s); @@ -376,7 +397,7 @@ static int ctrl(struct svdir *s, char c) case 'x': /* exit */ if (s->islog) break; - s->want = W_EXIT; + s->sd_want = W_EXIT; update_status(s); /* FALLTHROUGH */ case 't': /* sig term */ @@ -397,12 +418,11 @@ static int ctrl(struct svdir *s, char c) case 'c': /* sig cont */ if (s->pid && !custom(s, c)) kill(s->pid, SIGCONT); - if (s->ctrl & C_PAUSE) - s->ctrl &= ~C_PAUSE; + s->ctrl &= ~C_PAUSE; update_status(s); break; case 'o': /* once */ - s->want = W_DOWN; + s->sd_want = W_DOWN; update_status(s); if (!s->pid) startservice(s); @@ -443,9 +463,7 @@ int runsv_main(int argc UNUSED_PARAM, char **argv) INIT_G(); - if (!argv[1] || argv[2]) - bb_show_usage(); - dir = argv[1]; + dir = single_argv(argv); xpiped_pair(selfpipe); close_on_exec_on(selfpipe.rd); @@ -454,19 +472,20 @@ int runsv_main(int argc UNUSED_PARAM, char **argv) ndelay_on(selfpipe.wr); sig_block(SIGCHLD); - bb_signals_recursive(1 << SIGCHLD, s_child); + bb_signals_recursive_norestart(1 << SIGCHLD, s_child); sig_block(SIGTERM); - bb_signals_recursive(1 << SIGTERM, s_term); + bb_signals_recursive_norestart(1 << SIGTERM, s_term); xchdir(dir); /* bss: svd[0].pid = 0; */ if (S_DOWN) svd[0].state = S_DOWN; /* otherwise already 0 (bss) */ if (C_NOOP) svd[0].ctrl = C_NOOP; - if (W_UP) svd[0].want = W_UP; + if (W_UP) svd[0].sd_want = W_UP; /* bss: svd[0].islog = 0; */ /* bss: svd[1].pid = 0; */ gettimeofday_ns(&svd[0].start); - if (stat("down", &s) != -1) svd[0].want = W_DOWN; + if (stat("down", &s) != -1) + svd[0].sd_want = W_DOWN; if (stat("log", &s) == -1) { if (errno != ENOENT) @@ -479,11 +498,11 @@ int runsv_main(int argc UNUSED_PARAM, char **argv) haslog = 1; svd[1].state = S_DOWN; svd[1].ctrl = C_NOOP; - svd[1].want = W_UP; + svd[1].sd_want = W_UP; svd[1].islog = 1; gettimeofday_ns(&svd[1].start); if (stat("log/down", &s) != -1) - svd[1].want = W_DOWN; + svd[1].sd_want = W_DOWN; xpiped_pair(logpipe); close_on_exec_on(logpipe.rd); close_on_exec_on(logpipe.wr); @@ -561,10 +580,10 @@ int runsv_main(int argc UNUSED_PARAM, char **argv) char ch; if (haslog) - if (!svd[1].pid && svd[1].want == W_UP) + if (!svd[1].pid && svd[1].sd_want == W_UP) startservice(&svd[1]); if (!svd[0].pid) - if (svd[0].want == W_UP || svd[0].state == S_FINISH) + if (svd[0].sd_want == W_UP || svd[0].state == S_FINISH) startservice(&svd[0]); x[0].fd = selfpipe.rd; @@ -584,7 +603,7 @@ int runsv_main(int argc UNUSED_PARAM, char **argv) continue; for (;;) { - int child; + pid_t child; int wstat; child = wait_any_nohang(&wstat); @@ -593,9 +612,10 @@ int runsv_main(int argc UNUSED_PARAM, char **argv) if ((child == -1) && (errno != EINTR)) break; if (child == svd[0].pid) { + svd[0].wstat = wstat; svd[0].pid = 0; pidchanged = 1; - svd[0].ctrl &=~ C_TERM; + svd[0].ctrl &= ~C_TERM; if (svd[0].state != S_FINISH) { fd = open_read("finish"); if (fd != -1) { @@ -614,6 +634,7 @@ int runsv_main(int argc UNUSED_PARAM, char **argv) } if (haslog) { if (child == svd[1].pid) { + svd[0].wstat = wstat; svd[1].pid = 0; pidchanged = 1; svd[1].state = S_DOWN; @@ -637,11 +658,11 @@ int runsv_main(int argc UNUSED_PARAM, char **argv) sigterm = 0; } - if (svd[0].want == W_EXIT && svd[0].state == S_DOWN) { + if (svd[0].sd_want == W_EXIT && svd[0].state == S_DOWN) { if (svd[1].pid == 0) _exit(EXIT_SUCCESS); - if (svd[1].want != W_EXIT) { - svd[1].want = W_EXIT; + if (svd[1].sd_want != W_EXIT) { + svd[1].sd_want = W_EXIT; /* stopservice(&svd[1]); */ update_status(&svd[1]); close(logpipe.wr);