aboutsummaryrefslogtreecommitdiff
path: root/shell/ash.c
diff options
context:
space:
mode:
authorDenys Vlasenko <vda.linux@googlemail.com>2011-02-21 03:22:20 +0100
committerDenys Vlasenko <vda.linux@googlemail.com>2011-02-21 03:22:20 +0100
commitb12553faa8991e11c11f70a81f1d9d44078c7645 (patch)
tree4154d5308a9bad6bb46e2014cb6cc997dc9a92cd /shell/ash.c
parent55988aed472d9cd362f9a50f4999b5e47ca33abe (diff)
downloadbusybox-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.c58
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)
3783static int FAST_FUNC 3783static int FAST_FUNC
3784killcmd(int argc, char **argv) 3784killcmd(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);