xref: /freebsd/contrib/llvm-project/compiler-rt/lib/msan/msan_linux.cpp (revision bc5304a006238115291e7568583632889dffbab9)
1 //===-- msan_linux.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 MemorySanitizer.
10 //
11 // Linux-, NetBSD- and FreeBSD-specific code.
12 //===----------------------------------------------------------------------===//
13 
14 #include "sanitizer_common/sanitizer_platform.h"
15 #if SANITIZER_FREEBSD || SANITIZER_LINUX || SANITIZER_NETBSD
16 
17 #include "msan.h"
18 #include "msan_report.h"
19 #include "msan_thread.h"
20 
21 #include <elf.h>
22 #include <link.h>
23 #include <pthread.h>
24 #include <stdio.h>
25 #include <stdlib.h>
26 #include <signal.h>
27 #include <unistd.h>
28 #include <unwind.h>
29 #include <sys/time.h>
30 #include <sys/resource.h>
31 
32 #include "sanitizer_common/sanitizer_common.h"
33 #include "sanitizer_common/sanitizer_procmaps.h"
34 
35 namespace __msan {
36 
37 void ReportMapRange(const char *descr, uptr beg, uptr size) {
38   if (size > 0) {
39     uptr end = beg + size - 1;
40     VPrintf(1, "%s : %p - %p\n", descr, beg, end);
41   }
42 }
43 
44 static bool CheckMemoryRangeAvailability(uptr beg, uptr size) {
45   if (size > 0) {
46     uptr end = beg + size - 1;
47     if (!MemoryRangeIsAvailable(beg, end)) {
48       Printf("FATAL: Memory range %p - %p is not available.\n", beg, end);
49       return false;
50     }
51   }
52   return true;
53 }
54 
55 static bool ProtectMemoryRange(uptr beg, uptr size, const char *name) {
56   if (size > 0) {
57     void *addr = MmapFixedNoAccess(beg, size, name);
58     if (beg == 0 && addr) {
59       // Depending on the kernel configuration, we may not be able to protect
60       // the page at address zero.
61       uptr gap = 16 * GetPageSizeCached();
62       beg += gap;
63       size -= gap;
64       addr = MmapFixedNoAccess(beg, size, name);
65     }
66     if ((uptr)addr != beg) {
67       uptr end = beg + size - 1;
68       Printf("FATAL: Cannot protect memory range %p - %p (%s).\n", beg, end,
69              name);
70       return false;
71     }
72   }
73   return true;
74 }
75 
76 static void CheckMemoryLayoutSanity() {
77   uptr prev_end = 0;
78   for (unsigned i = 0; i < kMemoryLayoutSize; ++i) {
79     uptr start = kMemoryLayout[i].start;
80     uptr end = kMemoryLayout[i].end;
81     MappingDesc::Type type = kMemoryLayout[i].type;
82     CHECK_LT(start, end);
83     CHECK_EQ(prev_end, start);
84     CHECK(addr_is_type(start, type));
85     CHECK(addr_is_type((start + end) / 2, type));
86     CHECK(addr_is_type(end - 1, type));
87     if (type == MappingDesc::APP) {
88       uptr addr = start;
89       CHECK(MEM_IS_SHADOW(MEM_TO_SHADOW(addr)));
90       CHECK(MEM_IS_ORIGIN(MEM_TO_ORIGIN(addr)));
91       CHECK_EQ(MEM_TO_ORIGIN(addr), SHADOW_TO_ORIGIN(MEM_TO_SHADOW(addr)));
92 
93       addr = (start + end) / 2;
94       CHECK(MEM_IS_SHADOW(MEM_TO_SHADOW(addr)));
95       CHECK(MEM_IS_ORIGIN(MEM_TO_ORIGIN(addr)));
96       CHECK_EQ(MEM_TO_ORIGIN(addr), SHADOW_TO_ORIGIN(MEM_TO_SHADOW(addr)));
97 
98       addr = end - 1;
99       CHECK(MEM_IS_SHADOW(MEM_TO_SHADOW(addr)));
100       CHECK(MEM_IS_ORIGIN(MEM_TO_ORIGIN(addr)));
101       CHECK_EQ(MEM_TO_ORIGIN(addr), SHADOW_TO_ORIGIN(MEM_TO_SHADOW(addr)));
102     }
103     prev_end = end;
104   }
105 }
106 
107 bool InitShadow(bool init_origins) {
108   // Let user know mapping parameters first.
109   VPrintf(1, "__msan_init %p\n", &__msan_init);
110   for (unsigned i = 0; i < kMemoryLayoutSize; ++i)
111     VPrintf(1, "%s: %zx - %zx\n", kMemoryLayout[i].name, kMemoryLayout[i].start,
112             kMemoryLayout[i].end - 1);
113 
114   CheckMemoryLayoutSanity();
115 
116   if (!MEM_IS_APP(&__msan_init)) {
117     Printf("FATAL: Code %p is out of application range. Non-PIE build?\n",
118            (uptr)&__msan_init);
119     return false;
120   }
121 
122   const uptr maxVirtualAddress = GetMaxUserVirtualAddress();
123 
124   for (unsigned i = 0; i < kMemoryLayoutSize; ++i) {
125     uptr start = kMemoryLayout[i].start;
126     uptr end = kMemoryLayout[i].end;
127     uptr size = end - start;
128     MappingDesc::Type type = kMemoryLayout[i].type;
129 
130     // Check if the segment should be mapped based on platform constraints.
131     if (start >= maxVirtualAddress)
132       continue;
133 
134     bool map = type == MappingDesc::SHADOW ||
135                (init_origins && type == MappingDesc::ORIGIN);
136     bool protect = type == MappingDesc::INVALID ||
137                    (!init_origins && type == MappingDesc::ORIGIN);
138     CHECK(!(map && protect));
139     if (!map && !protect)
140       CHECK(type == MappingDesc::APP);
141     if (map) {
142       if (!CheckMemoryRangeAvailability(start, size))
143         return false;
144       if (!MmapFixedSuperNoReserve(start, size, kMemoryLayout[i].name))
145         return false;
146       if (common_flags()->use_madv_dontdump)
147         DontDumpShadowMemory(start, size);
148     }
149     if (protect) {
150       if (!CheckMemoryRangeAvailability(start, size))
151         return false;
152       if (!ProtectMemoryRange(start, size, kMemoryLayout[i].name))
153         return false;
154     }
155   }
156 
157   return true;
158 }
159 
160 static void MsanAtExit(void) {
161   if (flags()->print_stats && (flags()->atexit || msan_report_count > 0))
162     ReportStats();
163   if (msan_report_count > 0) {
164     ReportAtExitStatistics();
165     if (common_flags()->exitcode)
166       internal__exit(common_flags()->exitcode);
167   }
168 }
169 
170 void InstallAtExitHandler() {
171   atexit(MsanAtExit);
172 }
173 
174 // ---------------------- TSD ---------------- {{{1
175 
176 #if SANITIZER_NETBSD
177 // Thread Static Data cannot be used in early init on NetBSD.
178 // Reuse the MSan TSD API for compatibility with existing code
179 // with an alternative implementation.
180 
181 static void (*tsd_destructor)(void *tsd) = nullptr;
182 
183 struct tsd_key {
184   tsd_key() : key(nullptr) {}
185   ~tsd_key() {
186     CHECK(tsd_destructor);
187     if (key)
188       (*tsd_destructor)(key);
189   }
190   MsanThread *key;
191 };
192 
193 static thread_local struct tsd_key key;
194 
195 void MsanTSDInit(void (*destructor)(void *tsd)) {
196   CHECK(!tsd_destructor);
197   tsd_destructor = destructor;
198 }
199 
200 MsanThread *GetCurrentThread() {
201   CHECK(tsd_destructor);
202   return key.key;
203 }
204 
205 void SetCurrentThread(MsanThread *tsd) {
206   CHECK(tsd_destructor);
207   CHECK(tsd);
208   CHECK(!key.key);
209   key.key = tsd;
210 }
211 
212 void MsanTSDDtor(void *tsd) {
213   CHECK(tsd_destructor);
214   CHECK_EQ(key.key, tsd);
215   key.key = nullptr;
216   // Make sure that signal handler can not see a stale current thread pointer.
217   atomic_signal_fence(memory_order_seq_cst);
218   MsanThread::TSDDtor(tsd);
219 }
220 #else
221 static pthread_key_t tsd_key;
222 static bool tsd_key_inited = false;
223 
224 void MsanTSDInit(void (*destructor)(void *tsd)) {
225   CHECK(!tsd_key_inited);
226   tsd_key_inited = true;
227   CHECK_EQ(0, pthread_key_create(&tsd_key, destructor));
228 }
229 
230 static THREADLOCAL MsanThread* msan_current_thread;
231 
232 MsanThread *GetCurrentThread() {
233   return msan_current_thread;
234 }
235 
236 void SetCurrentThread(MsanThread *t) {
237   // Make sure we do not reset the current MsanThread.
238   CHECK_EQ(0, msan_current_thread);
239   msan_current_thread = t;
240   // Make sure that MsanTSDDtor gets called at the end.
241   CHECK(tsd_key_inited);
242   pthread_setspecific(tsd_key, (void *)t);
243 }
244 
245 void MsanTSDDtor(void *tsd) {
246   MsanThread *t = (MsanThread*)tsd;
247   if (t->destructor_iterations_ > 1) {
248     t->destructor_iterations_--;
249     CHECK_EQ(0, pthread_setspecific(tsd_key, tsd));
250     return;
251   }
252   msan_current_thread = nullptr;
253   // Make sure that signal handler can not see a stale current thread pointer.
254   atomic_signal_fence(memory_order_seq_cst);
255   MsanThread::TSDDtor(tsd);
256 }
257 #endif
258 
259 } // namespace __msan
260 
261 #endif // SANITIZER_FREEBSD || SANITIZER_LINUX || SANITIZER_NETBSD
262