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 180b57cec5SDimitry Andric #if defined(__powerpc64__) 190b57cec5SDimitry Andric #define SEPARATOR ; 200b57cec5SDimitry Andric #define PPC64_OFFS_SRR0 0 210b57cec5SDimitry Andric #define PPC64_OFFS_CR 272 220b57cec5SDimitry Andric #define PPC64_OFFS_XER 280 230b57cec5SDimitry Andric #define PPC64_OFFS_LR 288 240b57cec5SDimitry Andric #define PPC64_OFFS_CTR 296 250b57cec5SDimitry Andric #define PPC64_OFFS_VRSAVE 304 260b57cec5SDimitry Andric #define PPC64_OFFS_FP 312 270b57cec5SDimitry Andric #define PPC64_OFFS_V 824 285ffd83dbSDimitry Andric #elif defined(__APPLE__) && defined(__aarch64__) 290b57cec5SDimitry Andric #define SEPARATOR %% 30*fe6060f1SDimitry Andric #elif defined(__riscv) 31*fe6060f1SDimitry Andric # define RISCV_ISIZE (__riscv_xlen / 8) 32*fe6060f1SDimitry Andric # define RISCV_FOFFSET (RISCV_ISIZE * 32) 33*fe6060f1SDimitry Andric # if defined(__riscv_flen) 34*fe6060f1SDimitry Andric # define RISCV_FSIZE (__riscv_flen / 8) 35*fe6060f1SDimitry Andric # endif 36*fe6060f1SDimitry Andric 37*fe6060f1SDimitry Andric # if __riscv_xlen == 64 38*fe6060f1SDimitry Andric # define ILOAD ld 39*fe6060f1SDimitry Andric # define ISTORE sd 40*fe6060f1SDimitry Andric # elif __riscv_xlen == 32 41*fe6060f1SDimitry Andric # define ILOAD lw 42*fe6060f1SDimitry Andric # define ISTORE sw 43*fe6060f1SDimitry Andric # else 44*fe6060f1SDimitry Andric # error "Unsupported __riscv_xlen" 45*fe6060f1SDimitry Andric # endif 46*fe6060f1SDimitry Andric 47*fe6060f1SDimitry Andric # if defined(__riscv_flen) 48*fe6060f1SDimitry Andric # if __riscv_flen == 64 49*fe6060f1SDimitry Andric # define FLOAD fld 50*fe6060f1SDimitry Andric # define FSTORE fsd 51*fe6060f1SDimitry Andric # elif __riscv_flen == 32 52*fe6060f1SDimitry Andric # define FLOAD flw 53*fe6060f1SDimitry Andric # define FSTORE fsw 54*fe6060f1SDimitry Andric # else 55*fe6060f1SDimitry Andric # error "Unsupported __riscv_flen" 56*fe6060f1SDimitry Andric # endif 57*fe6060f1SDimitry Andric # endif 58*fe6060f1SDimitry Andric # define SEPARATOR ; 590b57cec5SDimitry Andric #else 600b57cec5SDimitry Andric #define SEPARATOR ; 610b57cec5SDimitry Andric #endif 620b57cec5SDimitry Andric 630b57cec5SDimitry Andric #if defined(__powerpc64__) && (!defined(_CALL_ELF) || _CALL_ELF == 1) 640b57cec5SDimitry Andric #define PPC64_OPD1 .section .opd,"aw",@progbits SEPARATOR 650b57cec5SDimitry Andric #define PPC64_OPD2 SEPARATOR \ 660b57cec5SDimitry Andric .p2align 3 SEPARATOR \ 670b57cec5SDimitry Andric .quad .Lfunc_begin0 SEPARATOR \ 680b57cec5SDimitry Andric .quad .TOC.@tocbase SEPARATOR \ 690b57cec5SDimitry Andric .quad 0 SEPARATOR \ 700b57cec5SDimitry Andric .text SEPARATOR \ 710b57cec5SDimitry Andric .Lfunc_begin0: 720b57cec5SDimitry Andric #else 730b57cec5SDimitry Andric #define PPC64_OPD1 740b57cec5SDimitry Andric #define PPC64_OPD2 750b57cec5SDimitry Andric #endif 760b57cec5SDimitry Andric 77e8d8bef9SDimitry Andric #if defined(__ARM_FEATURE_BTI_DEFAULT) 78e8d8bef9SDimitry Andric .pushsection ".note.gnu.property", "a" SEPARATOR \ 79e8d8bef9SDimitry Andric .balign 8 SEPARATOR \ 80e8d8bef9SDimitry Andric .long 4 SEPARATOR \ 81e8d8bef9SDimitry Andric .long 0x10 SEPARATOR \ 82e8d8bef9SDimitry Andric .long 0x5 SEPARATOR \ 83e8d8bef9SDimitry Andric .asciz "GNU" SEPARATOR \ 84e8d8bef9SDimitry Andric .long 0xc0000000 SEPARATOR /* GNU_PROPERTY_AARCH64_FEATURE_1_AND */ \ 85e8d8bef9SDimitry Andric .long 4 SEPARATOR \ 86e8d8bef9SDimitry Andric .long 3 SEPARATOR /* GNU_PROPERTY_AARCH64_FEATURE_1_BTI AND */ \ 87e8d8bef9SDimitry Andric /* GNU_PROPERTY_AARCH64_FEATURE_1_PAC */ \ 88e8d8bef9SDimitry Andric .long 0 SEPARATOR \ 89e8d8bef9SDimitry Andric .popsection SEPARATOR 90e8d8bef9SDimitry Andric #define AARCH64_BTI bti c 91e8d8bef9SDimitry Andric #else 92e8d8bef9SDimitry Andric #define AARCH64_BTI 93e8d8bef9SDimitry Andric #endif 94e8d8bef9SDimitry Andric 950b57cec5SDimitry Andric #define GLUE2(a, b) a ## b 960b57cec5SDimitry Andric #define GLUE(a, b) GLUE2(a, b) 970b57cec5SDimitry Andric #define SYMBOL_NAME(name) GLUE(__USER_LABEL_PREFIX__, name) 980b57cec5SDimitry Andric 990b57cec5SDimitry Andric #if defined(__APPLE__) 1000b57cec5SDimitry Andric 1010b57cec5SDimitry Andric #define SYMBOL_IS_FUNC(name) 1020b57cec5SDimitry Andric #define HIDDEN_SYMBOL(name) .private_extern name 103*fe6060f1SDimitry Andric #if defined(_LIBUNWIND_HIDE_SYMBOLS) 104*fe6060f1SDimitry Andric #define EXPORT_SYMBOL(name) HIDDEN_SYMBOL(name) 105*fe6060f1SDimitry Andric #else 106*fe6060f1SDimitry Andric #define EXPORT_SYMBOL(name) 107*fe6060f1SDimitry Andric #endif 1080b57cec5SDimitry Andric #define WEAK_ALIAS(name, aliasname) \ 1090b57cec5SDimitry Andric .globl SYMBOL_NAME(aliasname) SEPARATOR \ 110*fe6060f1SDimitry Andric EXPORT_SYMBOL(SYMBOL_NAME(aliasname)) SEPARATOR \ 1110b57cec5SDimitry Andric SYMBOL_NAME(aliasname) = SYMBOL_NAME(name) 1120b57cec5SDimitry Andric 1130b57cec5SDimitry Andric #define NO_EXEC_STACK_DIRECTIVE 1140b57cec5SDimitry Andric 1150b57cec5SDimitry Andric #elif defined(__ELF__) 1160b57cec5SDimitry Andric 1170b57cec5SDimitry Andric #if defined(__arm__) 1180b57cec5SDimitry Andric #define SYMBOL_IS_FUNC(name) .type name,%function 1190b57cec5SDimitry Andric #else 1200b57cec5SDimitry Andric #define SYMBOL_IS_FUNC(name) .type name,@function 1210b57cec5SDimitry Andric #endif 1220b57cec5SDimitry Andric #define HIDDEN_SYMBOL(name) .hidden name 123*fe6060f1SDimitry Andric #if defined(_LIBUNWIND_HIDE_SYMBOLS) 124*fe6060f1SDimitry Andric #define EXPORT_SYMBOL(name) HIDDEN_SYMBOL(name) 125*fe6060f1SDimitry Andric #else 126*fe6060f1SDimitry Andric #define EXPORT_SYMBOL(name) 127*fe6060f1SDimitry Andric #endif 1280b57cec5SDimitry Andric #define WEAK_SYMBOL(name) .weak name 1295ffd83dbSDimitry Andric 1305ffd83dbSDimitry Andric #if defined(__hexagon__) 1315ffd83dbSDimitry Andric #define WEAK_ALIAS(name, aliasname) \ 132*fe6060f1SDimitry Andric EXPORT_SYMBOL(SYMBOL_NAME(aliasname)) SEPARATOR \ 133*fe6060f1SDimitry Andric WEAK_SYMBOL(SYMBOL_NAME(aliasname)) SEPARATOR \ 1345ffd83dbSDimitry Andric .equiv SYMBOL_NAME(aliasname), SYMBOL_NAME(name) 1355ffd83dbSDimitry Andric #else 1360b57cec5SDimitry Andric #define WEAK_ALIAS(name, aliasname) \ 137*fe6060f1SDimitry Andric EXPORT_SYMBOL(SYMBOL_NAME(aliasname)) SEPARATOR \ 138*fe6060f1SDimitry Andric WEAK_SYMBOL(SYMBOL_NAME(aliasname)) SEPARATOR \ 1390b57cec5SDimitry Andric SYMBOL_NAME(aliasname) = SYMBOL_NAME(name) 1405ffd83dbSDimitry Andric #endif 1410b57cec5SDimitry Andric 1420b57cec5SDimitry Andric #if defined(__GNU__) || defined(__FreeBSD__) || defined(__Fuchsia__) || \ 1430b57cec5SDimitry Andric defined(__linux__) 1440b57cec5SDimitry Andric #define NO_EXEC_STACK_DIRECTIVE .section .note.GNU-stack,"",%progbits 1450b57cec5SDimitry Andric #else 1460b57cec5SDimitry Andric #define NO_EXEC_STACK_DIRECTIVE 1470b57cec5SDimitry Andric #endif 1480b57cec5SDimitry Andric 1490b57cec5SDimitry Andric #elif defined(_WIN32) 1500b57cec5SDimitry Andric 1510b57cec5SDimitry Andric #define SYMBOL_IS_FUNC(name) \ 1520b57cec5SDimitry Andric .def name SEPARATOR \ 1530b57cec5SDimitry Andric .scl 2 SEPARATOR \ 1540b57cec5SDimitry Andric .type 32 SEPARATOR \ 1550b57cec5SDimitry Andric .endef 1560b57cec5SDimitry Andric #define EXPORT_SYMBOL2(name) \ 1570b57cec5SDimitry Andric .section .drectve,"yn" SEPARATOR \ 1580b57cec5SDimitry Andric .ascii "-export:", #name, "\0" SEPARATOR \ 1590b57cec5SDimitry Andric .text 160*fe6060f1SDimitry Andric #if defined(_LIBUNWIND_HIDE_SYMBOLS) 1610b57cec5SDimitry Andric #define EXPORT_SYMBOL(name) 1620b57cec5SDimitry Andric #else 1630b57cec5SDimitry Andric #define EXPORT_SYMBOL(name) EXPORT_SYMBOL2(name) 1640b57cec5SDimitry Andric #endif 1650b57cec5SDimitry Andric #define HIDDEN_SYMBOL(name) 1660b57cec5SDimitry Andric 1670b57cec5SDimitry Andric #if defined(__MINGW32__) 1680b57cec5SDimitry Andric #define WEAK_ALIAS(name, aliasname) \ 1690b57cec5SDimitry Andric .globl SYMBOL_NAME(aliasname) SEPARATOR \ 1700b57cec5SDimitry Andric EXPORT_SYMBOL(aliasname) SEPARATOR \ 1710b57cec5SDimitry Andric SYMBOL_NAME(aliasname) = SYMBOL_NAME(name) 1720b57cec5SDimitry Andric #else 1730b57cec5SDimitry Andric #define WEAK_ALIAS3(name, aliasname) \ 1740b57cec5SDimitry Andric .section .drectve,"yn" SEPARATOR \ 1750b57cec5SDimitry Andric .ascii "-alternatename:", #aliasname, "=", #name, "\0" SEPARATOR \ 1760b57cec5SDimitry Andric .text 1770b57cec5SDimitry Andric #define WEAK_ALIAS2(name, aliasname) \ 1780b57cec5SDimitry Andric WEAK_ALIAS3(name, aliasname) 1790b57cec5SDimitry Andric #define WEAK_ALIAS(name, aliasname) \ 1800b57cec5SDimitry Andric EXPORT_SYMBOL(SYMBOL_NAME(aliasname)) SEPARATOR \ 1810b57cec5SDimitry Andric WEAK_ALIAS2(SYMBOL_NAME(name), SYMBOL_NAME(aliasname)) 1820b57cec5SDimitry Andric #endif 1830b57cec5SDimitry Andric 1840b57cec5SDimitry Andric #define NO_EXEC_STACK_DIRECTIVE 1850b57cec5SDimitry Andric 1860b57cec5SDimitry Andric #elif defined(__sparc__) 1870b57cec5SDimitry Andric 1880b57cec5SDimitry Andric #else 1890b57cec5SDimitry Andric 1900b57cec5SDimitry Andric #error Unsupported target 1910b57cec5SDimitry Andric 1920b57cec5SDimitry Andric #endif 1930b57cec5SDimitry Andric 1940b57cec5SDimitry Andric #define DEFINE_LIBUNWIND_FUNCTION(name) \ 1950b57cec5SDimitry Andric .globl SYMBOL_NAME(name) SEPARATOR \ 1960b57cec5SDimitry Andric HIDDEN_SYMBOL(SYMBOL_NAME(name)) SEPARATOR \ 1970b57cec5SDimitry Andric SYMBOL_IS_FUNC(SYMBOL_NAME(name)) SEPARATOR \ 1980b57cec5SDimitry Andric PPC64_OPD1 \ 1990b57cec5SDimitry Andric SYMBOL_NAME(name): \ 200e8d8bef9SDimitry Andric PPC64_OPD2 \ 201e8d8bef9SDimitry Andric AARCH64_BTI 2020b57cec5SDimitry Andric 2030b57cec5SDimitry Andric #if defined(__arm__) 2040b57cec5SDimitry Andric #if !defined(__ARM_ARCH) 2050b57cec5SDimitry Andric #define __ARM_ARCH 4 2060b57cec5SDimitry Andric #endif 2070b57cec5SDimitry Andric 2080b57cec5SDimitry Andric #if defined(__ARM_ARCH_4T__) || __ARM_ARCH >= 5 2090b57cec5SDimitry Andric #define ARM_HAS_BX 2100b57cec5SDimitry Andric #endif 2110b57cec5SDimitry Andric 2120b57cec5SDimitry Andric #ifdef ARM_HAS_BX 2130b57cec5SDimitry Andric #define JMP(r) bx r 2140b57cec5SDimitry Andric #else 2150b57cec5SDimitry Andric #define JMP(r) mov pc, r 2160b57cec5SDimitry Andric #endif 2170b57cec5SDimitry Andric #endif /* __arm__ */ 2180b57cec5SDimitry Andric 219*fe6060f1SDimitry Andric #if defined(__ppc__) || defined(__powerpc64__) 220*fe6060f1SDimitry Andric #define PPC_LEFT_SHIFT(index) << (index) 221*fe6060f1SDimitry Andric #endif 222*fe6060f1SDimitry Andric 2230b57cec5SDimitry Andric #endif /* UNWIND_ASSEMBLY_H */ 224