1 //===-- asan_posix.cpp ----------------------------------------------------===//
2 //
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6 //
7 //===----------------------------------------------------------------------===//
8 //
9 // This file is a part of AddressSanitizer, an address sanity checker.
10 //
11 // Posix-specific details.
12 //===----------------------------------------------------------------------===//
13
14 #include "sanitizer_common/sanitizer_platform.h"
15 #if SANITIZER_POSIX
16
17 #include "asan_internal.h"
18 #include "asan_interceptors.h"
19 #include "asan_mapping.h"
20 #include "asan_poisoning.h"
21 #include "asan_report.h"
22 #include "asan_stack.h"
23 #include "sanitizer_common/sanitizer_libc.h"
24 #include "sanitizer_common/sanitizer_posix.h"
25 #include "sanitizer_common/sanitizer_procmaps.h"
26
27 #include <pthread.h>
28 #include <signal.h>
29 #include <stdlib.h>
30 #include <sys/time.h>
31 #include <sys/resource.h>
32 #include <unistd.h>
33
34 namespace __asan {
35
AsanOnDeadlySignal(int signo,void * siginfo,void * context)36 void AsanOnDeadlySignal(int signo, void *siginfo, void *context) {
37 StartReportDeadlySignal();
38 SignalContext sig(siginfo, context);
39 ReportDeadlySignal(sig);
40 }
41
PlatformUnpoisonStacks()42 bool PlatformUnpoisonStacks() {
43 stack_t signal_stack;
44 CHECK_EQ(0, sigaltstack(nullptr, &signal_stack));
45 uptr sigalt_bottom = (uptr)signal_stack.ss_sp;
46 uptr sigalt_top = (uptr)((char *)signal_stack.ss_sp + signal_stack.ss_size);
47 // If we're executing on the signal alternate stack AND the Linux flag
48 // SS_AUTODISARM was used, then we cannot get the signal alternate stack
49 // bounds from sigaltstack -- sigaltstack's output looks just as if no
50 // alternate stack has ever been set up.
51 // We're always unpoisoning the signal alternate stack to support jumping
52 // between the default stack and signal alternate stack.
53 if (signal_stack.ss_flags != SS_DISABLE)
54 UnpoisonStack(sigalt_bottom, sigalt_top, "sigalt");
55
56 if (signal_stack.ss_flags != SS_ONSTACK)
57 return false;
58
59 // Since we're on the signal altnerate stack, we cannot find the DEFAULT
60 // stack bottom using a local variable.
61 uptr default_bottom, tls_addr, tls_size, stack_size;
62 GetThreadStackAndTls(/*main=*/false, &default_bottom, &stack_size, &tls_addr,
63 &tls_size);
64 UnpoisonStack(default_bottom, default_bottom + stack_size, "default");
65 return true;
66 }
67
68 // ---------------------- TSD ---------------- {{{1
69
70 #if SANITIZER_NETBSD && !ASAN_DYNAMIC
71 // Thread Static Data cannot be used in early static ASan init on NetBSD.
72 // Reuse the Asan TSD API for compatibility with existing code
73 // with an alternative implementation.
74
75 static void (*tsd_destructor)(void *tsd) = nullptr;
76
77 struct tsd_key {
tsd_key__asan::tsd_key78 tsd_key() : key(nullptr) {}
~tsd_key__asan::tsd_key79 ~tsd_key() {
80 CHECK(tsd_destructor);
81 if (key)
82 (*tsd_destructor)(key);
83 }
84 void *key;
85 };
86
87 static thread_local struct tsd_key key;
88
AsanTSDInit(void (* destructor)(void * tsd))89 void AsanTSDInit(void (*destructor)(void *tsd)) {
90 CHECK(!tsd_destructor);
91 tsd_destructor = destructor;
92 }
93
AsanTSDGet()94 void *AsanTSDGet() {
95 CHECK(tsd_destructor);
96 return key.key;
97 }
98
AsanTSDSet(void * tsd)99 void AsanTSDSet(void *tsd) {
100 CHECK(tsd_destructor);
101 CHECK(tsd);
102 CHECK(!key.key);
103 key.key = tsd;
104 }
105
PlatformTSDDtor(void * tsd)106 void PlatformTSDDtor(void *tsd) {
107 CHECK(tsd_destructor);
108 CHECK_EQ(key.key, tsd);
109 key.key = nullptr;
110 // Make sure that signal handler can not see a stale current thread pointer.
111 atomic_signal_fence(memory_order_seq_cst);
112 AsanThread::TSDDtor(tsd);
113 }
114 #else
115 static pthread_key_t tsd_key;
116 static bool tsd_key_inited = false;
AsanTSDInit(void (* destructor)(void * tsd))117 void AsanTSDInit(void (*destructor)(void *tsd)) {
118 CHECK(!tsd_key_inited);
119 tsd_key_inited = true;
120 CHECK_EQ(0, pthread_key_create(&tsd_key, destructor));
121 }
122
AsanTSDGet()123 void *AsanTSDGet() {
124 CHECK(tsd_key_inited);
125 return pthread_getspecific(tsd_key);
126 }
127
AsanTSDSet(void * tsd)128 void AsanTSDSet(void *tsd) {
129 CHECK(tsd_key_inited);
130 pthread_setspecific(tsd_key, tsd);
131 }
132
PlatformTSDDtor(void * tsd)133 void PlatformTSDDtor(void *tsd) {
134 AsanThreadContext *context = (AsanThreadContext*)tsd;
135 if (context->destructor_iterations > 1) {
136 context->destructor_iterations--;
137 CHECK_EQ(0, pthread_setspecific(tsd_key, tsd));
138 return;
139 }
140 AsanThread::TSDDtor(tsd);
141 }
142 #endif
143 } // namespace __asan
144
145 #endif // SANITIZER_POSIX
146