Home | History | Annotate | Download | only in include
      1 /* libunwind - a platform-independent unwind library
      2    Copyright (C) 2008 CodeSourcery
      3 
      4 This file is part of libunwind.
      5 
      6 Permission is hereby granted, free of charge, to any person obtaining
      7 a copy of this software and associated documentation files (the
      8 "Software"), to deal in the Software without restriction, including
      9 without limitation the rights to use, copy, modify, merge, publish,
     10 distribute, sublicense, and/or sell copies of the Software, and to
     11 permit persons to whom the Software is furnished to do so, subject to
     12 the following conditions:
     13 
     14 The above copyright notice and this permission notice shall be
     15 included in all copies or substantial portions of the Software.
     16 
     17 THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
     18 EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
     19 MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
     20 NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
     21 LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
     22 OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
     23 WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.  */
     24 
     25 #ifndef LIBUNWIND_H
     26 #define LIBUNWIND_H
     27 
     28 #if defined(__cplusplus) || defined(c_plusplus)
     29 extern "C" {
     30 #endif
     31 
     32 #include <inttypes.h>
     33 #include <stddef.h>
     34 
     35 #define UNW_TARGET	arm
     36 #define UNW_TARGET_ARM	1
     37 
     38 #define _U_TDEP_QP_TRUE	0	/* see libunwind-dynamic.h  */
     39 
     40 /* This needs to be big enough to accommodate "struct cursor", while
     41    leaving some slack for future expansion.  Changing this value will
     42    require recompiling all users of this library.  Stack allocation is
     43    relatively cheap and unwind-state copying is relatively rare, so we
     44    want to err on making it rather too big than too small.  */
     45 
     46 /* FIXME for ARM. Too big?  What do other things use for similar tasks?  */
     47 #define UNW_TDEP_CURSOR_LEN	4096
     48 
     49 typedef uint32_t unw_word_t;
     50 typedef int32_t unw_sword_t;
     51 
     52 typedef long double unw_tdep_fpreg_t;
     53 
     54 typedef enum
     55   {
     56     UNW_ARM_R0,
     57     UNW_ARM_R1,
     58     UNW_ARM_R2,
     59     UNW_ARM_R3,
     60     UNW_ARM_R4,
     61     UNW_ARM_R5,
     62     UNW_ARM_R6,
     63     UNW_ARM_R7,
     64     UNW_ARM_R8,
     65     UNW_ARM_R9,
     66     UNW_ARM_R10,
     67     UNW_ARM_R11,
     68     UNW_ARM_R12,
     69     UNW_ARM_R13,
     70     UNW_ARM_R14,
     71     UNW_ARM_R15,
     72 
     73     /* VFPv2 s0-s31 (obsolescent numberings).  */
     74     UNW_ARM_S0 = 64,
     75     UNW_ARM_S1,
     76     UNW_ARM_S2,
     77     UNW_ARM_S3,
     78     UNW_ARM_S4,
     79     UNW_ARM_S5,
     80     UNW_ARM_S6,
     81     UNW_ARM_S7,
     82     UNW_ARM_S8,
     83     UNW_ARM_S9,
     84     UNW_ARM_S10,
     85     UNW_ARM_S11,
     86     UNW_ARM_S12,
     87     UNW_ARM_S13,
     88     UNW_ARM_S14,
     89     UNW_ARM_S15,
     90     UNW_ARM_S16,
     91     UNW_ARM_S17,
     92     UNW_ARM_S18,
     93     UNW_ARM_S19,
     94     UNW_ARM_S20,
     95     UNW_ARM_S21,
     96     UNW_ARM_S22,
     97     UNW_ARM_S23,
     98     UNW_ARM_S24,
     99     UNW_ARM_S25,
    100     UNW_ARM_S26,
    101     UNW_ARM_S27,
    102     UNW_ARM_S28,
    103     UNW_ARM_S29,
    104     UNW_ARM_S30,
    105     UNW_ARM_S31,
    106 
    107     /* FPA register numberings.  */
    108     UNW_ARM_F0 = 96,
    109     UNW_ARM_F1,
    110     UNW_ARM_F2,
    111     UNW_ARM_F3,
    112     UNW_ARM_F4,
    113     UNW_ARM_F5,
    114     UNW_ARM_F6,
    115     UNW_ARM_F7,
    116 
    117     /* iWMMXt GR register numberings.  */
    118     UNW_ARM_wCGR0 = 104,
    119     UNW_ARM_wCGR1,
    120     UNW_ARM_wCGR2,
    121     UNW_ARM_wCGR3,
    122     UNW_ARM_wCGR4,
    123     UNW_ARM_wCGR5,
    124     UNW_ARM_wCGR6,
    125     UNW_ARM_wCGR7,
    126 
    127     /* iWMMXt register numberings.  */
    128     UNW_ARM_wR0 = 112,
    129     UNW_ARM_wR1,
    130     UNW_ARM_wR2,
    131     UNW_ARM_wR3,
    132     UNW_ARM_wR4,
    133     UNW_ARM_wR5,
    134     UNW_ARM_wR6,
    135     UNW_ARM_wR7,
    136     UNW_ARM_wR8,
    137     UNW_ARM_wR9,
    138     UNW_ARM_wR10,
    139     UNW_ARM_wR11,
    140     UNW_ARM_wR12,
    141     UNW_ARM_wR13,
    142     UNW_ARM_wR14,
    143     UNW_ARM_wR15,
    144 
    145     /* Two-byte encodings from here on.  */
    146 
    147     /* SPSR.  */
    148     UNW_ARM_SPSR = 128,
    149     UNW_ARM_SPSR_FIQ,
    150     UNW_ARM_SPSR_IRQ,
    151     UNW_ARM_SPSR_ABT,
    152     UNW_ARM_SPSR_UND,
    153     UNW_ARM_SPSR_SVC,
    154 
    155     /* User mode registers.  */
    156     UNW_ARM_R8_USR = 144,
    157     UNW_ARM_R9_USR,
    158     UNW_ARM_R10_USR,
    159     UNW_ARM_R11_USR,
    160     UNW_ARM_R12_USR,
    161     UNW_ARM_R13_USR,
    162     UNW_ARM_R14_USR,
    163 
    164     /* FIQ registers.  */
    165     UNW_ARM_R8_FIQ = 151,
    166     UNW_ARM_R9_FIQ,
    167     UNW_ARM_R10_FIQ,
    168     UNW_ARM_R11_FIQ,
    169     UNW_ARM_R12_FIQ,
    170     UNW_ARM_R13_FIQ,
    171     UNW_ARM_R14_FIQ,
    172 
    173     /* IRQ registers.  */
    174     UNW_ARM_R13_IRQ = 158,
    175     UNW_ARM_R14_IRQ,
    176 
    177     /* ABT registers.  */
    178     UNW_ARM_R13_ABT = 160,
    179     UNW_ARM_R14_ABT,
    180 
    181     /* UND registers.  */
    182     UNW_ARM_R13_UND = 162,
    183     UNW_ARM_R14_UND,
    184 
    185     /* SVC registers.  */
    186     UNW_ARM_R13_SVC = 164,
    187     UNW_ARM_R14_SVC,
    188 
    189     /* iWMMXt control registers.  */
    190     UNW_ARM_wC0 = 192,
    191     UNW_ARM_wC1,
    192     UNW_ARM_wC2,
    193     UNW_ARM_wC3,
    194     UNW_ARM_wC4,
    195     UNW_ARM_wC5,
    196     UNW_ARM_wC6,
    197     UNW_ARM_wC7,
    198 
    199     /* VFPv3/Neon 64-bit registers.  */
    200     UNW_ARM_D0 = 256,
    201     UNW_ARM_D1,
    202     UNW_ARM_D2,
    203     UNW_ARM_D3,
    204     UNW_ARM_D4,
    205     UNW_ARM_D5,
    206     UNW_ARM_D6,
    207     UNW_ARM_D7,
    208     UNW_ARM_D8,
    209     UNW_ARM_D9,
    210     UNW_ARM_D10,
    211     UNW_ARM_D11,
    212     UNW_ARM_D12,
    213     UNW_ARM_D13,
    214     UNW_ARM_D14,
    215     UNW_ARM_D15,
    216     UNW_ARM_D16,
    217     UNW_ARM_D17,
    218     UNW_ARM_D18,
    219     UNW_ARM_D19,
    220     UNW_ARM_D20,
    221     UNW_ARM_D21,
    222     UNW_ARM_D22,
    223     UNW_ARM_D23,
    224     UNW_ARM_D24,
    225     UNW_ARM_D25,
    226     UNW_ARM_D26,
    227     UNW_ARM_D27,
    228     UNW_ARM_D28,
    229     UNW_ARM_D29,
    230     UNW_ARM_D30,
    231     UNW_ARM_D31,
    232 
    233     /* For ARM, the CFA is the value of SP (r13) at the call site in the
    234        previous frame.  */
    235     UNW_ARM_CFA,
    236 
    237     UNW_TDEP_LAST_REG = UNW_ARM_D31,
    238 
    239     UNW_TDEP_IP = UNW_ARM_R14,  /* A little white lie.  */
    240     UNW_TDEP_SP = UNW_ARM_R13,
    241     UNW_TDEP_EH = UNW_ARM_R0   /* FIXME.  */
    242   }
    243 arm_regnum_t;
    244 
    245 #define UNW_TDEP_NUM_EH_REGS	2	/* FIXME for ARM.  */
    246 
    247 typedef struct unw_tdep_save_loc
    248   {
    249     /* Additional target-dependent info on a save location.  */
    250   }
    251 unw_tdep_save_loc_t;
    252 
    253 /* On ARM, we define our own unw_tdep_context instead of using ucontext_t.
    254    This allows us to support systems that don't support getcontext and
    255    therefore do not define ucontext_t.  */
    256 typedef struct unw_tdep_context
    257   {
    258     unsigned long regs[16];
    259   }
    260 unw_tdep_context_t;
    261 
    262 /* There is no getcontext() on ARM.  Use a stub version which only saves GP
    263    registers.  FIXME: Not ideal, may not be sufficient for all libunwind
    264    use cases.  Stores pc+8, which is only approximately correct, really.  */
    265 #ifndef __thumb__
    266 #define unw_tdep_getcontext(uc) (({					\
    267   unw_tdep_context_t *unw_ctx = (uc);					\
    268   register unsigned long *unw_base asm ("r0") = unw_ctx->regs;		\
    269   __asm__ __volatile__ (						\
    270     "stmia %[base], {r0-r15}"						\
    271     : : [base] "r" (unw_base) : "memory");				\
    272   }), 0)
    273 #else /* __thumb__ */
    274 #define unw_tdep_getcontext(uc) (({					\
    275   unw_tdep_context_t *unw_ctx = (uc);					\
    276   register unsigned long *unw_base asm ("r0") = unw_ctx->regs;		\
    277   __asm__ __volatile__ (						\
    278     ".align 2\nbx pc\nnop\n.code 32\n"					\
    279     "stmia %[base], {r0-r15}\n"						\
    280     "orr %[base], pc, #1\nbx %[base]"					\
    281     : [base] "+r" (unw_base) : : "memory", "cc");			\
    282   }), 0)
    283 #endif
    284 
    285 #include "libunwind-dynamic.h"
    286 
    287 typedef struct
    288   {
    289     /* no arm-specific auxiliary proc-info */
    290   }
    291 unw_tdep_proc_info_t;
    292 
    293 #include "libunwind-common.h"
    294 
    295 #define unw_tdep_is_fpreg		UNW_ARCH_OBJ(is_fpreg)
    296 extern int unw_tdep_is_fpreg (int);
    297 
    298 #if defined(__cplusplus) || defined(c_plusplus)
    299 }
    300 #endif
    301 
    302 #endif /* LIBUNWIND_H */
    303