diff options
author | Denys Vlasenko <vda.linux@googlemail.com> | 2011-02-21 03:22:20 +0100 |
---|---|---|
committer | Denys Vlasenko <vda.linux@googlemail.com> | 2011-02-21 03:22:20 +0100 |
commit | b12553faa8991e11c11f70a81f1d9d44078c7645 (patch) | |
tree | 4154d5308a9bad6bb46e2014cb6cc997dc9a92cd /shell/ash.c | |
parent | 55988aed472d9cd362f9a50f4999b5e47ca33abe (diff) | |
download | busybox-w32-b12553faa8991e11c11f70a81f1d9d44078c7645.tar.gz busybox-w32-b12553faa8991e11c11f70a81f1d9d44078c7645.tar.bz2 busybox-w32-b12553faa8991e11c11f70a81f1d9d44078c7645.zip |
ash: fix ash-signals/signal8 testcase failure
function old new delta
killcmd 109 224 +115
kill_main 882 910 +28
changepath 194 195 +1
------------------------------------------------------------------------------
(add/remove: 0/0 grow/shrink: 3/0 up/down: 144/0) Total: 144 bytes
Signed-off-by: Denys Vlasenko <vda.linux@googlemail.com>
Diffstat (limited to 'shell/ash.c')
-rw-r--r-- | shell/ash.c | 58 |
1 files changed, 46 insertions, 12 deletions
diff --git a/shell/ash.c b/shell/ash.c index cccd6dd79..98d2c7c29 100644 --- a/shell/ash.c +++ b/shell/ash.c | |||
@@ -3783,18 +3783,51 @@ setjobctl(int on) | |||
3783 | static int FAST_FUNC | 3783 | static int FAST_FUNC |
3784 | killcmd(int argc, char **argv) | 3784 | killcmd(int argc, char **argv) |
3785 | { | 3785 | { |
3786 | int i = 1; | ||
3787 | if (argv[1] && strcmp(argv[1], "-l") != 0) { | 3786 | if (argv[1] && strcmp(argv[1], "-l") != 0) { |
3787 | int i = 1; | ||
3788 | do { | 3788 | do { |
3789 | if (argv[i][0] == '%') { | 3789 | if (argv[i][0] == '%') { |
3790 | struct job *jp = getjob(argv[i], 0); | 3790 | /* |
3791 | unsigned pid = jp->ps[0].ps_pid; | 3791 | * "kill %N" - job kill |
3792 | /* Enough space for ' -NNN<nul>' */ | 3792 | * Converting to pgrp / pid kill |
3793 | argv[i] = alloca(sizeof(int)*3 + 3); | 3793 | */ |
3794 | /* kill_main has matching code to expect | 3794 | struct job *jp; |
3795 | * leading space. Needed to not confuse | 3795 | char *dst; |
3796 | * negative pids with "kill -SIGNAL_NO" syntax */ | 3796 | int j, n; |
3797 | sprintf(argv[i], " -%u", pid); | 3797 | |
3798 | jp = getjob(argv[i], 0); | ||
3799 | /* | ||
3800 | * In jobs started under job control, we signal | ||
3801 | * entire process group by kill -PGRP_ID. | ||
3802 | * This happens, f.e., in interactive shell. | ||
3803 | * | ||
3804 | * Otherwise, we signal each child via | ||
3805 | * kill PID1 PID2 PID3. | ||
3806 | * Testcases: | ||
3807 | * sh -c 'sleep 1|sleep 1 & kill %1' | ||
3808 | * sh -c 'true|sleep 2 & sleep 1; kill %1' | ||
3809 | * sh -c 'true|sleep 1 & sleep 2; kill %1' | ||
3810 | */ | ||
3811 | n = jp->nprocs; /* can't be 0 (I hope) */ | ||
3812 | if (jp->jobctl) | ||
3813 | n = 1; | ||
3814 | dst = alloca(n * sizeof(int)*4); | ||
3815 | argv[i] = dst; | ||
3816 | for (j = 0; j < n; j++) { | ||
3817 | struct procstat *ps = &jp->ps[j]; | ||
3818 | /* Skip non-running and not-stopped members | ||
3819 | * (i.e. dead members) of the job | ||
3820 | */ | ||
3821 | if (ps->ps_status != -1 && !WIFSTOPPED(ps->ps_status)) | ||
3822 | continue; | ||
3823 | /* | ||
3824 | * kill_main has matching code to expect | ||
3825 | * leading space. Needed to not confuse | ||
3826 | * negative pids with "kill -SIGNAL_NO" syntax | ||
3827 | */ | ||
3828 | dst += sprintf(dst, jp->jobctl ? " -%u" : " %u", (int)ps->ps_pid); | ||
3829 | } | ||
3830 | *dst = '\0'; | ||
3798 | } | 3831 | } |
3799 | } while (argv[++i]); | 3832 | } while (argv[++i]); |
3800 | } | 3833 | } |
@@ -4227,8 +4260,9 @@ waitcmd(int argc UNUSED_PARAM, char **argv) | |||
4227 | break; | 4260 | break; |
4228 | job = job->prev_job; | 4261 | job = job->prev_job; |
4229 | } | 4262 | } |
4230 | } else | 4263 | } else { |
4231 | job = getjob(*argv, 0); | 4264 | job = getjob(*argv, 0); |
4265 | } | ||
4232 | /* loop until process terminated or stopped */ | 4266 | /* loop until process terminated or stopped */ |
4233 | while (job->state == JOBRUNNING) | 4267 | while (job->state == JOBRUNNING) |
4234 | blocking_wait_with_raise_on_sig(); | 4268 | blocking_wait_with_raise_on_sig(); |
@@ -4724,7 +4758,7 @@ forkchild(struct job *jp, union node *n, int mode) | |||
4724 | #if JOBS | 4758 | #if JOBS |
4725 | /* do job control only in root shell */ | 4759 | /* do job control only in root shell */ |
4726 | doing_jobctl = 0; | 4760 | doing_jobctl = 0; |
4727 | if (mode != FORK_NOJOB && jp->jobctl && !oldlvl) { | 4761 | if (mode != FORK_NOJOB && jp->jobctl && oldlvl == 0) { |
4728 | pid_t pgrp; | 4762 | pid_t pgrp; |
4729 | 4763 | ||
4730 | if (jp->nprocs == 0) | 4764 | if (jp->nprocs == 0) |
@@ -4750,7 +4784,7 @@ forkchild(struct job *jp, union node *n, int mode) | |||
4750 | ash_msg_and_raise_error("can't open '%s'", bb_dev_null); | 4784 | ash_msg_and_raise_error("can't open '%s'", bb_dev_null); |
4751 | } | 4785 | } |
4752 | } | 4786 | } |
4753 | if (!oldlvl) { | 4787 | if (oldlvl == 0) { |
4754 | if (iflag) { /* why if iflag only? */ | 4788 | if (iflag) { /* why if iflag only? */ |
4755 | setsignal(SIGINT); | 4789 | setsignal(SIGINT); |
4756 | setsignal(SIGTERM); | 4790 | setsignal(SIGTERM); |