hush: make "exit" in trap use pre-trap exitcode - fix for nested trap
authorDenys Vlasenko <vda.linux@googlemail.com>
Fri, 21 Feb 2020 01:55:53 +0000 (02:55 +0100)
committerDenys Vlasenko <vda.linux@googlemail.com>
Fri, 21 Feb 2020 01:55:53 +0000 (02:55 +0100)
function                                             old     new   delta
check_and_run_traps                                  276     278      +2

Signed-off-by: Denys Vlasenko <vda.linux@googlemail.com>
shell/ash_test/ash-misc/exitcode_trap6.right [new file with mode: 0644]
shell/ash_test/ash-misc/exitcode_trap6.tests [new file with mode: 0755]
shell/hush.c
shell/hush_test/hush-misc/exitcode_trap6.right [new file with mode: 0644]
shell/hush_test/hush-misc/exitcode_trap6.tests [new file with mode: 0755]

diff --git a/shell/ash_test/ash-misc/exitcode_trap6.right b/shell/ash_test/ash-misc/exitcode_trap6.right
new file mode 100644 (file)
index 0000000..b76c190
--- /dev/null
@@ -0,0 +1,2 @@
+INT
+42:42
diff --git a/shell/ash_test/ash-misc/exitcode_trap6.tests b/shell/ash_test/ash-misc/exitcode_trap6.tests
new file mode 100755 (executable)
index 0000000..15fb99d
--- /dev/null
@@ -0,0 +1,11 @@
+# "exit" in trap should not use last command's exitcode,
+# but exitcode on entering the trap.
+# Nested trap should not interfere with this.
+$THIS_SH -c '
+ trap "echo INT" int
+ trap "kill -int $$;exit" term
+ kill $$ &
+ (exit 42)
+ wait
+'
+echo 42:$?
index b881b001abe3956eb8502cf1f28414af06044a3b..357a354e2b8760492c5ba4c33372895f27ab7515 100644 (file)
@@ -2107,16 +2107,18 @@ static int check_and_run_traps(void)
                        if (G_traps[sig][0]) {
                                /* We have user-defined handler */
                                smalluint save_rcode;
+                               int save_pre;
                                char *argv[3];
                                /* argv[0] is unused */
                                argv[1] = xstrdup(G_traps[sig]);
                                /* why strdup? trap can modify itself: trap 'trap "echo oops" INT' INT */
                                argv[2] = NULL;
+                               save_pre = G.pre_trap_exitcode;
                                G.pre_trap_exitcode = save_rcode = G.last_exitcode;
                                builtin_eval(argv);
                                free(argv[1]);
+                               G.pre_trap_exitcode = save_pre;
                                G.last_exitcode = save_rcode;
-                               G.pre_trap_exitcode = -1;
 # if ENABLE_HUSH_FUNCTIONS
                                if (G.return_exitcode >= 0) {
                                        debug_printf_exec("trap exitcode:%d\n", G.return_exitcode);
diff --git a/shell/hush_test/hush-misc/exitcode_trap6.right b/shell/hush_test/hush-misc/exitcode_trap6.right
new file mode 100644 (file)
index 0000000..b76c190
--- /dev/null
@@ -0,0 +1,2 @@
+INT
+42:42
diff --git a/shell/hush_test/hush-misc/exitcode_trap6.tests b/shell/hush_test/hush-misc/exitcode_trap6.tests
new file mode 100755 (executable)
index 0000000..15fb99d
--- /dev/null
@@ -0,0 +1,11 @@
+# "exit" in trap should not use last command's exitcode,
+# but exitcode on entering the trap.
+# Nested trap should not interfere with this.
+$THIS_SH -c '
+ trap "echo INT" int
+ trap "kill -int $$;exit" term
+ kill $$ &
+ (exit 42)
+ wait
+'
+echo 42:$?