1 //===-- sanitizer_platform.h ------------------------------------*- C++ -*-===//
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 // Common platform macros.
10 //===----------------------------------------------------------------------===//
11 
12 #ifndef SANITIZER_PLATFORM_H
13 #define SANITIZER_PLATFORM_H
14 
15 #if !defined(__linux__) && !defined(__FreeBSD__) && !defined(__NetBSD__) && \
16     !defined(__APPLE__) && !defined(_WIN32) && !defined(__Fuchsia__) &&     \
17     !(defined(__sun__) && defined(__svr4__))
18 #  error "This operating system is not supported"
19 #endif
20 
21 // Get __GLIBC__ on a glibc platform. Exclude Android: features.h includes C
22 // function declarations into a .S file which doesn't compile.
23 // https://crbug.com/1162741
24 #if __has_include(<features.h>) && !defined(__ANDROID__)
25 #  include <features.h>
26 #endif
27 
28 #if defined(__linux__)
29 #  define SANITIZER_LINUX 1
30 #else
31 #  define SANITIZER_LINUX 0
32 #endif
33 
34 #if defined(__GLIBC__)
35 #  define SANITIZER_GLIBC 1
36 #else
37 #  define SANITIZER_GLIBC 0
38 #endif
39 
40 #if defined(__FreeBSD__)
41 #  define SANITIZER_FREEBSD 1
42 #else
43 #  define SANITIZER_FREEBSD 0
44 #endif
45 
46 #if defined(__NetBSD__)
47 #  define SANITIZER_NETBSD 1
48 #else
49 #  define SANITIZER_NETBSD 0
50 #endif
51 
52 #if defined(__sun__) && defined(__svr4__)
53 #  define SANITIZER_SOLARIS 1
54 #else
55 #  define SANITIZER_SOLARIS 0
56 #endif
57 
58 // - SANITIZER_APPLE: all Apple code
59 //   - TARGET_OS_OSX: macOS
60 //   - SANITIZER_IOS: devices (iOS and iOS-like)
61 //     - SANITIZER_WATCHOS
62 //     - SANITIZER_TVOS
63 //   - SANITIZER_IOSSIM: simulators (iOS and iOS-like)
64 //   - SANITIZER_DRIVERKIT
65 #if defined(__APPLE__)
66 #  define SANITIZER_APPLE 1
67 #  include <TargetConditionals.h>
68 #  if TARGET_OS_OSX
69 #    define SANITIZER_OSX 1
70 #  else
71 #    define SANITIZER_OSX 0
72 #  endif
73 #  if TARGET_OS_IPHONE
74 #    define SANITIZER_IOS 1
75 #  else
76 #    define SANITIZER_IOS 0
77 #  endif
78 #  if TARGET_OS_WATCH
79 #    define SANITIZER_WATCHOS 1
80 #  else
81 #    define SANITIZER_WATCHOS 0
82 #  endif
83 #  if TARGET_OS_TV
84 #    define SANITIZER_TVOS 1
85 #  else
86 #    define SANITIZER_TVOS 0
87 #  endif
88 #  if TARGET_OS_SIMULATOR
89 #    define SANITIZER_IOSSIM 1
90 #  else
91 #    define SANITIZER_IOSSIM 0
92 #  endif
93 #  if defined(TARGET_OS_DRIVERKIT) && TARGET_OS_DRIVERKIT
94 #    define SANITIZER_DRIVERKIT 1
95 #  else
96 #    define SANITIZER_DRIVERKIT 0
97 #  endif
98 #else
99 #  define SANITIZER_APPLE 0
100 #  define SANITIZER_OSX 0
101 #  define SANITIZER_IOS 0
102 #  define SANITIZER_WATCHOS 0
103 #  define SANITIZER_TVOS 0
104 #  define SANITIZER_IOSSIM 0
105 #  define SANITIZER_DRIVERKIT 0
106 #endif
107 
108 #if defined(_WIN32)
109 #  define SANITIZER_WINDOWS 1
110 #else
111 #  define SANITIZER_WINDOWS 0
112 #endif
113 
114 #if defined(_WIN64)
115 #  define SANITIZER_WINDOWS64 1
116 #else
117 #  define SANITIZER_WINDOWS64 0
118 #endif
119 
120 #if defined(__ANDROID__)
121 #  define SANITIZER_ANDROID 1
122 #else
123 #  define SANITIZER_ANDROID 0
124 #endif
125 
126 #if defined(__Fuchsia__)
127 #  define SANITIZER_FUCHSIA 1
128 #else
129 #  define SANITIZER_FUCHSIA 0
130 #endif
131 
132 // Assume linux that is not glibc or android is musl libc.
133 #if SANITIZER_LINUX && !SANITIZER_GLIBC && !SANITIZER_ANDROID
134 #  define SANITIZER_MUSL 1
135 #else
136 #  define SANITIZER_MUSL 0
137 #endif
138 
139 #define SANITIZER_POSIX                                     \
140   (SANITIZER_FREEBSD || SANITIZER_LINUX || SANITIZER_APPLE || \
141    SANITIZER_NETBSD || SANITIZER_SOLARIS)
142 
143 #if __LP64__ || defined(_WIN64)
144 #  define SANITIZER_WORDSIZE 64
145 #else
146 #  define SANITIZER_WORDSIZE 32
147 #endif
148 
149 #if SANITIZER_WORDSIZE == 64
150 #  define FIRST_32_SECOND_64(a, b) (b)
151 #else
152 #  define FIRST_32_SECOND_64(a, b) (a)
153 #endif
154 
155 #if defined(__x86_64__) && !defined(_LP64)
156 #  define SANITIZER_X32 1
157 #else
158 #  define SANITIZER_X32 0
159 #endif
160 
161 #if defined(__x86_64__) || defined(_M_X64)
162 #  define SANITIZER_X64 1
163 #else
164 #  define SANITIZER_X64 0
165 #endif
166 
167 #if defined(__i386__) || defined(_M_IX86)
168 #  define SANITIZER_I386 1
169 #else
170 #  define SANITIZER_I386 0
171 #endif
172 
173 #if defined(__mips__)
174 #  define SANITIZER_MIPS 1
175 #  if defined(__mips64) && _MIPS_SIM == _ABI64
176 #    define SANITIZER_MIPS32 0
177 #    define SANITIZER_MIPS64 1
178 #  else
179 #    define SANITIZER_MIPS32 1
180 #    define SANITIZER_MIPS64 0
181 #  endif
182 #else
183 #  define SANITIZER_MIPS 0
184 #  define SANITIZER_MIPS32 0
185 #  define SANITIZER_MIPS64 0
186 #endif
187 
188 #if defined(__s390__)
189 #  define SANITIZER_S390 1
190 #  if defined(__s390x__)
191 #    define SANITIZER_S390_31 0
192 #    define SANITIZER_S390_64 1
193 #  else
194 #    define SANITIZER_S390_31 1
195 #    define SANITIZER_S390_64 0
196 #  endif
197 #else
198 #  define SANITIZER_S390 0
199 #  define SANITIZER_S390_31 0
200 #  define SANITIZER_S390_64 0
201 #endif
202 
203 #if defined(__sparc__)
204 #  define SANITIZER_SPARC 1
205 #  if defined(__arch64__)
206 #    define SANITIZER_SPARC32 0
207 #    define SANITIZER_SPARC64 1
208 #  else
209 #    define SANITIZER_SPARC32 1
210 #    define SANITIZER_SPARC64 0
211 #  endif
212 #else
213 #  define SANITIZER_SPARC 0
214 #  define SANITIZER_SPARC32 0
215 #  define SANITIZER_SPARC64 0
216 #endif
217 
218 #if defined(__powerpc__)
219 #  define SANITIZER_PPC 1
220 #  if defined(__powerpc64__)
221 #    define SANITIZER_PPC32 0
222 #    define SANITIZER_PPC64 1
223 // 64-bit PPC has two ABIs (v1 and v2).  The old powerpc64 target is
224 // big-endian, and uses v1 ABI (known for its function descriptors),
225 // while the new powerpc64le target is little-endian and uses v2.
226 // In theory, you could convince gcc to compile for their evil twins
227 // (eg. big-endian v2), but you won't find such combinations in the wild
228 // (it'd require bootstrapping a whole system, which would be quite painful
229 // - there's no target triple for that).  LLVM doesn't support them either.
230 #    if _CALL_ELF == 2
231 #      define SANITIZER_PPC64V1 0
232 #      define SANITIZER_PPC64V2 1
233 #    else
234 #      define SANITIZER_PPC64V1 1
235 #      define SANITIZER_PPC64V2 0
236 #    endif
237 #  else
238 #    define SANITIZER_PPC32 1
239 #    define SANITIZER_PPC64 0
240 #    define SANITIZER_PPC64V1 0
241 #    define SANITIZER_PPC64V2 0
242 #  endif
243 #else
244 #  define SANITIZER_PPC 0
245 #  define SANITIZER_PPC32 0
246 #  define SANITIZER_PPC64 0
247 #  define SANITIZER_PPC64V1 0
248 #  define SANITIZER_PPC64V2 0
249 #endif
250 
251 #if defined(__arm__) || defined(_M_ARM)
252 #  define SANITIZER_ARM 1
253 #else
254 #  define SANITIZER_ARM 0
255 #endif
256 
257 #if defined(__aarch64__) || defined(_M_ARM64)
258 #  define SANITIZER_ARM64 1
259 #else
260 #  define SANITIZER_ARM64 0
261 #endif
262 
263 #if SANITIZER_SOLARIS && SANITIZER_WORDSIZE == 32
264 #  define SANITIZER_SOLARIS32 1
265 #else
266 #  define SANITIZER_SOLARIS32 0
267 #endif
268 
269 #if defined(__riscv) && (__riscv_xlen == 64)
270 #  define SANITIZER_RISCV64 1
271 #else
272 #  define SANITIZER_RISCV64 0
273 #endif
274 
275 #if defined(__loongarch_lp64)
276 #  define SANITIZER_LOONGARCH64 1
277 #else
278 #  define SANITIZER_LOONGARCH64 0
279 #endif
280 
281 // By default we allow to use SizeClassAllocator64 on 64-bit platform.
282 // But in some cases SizeClassAllocator64 does not work well and we need to
283 // fallback to SizeClassAllocator32.
284 // For such platforms build this code with -DSANITIZER_CAN_USE_ALLOCATOR64=0 or
285 // change the definition of SANITIZER_CAN_USE_ALLOCATOR64 here.
286 #ifndef SANITIZER_CAN_USE_ALLOCATOR64
287 #  if SANITIZER_RISCV64 || SANITIZER_IOS
288 #    define SANITIZER_CAN_USE_ALLOCATOR64 0
289 #  elif defined(__mips64) || defined(__hexagon__)
290 #    define SANITIZER_CAN_USE_ALLOCATOR64 0
291 #  else
292 #    define SANITIZER_CAN_USE_ALLOCATOR64 (SANITIZER_WORDSIZE == 64)
293 #  endif
294 #endif
295 
296 // The range of addresses which can be returned my mmap.
297 // FIXME: this value should be different on different platforms.  Larger values
298 // will still work but will consume more memory for TwoLevelByteMap.
299 #if defined(__mips__)
300 #  if SANITIZER_GO && defined(__mips64)
301 #    define SANITIZER_MMAP_RANGE_SIZE FIRST_32_SECOND_64(1ULL << 32, 1ULL << 47)
302 #  else
303 #    define SANITIZER_MMAP_RANGE_SIZE FIRST_32_SECOND_64(1ULL << 32, 1ULL << 40)
304 #  endif
305 #elif SANITIZER_RISCV64
306 #  define SANITIZER_MMAP_RANGE_SIZE FIRST_32_SECOND_64(1ULL << 32, 1ULL << 38)
307 #elif defined(__aarch64__)
308 #  if SANITIZER_APPLE
309 #    if SANITIZER_OSX || SANITIZER_IOSSIM
310 #      define SANITIZER_MMAP_RANGE_SIZE \
311         FIRST_32_SECOND_64(1ULL << 32, 1ULL << 47)
312 #    else
313 // Darwin iOS/ARM64 has a 36-bit VMA, 64GiB VM
314 #      define SANITIZER_MMAP_RANGE_SIZE \
315         FIRST_32_SECOND_64(1ULL << 32, 1ULL << 36)
316 #    endif
317 #  else
318 #    define SANITIZER_MMAP_RANGE_SIZE FIRST_32_SECOND_64(1ULL << 32, 1ULL << 48)
319 #  endif
320 #elif defined(__sparc__)
321 #  define SANITIZER_MMAP_RANGE_SIZE FIRST_32_SECOND_64(1ULL << 32, 1ULL << 52)
322 #else
323 #  define SANITIZER_MMAP_RANGE_SIZE FIRST_32_SECOND_64(1ULL << 32, 1ULL << 47)
324 #endif
325 
326 // Whether the addresses are sign-extended from the VMA range to the word.
327 // The SPARC64 Linux port implements this to split the VMA space into two
328 // non-contiguous halves with a huge hole in the middle.
329 #if defined(__sparc__) && SANITIZER_WORDSIZE == 64
330 #  define SANITIZER_SIGN_EXTENDED_ADDRESSES 1
331 #else
332 #  define SANITIZER_SIGN_EXTENDED_ADDRESSES 0
333 #endif
334 
335 // udi16 syscalls can only be used when the following conditions are
336 // met:
337 // * target is one of arm32, x86-32, sparc32, sh or m68k
338 // * libc version is libc5, glibc-2.0, glibc-2.1 or glibc-2.2 to 2.15
339 //   built against > linux-2.2 kernel headers
340 // Since we don't want to include libc headers here, we check the
341 // target only.
342 #if defined(__arm__) || SANITIZER_X32 || defined(__sparc__)
343 #  define SANITIZER_USES_UID16_SYSCALLS 1
344 #else
345 #  define SANITIZER_USES_UID16_SYSCALLS 0
346 #endif
347 
348 #if defined(__mips__)
349 #  define SANITIZER_POINTER_FORMAT_LENGTH FIRST_32_SECOND_64(8, 10)
350 #else
351 #  define SANITIZER_POINTER_FORMAT_LENGTH FIRST_32_SECOND_64(8, 12)
352 #endif
353 
354 /// \macro MSC_PREREQ
355 /// \brief Is the compiler MSVC of at least the specified version?
356 /// The common \param version values to check for are:
357 ///  * 1800: Microsoft Visual Studio 2013 / 12.0
358 ///  * 1900: Microsoft Visual Studio 2015 / 14.0
359 #ifdef _MSC_VER
360 #  define MSC_PREREQ(version) (_MSC_VER >= (version))
361 #else
362 #  define MSC_PREREQ(version) 0
363 #endif
364 
365 #if SANITIZER_APPLE && defined(__x86_64__)
366 #  define SANITIZER_NON_UNIQUE_TYPEINFO 0
367 #else
368 #  define SANITIZER_NON_UNIQUE_TYPEINFO 1
369 #endif
370 
371 // On linux, some architectures had an ABI transition from 64-bit long double
372 // (ie. same as double) to 128-bit long double.  On those, glibc symbols
373 // involving long doubles come in two versions, and we need to pass the
374 // correct one to dlvsym when intercepting them.
375 #if SANITIZER_LINUX && (SANITIZER_S390 || SANITIZER_PPC32 || SANITIZER_PPC64V1)
376 #  define SANITIZER_NLDBL_VERSION "GLIBC_2.4"
377 #endif
378 
379 #if SANITIZER_GO == 0
380 #  define SANITIZER_GO 0
381 #endif
382 
383 // On PowerPC and ARM Thumb, calling pthread_exit() causes LSan to detect leaks.
384 // pthread_exit() performs unwinding that leads to dlopen'ing libgcc_s.so.
385 // dlopen mallocs "libgcc_s.so" string which confuses LSan, it fails to realize
386 // that this allocation happens in dynamic linker and should be ignored.
387 #if SANITIZER_PPC || defined(__thumb__)
388 #  define SANITIZER_SUPPRESS_LEAK_ON_PTHREAD_EXIT 1
389 #else
390 #  define SANITIZER_SUPPRESS_LEAK_ON_PTHREAD_EXIT 0
391 #endif
392 
393 #if SANITIZER_FREEBSD || SANITIZER_APPLE || SANITIZER_NETBSD || SANITIZER_SOLARIS
394 #  define SANITIZER_MADVISE_DONTNEED MADV_FREE
395 #else
396 #  define SANITIZER_MADVISE_DONTNEED MADV_DONTNEED
397 #endif
398 
399 // Older gcc have issues aligning to a constexpr, and require an integer.
400 // See https://gcc.gnu.org/bugzilla/show_bug.cgi?id=56859 among others.
401 #if defined(__powerpc__) || defined(__powerpc64__)
402 #  define SANITIZER_CACHE_LINE_SIZE 128
403 #else
404 #  define SANITIZER_CACHE_LINE_SIZE 64
405 #endif
406 
407 // Enable offline markup symbolizer for Fuchsia.
408 #if SANITIZER_FUCHSIA
409 #  define SANITIZER_SYMBOLIZER_MARKUP 1
410 #else
411 #  define SANITIZER_SYMBOLIZER_MARKUP 0
412 #endif
413 
414 // Enable ability to support sanitizer initialization that is
415 // compatible with the sanitizer library being loaded via
416 // `dlopen()`.
417 #if SANITIZER_APPLE
418 #  define SANITIZER_SUPPORTS_INIT_FOR_DLOPEN 1
419 #else
420 #  define SANITIZER_SUPPORTS_INIT_FOR_DLOPEN 0
421 #endif
422 
423 // SANITIZER_SUPPORTS_THREADLOCAL
424 // 1 - THREADLOCAL macro is supported by target
425 // 0 - THREADLOCAL macro is not supported by target
426 #ifndef __has_feature
427 // TODO: Support other compilers here
428 #  define SANITIZER_SUPPORTS_THREADLOCAL 1
429 #else
430 #  if __has_feature(tls)
431 #    define SANITIZER_SUPPORTS_THREADLOCAL 1
432 #  else
433 #    define SANITIZER_SUPPORTS_THREADLOCAL 0
434 #  endif
435 #endif
436 
437 #if defined(__thumb__) && defined(__linux__)
438 // Workaround for
439 // https://lab.llvm.org/buildbot/#/builders/clang-thumbv7-full-2stage
440 // or
441 // https://lab.llvm.org/staging/#/builders/clang-thumbv7-full-2stage
442 // It fails *rss_limit_mb_test* without meaningful errors.
443 #  define SANITIZER_START_BACKGROUND_THREAD_IN_ASAN_INTERNAL 1
444 #else
445 #  define SANITIZER_START_BACKGROUND_THREAD_IN_ASAN_INTERNAL 0
446 #endif
447 
448 #endif  // SANITIZER_PLATFORM_H
449