1 //===----------------------------------------------------------------------===//
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
8 // Defines macros used within libunwind project.
10 //===----------------------------------------------------------------------===//
13 #ifndef LIBUNWIND_CONFIG_H
14 #define LIBUNWIND_CONFIG_H
21 #include <__libunwind_config.h>
23 // Platform specific configuration defines.
26 #define _LIBUNWIND_SUPPORT_COMPACT_UNWIND 1
28 #define _LIBUNWIND_SUPPORT_COMPACT_UNWIND 1
29 #define _LIBUNWIND_SUPPORT_DWARF_UNWIND 1
33 #define _LIBUNWIND_SUPPORT_SEH_UNWIND 1
35 #define _LIBUNWIND_SUPPORT_DWARF_UNWIND 1
37 #elif defined(_LIBUNWIND_IS_BAREMETAL)
38 #if !defined(_LIBUNWIND_ARM_EHABI)
39 #define _LIBUNWIND_SUPPORT_DWARF_UNWIND 1
40 #define _LIBUNWIND_SUPPORT_DWARF_INDEX 1
42 #elif defined(__BIONIC__) && defined(_LIBUNWIND_ARM_EHABI)
43 // For ARM EHABI, Bionic didn't implement dl_iterate_phdr until API 21. After
44 // API 21, dl_iterate_phdr exists, but dl_unwind_find_exidx is much faster.
45 #define _LIBUNWIND_USE_DL_UNWIND_FIND_EXIDX 1
47 // The traceback table at the end of each function is used for unwinding.
48 #define _LIBUNWIND_SUPPORT_TBTAB_UNWIND 1
50 // Assume an ELF system with a dl_iterate_phdr function.
51 #define _LIBUNWIND_USE_DL_ITERATE_PHDR 1
52 #if !defined(_LIBUNWIND_ARM_EHABI)
53 #define _LIBUNWIND_SUPPORT_DWARF_UNWIND 1
54 #define _LIBUNWIND_SUPPORT_DWARF_INDEX 1
58 #if defined(_LIBUNWIND_HIDE_SYMBOLS)
59 // The CMake file passes -fvisibility=hidden to control ELF/Mach-O visibility.
60 #define _LIBUNWIND_EXPORT
61 #define _LIBUNWIND_HIDDEN
63 #if !defined(__ELF__) && !defined(__MACH__) && !defined(_AIX)
64 #define _LIBUNWIND_EXPORT __declspec(dllexport)
65 #define _LIBUNWIND_HIDDEN
67 #define _LIBUNWIND_EXPORT __attribute__((visibility("default")))
68 #define _LIBUNWIND_HIDDEN __attribute__((visibility("hidden")))
73 #define XSTR(a) STR(a)
74 #define SYMBOL_NAME(name) XSTR(__USER_LABEL_PREFIX__) #name
76 #if defined(__APPLE__)
77 #if defined(_LIBUNWIND_HIDE_SYMBOLS)
78 #define _LIBUNWIND_ALIAS_VISIBILITY(name) __asm__(".private_extern " name);
80 #define _LIBUNWIND_ALIAS_VISIBILITY(name)
82 #define _LIBUNWIND_WEAK_ALIAS(name, aliasname) \
83 __asm__(".globl " SYMBOL_NAME(aliasname)); \
84 __asm__(SYMBOL_NAME(aliasname) " = " SYMBOL_NAME(name)); \
85 _LIBUNWIND_ALIAS_VISIBILITY(SYMBOL_NAME(aliasname))
86 #elif defined(__ELF__) || defined(_AIX)
87 #define _LIBUNWIND_WEAK_ALIAS(name, aliasname) \
88 extern "C" _LIBUNWIND_EXPORT __typeof(name) aliasname \
89 __attribute__((weak, alias(#name)));
91 #if defined(__MINGW32__)
92 #define _LIBUNWIND_WEAK_ALIAS(name, aliasname) \
93 extern "C" _LIBUNWIND_EXPORT __typeof(name) aliasname \
94 __attribute__((alias(#name)));
96 #define _LIBUNWIND_WEAK_ALIAS(name, aliasname) \
97 __pragma(comment(linker, "/alternatename:" SYMBOL_NAME(aliasname) "=" \
99 extern "C" _LIBUNWIND_EXPORT __typeof(name) aliasname;
102 #error Unsupported target
105 // Apple/armv7k defaults to DWARF/Compact unwinding, but its libunwind also
106 // needs to include the SJLJ APIs.
107 #if (defined(__APPLE__) && defined(__arm__)) || defined(__USING_SJLJ_EXCEPTIONS__)
108 #define _LIBUNWIND_BUILD_SJLJ_APIS
111 #if defined(__i386__) || defined(__x86_64__) || defined(__powerpc__)
112 #define _LIBUNWIND_SUPPORT_FRAME_APIS
115 #if defined(__i386__) || defined(__x86_64__) || defined(__powerpc__) || \
116 (!defined(__APPLE__) && defined(__arm__)) || defined(__aarch64__) || \
117 defined(__mips__) || defined(__riscv) || defined(__hexagon__) || \
118 defined(__sparc__) || defined(__s390x__) || defined(__loongarch__)
119 #if !defined(_LIBUNWIND_BUILD_SJLJ_APIS)
120 #define _LIBUNWIND_BUILD_ZERO_COST_APIS
124 #ifndef _LIBUNWIND_REMEMBER_HEAP_ALLOC
125 #if defined(_LIBUNWIND_REMEMBER_STACK_ALLOC) || defined(__APPLE__) || \
126 defined(__linux__) || defined(__ANDROID__) || defined(__MINGW32__) || \
127 defined(_LIBUNWIND_IS_BAREMETAL)
128 #define _LIBUNWIND_REMEMBER_ALLOC(_size) alloca(_size)
129 #define _LIBUNWIND_REMEMBER_FREE(_ptr) \
132 #elif defined(_WIN32)
133 #define _LIBUNWIND_REMEMBER_ALLOC(_size) _malloca(_size)
134 #define _LIBUNWIND_REMEMBER_FREE(_ptr) _freea(_ptr)
135 #define _LIBUNWIND_REMEMBER_CLEANUP_NEEDED
137 #define _LIBUNWIND_REMEMBER_ALLOC(_size) malloc(_size)
138 #define _LIBUNWIND_REMEMBER_FREE(_ptr) free(_ptr)
139 #define _LIBUNWIND_REMEMBER_CLEANUP_NEEDED
141 #else /* _LIBUNWIND_REMEMBER_HEAP_ALLOC */
142 #define _LIBUNWIND_REMEMBER_ALLOC(_size) malloc(_size)
143 #define _LIBUNWIND_REMEMBER_FREE(_ptr) free(_ptr)
144 #define _LIBUNWIND_REMEMBER_CLEANUP_NEEDED
147 #if defined(NDEBUG) && defined(_LIBUNWIND_IS_BAREMETAL)
148 #define _LIBUNWIND_ABORT(msg) \
153 #define _LIBUNWIND_ABORT(msg) \
155 fprintf(stderr, "libunwind: %s - %s\n", __func__, msg); \
161 #if defined(NDEBUG) && defined(_LIBUNWIND_IS_BAREMETAL)
162 #define _LIBUNWIND_LOG0(msg)
163 #define _LIBUNWIND_LOG(msg, ...)
165 #define _LIBUNWIND_LOG0(msg) do { \
166 fprintf(stderr, "libunwind: " msg "\n"); \
169 #define _LIBUNWIND_LOG(msg, ...) do { \
170 fprintf(stderr, "libunwind: " msg "\n", __VA_ARGS__); \
176 #define _LIBUNWIND_LOG_IF_FALSE(x) x
178 #define _LIBUNWIND_LOG_IF_FALSE(x) \
182 _LIBUNWIND_LOG("" #x " failed in %s", __FUNCTION__); \
186 // Macros that define away in non-Debug builds
188 #define _LIBUNWIND_DEBUG_LOG(msg, ...)
189 #define _LIBUNWIND_TRACE_API(msg, ...)
190 #define _LIBUNWIND_TRACING_UNWINDING (0)
191 #define _LIBUNWIND_TRACING_DWARF (0)
192 #define _LIBUNWIND_TRACE_UNWINDING(msg, ...)
193 #define _LIBUNWIND_TRACE_DWARF(...)
198 extern bool logAPIs(void);
199 extern bool logUnwinding(void);
200 extern bool logDWARF(void);
204 #define _LIBUNWIND_DEBUG_LOG(msg, ...) _LIBUNWIND_LOG(msg, __VA_ARGS__)
205 #define _LIBUNWIND_TRACE_API(msg, ...) \
208 _LIBUNWIND_LOG(msg, __VA_ARGS__); \
210 #define _LIBUNWIND_TRACING_UNWINDING logUnwinding()
211 #define _LIBUNWIND_TRACING_DWARF logDWARF()
212 #define _LIBUNWIND_TRACE_UNWINDING(msg, ...) \
214 if (logUnwinding()) \
215 _LIBUNWIND_LOG(msg, __VA_ARGS__); \
217 #define _LIBUNWIND_TRACE_DWARF(...) \
220 fprintf(stderr, __VA_ARGS__); \
225 // Used to fit UnwindCursor and Registers_xxx types against unw_context_t /
226 // unw_cursor_t sized memory blocks.
227 #if defined(_LIBUNWIND_IS_NATIVE_ONLY)
232 template <typename _Type
, typename _Mem
>
234 template <typename T
>
236 static const size_t count
=
237 (sizeof(T
) + sizeof(uint64_t) - 1) / sizeof(uint64_t);
239 static const bool does_fit
=
240 (blk_count
<_Type
>::count COMP_OP blk_count
<_Mem
>::count
);
243 #endif // __cplusplus
245 #endif // LIBUNWIND_CONFIG_H