1 /* Header file for the ARM EABI unwinder
2    Copyright (C) 2003-2022 Free Software Foundation, Inc.
3 
4    This file is free software; you can redistribute it and/or modify it
5    under the terms of the GNU General Public License as published by the
6    Free Software Foundation; either version 2, or (at your option) any
7    later version.
8 
9    In addition to the permissions in the GNU General Public License, the
10    Free Software Foundation gives you unlimited permission to link the
11    compiled version of this file into combinations with other programs,
12    and to distribute those combinations without any restriction coming
13    from the use of this file.  (The General Public License restrictions
14    do apply in other respects; for example, they cover modification of
15    the file, and distribution when not linked into a combine
16    executable.)
17 
18    This file is distributed in the hope that it will be useful, but
19    WITHOUT ANY WARRANTY; without even the implied warranty of
20    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
21    General Public License for more details.
22 
23    You should have received a copy of the GNU General Public License
24    along with this program.  If not, see
25    <https://www.gnu.org/licenses/>.  */
26 
27 /* Language-independent unwinder header public defines.  This contains both
28    ABI defined objects, and GNU support routines.  */
29 
30 #ifndef UNWIND_ARM_H
31 #define UNWIND_ARM_H
32 
33 #define __ARM_EABI_UNWINDER__ 1
34 
35 #ifdef __cplusplus
36 extern "C" {
37 #endif
38   typedef unsigned _Unwind_Word __attribute__((__mode__(__word__)));
39   typedef signed _Unwind_Sword __attribute__((__mode__(__word__)));
40   typedef unsigned _Unwind_Ptr __attribute__((__mode__(__pointer__)));
41   typedef unsigned _Unwind_Internal_Ptr __attribute__((__mode__(__pointer__)));
42   typedef _Unwind_Word _uw;
43   typedef unsigned _uw64 __attribute__((mode(__DI__)));
44   typedef unsigned _uw16 __attribute__((mode(__HI__)));
45   typedef unsigned _uw8 __attribute__((mode(__QI__)));
46 
47   typedef enum
48     {
49       _URC_OK = 0,       /* operation completed successfully */
50       _URC_FOREIGN_EXCEPTION_CAUGHT = 1,
51       _URC_END_OF_STACK = 5,
52       _URC_HANDLER_FOUND = 6,
53       _URC_INSTALL_CONTEXT = 7,
54       _URC_CONTINUE_UNWIND = 8,
55       _URC_FAILURE = 9   /* unspecified failure of some kind */
56     }
57   _Unwind_Reason_Code;
58 
59   typedef enum
60     {
61       _US_VIRTUAL_UNWIND_FRAME = 0,
62       _US_UNWIND_FRAME_STARTING = 1,
63       _US_UNWIND_FRAME_RESUME = 2,
64       _US_ACTION_MASK = 3,
65       _US_FORCE_UNWIND = 8,
66       _US_END_OF_STACK = 16
67     }
68   _Unwind_State;
69 
70   /* Provided only for for compatibility with existing code.  */
71   typedef int _Unwind_Action;
72 #define _UA_SEARCH_PHASE	1
73 #define _UA_CLEANUP_PHASE	2
74 #define _UA_HANDLER_FRAME	4
75 #define _UA_FORCE_UNWIND	8
76 #define _UA_END_OF_STACK	16
77 #define _URC_NO_REASON 	_URC_OK
78 
79   typedef struct _Unwind_Control_Block _Unwind_Control_Block;
80   typedef struct _Unwind_Context _Unwind_Context;
81   typedef _uw _Unwind_EHT_Header;
82 
83 
84   /* UCB: */
85 
86   struct _Unwind_Control_Block
87     {
88 #ifdef _LIBC
89       /* For the benefit of code which assumes this is a scalar.  All
90 	 glibc ever does is clear it.  */
91       _uw64 exception_class;
92 #else
93       char exception_class[8];
94 #endif
95       void (*exception_cleanup)(_Unwind_Reason_Code, _Unwind_Control_Block *);
96       /* Unwinder cache, private fields for the unwinder's use */
97       struct
98 	{
99 	  _uw reserved1;  /* Forced unwind stop fn, 0 if not forced */
100 	  _uw reserved2;  /* Personality routine address */
101 	  _uw reserved3;  /* Saved callsite address */
102 	  _uw reserved4;  /* Forced unwind stop arg */
103 	  _uw reserved5;
104 	}
105       unwinder_cache;
106       /* Propagation barrier cache (valid after phase 1): */
107       struct
108 	{
109 	  _uw sp;
110 	  _uw bitpattern[5];
111 	}
112       barrier_cache;
113       /* Cleanup cache (preserved over cleanup): */
114       struct
115 	{
116 	  _uw bitpattern[4];
117 	}
118       cleanup_cache;
119       /* Pr cache (for pr's benefit): */
120       struct
121 	{
122 	  _uw fnstart;			/* function start address */
123 	  _Unwind_EHT_Header *ehtp;	/* pointer to EHT entry header word */
124 	  _uw additional;		/* additional data */
125 	  _uw reserved1;
126 	}
127       pr_cache;
128       long long int :0;	/* Force alignment to 8-byte boundary */
129     };
130 
131   /* Virtual Register Set*/
132 
133   typedef enum
134     {
135       _UVRSC_CORE = 0,      /* integer register */
136       _UVRSC_VFP = 1,       /* vfp */
137       _UVRSC_FPA = 2,       /* fpa */
138       _UVRSC_WMMXD = 3,     /* Intel WMMX data register */
139       _UVRSC_WMMXC = 4      /* Intel WMMX control register */
140     }
141   _Unwind_VRS_RegClass;
142 
143   typedef enum
144     {
145       _UVRSD_UINT32 = 0,
146       _UVRSD_VFPX = 1,
147       _UVRSD_FPAX = 2,
148       _UVRSD_UINT64 = 3,
149       _UVRSD_FLOAT = 4,
150       _UVRSD_DOUBLE = 5
151     }
152   _Unwind_VRS_DataRepresentation;
153 
154   typedef enum
155     {
156       _UVRSR_OK = 0,
157       _UVRSR_NOT_IMPLEMENTED = 1,
158       _UVRSR_FAILED = 2
159     }
160   _Unwind_VRS_Result;
161 
162   /* Frame unwinding state.  */
163   typedef struct
164     {
165       /* The current word (bytes packed msb first).  */
166       _uw data;
167       /* Pointer to the next word of data.  */
168       _uw *next;
169       /* The number of bytes left in this word.  */
170       _uw8 bytes_left;
171       /* The number of words pointed to by ptr.  */
172       _uw8 words_left;
173     }
174   __gnu_unwind_state;
175 
176   typedef _Unwind_Reason_Code (*personality_routine) (_Unwind_State,
177       _Unwind_Control_Block *, _Unwind_Context *);
178 
179   _Unwind_VRS_Result _Unwind_VRS_Set(_Unwind_Context *, _Unwind_VRS_RegClass,
180                                      _uw, _Unwind_VRS_DataRepresentation,
181                                      void *);
182 
183   _Unwind_VRS_Result _Unwind_VRS_Get(_Unwind_Context *, _Unwind_VRS_RegClass,
184                                      _uw, _Unwind_VRS_DataRepresentation,
185                                      void *);
186 
187   _Unwind_VRS_Result _Unwind_VRS_Pop(_Unwind_Context *, _Unwind_VRS_RegClass,
188                                      _uw, _Unwind_VRS_DataRepresentation);
189 
190 
191   /* Support functions for the PR.  */
192 #define _Unwind_Exception _Unwind_Control_Block
193   typedef char _Unwind_Exception_Class[8];
194 
195   void * _Unwind_GetLanguageSpecificData (_Unwind_Context *);
196   _Unwind_Ptr _Unwind_GetRegionStart (_Unwind_Context *);
197 
198   /* These two should never be used.  */
199   _Unwind_Ptr _Unwind_GetDataRelBase (_Unwind_Context *);
200   _Unwind_Ptr _Unwind_GetTextRelBase (_Unwind_Context *);
201 
202   /* Interface functions: */
203   _Unwind_Reason_Code _Unwind_RaiseException(_Unwind_Control_Block *ucbp);
204   void __attribute__((noreturn)) _Unwind_Resume(_Unwind_Control_Block *ucbp);
205   _Unwind_Reason_Code _Unwind_Resume_or_Rethrow (_Unwind_Control_Block *ucbp);
206 
207   typedef _Unwind_Reason_Code (*_Unwind_Stop_Fn)
208        (int, _Unwind_Action, _Unwind_Exception_Class,
209 	_Unwind_Control_Block *, struct _Unwind_Context *, void *);
210   _Unwind_Reason_Code _Unwind_ForcedUnwind (_Unwind_Control_Block *,
211 					    _Unwind_Stop_Fn, void *);
212   _Unwind_Word _Unwind_GetCFA (struct _Unwind_Context *);
213   void _Unwind_Complete(_Unwind_Control_Block *ucbp);
214   void _Unwind_DeleteException (_Unwind_Exception *);
215 
216   _Unwind_Reason_Code __gnu_unwind_frame (_Unwind_Control_Block *,
217 					  _Unwind_Context *);
218   _Unwind_Reason_Code __gnu_unwind_execute (_Unwind_Context *,
219 					    __gnu_unwind_state *);
220 
221   /* Decode an R_ARM_TARGET2 relocation.  */
222   static inline _Unwind_Word
_Unwind_decode_target2(_Unwind_Word ptr)223   _Unwind_decode_target2 (_Unwind_Word ptr)
224     {
225       _Unwind_Word tmp;
226 
227       tmp = *(_Unwind_Word *) ptr;
228       /* Zero values are always NULL.  */
229       if (!tmp)
230 	return 0;
231 
232 #if defined(linux) || defined(__NetBSD__)
233       /* Pc-relative indirect.  */
234       tmp += ptr;
235       tmp = *(_Unwind_Word *) tmp;
236 #elif defined(__symbian__)
237       /* Absolute pointer.  Nothing more to do.  */
238 #else
239       /* Pc-relative pointer.  */
240       tmp += ptr;
241 #endif
242       return tmp;
243     }
244 
245   static inline _Unwind_Word
_Unwind_GetGR(_Unwind_Context * context,int regno)246   _Unwind_GetGR (_Unwind_Context *context, int regno)
247     {
248       _uw val;
249       _Unwind_VRS_Get (context, _UVRSC_CORE, regno, _UVRSD_UINT32, &val);
250       return val;
251     }
252 
253   /* Return the address of the instruction, not the actual IP value.  */
254 #define _Unwind_GetIP(context) \
255   (_Unwind_GetGR (context, 15) & ~(_Unwind_Word)1)
256 
257   static inline void
_Unwind_SetGR(_Unwind_Context * context,int regno,_Unwind_Word val)258   _Unwind_SetGR (_Unwind_Context *context, int regno, _Unwind_Word val)
259     {
260       _Unwind_VRS_Set (context, _UVRSC_CORE, regno, _UVRSD_UINT32, &val);
261     }
262 
263   /* The dwarf unwinder doesn't understand arm/thumb state.  We assume the
264      landing pad uses the same instruction set as the call site.  */
265 #define _Unwind_SetIP(context, val) \
266   _Unwind_SetGR (context, 15, val | (_Unwind_GetGR (context, 15) & 1))
267 
268 typedef _Unwind_Reason_Code (*_Unwind_Trace_Fn)
269      (struct _Unwind_Context *, void *);
270 
271 extern _Unwind_Reason_Code _Unwind_Backtrace (_Unwind_Trace_Fn, void *);
272 
273 #ifdef __cplusplus
274 }   /* extern "C" */
275 #endif
276 
277 #endif /* defined UNWIND_ARM_H */
278