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