X-Git-Url: http://nsz.repo.hu/git/?a=blobdiff_plain;f=src%2Fexit%2Fabort.c;h=e1980f10a5b6488a8a82a98a6a987e6db7997e3b;hb=a63c0104e496f7ba78b64be3cd299b41e8cd427f;hp=9a1c3d40c1304d24da88e390c9a18a74b12086e6;hpb=0b44a0315b47dd8eced9f3b7f31580cf14bbfc01;p=musl diff --git a/src/exit/abort.c b/src/exit/abort.c index 9a1c3d40..e1980f10 100644 --- a/src/exit/abort.c +++ b/src/exit/abort.c @@ -1,8 +1,32 @@ #include #include +#include "syscall.h" +#include "pthread_impl.h" +#include "atomic.h" +#include "lock.h" +#include "ksigaction.h" -void abort(void) +hidden volatile int __abort_lock[1]; + +_Noreturn void abort(void) { raise(SIGABRT); - for (;;); + + /* If there was a SIGABRT handler installed and it returned, or if + * SIGABRT was blocked or ignored, take an AS-safe lock to prevent + * sigaction from installing a new SIGABRT handler, uninstall any + * handler that may be present, and re-raise the signal to generate + * the default action of abnormal termination. */ + __block_all_sigs(0); + LOCK(__abort_lock); + __syscall(SYS_rt_sigaction, SIGABRT, + &(struct k_sigaction){.handler = SIG_DFL}, 0, _NSIG/8); + __syscall(SYS_tkill, __pthread_self()->tid, SIGABRT); + __syscall(SYS_rt_sigprocmask, SIG_UNBLOCK, + &(long[_NSIG/(8*sizeof(long))]){1UL<<(SIGABRT-1)}, 0, _NSIG/8); + + /* Beyond this point should be unreachable. */ + a_crash(); + raise(SIGKILL); + _Exit(127); }