summaryrefslogtreecommitdiffstats
path: root/signal.c
diff options
context:
space:
mode:
authorKaz Kylheku <kaz@kylheku.com>2021-06-24 22:01:38 -0700
committerKaz Kylheku <kaz@kylheku.com>2021-06-24 22:01:38 -0700
commit817cd5991d6304a2368824acbfc0b0e95c2aa7a7 (patch)
tree4aa5e7b112cdf41d2a713e42562f9699c8839dd3 /signal.c
parent1c1685861e0d304f60f0669c6dbd5005349262e4 (diff)
downloadtxr-817cd5991d6304a2368824acbfc0b0e95c2aa7a7.tar.gz
txr-817cd5991d6304a2368824acbfc0b0e95c2aa7a7.tar.bz2
txr-817cd5991d6304a2368824acbfc0b0e95c2aa7a7.zip
signals: disable stack overflow in handler.
* signal.c (sig_handler): For a is_cpu_exception signal, we temporarily disable the stack limit. It might be executing on the sigaltstack buffer, which is almost certainly below the stack limit. * tests/012/stack.tl: New test case. We raise a SIGSEGV and check that in the handler, the stack limit is disabled, and that we can executed code.
Diffstat (limited to 'signal.c')
-rw-r--r--signal.c4
1 files changed, 4 insertions, 0 deletions
diff --git a/signal.c b/signal.c
index 53e4e499..ad5c9849 100644
--- a/signal.c
+++ b/signal.c
@@ -81,11 +81,14 @@ static void sig_handler(int sig)
int exc = is_cpu_exception(sig);
int ic = interrupt_count++;
int in_interrupt = ic > 0;
+ val *stack_lim;
if (exc) {
gc = gc_state(0);
+ stack_lim = gc_stack_limit;
as = async_sig_enabled;
async_sig_enabled = 1;
+ gc_stack_limit = 0;
}
sig_reload_cache();
@@ -109,6 +112,7 @@ static void sig_handler(int sig)
}
if (exc) {
+ gc_stack_limit = stack_lim;
async_sig_enabled = as;
gc_state(gc);
}