1 | // -*- C++ -*- |
2 | //===----------------------------------------------------------------------===// |
3 | // |
4 | // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. |
5 | // See https://llvm.org/LICENSE.txt for license information. |
6 | // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception |
7 | // |
8 | //===----------------------------------------------------------------------===// |
9 | |
10 | // Ensure that the unwinder can cope with the signal handler. |
11 | // REQUIRES: target={{(aarch64|riscv64|s390x|x86_64)-.+linux.*}} |
12 | |
13 | // TODO: Figure out why this fails with Memory Sanitizer. |
14 | // XFAIL: msan |
15 | |
16 | #undef NDEBUG |
17 | #include <assert.h> |
18 | #include <dlfcn.h> |
19 | #include <signal.h> |
20 | #include <stdio.h> |
21 | #include <stdlib.h> |
22 | #include <string.h> |
23 | #include <sys/types.h> |
24 | #include <unistd.h> |
25 | #include <unwind.h> |
26 | |
27 | _Unwind_Reason_Code frame_handler(struct _Unwind_Context* ctx, void* arg) { |
28 | (void)arg; |
29 | Dl_info info = { .dli_fname: 0, .dli_fbase: 0, .dli_sname: 0, .dli_saddr: 0 }; |
30 | |
31 | // Unwind util the main is reached, above frames depend on the platform and |
32 | // architecture. |
33 | if (dladdr(address: reinterpret_cast<void *>(_Unwind_GetIP(ctx)), info: &info) && |
34 | info.dli_sname && !strcmp(s1: "main" , s2: info.dli_sname)) { |
35 | _Exit(status: 0); |
36 | } |
37 | return _URC_NO_REASON; |
38 | } |
39 | |
40 | void signal_handler(int signum) { |
41 | (void)signum; |
42 | _Unwind_Backtrace(frame_handler, NULL); |
43 | _Exit(status: -1); |
44 | } |
45 | |
46 | int main(int, char**) { |
47 | signal(SIGUSR1, handler: signal_handler); |
48 | kill(pid: getpid(), SIGUSR1); |
49 | return -2; |
50 | } |
51 | |