xref: /aosp_15_r20/external/abseil-cpp/absl/debugging/failure_signal_handler.cc (revision 9356374a3709195abf420251b3e825997ff56c0f)
1*9356374aSAndroid Build Coastguard Worker //
2*9356374aSAndroid Build Coastguard Worker // Copyright 2018 The Abseil Authors.
3*9356374aSAndroid Build Coastguard Worker //
4*9356374aSAndroid Build Coastguard Worker // Licensed under the Apache License, Version 2.0 (the "License");
5*9356374aSAndroid Build Coastguard Worker // you may not use this file except in compliance with the License.
6*9356374aSAndroid Build Coastguard Worker // You may obtain a copy of the License at
7*9356374aSAndroid Build Coastguard Worker //
8*9356374aSAndroid Build Coastguard Worker //      https://www.apache.org/licenses/LICENSE-2.0
9*9356374aSAndroid Build Coastguard Worker //
10*9356374aSAndroid Build Coastguard Worker // Unless required by applicable law or agreed to in writing, software
11*9356374aSAndroid Build Coastguard Worker // distributed under the License is distributed on an "AS IS" BASIS,
12*9356374aSAndroid Build Coastguard Worker // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*9356374aSAndroid Build Coastguard Worker // See the License for the specific language governing permissions and
14*9356374aSAndroid Build Coastguard Worker // limitations under the License.
15*9356374aSAndroid Build Coastguard Worker //
16*9356374aSAndroid Build Coastguard Worker 
17*9356374aSAndroid Build Coastguard Worker #include "absl/debugging/failure_signal_handler.h"
18*9356374aSAndroid Build Coastguard Worker 
19*9356374aSAndroid Build Coastguard Worker #include "absl/base/config.h"
20*9356374aSAndroid Build Coastguard Worker 
21*9356374aSAndroid Build Coastguard Worker #ifdef _WIN32
22*9356374aSAndroid Build Coastguard Worker #include <windows.h>
23*9356374aSAndroid Build Coastguard Worker #else
24*9356374aSAndroid Build Coastguard Worker #include <sched.h>
25*9356374aSAndroid Build Coastguard Worker #include <unistd.h>
26*9356374aSAndroid Build Coastguard Worker #endif
27*9356374aSAndroid Build Coastguard Worker 
28*9356374aSAndroid Build Coastguard Worker #ifdef __APPLE__
29*9356374aSAndroid Build Coastguard Worker #include <TargetConditionals.h>
30*9356374aSAndroid Build Coastguard Worker #endif
31*9356374aSAndroid Build Coastguard Worker 
32*9356374aSAndroid Build Coastguard Worker #ifdef ABSL_HAVE_MMAP
33*9356374aSAndroid Build Coastguard Worker #include <sys/mman.h>
34*9356374aSAndroid Build Coastguard Worker #if defined(MAP_ANON) && !defined(MAP_ANONYMOUS)
35*9356374aSAndroid Build Coastguard Worker #define MAP_ANONYMOUS MAP_ANON
36*9356374aSAndroid Build Coastguard Worker #endif
37*9356374aSAndroid Build Coastguard Worker #endif
38*9356374aSAndroid Build Coastguard Worker 
39*9356374aSAndroid Build Coastguard Worker #ifdef __linux__
40*9356374aSAndroid Build Coastguard Worker #include <sys/prctl.h>
41*9356374aSAndroid Build Coastguard Worker #endif
42*9356374aSAndroid Build Coastguard Worker 
43*9356374aSAndroid Build Coastguard Worker #include <algorithm>
44*9356374aSAndroid Build Coastguard Worker #include <atomic>
45*9356374aSAndroid Build Coastguard Worker #include <cerrno>
46*9356374aSAndroid Build Coastguard Worker #include <csignal>
47*9356374aSAndroid Build Coastguard Worker #include <cstdio>
48*9356374aSAndroid Build Coastguard Worker #include <cstring>
49*9356374aSAndroid Build Coastguard Worker #include <ctime>
50*9356374aSAndroid Build Coastguard Worker 
51*9356374aSAndroid Build Coastguard Worker #include "absl/base/attributes.h"
52*9356374aSAndroid Build Coastguard Worker #include "absl/base/internal/raw_logging.h"
53*9356374aSAndroid Build Coastguard Worker #include "absl/base/internal/sysinfo.h"
54*9356374aSAndroid Build Coastguard Worker #include "absl/debugging/internal/examine_stack.h"
55*9356374aSAndroid Build Coastguard Worker #include "absl/debugging/stacktrace.h"
56*9356374aSAndroid Build Coastguard Worker 
57*9356374aSAndroid Build Coastguard Worker #if !defined(_WIN32) && !defined(__wasi__)
58*9356374aSAndroid Build Coastguard Worker #define ABSL_HAVE_SIGACTION
59*9356374aSAndroid Build Coastguard Worker // Apple WatchOS and TVOS don't allow sigaltstack
60*9356374aSAndroid Build Coastguard Worker // Apple macOS has sigaltstack, but using it makes backtrace() unusable.
61*9356374aSAndroid Build Coastguard Worker #if !(defined(TARGET_OS_OSX) && TARGET_OS_OSX) &&     \
62*9356374aSAndroid Build Coastguard Worker     !(defined(TARGET_OS_WATCH) && TARGET_OS_WATCH) && \
63*9356374aSAndroid Build Coastguard Worker     !(defined(TARGET_OS_TV) && TARGET_OS_TV) && !defined(__QNX__)
64*9356374aSAndroid Build Coastguard Worker #define ABSL_HAVE_SIGALTSTACK
65*9356374aSAndroid Build Coastguard Worker #endif
66*9356374aSAndroid Build Coastguard Worker #endif
67*9356374aSAndroid Build Coastguard Worker 
68*9356374aSAndroid Build Coastguard Worker namespace absl {
69*9356374aSAndroid Build Coastguard Worker ABSL_NAMESPACE_BEGIN
70*9356374aSAndroid Build Coastguard Worker 
71*9356374aSAndroid Build Coastguard Worker ABSL_CONST_INIT static FailureSignalHandlerOptions fsh_options;
72*9356374aSAndroid Build Coastguard Worker 
73*9356374aSAndroid Build Coastguard Worker // Resets the signal handler for signo to the default action for that
74*9356374aSAndroid Build Coastguard Worker // signal, then raises the signal.
RaiseToDefaultHandler(int signo)75*9356374aSAndroid Build Coastguard Worker static void RaiseToDefaultHandler(int signo) {
76*9356374aSAndroid Build Coastguard Worker   signal(signo, SIG_DFL);
77*9356374aSAndroid Build Coastguard Worker   raise(signo);
78*9356374aSAndroid Build Coastguard Worker }
79*9356374aSAndroid Build Coastguard Worker 
80*9356374aSAndroid Build Coastguard Worker struct FailureSignalData {
81*9356374aSAndroid Build Coastguard Worker   const int signo;
82*9356374aSAndroid Build Coastguard Worker   const char* const as_string;
83*9356374aSAndroid Build Coastguard Worker #ifdef ABSL_HAVE_SIGACTION
84*9356374aSAndroid Build Coastguard Worker   struct sigaction previous_action;
85*9356374aSAndroid Build Coastguard Worker   // StructSigaction is used to silence -Wmissing-field-initializers.
86*9356374aSAndroid Build Coastguard Worker   using StructSigaction = struct sigaction;
87*9356374aSAndroid Build Coastguard Worker #define FSD_PREVIOUS_INIT FailureSignalData::StructSigaction()
88*9356374aSAndroid Build Coastguard Worker #else
89*9356374aSAndroid Build Coastguard Worker   void (*previous_handler)(int);
90*9356374aSAndroid Build Coastguard Worker #define FSD_PREVIOUS_INIT SIG_DFL
91*9356374aSAndroid Build Coastguard Worker #endif
92*9356374aSAndroid Build Coastguard Worker };
93*9356374aSAndroid Build Coastguard Worker 
94*9356374aSAndroid Build Coastguard Worker ABSL_CONST_INIT static FailureSignalData failure_signal_data[] = {
95*9356374aSAndroid Build Coastguard Worker     {SIGSEGV, "SIGSEGV", FSD_PREVIOUS_INIT},
96*9356374aSAndroid Build Coastguard Worker     {SIGILL, "SIGILL", FSD_PREVIOUS_INIT},
97*9356374aSAndroid Build Coastguard Worker     {SIGFPE, "SIGFPE", FSD_PREVIOUS_INIT},
98*9356374aSAndroid Build Coastguard Worker     {SIGABRT, "SIGABRT", FSD_PREVIOUS_INIT},
99*9356374aSAndroid Build Coastguard Worker     {SIGTERM, "SIGTERM", FSD_PREVIOUS_INIT},
100*9356374aSAndroid Build Coastguard Worker #ifndef _WIN32
101*9356374aSAndroid Build Coastguard Worker     {SIGBUS, "SIGBUS", FSD_PREVIOUS_INIT},
102*9356374aSAndroid Build Coastguard Worker     {SIGTRAP, "SIGTRAP", FSD_PREVIOUS_INIT},
103*9356374aSAndroid Build Coastguard Worker #endif
104*9356374aSAndroid Build Coastguard Worker };
105*9356374aSAndroid Build Coastguard Worker 
106*9356374aSAndroid Build Coastguard Worker #undef FSD_PREVIOUS_INIT
107*9356374aSAndroid Build Coastguard Worker 
RaiseToPreviousHandler(int signo)108*9356374aSAndroid Build Coastguard Worker static void RaiseToPreviousHandler(int signo) {
109*9356374aSAndroid Build Coastguard Worker   // Search for the previous handler.
110*9356374aSAndroid Build Coastguard Worker   for (const auto& it : failure_signal_data) {
111*9356374aSAndroid Build Coastguard Worker     if (it.signo == signo) {
112*9356374aSAndroid Build Coastguard Worker #ifdef ABSL_HAVE_SIGACTION
113*9356374aSAndroid Build Coastguard Worker       sigaction(signo, &it.previous_action, nullptr);
114*9356374aSAndroid Build Coastguard Worker #else
115*9356374aSAndroid Build Coastguard Worker       signal(signo, it.previous_handler);
116*9356374aSAndroid Build Coastguard Worker #endif
117*9356374aSAndroid Build Coastguard Worker       raise(signo);
118*9356374aSAndroid Build Coastguard Worker       return;
119*9356374aSAndroid Build Coastguard Worker     }
120*9356374aSAndroid Build Coastguard Worker   }
121*9356374aSAndroid Build Coastguard Worker 
122*9356374aSAndroid Build Coastguard Worker   // Not found, use the default handler.
123*9356374aSAndroid Build Coastguard Worker   RaiseToDefaultHandler(signo);
124*9356374aSAndroid Build Coastguard Worker }
125*9356374aSAndroid Build Coastguard Worker 
126*9356374aSAndroid Build Coastguard Worker namespace debugging_internal {
127*9356374aSAndroid Build Coastguard Worker 
FailureSignalToString(int signo)128*9356374aSAndroid Build Coastguard Worker const char* FailureSignalToString(int signo) {
129*9356374aSAndroid Build Coastguard Worker   for (const auto& it : failure_signal_data) {
130*9356374aSAndroid Build Coastguard Worker     if (it.signo == signo) {
131*9356374aSAndroid Build Coastguard Worker       return it.as_string;
132*9356374aSAndroid Build Coastguard Worker     }
133*9356374aSAndroid Build Coastguard Worker   }
134*9356374aSAndroid Build Coastguard Worker   return "";
135*9356374aSAndroid Build Coastguard Worker }
136*9356374aSAndroid Build Coastguard Worker 
137*9356374aSAndroid Build Coastguard Worker }  // namespace debugging_internal
138*9356374aSAndroid Build Coastguard Worker 
139*9356374aSAndroid Build Coastguard Worker #ifdef ABSL_HAVE_SIGALTSTACK
140*9356374aSAndroid Build Coastguard Worker 
SetupAlternateStackOnce()141*9356374aSAndroid Build Coastguard Worker static bool SetupAlternateStackOnce() {
142*9356374aSAndroid Build Coastguard Worker #if defined(__wasm__) || defined(__asjms__)
143*9356374aSAndroid Build Coastguard Worker   const size_t page_mask = getpagesize() - 1;
144*9356374aSAndroid Build Coastguard Worker #else
145*9356374aSAndroid Build Coastguard Worker   const size_t page_mask = static_cast<size_t>(sysconf(_SC_PAGESIZE)) - 1;
146*9356374aSAndroid Build Coastguard Worker #endif
147*9356374aSAndroid Build Coastguard Worker   size_t stack_size =
148*9356374aSAndroid Build Coastguard Worker       (std::max(static_cast<size_t>(SIGSTKSZ), size_t{65536}) + page_mask) &
149*9356374aSAndroid Build Coastguard Worker       ~page_mask;
150*9356374aSAndroid Build Coastguard Worker #if defined(ABSL_HAVE_ADDRESS_SANITIZER) || \
151*9356374aSAndroid Build Coastguard Worker     defined(ABSL_HAVE_MEMORY_SANITIZER) || defined(ABSL_HAVE_THREAD_SANITIZER)
152*9356374aSAndroid Build Coastguard Worker   // Account for sanitizer instrumentation requiring additional stack space.
153*9356374aSAndroid Build Coastguard Worker   stack_size *= 5;
154*9356374aSAndroid Build Coastguard Worker #endif
155*9356374aSAndroid Build Coastguard Worker 
156*9356374aSAndroid Build Coastguard Worker   stack_t sigstk;
157*9356374aSAndroid Build Coastguard Worker   memset(&sigstk, 0, sizeof(sigstk));
158*9356374aSAndroid Build Coastguard Worker   sigstk.ss_size = stack_size;
159*9356374aSAndroid Build Coastguard Worker 
160*9356374aSAndroid Build Coastguard Worker #ifdef ABSL_HAVE_MMAP
161*9356374aSAndroid Build Coastguard Worker #ifndef MAP_STACK
162*9356374aSAndroid Build Coastguard Worker #define MAP_STACK 0
163*9356374aSAndroid Build Coastguard Worker #endif
164*9356374aSAndroid Build Coastguard Worker   sigstk.ss_sp = mmap(nullptr, sigstk.ss_size, PROT_READ | PROT_WRITE,
165*9356374aSAndroid Build Coastguard Worker                       MAP_PRIVATE | MAP_ANONYMOUS | MAP_STACK, -1, 0);
166*9356374aSAndroid Build Coastguard Worker   if (sigstk.ss_sp == MAP_FAILED) {
167*9356374aSAndroid Build Coastguard Worker     ABSL_RAW_LOG(FATAL, "mmap() for alternate signal stack failed");
168*9356374aSAndroid Build Coastguard Worker   }
169*9356374aSAndroid Build Coastguard Worker #else
170*9356374aSAndroid Build Coastguard Worker   sigstk.ss_sp = malloc(sigstk.ss_size);
171*9356374aSAndroid Build Coastguard Worker   if (sigstk.ss_sp == nullptr) {
172*9356374aSAndroid Build Coastguard Worker     ABSL_RAW_LOG(FATAL, "malloc() for alternate signal stack failed");
173*9356374aSAndroid Build Coastguard Worker   }
174*9356374aSAndroid Build Coastguard Worker #endif
175*9356374aSAndroid Build Coastguard Worker 
176*9356374aSAndroid Build Coastguard Worker   if (sigaltstack(&sigstk, nullptr) != 0) {
177*9356374aSAndroid Build Coastguard Worker     ABSL_RAW_LOG(FATAL, "sigaltstack() failed with errno=%d", errno);
178*9356374aSAndroid Build Coastguard Worker   }
179*9356374aSAndroid Build Coastguard Worker 
180*9356374aSAndroid Build Coastguard Worker #ifdef __linux__
181*9356374aSAndroid Build Coastguard Worker #if defined(PR_SET_VMA) && defined(PR_SET_VMA_ANON_NAME)
182*9356374aSAndroid Build Coastguard Worker   // Make a best-effort attempt to name the allocated region in
183*9356374aSAndroid Build Coastguard Worker   // /proc/$PID/smaps.
184*9356374aSAndroid Build Coastguard Worker   //
185*9356374aSAndroid Build Coastguard Worker   // The call to prctl() may fail if the kernel was not configured with the
186*9356374aSAndroid Build Coastguard Worker   // CONFIG_ANON_VMA_NAME kernel option.  This is OK since the call is
187*9356374aSAndroid Build Coastguard Worker   // primarily a debugging aid.
188*9356374aSAndroid Build Coastguard Worker   prctl(PR_SET_VMA, PR_SET_VMA_ANON_NAME, sigstk.ss_sp, sigstk.ss_size,
189*9356374aSAndroid Build Coastguard Worker         "absl-signalstack");
190*9356374aSAndroid Build Coastguard Worker #endif
191*9356374aSAndroid Build Coastguard Worker #endif  // __linux__
192*9356374aSAndroid Build Coastguard Worker 
193*9356374aSAndroid Build Coastguard Worker   return true;
194*9356374aSAndroid Build Coastguard Worker }
195*9356374aSAndroid Build Coastguard Worker 
196*9356374aSAndroid Build Coastguard Worker #endif
197*9356374aSAndroid Build Coastguard Worker 
198*9356374aSAndroid Build Coastguard Worker #ifdef ABSL_HAVE_SIGACTION
199*9356374aSAndroid Build Coastguard Worker 
200*9356374aSAndroid Build Coastguard Worker // Sets up an alternate stack for signal handlers once.
201*9356374aSAndroid Build Coastguard Worker // Returns the appropriate flag for sig_action.sa_flags
202*9356374aSAndroid Build Coastguard Worker // if the system supports using an alternate stack.
MaybeSetupAlternateStack()203*9356374aSAndroid Build Coastguard Worker static int MaybeSetupAlternateStack() {
204*9356374aSAndroid Build Coastguard Worker #ifdef ABSL_HAVE_SIGALTSTACK
205*9356374aSAndroid Build Coastguard Worker   ABSL_ATTRIBUTE_UNUSED static const bool kOnce = SetupAlternateStackOnce();
206*9356374aSAndroid Build Coastguard Worker   return SA_ONSTACK;
207*9356374aSAndroid Build Coastguard Worker #else
208*9356374aSAndroid Build Coastguard Worker   return 0;
209*9356374aSAndroid Build Coastguard Worker #endif
210*9356374aSAndroid Build Coastguard Worker }
211*9356374aSAndroid Build Coastguard Worker 
InstallOneFailureHandler(FailureSignalData * data,void (* handler)(int,siginfo_t *,void *))212*9356374aSAndroid Build Coastguard Worker static void InstallOneFailureHandler(FailureSignalData* data,
213*9356374aSAndroid Build Coastguard Worker                                      void (*handler)(int, siginfo_t*, void*)) {
214*9356374aSAndroid Build Coastguard Worker   struct sigaction act;
215*9356374aSAndroid Build Coastguard Worker   memset(&act, 0, sizeof(act));
216*9356374aSAndroid Build Coastguard Worker   sigemptyset(&act.sa_mask);
217*9356374aSAndroid Build Coastguard Worker   act.sa_flags |= SA_SIGINFO;
218*9356374aSAndroid Build Coastguard Worker   // SA_NODEFER is required to handle SIGABRT from
219*9356374aSAndroid Build Coastguard Worker   // ImmediateAbortSignalHandler().
220*9356374aSAndroid Build Coastguard Worker   act.sa_flags |= SA_NODEFER;
221*9356374aSAndroid Build Coastguard Worker   if (fsh_options.use_alternate_stack) {
222*9356374aSAndroid Build Coastguard Worker     act.sa_flags |= MaybeSetupAlternateStack();
223*9356374aSAndroid Build Coastguard Worker   }
224*9356374aSAndroid Build Coastguard Worker   act.sa_sigaction = handler;
225*9356374aSAndroid Build Coastguard Worker   ABSL_RAW_CHECK(sigaction(data->signo, &act, &data->previous_action) == 0,
226*9356374aSAndroid Build Coastguard Worker                  "sigaction() failed");
227*9356374aSAndroid Build Coastguard Worker }
228*9356374aSAndroid Build Coastguard Worker 
229*9356374aSAndroid Build Coastguard Worker #else
230*9356374aSAndroid Build Coastguard Worker 
InstallOneFailureHandler(FailureSignalData * data,void (* handler)(int))231*9356374aSAndroid Build Coastguard Worker static void InstallOneFailureHandler(FailureSignalData* data,
232*9356374aSAndroid Build Coastguard Worker                                      void (*handler)(int)) {
233*9356374aSAndroid Build Coastguard Worker   data->previous_handler = signal(data->signo, handler);
234*9356374aSAndroid Build Coastguard Worker   ABSL_RAW_CHECK(data->previous_handler != SIG_ERR, "signal() failed");
235*9356374aSAndroid Build Coastguard Worker }
236*9356374aSAndroid Build Coastguard Worker 
237*9356374aSAndroid Build Coastguard Worker #endif
238*9356374aSAndroid Build Coastguard Worker 
WriteSignalMessage(int signo,int cpu,void (* writerfn)(const char *))239*9356374aSAndroid Build Coastguard Worker static void WriteSignalMessage(int signo, int cpu,
240*9356374aSAndroid Build Coastguard Worker                                void (*writerfn)(const char*)) {
241*9356374aSAndroid Build Coastguard Worker   char buf[96];
242*9356374aSAndroid Build Coastguard Worker   char on_cpu[32] = {0};
243*9356374aSAndroid Build Coastguard Worker   if (cpu != -1) {
244*9356374aSAndroid Build Coastguard Worker     snprintf(on_cpu, sizeof(on_cpu), " on cpu %d", cpu);
245*9356374aSAndroid Build Coastguard Worker   }
246*9356374aSAndroid Build Coastguard Worker   const char* const signal_string =
247*9356374aSAndroid Build Coastguard Worker       debugging_internal::FailureSignalToString(signo);
248*9356374aSAndroid Build Coastguard Worker   if (signal_string != nullptr && signal_string[0] != '\0') {
249*9356374aSAndroid Build Coastguard Worker     snprintf(buf, sizeof(buf), "*** %s received at time=%ld%s ***\n",
250*9356374aSAndroid Build Coastguard Worker              signal_string,
251*9356374aSAndroid Build Coastguard Worker              static_cast<long>(time(nullptr)),  // NOLINT(runtime/int)
252*9356374aSAndroid Build Coastguard Worker              on_cpu);
253*9356374aSAndroid Build Coastguard Worker   } else {
254*9356374aSAndroid Build Coastguard Worker     snprintf(buf, sizeof(buf), "*** Signal %d received at time=%ld%s ***\n",
255*9356374aSAndroid Build Coastguard Worker              signo, static_cast<long>(time(nullptr)),  // NOLINT(runtime/int)
256*9356374aSAndroid Build Coastguard Worker              on_cpu);
257*9356374aSAndroid Build Coastguard Worker   }
258*9356374aSAndroid Build Coastguard Worker   writerfn(buf);
259*9356374aSAndroid Build Coastguard Worker }
260*9356374aSAndroid Build Coastguard Worker 
261*9356374aSAndroid Build Coastguard Worker // `void*` might not be big enough to store `void(*)(const char*)`.
262*9356374aSAndroid Build Coastguard Worker struct WriterFnStruct {
263*9356374aSAndroid Build Coastguard Worker   void (*writerfn)(const char*);
264*9356374aSAndroid Build Coastguard Worker };
265*9356374aSAndroid Build Coastguard Worker 
266*9356374aSAndroid Build Coastguard Worker // Many of the absl::debugging_internal::Dump* functions in
267*9356374aSAndroid Build Coastguard Worker // examine_stack.h take a writer function pointer that has a void* arg
268*9356374aSAndroid Build Coastguard Worker // for historical reasons. failure_signal_handler_writer only takes a
269*9356374aSAndroid Build Coastguard Worker // data pointer. This function converts between these types.
WriterFnWrapper(const char * data,void * arg)270*9356374aSAndroid Build Coastguard Worker static void WriterFnWrapper(const char* data, void* arg) {
271*9356374aSAndroid Build Coastguard Worker   static_cast<WriterFnStruct*>(arg)->writerfn(data);
272*9356374aSAndroid Build Coastguard Worker }
273*9356374aSAndroid Build Coastguard Worker 
274*9356374aSAndroid Build Coastguard Worker // Convenient wrapper around DumpPCAndFrameSizesAndStackTrace() for signal
275*9356374aSAndroid Build Coastguard Worker // handlers. "noinline" so that GetStackFrames() skips the top-most stack
276*9356374aSAndroid Build Coastguard Worker // frame for this function.
WriteStackTrace(void * ucontext,bool symbolize_stacktrace,void (* writerfn)(const char *,void *),void * writerfn_arg)277*9356374aSAndroid Build Coastguard Worker ABSL_ATTRIBUTE_NOINLINE static void WriteStackTrace(
278*9356374aSAndroid Build Coastguard Worker     void* ucontext, bool symbolize_stacktrace,
279*9356374aSAndroid Build Coastguard Worker     void (*writerfn)(const char*, void*), void* writerfn_arg) {
280*9356374aSAndroid Build Coastguard Worker   constexpr int kNumStackFrames = 32;
281*9356374aSAndroid Build Coastguard Worker   void* stack[kNumStackFrames];
282*9356374aSAndroid Build Coastguard Worker   int frame_sizes[kNumStackFrames];
283*9356374aSAndroid Build Coastguard Worker   int min_dropped_frames;
284*9356374aSAndroid Build Coastguard Worker   int depth = absl::GetStackFramesWithContext(
285*9356374aSAndroid Build Coastguard Worker       stack, frame_sizes, kNumStackFrames,
286*9356374aSAndroid Build Coastguard Worker       1,  // Do not include this function in stack trace.
287*9356374aSAndroid Build Coastguard Worker       ucontext, &min_dropped_frames);
288*9356374aSAndroid Build Coastguard Worker   absl::debugging_internal::DumpPCAndFrameSizesAndStackTrace(
289*9356374aSAndroid Build Coastguard Worker       absl::debugging_internal::GetProgramCounter(ucontext), stack, frame_sizes,
290*9356374aSAndroid Build Coastguard Worker       depth, min_dropped_frames, symbolize_stacktrace, writerfn, writerfn_arg);
291*9356374aSAndroid Build Coastguard Worker }
292*9356374aSAndroid Build Coastguard Worker 
293*9356374aSAndroid Build Coastguard Worker // Called by AbslFailureSignalHandler() to write the failure info. It is
294*9356374aSAndroid Build Coastguard Worker // called once with writerfn set to WriteToStderr() and then possibly
295*9356374aSAndroid Build Coastguard Worker // with writerfn set to the user provided function.
WriteFailureInfo(int signo,void * ucontext,int cpu,void (* writerfn)(const char *))296*9356374aSAndroid Build Coastguard Worker static void WriteFailureInfo(int signo, void* ucontext, int cpu,
297*9356374aSAndroid Build Coastguard Worker                              void (*writerfn)(const char*)) {
298*9356374aSAndroid Build Coastguard Worker   WriterFnStruct writerfn_struct{writerfn};
299*9356374aSAndroid Build Coastguard Worker   WriteSignalMessage(signo, cpu, writerfn);
300*9356374aSAndroid Build Coastguard Worker   WriteStackTrace(ucontext, fsh_options.symbolize_stacktrace, WriterFnWrapper,
301*9356374aSAndroid Build Coastguard Worker                   &writerfn_struct);
302*9356374aSAndroid Build Coastguard Worker }
303*9356374aSAndroid Build Coastguard Worker 
304*9356374aSAndroid Build Coastguard Worker // absl::SleepFor() can't be used here since AbslInternalSleepFor()
305*9356374aSAndroid Build Coastguard Worker // may be overridden to do something that isn't async-signal-safe on
306*9356374aSAndroid Build Coastguard Worker // some platforms.
PortableSleepForSeconds(int seconds)307*9356374aSAndroid Build Coastguard Worker static void PortableSleepForSeconds(int seconds) {
308*9356374aSAndroid Build Coastguard Worker #ifdef _WIN32
309*9356374aSAndroid Build Coastguard Worker   Sleep(static_cast<DWORD>(seconds * 1000));
310*9356374aSAndroid Build Coastguard Worker #else
311*9356374aSAndroid Build Coastguard Worker   struct timespec sleep_time;
312*9356374aSAndroid Build Coastguard Worker   sleep_time.tv_sec = seconds;
313*9356374aSAndroid Build Coastguard Worker   sleep_time.tv_nsec = 0;
314*9356374aSAndroid Build Coastguard Worker   while (nanosleep(&sleep_time, &sleep_time) != 0 && errno == EINTR) {
315*9356374aSAndroid Build Coastguard Worker   }
316*9356374aSAndroid Build Coastguard Worker #endif
317*9356374aSAndroid Build Coastguard Worker }
318*9356374aSAndroid Build Coastguard Worker 
319*9356374aSAndroid Build Coastguard Worker #ifdef ABSL_HAVE_ALARM
320*9356374aSAndroid Build Coastguard Worker // AbslFailureSignalHandler() installs this as a signal handler for
321*9356374aSAndroid Build Coastguard Worker // SIGALRM, then sets an alarm to be delivered to the program after a
322*9356374aSAndroid Build Coastguard Worker // set amount of time. If AbslFailureSignalHandler() hangs for more than
323*9356374aSAndroid Build Coastguard Worker // the alarm timeout, ImmediateAbortSignalHandler() will abort the
324*9356374aSAndroid Build Coastguard Worker // program.
ImmediateAbortSignalHandler(int)325*9356374aSAndroid Build Coastguard Worker static void ImmediateAbortSignalHandler(int) { RaiseToDefaultHandler(SIGABRT); }
326*9356374aSAndroid Build Coastguard Worker #endif
327*9356374aSAndroid Build Coastguard Worker 
328*9356374aSAndroid Build Coastguard Worker // absl::base_internal::GetTID() returns pid_t on most platforms, but
329*9356374aSAndroid Build Coastguard Worker // returns absl::base_internal::pid_t on Windows.
330*9356374aSAndroid Build Coastguard Worker using GetTidType = decltype(absl::base_internal::GetTID());
331*9356374aSAndroid Build Coastguard Worker ABSL_CONST_INIT static std::atomic<GetTidType> failed_tid(0);
332*9356374aSAndroid Build Coastguard Worker 
333*9356374aSAndroid Build Coastguard Worker #ifndef ABSL_HAVE_SIGACTION
AbslFailureSignalHandler(int signo)334*9356374aSAndroid Build Coastguard Worker static void AbslFailureSignalHandler(int signo) {
335*9356374aSAndroid Build Coastguard Worker   void* ucontext = nullptr;
336*9356374aSAndroid Build Coastguard Worker #else
337*9356374aSAndroid Build Coastguard Worker static void AbslFailureSignalHandler(int signo, siginfo_t*, void* ucontext) {
338*9356374aSAndroid Build Coastguard Worker #endif
339*9356374aSAndroid Build Coastguard Worker 
340*9356374aSAndroid Build Coastguard Worker   const GetTidType this_tid = absl::base_internal::GetTID();
341*9356374aSAndroid Build Coastguard Worker   GetTidType previous_failed_tid = 0;
342*9356374aSAndroid Build Coastguard Worker   if (!failed_tid.compare_exchange_strong(previous_failed_tid, this_tid,
343*9356374aSAndroid Build Coastguard Worker                                           std::memory_order_acq_rel,
344*9356374aSAndroid Build Coastguard Worker                                           std::memory_order_relaxed)) {
345*9356374aSAndroid Build Coastguard Worker     ABSL_RAW_LOG(
346*9356374aSAndroid Build Coastguard Worker         ERROR,
347*9356374aSAndroid Build Coastguard Worker         "Signal %d raised at PC=%p while already in AbslFailureSignalHandler()",
348*9356374aSAndroid Build Coastguard Worker         signo, absl::debugging_internal::GetProgramCounter(ucontext));
349*9356374aSAndroid Build Coastguard Worker     if (this_tid != previous_failed_tid) {
350*9356374aSAndroid Build Coastguard Worker       // Another thread is already in AbslFailureSignalHandler(), so wait
351*9356374aSAndroid Build Coastguard Worker       // a bit for it to finish. If the other thread doesn't kill us,
352*9356374aSAndroid Build Coastguard Worker       // we do so after sleeping.
353*9356374aSAndroid Build Coastguard Worker       PortableSleepForSeconds(3);
354*9356374aSAndroid Build Coastguard Worker       RaiseToDefaultHandler(signo);
355*9356374aSAndroid Build Coastguard Worker       // The recursively raised signal may be blocked until we return.
356*9356374aSAndroid Build Coastguard Worker       return;
357*9356374aSAndroid Build Coastguard Worker     }
358*9356374aSAndroid Build Coastguard Worker   }
359*9356374aSAndroid Build Coastguard Worker 
360*9356374aSAndroid Build Coastguard Worker   // Increase the chance that the CPU we report was the same CPU on which the
361*9356374aSAndroid Build Coastguard Worker   // signal was received by doing this as early as possible, i.e. after
362*9356374aSAndroid Build Coastguard Worker   // verifying that this is not a recursive signal handler invocation.
363*9356374aSAndroid Build Coastguard Worker   int my_cpu = -1;
364*9356374aSAndroid Build Coastguard Worker #ifdef ABSL_HAVE_SCHED_GETCPU
365*9356374aSAndroid Build Coastguard Worker   my_cpu = sched_getcpu();
366*9356374aSAndroid Build Coastguard Worker #endif
367*9356374aSAndroid Build Coastguard Worker 
368*9356374aSAndroid Build Coastguard Worker #ifdef ABSL_HAVE_ALARM
369*9356374aSAndroid Build Coastguard Worker   // Set an alarm to abort the program in case this code hangs or deadlocks.
370*9356374aSAndroid Build Coastguard Worker   if (fsh_options.alarm_on_failure_secs > 0) {
371*9356374aSAndroid Build Coastguard Worker     alarm(0);  // Cancel any existing alarms.
372*9356374aSAndroid Build Coastguard Worker     signal(SIGALRM, ImmediateAbortSignalHandler);
373*9356374aSAndroid Build Coastguard Worker     alarm(static_cast<unsigned int>(fsh_options.alarm_on_failure_secs));
374*9356374aSAndroid Build Coastguard Worker   }
375*9356374aSAndroid Build Coastguard Worker #endif
376*9356374aSAndroid Build Coastguard Worker 
377*9356374aSAndroid Build Coastguard Worker   // First write to stderr.
378*9356374aSAndroid Build Coastguard Worker   WriteFailureInfo(
379*9356374aSAndroid Build Coastguard Worker       signo, ucontext, my_cpu, +[](const char* data) {
380*9356374aSAndroid Build Coastguard Worker         absl::raw_log_internal::AsyncSignalSafeWriteError(data, strlen(data));
381*9356374aSAndroid Build Coastguard Worker       });
382*9356374aSAndroid Build Coastguard Worker 
383*9356374aSAndroid Build Coastguard Worker   // Riskier code (because it is less likely to be async-signal-safe)
384*9356374aSAndroid Build Coastguard Worker   // goes after this point.
385*9356374aSAndroid Build Coastguard Worker   if (fsh_options.writerfn != nullptr) {
386*9356374aSAndroid Build Coastguard Worker     WriteFailureInfo(signo, ucontext, my_cpu, fsh_options.writerfn);
387*9356374aSAndroid Build Coastguard Worker     fsh_options.writerfn(nullptr);
388*9356374aSAndroid Build Coastguard Worker   }
389*9356374aSAndroid Build Coastguard Worker 
390*9356374aSAndroid Build Coastguard Worker   if (fsh_options.call_previous_handler) {
391*9356374aSAndroid Build Coastguard Worker     RaiseToPreviousHandler(signo);
392*9356374aSAndroid Build Coastguard Worker   } else {
393*9356374aSAndroid Build Coastguard Worker     RaiseToDefaultHandler(signo);
394*9356374aSAndroid Build Coastguard Worker   }
395*9356374aSAndroid Build Coastguard Worker }
396*9356374aSAndroid Build Coastguard Worker 
397*9356374aSAndroid Build Coastguard Worker void InstallFailureSignalHandler(const FailureSignalHandlerOptions& options) {
398*9356374aSAndroid Build Coastguard Worker   fsh_options = options;
399*9356374aSAndroid Build Coastguard Worker   for (auto& it : failure_signal_data) {
400*9356374aSAndroid Build Coastguard Worker     InstallOneFailureHandler(&it, AbslFailureSignalHandler);
401*9356374aSAndroid Build Coastguard Worker   }
402*9356374aSAndroid Build Coastguard Worker }
403*9356374aSAndroid Build Coastguard Worker 
404*9356374aSAndroid Build Coastguard Worker ABSL_NAMESPACE_END
405*9356374aSAndroid Build Coastguard Worker }  // namespace absl
406