1 //===--------------------------- libuwind.cpp -----------------------------===//
2 //
3 // The LLVM Compiler Infrastructure
4 //
5 // This file is dual licensed under the MIT and the University of Illinois Open
6 // Source Licenses. See LICENSE.TXT for details.
7 //
8 //
9 // Implements C++ ABI Exception Handling Level 1 as documented at:
10 // http://mentorembedded.github.io/cxx-abi/abi-eh.html
11 //
12 //===----------------------------------------------------------------------===//
13
14 #define _UNWIND_GCC_EXTENSIONS
15
16 #include <unwind.h>
17
18 #include "UnwindCursor.hpp"
19
20 using namespace _Unwind;
21
22 typedef CFI_Parser<LocalAddressSpace, NativeUnwindRegisters> MyCFIParser;
23
24 // Internal object representing the address space of this process.
25 static LocalAddressSpace sThisAddressSpace(MyCFIParser::findPCRange);
26
27 typedef UnwindCursor<LocalAddressSpace, NativeUnwindRegisters> ThisUnwindCursor;
28
unwind_phase1(ThisUnwindCursor & cursor,struct _Unwind_Exception * exc)29 static _Unwind_Reason_Code unwind_phase1(ThisUnwindCursor &cursor,
30 struct _Unwind_Exception *exc) {
31 cursor.setInfoBasedOnIPRegister();
32
33 // Walk frames looking for a place to stop.
34 for (;;) {
35 // Get next frame.
36 // First frame is _Unwind_RaiseException and skipped.
37 switch (cursor.step()) {
38 case UNW_STEP_END:
39 return _URC_END_OF_STACK;
40 case UNW_STEP_FAILED:
41 return _URC_FATAL_PHASE1_ERROR;
42 case UNW_STEP_SUCCESS:
43 break;
44 }
45
46 // Check if there is a personality routine for this frame.
47 unw_proc_info_t frameInfo;
48 cursor.getInfo(&frameInfo);
49 if (frameInfo.end_ip == 0)
50 return _URC_FATAL_PHASE1_ERROR;
51
52 if (frameInfo.handler == 0)
53 continue; // No personality routine, so try next frame.
54
55 __personality_routine p = (__personality_routine)(frameInfo.handler);
56 _Unwind_Reason_Code result = (*p)(1, _UA_SEARCH_PHASE, exc->exception_class,
57 exc, (struct _Unwind_Context *)(&cursor));
58
59 switch (result) {
60 case _URC_HANDLER_FOUND:
61 // This is either a catch clause or a local variable
62 // with destructor.
63 // Stop search and remember the frame for phase 2.
64 exc->private_2 = cursor.getSP();
65 return _URC_NO_REASON;
66
67 case _URC_CONTINUE_UNWIND:
68 // Continue unwinding
69 break;
70
71 default:
72 // Bad personality routine.
73 return _URC_FATAL_PHASE1_ERROR;
74 }
75 }
76 }
77
unwind_phase2(ThisUnwindCursor & cursor,struct _Unwind_Exception * exc)78 static _Unwind_Reason_Code unwind_phase2(ThisUnwindCursor &cursor,
79 struct _Unwind_Exception *exc) {
80 cursor.setInfoBasedOnIPRegister();
81
82 // Walk frames until the frame selected in phase 1 is reached.
83 for (;;) {
84 // Get next frame.
85 // First frame is _Unwind_RaiseException and skipped.
86 switch (cursor.step()) {
87 case UNW_STEP_END:
88 return _URC_END_OF_STACK;
89 case UNW_STEP_FAILED:
90 return _URC_FATAL_PHASE2_ERROR;
91 case UNW_STEP_SUCCESS:
92 break;
93 }
94
95 unw_proc_info_t frameInfo;
96 cursor.getInfo(&frameInfo);
97 if (frameInfo.end_ip == 0)
98 return _URC_FATAL_PHASE2_ERROR;
99
100 if (frameInfo.handler == 0)
101 continue; // No personality routine, continue.
102
103 uintptr_t sp = cursor.getSP();
104
105 _Unwind_Action action = _UA_CLEANUP_PHASE;
106 // If this frame was selected in phase 1,
107 // inform the personality routine.
108 if (sp == exc->private_2)
109 action = (_Unwind_Action)(action | _UA_HANDLER_FRAME);
110 __personality_routine p = (__personality_routine)(frameInfo.handler);
111 _Unwind_Reason_Code result = (*p)(1, action, exc->exception_class, exc,
112 (struct _Unwind_Context *)(&cursor));
113 switch (result) {
114 case _URC_CONTINUE_UNWIND:
115 // Continue unwinding unless the selected frame passed.
116 if (sp == exc->private_2)
117 return _URC_FATAL_PHASE2_ERROR;
118 break;
119 case _URC_INSTALL_CONTEXT:
120 // Transfer control to landing pad.
121 cursor.jumpto();
122 __builtin_unreachable();
123 default:
124 // Bad personality routine.
125 return _URC_FATAL_PHASE2_ERROR;
126 }
127 }
128 }
129
unwind_phase2_forced(ThisUnwindCursor & cursor,struct _Unwind_Exception * exc,_Unwind_Stop_Fn stop,void * stop_arg)130 static _Unwind_Reason_Code unwind_phase2_forced(ThisUnwindCursor &cursor,
131 struct _Unwind_Exception *exc,
132 _Unwind_Stop_Fn stop,
133 void *stop_arg) {
134 _Unwind_Action action;
135 cursor.setInfoBasedOnIPRegister();
136
137 // Walk frames until the frame selected in phase 1 is reached.
138 for (;;) {
139 // Get next frame.
140 // First frame is _Unwind_RaiseException and skipped.
141 switch (cursor.step()) {
142 case UNW_STEP_END:
143 case UNW_STEP_FAILED:
144 // End of stack or error condition.
145 // Call the stop function one last time.
146 action = (_Unwind_Action)(_UA_FORCE_UNWIND | _UA_CLEANUP_PHASE |
147 _UA_END_OF_STACK);
148 (*stop)(1, action, exc->exception_class, exc,
149 (struct _Unwind_Context *)(&cursor), stop_arg);
150
151 // Didn't stop at the expected frame, so return error.
152 return _URC_FATAL_PHASE2_ERROR;
153
154 case UNW_STEP_SUCCESS:
155 break;
156 }
157
158 unw_proc_info_t frameInfo;
159 cursor.getInfo(&frameInfo);
160 if (frameInfo.end_ip == 0)
161 return _URC_FATAL_PHASE2_ERROR;
162
163 // Call stop function for each frame
164 action = (_Unwind_Action)(_UA_FORCE_UNWIND | _UA_CLEANUP_PHASE);
165 _Unwind_Reason_Code result =
166 (*stop)(1, action, exc->exception_class, exc,
167 (struct _Unwind_Context *)(&cursor), stop_arg);
168 if (result != _URC_NO_REASON)
169 return _URC_FATAL_PHASE2_ERROR;
170
171 if (frameInfo.handler == 0)
172 continue; // No personality routine, continue.
173
174 __personality_routine p = (__personality_routine)(frameInfo.handler);
175 result = (*p)(1, action, exc->exception_class, exc,
176 (struct _Unwind_Context *)(&cursor));
177
178 switch (result) {
179 case _URC_CONTINUE_UNWIND:
180 // Destructors called, continue.
181 break;
182 case _URC_INSTALL_CONTEXT:
183 // Transfer control to landing pad.
184 cursor.jumpto();
185 __builtin_unreachable();
186 default:
187 // Bad personality routine.
188 return _URC_FATAL_PHASE2_ERROR;
189 }
190 }
191 }
192
_Unwind_RaiseException(struct _Unwind_Exception * exc)193 _Unwind_Reason_Code _Unwind_RaiseException(struct _Unwind_Exception *exc) {
194 NativeUnwindRegisters registers;
195 ThisUnwindCursor cursor1(registers, sThisAddressSpace);
196 ThisUnwindCursor cursor2(registers, sThisAddressSpace);
197
198 // Mark this as a non-forced unwind for _Unwind_Resume().
199 exc->private_1 = 0;
200 exc->private_2 = 0;
201
202 // Phase 1: searching.
203 _Unwind_Reason_Code phase1 = unwind_phase1(cursor1, exc);
204 if (phase1 != _URC_NO_REASON)
205 return phase1;
206
207 // Phase 2: cleaning up.
208 return unwind_phase2(cursor2, exc);
209 }
210
_Unwind_ForcedUnwind(struct _Unwind_Exception * exc,_Unwind_Stop_Fn stop,void * stop_arg)211 _Unwind_Reason_Code _Unwind_ForcedUnwind(struct _Unwind_Exception *exc,
212 _Unwind_Stop_Fn stop, void *stop_arg) {
213 NativeUnwindRegisters registers;
214 ThisUnwindCursor cursor(registers, sThisAddressSpace);
215
216 // Mark this as forced unwind for _Unwind_Resume().
217 exc->private_1 = (uintptr_t)stop;
218 exc->private_2 = (uintptr_t)stop_arg;
219
220 return unwind_phase2_forced(cursor, exc, stop, stop_arg);
221 }
222
_Unwind_Resume(struct _Unwind_Exception * exc)223 void _Unwind_Resume(struct _Unwind_Exception *exc) {
224 NativeUnwindRegisters registers;
225 ThisUnwindCursor cursor(registers, sThisAddressSpace);
226
227 if (exc->private_1 != 0)
228 unwind_phase2_forced(cursor, exc, (_Unwind_Stop_Fn)exc->private_1,
229 (void *)exc->private_2);
230 else
231 unwind_phase2(cursor, exc);
232 abort();
233 }
234
_Unwind_Resume_or_Rethrow(struct _Unwind_Exception * exc)235 _Unwind_Reason_Code _Unwind_Resume_or_Rethrow(struct _Unwind_Exception *exc) {
236 // This is a re-throw, if this is a non-forced unwind
237 // and the stopping place was found.
238 // In that case, call _Unwind_RaiseException() as if
239 // it was a new exception.
240
241 if (exc->private_1 != 0)
242 _Unwind_Resume(exc);
243
244 // This can return if there is no catch clause.
245 // In that case, __cxa_rethrow is expected to call std::terminate().
246 return _Unwind_RaiseException(exc);
247 }
248
_Unwind_DeleteException(struct _Unwind_Exception * exc)249 void _Unwind_DeleteException(struct _Unwind_Exception *exc) {
250 if (exc->exception_cleanup != NULL)
251 (*exc->exception_cleanup)(_URC_FOREIGN_EXCEPTION_CAUGHT, exc);
252 }
253
_Unwind_GetGR(struct _Unwind_Context * context,int index)254 uintptr_t _Unwind_GetGR(struct _Unwind_Context *context, int index) {
255 ThisUnwindCursor *cursor = (ThisUnwindCursor *)context;
256 return cursor->getReg(index);
257 }
258
_Unwind_SetGR(struct _Unwind_Context * context,int index,uintptr_t new_value)259 void _Unwind_SetGR(struct _Unwind_Context *context, int index,
260 uintptr_t new_value) {
261 ThisUnwindCursor *cursor = (ThisUnwindCursor *)context;
262 cursor->setReg(index, new_value);
263 }
264
_Unwind_GetIP(struct _Unwind_Context * context)265 uintptr_t _Unwind_GetIP(struct _Unwind_Context *context) {
266 ThisUnwindCursor *cursor = (ThisUnwindCursor *)context;
267 return cursor->getIP();
268 }
269
_Unwind_GetIPInfo(struct _Unwind_Context * context,int * isSignalFrame)270 uintptr_t _Unwind_GetIPInfo(struct _Unwind_Context *context, int *isSignalFrame) {
271 ThisUnwindCursor *cursor = (ThisUnwindCursor *)context;
272 *isSignalFrame = cursor->isSignalFrame() ? 1 : 0;
273 return cursor->getIP();
274 }
275
_Unwind_SetIP(struct _Unwind_Context * context,uintptr_t new_value)276 void _Unwind_SetIP(struct _Unwind_Context *context, uintptr_t new_value) {
277 ThisUnwindCursor *cursor = (ThisUnwindCursor *)context;
278 cursor->setIP(new_value);
279 unw_proc_info_t info;
280 cursor->getInfo(&info);
281 cursor->setInfoBasedOnIPRegister(false);
282 if (info.extra_args)
283 cursor->setSP(cursor->getSP() + info.extra_args);
284 }
285
_Unwind_GetRegionStart(struct _Unwind_Context * context)286 uintptr_t _Unwind_GetRegionStart(struct _Unwind_Context *context) {
287 ThisUnwindCursor *cursor = (ThisUnwindCursor *)context;
288 unw_proc_info_t frameInfo;
289 cursor->getInfo(&frameInfo);
290 return frameInfo.end_ip ? frameInfo.start_ip : 0;
291 }
292
_Unwind_GetLanguageSpecificData(struct _Unwind_Context * context)293 uintptr_t _Unwind_GetLanguageSpecificData(struct _Unwind_Context *context) {
294 ThisUnwindCursor *cursor = (ThisUnwindCursor *)context;
295 unw_proc_info_t frameInfo;
296 cursor->getInfo(&frameInfo);
297 return frameInfo.end_ip ? frameInfo.lsda : 0;
298 }
299
_Unwind_Backtrace(_Unwind_Trace_Fn callback,void * ref)300 _Unwind_Reason_Code _Unwind_Backtrace(_Unwind_Trace_Fn callback, void *ref) {
301 NativeUnwindRegisters registers;
302 ThisUnwindCursor cursor(registers, sThisAddressSpace);
303 cursor.setInfoBasedOnIPRegister();
304
305 // Walk each frame.
306 while (true) {
307
308 // Ask libuwind to get next frame (skip over first frame which is
309 // _Unwind_Backtrace()).
310 if (cursor.step() != UNW_STEP_SUCCESS)
311 return _URC_END_OF_STACK;
312
313 // Call trace function with this frame.
314 _Unwind_Reason_Code result =
315 (*callback)((struct _Unwind_Context *)(&cursor), ref);
316 if (result != _URC_NO_REASON)
317 return result;
318 }
319 }
320
_Unwind_GetCFA(struct _Unwind_Context * context)321 uintptr_t _Unwind_GetCFA(struct _Unwind_Context *context) {
322 ThisUnwindCursor *cursor = (ThisUnwindCursor *)context;
323 return cursor->getSP();
324 }
325
_Unwind_FindEnclosingFunction(void * pc)326 void *_Unwind_FindEnclosingFunction(void *pc) {
327 NativeUnwindRegisters registers;
328 ThisUnwindCursor cursor(registers, sThisAddressSpace);
329
330 unw_proc_info_t info;
331 cursor.setIP((uintptr_t)pc);
332 cursor.setInfoBasedOnIPRegister();
333
334 cursor.getInfo(&info);
335 return info.end_ip ? (void *)info.start_ip : NULL;
336 }
337
_Unwind_Find_FDE(void * pc,struct dwarf_eh_bases * bases)338 void *_Unwind_Find_FDE(void *pc, struct dwarf_eh_bases *bases) {
339 NativeUnwindRegisters registers;
340 ThisUnwindCursor cursor(registers, sThisAddressSpace);
341
342 unw_proc_info_t info;
343 cursor.setIP((uintptr_t)pc);
344 cursor.setInfoBasedOnIPRegister();
345
346 cursor.getInfo(&info);
347 if (info.end_ip == 0)
348 return NULL;
349 bases->tbase = 0; /* Not supported */
350 bases->dbase = (void *)info.data_base;
351 bases->func = (void *)info.start_ip;
352 return (void *)info.unwind_info;
353 }
354
_Unwind_GetDataRelBase(struct _Unwind_Context * context)355 uintptr_t _Unwind_GetDataRelBase(struct _Unwind_Context *context) {
356 ThisUnwindCursor *cursor = (ThisUnwindCursor *)context;
357 unw_proc_info_t frameInfo;
358 cursor->getInfo(&frameInfo);
359 return frameInfo.data_base;
360 }
361
_Unwind_GetTextRelBase(struct _Unwind_Context * context)362 uintptr_t _Unwind_GetTextRelBase(struct _Unwind_Context *context) { return 0; }
363
__register_frame(const void * fde)364 void __register_frame(const void *fde) {
365 MyCFIParser::pint_t pcStart, pcEnd;
366
367 MyCFIParser::findPCRange(sThisAddressSpace, (uintptr_t)fde, pcStart, pcEnd);
368 if (pcEnd == 0)
369 return; // Bad FDE.
370
371 sThisAddressSpace.addFDE(pcStart, pcEnd, (uintptr_t)fde);
372 }
373
__register_frame_info(const void * ehframe,void * storage)374 void __register_frame_info(const void *ehframe, void *storage) {
375 sThisAddressSpace.setLazyReload();
376 }
377
__deregister_frame(const void * fde)378 void __deregister_frame(const void *fde) {
379 MyCFIParser::pint_t pcStart, pcEnd;
380
381 MyCFIParser::findPCRange(sThisAddressSpace, (uintptr_t)fde, pcStart, pcEnd);
382 if (pcEnd == 0)
383 return; // Bad FDE.
384
385 sThisAddressSpace.removeFDE(pcStart, pcEnd, (uintptr_t)fde);
386 }
387
__deregister_frame_info(const void * ehFrameStart)388 void *__deregister_frame_info(const void *ehFrameStart) {
389 sThisAddressSpace.removeDSO((LocalAddressSpace::pint_t)ehFrameStart);
390 return NULL;
391 }
392