xref: /freebsd/contrib/llvm-project/libunwind/src/assembly.h (revision 62987288060ff68c817b7056815aa9fb8ba8ecd7)
10b57cec5SDimitry Andric /* ===-- assembly.h - libUnwind assembler support macros -------------------===
20b57cec5SDimitry Andric  *
30b57cec5SDimitry Andric  * Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
40b57cec5SDimitry Andric  * See https://llvm.org/LICENSE.txt for license information.
50b57cec5SDimitry Andric  * SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
60b57cec5SDimitry Andric  *
70b57cec5SDimitry Andric  * ===----------------------------------------------------------------------===
80b57cec5SDimitry Andric  *
90b57cec5SDimitry Andric  * This file defines macros for use in libUnwind assembler source.
100b57cec5SDimitry Andric  * This file is not part of the interface of this library.
110b57cec5SDimitry Andric  *
120b57cec5SDimitry Andric  * ===----------------------------------------------------------------------===
130b57cec5SDimitry Andric  */
140b57cec5SDimitry Andric 
150b57cec5SDimitry Andric #ifndef UNWIND_ASSEMBLY_H
160b57cec5SDimitry Andric #define UNWIND_ASSEMBLY_H
170b57cec5SDimitry Andric 
18d781ede6SDimitry Andric #if defined(__linux__) && defined(__CET__)
19349cc55cSDimitry Andric #include <cet.h>
20349cc55cSDimitry Andric #define _LIBUNWIND_CET_ENDBR _CET_ENDBR
21349cc55cSDimitry Andric #else
22349cc55cSDimitry Andric #define _LIBUNWIND_CET_ENDBR
23349cc55cSDimitry Andric #endif
24349cc55cSDimitry Andric 
250b57cec5SDimitry Andric #if defined(__powerpc64__)
260b57cec5SDimitry Andric #define SEPARATOR ;
270b57cec5SDimitry Andric #define PPC64_OFFS_SRR0   0
280b57cec5SDimitry Andric #define PPC64_OFFS_CR     272
290b57cec5SDimitry Andric #define PPC64_OFFS_XER    280
300b57cec5SDimitry Andric #define PPC64_OFFS_LR     288
310b57cec5SDimitry Andric #define PPC64_OFFS_CTR    296
320b57cec5SDimitry Andric #define PPC64_OFFS_VRSAVE 304
330b57cec5SDimitry Andric #define PPC64_OFFS_FP     312
340b57cec5SDimitry Andric #define PPC64_OFFS_V      824
355ffd83dbSDimitry Andric #elif defined(__APPLE__) && defined(__aarch64__)
360b57cec5SDimitry Andric #define SEPARATOR %%
37fe6060f1SDimitry Andric #elif defined(__riscv)
38fe6060f1SDimitry Andric # define RISCV_ISIZE (__riscv_xlen / 8)
39fe6060f1SDimitry Andric # define RISCV_FOFFSET (RISCV_ISIZE * 32)
40fe6060f1SDimitry Andric # if defined(__riscv_flen)
41fe6060f1SDimitry Andric #  define RISCV_FSIZE (__riscv_flen / 8)
42fe6060f1SDimitry Andric # endif
43fe6060f1SDimitry Andric 
44fe6060f1SDimitry Andric # if __riscv_xlen == 64
45fe6060f1SDimitry Andric #  define ILOAD ld
46fe6060f1SDimitry Andric #  define ISTORE sd
47fe6060f1SDimitry Andric # elif __riscv_xlen == 32
48fe6060f1SDimitry Andric #  define ILOAD lw
49fe6060f1SDimitry Andric #  define ISTORE sw
50fe6060f1SDimitry Andric # else
51fe6060f1SDimitry Andric #  error "Unsupported __riscv_xlen"
52fe6060f1SDimitry Andric # endif
53fe6060f1SDimitry Andric 
54fe6060f1SDimitry Andric # if defined(__riscv_flen)
55fe6060f1SDimitry Andric #  if __riscv_flen == 64
56fe6060f1SDimitry Andric #   define FLOAD fld
57fe6060f1SDimitry Andric #   define FSTORE fsd
58fe6060f1SDimitry Andric #  elif __riscv_flen == 32
59fe6060f1SDimitry Andric #   define FLOAD flw
60fe6060f1SDimitry Andric #   define FSTORE fsw
61fe6060f1SDimitry Andric #  else
62fe6060f1SDimitry Andric #   error "Unsupported __riscv_flen"
63fe6060f1SDimitry Andric #  endif
64fe6060f1SDimitry Andric # endif
65fe6060f1SDimitry Andric # define SEPARATOR ;
660b57cec5SDimitry Andric #else
670b57cec5SDimitry Andric #define SEPARATOR ;
680b57cec5SDimitry Andric #endif
690b57cec5SDimitry Andric 
7081ad6265SDimitry Andric #if defined(__powerpc64__) && (!defined(_CALL_ELF) || _CALL_ELF == 1) &&       \
7181ad6265SDimitry Andric     !defined(_AIX)
720b57cec5SDimitry Andric #define PPC64_OPD1 .section .opd,"aw",@progbits SEPARATOR
730b57cec5SDimitry Andric #define PPC64_OPD2 SEPARATOR \
740b57cec5SDimitry Andric   .p2align 3 SEPARATOR \
750b57cec5SDimitry Andric   .quad .Lfunc_begin0 SEPARATOR \
760b57cec5SDimitry Andric   .quad .TOC.@tocbase SEPARATOR \
770b57cec5SDimitry Andric   .quad 0 SEPARATOR \
780b57cec5SDimitry Andric   .text SEPARATOR \
790b57cec5SDimitry Andric .Lfunc_begin0:
800b57cec5SDimitry Andric #else
810b57cec5SDimitry Andric #define PPC64_OPD1
820b57cec5SDimitry Andric #define PPC64_OPD2
830b57cec5SDimitry Andric #endif
840b57cec5SDimitry Andric 
85*62987288SDimitry Andric #if defined(__aarch64__)
86*62987288SDimitry Andric #if defined(__ARM_FEATURE_GCS_DEFAULT) && defined(__ARM_FEATURE_BTI_DEFAULT)
87*62987288SDimitry Andric // Set BTI, PAC, and GCS gnu property bits
88*62987288SDimitry Andric #define GNU_PROPERTY 7
89*62987288SDimitry Andric // We indirectly branch to __libunwind_Registers_arm64_jumpto from
90*62987288SDimitry Andric // __unw_phase2_resume, so we need to use bti jc.
91*62987288SDimitry Andric #define AARCH64_BTI bti jc
92*62987288SDimitry Andric #elif defined(__ARM_FEATURE_GCS_DEFAULT)
93*62987288SDimitry Andric // Set GCS gnu property bit
94*62987288SDimitry Andric #define GNU_PROPERTY 4
95*62987288SDimitry Andric #elif defined(__ARM_FEATURE_BTI_DEFAULT)
96*62987288SDimitry Andric // Set BTI and PAC gnu property bits
97*62987288SDimitry Andric #define GNU_PROPERTY 3
98*62987288SDimitry Andric #define AARCH64_BTI bti c
99*62987288SDimitry Andric #endif
100*62987288SDimitry Andric #ifdef GNU_PROPERTY
101e8d8bef9SDimitry Andric   .pushsection ".note.gnu.property", "a" SEPARATOR                             \
102e8d8bef9SDimitry Andric   .balign 8 SEPARATOR                                                          \
103e8d8bef9SDimitry Andric   .long 4 SEPARATOR                                                            \
104e8d8bef9SDimitry Andric   .long 0x10 SEPARATOR                                                         \
105e8d8bef9SDimitry Andric   .long 0x5 SEPARATOR                                                          \
106e8d8bef9SDimitry Andric   .asciz "GNU" SEPARATOR                                                       \
107e8d8bef9SDimitry Andric   .long 0xc0000000 SEPARATOR /* GNU_PROPERTY_AARCH64_FEATURE_1_AND */          \
108e8d8bef9SDimitry Andric   .long 4 SEPARATOR                                                            \
109*62987288SDimitry Andric   .long GNU_PROPERTY SEPARATOR                                                 \
110e8d8bef9SDimitry Andric   .long 0 SEPARATOR                                                            \
111e8d8bef9SDimitry Andric   .popsection SEPARATOR
112*62987288SDimitry Andric #endif
113*62987288SDimitry Andric #endif
114*62987288SDimitry Andric #if !defined(AARCH64_BTI)
115e8d8bef9SDimitry Andric #define AARCH64_BTI
116e8d8bef9SDimitry Andric #endif
117e8d8bef9SDimitry Andric 
1180eae32dcSDimitry Andric #if !defined(__aarch64__)
1190eae32dcSDimitry Andric #ifdef __ARM_FEATURE_PAC_DEFAULT
1200eae32dcSDimitry Andric   .eabi_attribute Tag_PAC_extension, 2
1210eae32dcSDimitry Andric   .eabi_attribute Tag_PACRET_use, 1
1220eae32dcSDimitry Andric #endif
1230eae32dcSDimitry Andric #ifdef __ARM_FEATURE_BTI_DEFAULT
1240eae32dcSDimitry Andric   .eabi_attribute Tag_BTI_extension, 1
1250eae32dcSDimitry Andric   .eabi_attribute Tag_BTI_use, 1
1260eae32dcSDimitry Andric #endif
1270eae32dcSDimitry Andric #endif
1280eae32dcSDimitry Andric 
1290b57cec5SDimitry Andric #define GLUE2(a, b) a ## b
1300b57cec5SDimitry Andric #define GLUE(a, b) GLUE2(a, b)
1310b57cec5SDimitry Andric #define SYMBOL_NAME(name) GLUE(__USER_LABEL_PREFIX__, name)
1320b57cec5SDimitry Andric 
1330b57cec5SDimitry Andric #if defined(__APPLE__)
1340b57cec5SDimitry Andric 
1350b57cec5SDimitry Andric #define SYMBOL_IS_FUNC(name)
1360b57cec5SDimitry Andric #define HIDDEN_SYMBOL(name) .private_extern name
137fe6060f1SDimitry Andric #if defined(_LIBUNWIND_HIDE_SYMBOLS)
138fe6060f1SDimitry Andric #define EXPORT_SYMBOL(name) HIDDEN_SYMBOL(name)
139fe6060f1SDimitry Andric #else
140fe6060f1SDimitry Andric #define EXPORT_SYMBOL(name)
141fe6060f1SDimitry Andric #endif
1420b57cec5SDimitry Andric #define WEAK_ALIAS(name, aliasname)                                            \
1430b57cec5SDimitry Andric   .globl SYMBOL_NAME(aliasname) SEPARATOR                                      \
144fe6060f1SDimitry Andric   EXPORT_SYMBOL(SYMBOL_NAME(aliasname)) SEPARATOR                              \
1450b57cec5SDimitry Andric   SYMBOL_NAME(aliasname) = SYMBOL_NAME(name)
1460b57cec5SDimitry Andric 
1470b57cec5SDimitry Andric #define NO_EXEC_STACK_DIRECTIVE
1480b57cec5SDimitry Andric 
1490b57cec5SDimitry Andric #elif defined(__ELF__)
1500b57cec5SDimitry Andric 
1510b57cec5SDimitry Andric #if defined(__arm__)
1520b57cec5SDimitry Andric #define SYMBOL_IS_FUNC(name) .type name,%function
1530b57cec5SDimitry Andric #else
1540b57cec5SDimitry Andric #define SYMBOL_IS_FUNC(name) .type name,@function
1550b57cec5SDimitry Andric #endif
1560b57cec5SDimitry Andric #define HIDDEN_SYMBOL(name) .hidden name
157fe6060f1SDimitry Andric #if defined(_LIBUNWIND_HIDE_SYMBOLS)
158fe6060f1SDimitry Andric #define EXPORT_SYMBOL(name) HIDDEN_SYMBOL(name)
159fe6060f1SDimitry Andric #else
160fe6060f1SDimitry Andric #define EXPORT_SYMBOL(name)
161fe6060f1SDimitry Andric #endif
1620b57cec5SDimitry Andric #define WEAK_SYMBOL(name) .weak name
1635ffd83dbSDimitry Andric 
1645ffd83dbSDimitry Andric #if defined(__hexagon__)
1655ffd83dbSDimitry Andric #define WEAK_ALIAS(name, aliasname)                                            \
166fe6060f1SDimitry Andric   EXPORT_SYMBOL(SYMBOL_NAME(aliasname)) SEPARATOR                              \
167fe6060f1SDimitry Andric   WEAK_SYMBOL(SYMBOL_NAME(aliasname)) SEPARATOR                                \
1685ffd83dbSDimitry Andric   .equiv SYMBOL_NAME(aliasname), SYMBOL_NAME(name)
1695ffd83dbSDimitry Andric #else
1700b57cec5SDimitry Andric #define WEAK_ALIAS(name, aliasname)                                            \
171fe6060f1SDimitry Andric   EXPORT_SYMBOL(SYMBOL_NAME(aliasname)) SEPARATOR                              \
172fe6060f1SDimitry Andric   WEAK_SYMBOL(SYMBOL_NAME(aliasname)) SEPARATOR                                \
1730b57cec5SDimitry Andric   SYMBOL_NAME(aliasname) = SYMBOL_NAME(name)
1745ffd83dbSDimitry Andric #endif
1750b57cec5SDimitry Andric 
1760b57cec5SDimitry Andric #if defined(__GNU__) || defined(__FreeBSD__) || defined(__Fuchsia__) || \
1770b57cec5SDimitry Andric     defined(__linux__)
1780b57cec5SDimitry Andric #define NO_EXEC_STACK_DIRECTIVE .section .note.GNU-stack,"",%progbits
1790b57cec5SDimitry Andric #else
1800b57cec5SDimitry Andric #define NO_EXEC_STACK_DIRECTIVE
1810b57cec5SDimitry Andric #endif
1820b57cec5SDimitry Andric 
1830b57cec5SDimitry Andric #elif defined(_WIN32)
1840b57cec5SDimitry Andric 
1850b57cec5SDimitry Andric #define SYMBOL_IS_FUNC(name)                                                   \
1860b57cec5SDimitry Andric   .def name SEPARATOR                                                          \
1870b57cec5SDimitry Andric     .scl 2 SEPARATOR                                                           \
1880b57cec5SDimitry Andric     .type 32 SEPARATOR                                                         \
1890b57cec5SDimitry Andric   .endef
1900b57cec5SDimitry Andric #define EXPORT_SYMBOL2(name)                                                   \
1910b57cec5SDimitry Andric   .section .drectve,"yn" SEPARATOR                                             \
1920b57cec5SDimitry Andric   .ascii "-export:", #name, "\0" SEPARATOR                                     \
1930b57cec5SDimitry Andric   .text
194fe6060f1SDimitry Andric #if defined(_LIBUNWIND_HIDE_SYMBOLS)
1950b57cec5SDimitry Andric #define EXPORT_SYMBOL(name)
1960b57cec5SDimitry Andric #else
1970b57cec5SDimitry Andric #define EXPORT_SYMBOL(name) EXPORT_SYMBOL2(name)
1980b57cec5SDimitry Andric #endif
1990b57cec5SDimitry Andric #define HIDDEN_SYMBOL(name)
2000b57cec5SDimitry Andric 
2010b57cec5SDimitry Andric #if defined(__MINGW32__)
2020b57cec5SDimitry Andric #define WEAK_ALIAS(name, aliasname)                                            \
2030b57cec5SDimitry Andric   .globl SYMBOL_NAME(aliasname) SEPARATOR                                      \
2040b57cec5SDimitry Andric   EXPORT_SYMBOL(aliasname) SEPARATOR                                           \
2050b57cec5SDimitry Andric   SYMBOL_NAME(aliasname) = SYMBOL_NAME(name)
2060b57cec5SDimitry Andric #else
2070b57cec5SDimitry Andric #define WEAK_ALIAS3(name, aliasname)                                           \
2080b57cec5SDimitry Andric   .section .drectve,"yn" SEPARATOR                                             \
2090b57cec5SDimitry Andric   .ascii "-alternatename:", #aliasname, "=", #name, "\0" SEPARATOR             \
2100b57cec5SDimitry Andric   .text
2110b57cec5SDimitry Andric #define WEAK_ALIAS2(name, aliasname)                                           \
2120b57cec5SDimitry Andric   WEAK_ALIAS3(name, aliasname)
2130b57cec5SDimitry Andric #define WEAK_ALIAS(name, aliasname)                                            \
2140b57cec5SDimitry Andric   EXPORT_SYMBOL(SYMBOL_NAME(aliasname)) SEPARATOR                              \
2150b57cec5SDimitry Andric   WEAK_ALIAS2(SYMBOL_NAME(name), SYMBOL_NAME(aliasname))
2160b57cec5SDimitry Andric #endif
2170b57cec5SDimitry Andric 
2180b57cec5SDimitry Andric #define NO_EXEC_STACK_DIRECTIVE
2190b57cec5SDimitry Andric 
2200b57cec5SDimitry Andric #elif defined(__sparc__)
2210b57cec5SDimitry Andric 
22281ad6265SDimitry Andric #elif defined(_AIX)
22381ad6265SDimitry Andric 
22481ad6265SDimitry Andric #if defined(__powerpc64__)
22581ad6265SDimitry Andric #define VBYTE_LEN 8
22681ad6265SDimitry Andric #define CSECT_ALIGN 3
22781ad6265SDimitry Andric #else
22881ad6265SDimitry Andric #define VBYTE_LEN 4
22981ad6265SDimitry Andric #define CSECT_ALIGN 2
23081ad6265SDimitry Andric #endif
23181ad6265SDimitry Andric 
23281ad6265SDimitry Andric // clang-format off
23381ad6265SDimitry Andric #define DEFINE_LIBUNWIND_FUNCTION_AND_WEAK_ALIAS(name, aliasname)              \
23481ad6265SDimitry Andric   .csect .text[PR], 2 SEPARATOR                                                \
23581ad6265SDimitry Andric   .csect .name[PR], 2 SEPARATOR                                                \
23681ad6265SDimitry Andric   .globl name[DS] SEPARATOR                                                    \
23781ad6265SDimitry Andric   .globl .name[PR] SEPARATOR                                                   \
23881ad6265SDimitry Andric   .align 4 SEPARATOR                                                           \
23981ad6265SDimitry Andric   .csect name[DS], CSECT_ALIGN SEPARATOR                                       \
24081ad6265SDimitry Andric aliasname:                                                                     \
24181ad6265SDimitry Andric   .vbyte VBYTE_LEN, .name[PR] SEPARATOR                                        \
24281ad6265SDimitry Andric   .vbyte VBYTE_LEN, TOC[TC0] SEPARATOR                                         \
24381ad6265SDimitry Andric   .vbyte VBYTE_LEN, 0 SEPARATOR                                                \
24481ad6265SDimitry Andric   .weak  aliasname SEPARATOR                                                   \
24581ad6265SDimitry Andric   .weak  .aliasname SEPARATOR                                                  \
24681ad6265SDimitry Andric   .csect .name[PR], 2 SEPARATOR                                                \
24781ad6265SDimitry Andric .aliasname:                                                                    \
24881ad6265SDimitry Andric 
24981ad6265SDimitry Andric #define WEAK_ALIAS(name, aliasname)
25081ad6265SDimitry Andric #define NO_EXEC_STACK_DIRECTIVE
25181ad6265SDimitry Andric 
25281ad6265SDimitry Andric // clang-format on
2530b57cec5SDimitry Andric #else
2540b57cec5SDimitry Andric 
2550b57cec5SDimitry Andric #error Unsupported target
2560b57cec5SDimitry Andric 
2570b57cec5SDimitry Andric #endif
2580b57cec5SDimitry Andric 
25981ad6265SDimitry Andric #if defined(_AIX)
26081ad6265SDimitry Andric   // clang-format off
26181ad6265SDimitry Andric #define DEFINE_LIBUNWIND_FUNCTION(name)                                        \
26281ad6265SDimitry Andric   .globl name[DS] SEPARATOR                                                    \
26381ad6265SDimitry Andric   .globl .name SEPARATOR                                                       \
26481ad6265SDimitry Andric   .align 4 SEPARATOR                                                           \
26581ad6265SDimitry Andric   .csect name[DS], CSECT_ALIGN SEPARATOR                                       \
26681ad6265SDimitry Andric   .vbyte VBYTE_LEN, .name SEPARATOR                                            \
26781ad6265SDimitry Andric   .vbyte VBYTE_LEN, TOC[TC0] SEPARATOR                                         \
26881ad6265SDimitry Andric   .vbyte VBYTE_LEN, 0 SEPARATOR                                                \
26981ad6265SDimitry Andric   .csect .text[PR], 2 SEPARATOR                                                \
27081ad6265SDimitry Andric .name:
27181ad6265SDimitry Andric   // clang-format on
27281ad6265SDimitry Andric #else
2730b57cec5SDimitry Andric #define DEFINE_LIBUNWIND_FUNCTION(name)                                        \
2740b57cec5SDimitry Andric   .globl SYMBOL_NAME(name) SEPARATOR                                           \
2750b57cec5SDimitry Andric   HIDDEN_SYMBOL(SYMBOL_NAME(name)) SEPARATOR                                   \
2760b57cec5SDimitry Andric   SYMBOL_IS_FUNC(SYMBOL_NAME(name)) SEPARATOR                                  \
2770b57cec5SDimitry Andric   PPC64_OPD1                                                                   \
2780b57cec5SDimitry Andric   SYMBOL_NAME(name):                                                           \
279e8d8bef9SDimitry Andric   PPC64_OPD2                                                                   \
280e8d8bef9SDimitry Andric   AARCH64_BTI
28181ad6265SDimitry Andric #endif
2820b57cec5SDimitry Andric 
2830b57cec5SDimitry Andric #if defined(__arm__)
2840b57cec5SDimitry Andric #if !defined(__ARM_ARCH)
2850b57cec5SDimitry Andric #define __ARM_ARCH 4
2860b57cec5SDimitry Andric #endif
2870b57cec5SDimitry Andric 
2880b57cec5SDimitry Andric #if defined(__ARM_ARCH_4T__) || __ARM_ARCH >= 5
2890b57cec5SDimitry Andric #define ARM_HAS_BX
2900b57cec5SDimitry Andric #endif
2910b57cec5SDimitry Andric 
2920b57cec5SDimitry Andric #ifdef ARM_HAS_BX
2930b57cec5SDimitry Andric #define JMP(r) bx r
2940b57cec5SDimitry Andric #else
2950b57cec5SDimitry Andric #define JMP(r) mov pc, r
2960b57cec5SDimitry Andric #endif
2970b57cec5SDimitry Andric #endif /* __arm__ */
2980b57cec5SDimitry Andric 
2991fd87a68SDimitry Andric #if defined(__powerpc__)
300fe6060f1SDimitry Andric #define PPC_LEFT_SHIFT(index) << (index)
301fe6060f1SDimitry Andric #endif
302fe6060f1SDimitry Andric 
3030b57cec5SDimitry Andric #endif /* UNWIND_ASSEMBLY_H */
304