1*b5893f02SDimitry Andric //===--------------------- Unwind_AppleExtras.cpp -------------------------===//
2*b5893f02SDimitry Andric //
3*b5893f02SDimitry Andric //                     The LLVM Compiler Infrastructure
4*b5893f02SDimitry Andric //
5*b5893f02SDimitry Andric // This file is dual licensed under the MIT and the University of Illinois Open
6*b5893f02SDimitry Andric // Source Licenses. See LICENSE.TXT for details.
7*b5893f02SDimitry Andric //
8*b5893f02SDimitry Andric //
9*b5893f02SDimitry Andric //===----------------------------------------------------------------------===//
10*b5893f02SDimitry Andric 
11*b5893f02SDimitry Andric #include "config.h"
12*b5893f02SDimitry Andric #include "AddressSpace.hpp"
13*b5893f02SDimitry Andric #include "DwarfParser.hpp"
14*b5893f02SDimitry Andric 
15*b5893f02SDimitry Andric 
16*b5893f02SDimitry Andric // private keymgr stuff
17*b5893f02SDimitry Andric #define KEYMGR_GCC3_DW2_OBJ_LIST 302
18*b5893f02SDimitry Andric extern "C" {
19*b5893f02SDimitry Andric  extern void _keymgr_set_and_unlock_processwide_ptr(int key, void *ptr);
20*b5893f02SDimitry Andric  extern void *_keymgr_get_and_lock_processwide_ptr(int key);
21*b5893f02SDimitry Andric }
22*b5893f02SDimitry Andric 
23*b5893f02SDimitry Andric // undocumented libgcc "struct object"
24*b5893f02SDimitry Andric struct libgcc_object {
25*b5893f02SDimitry Andric   void          *start;
26*b5893f02SDimitry Andric   void          *unused1;
27*b5893f02SDimitry Andric   void          *unused2;
28*b5893f02SDimitry Andric   void          *fde;
29*b5893f02SDimitry Andric   unsigned long  encoding;
30*b5893f02SDimitry Andric   void          *fde_end;
31*b5893f02SDimitry Andric   libgcc_object *next;
32*b5893f02SDimitry Andric };
33*b5893f02SDimitry Andric 
34*b5893f02SDimitry Andric // undocumented libgcc "struct km_object_info" referenced by
35*b5893f02SDimitry Andric // KEYMGR_GCC3_DW2_OBJ_LIST
36*b5893f02SDimitry Andric struct libgcc_object_info {
37*b5893f02SDimitry Andric   libgcc_object   *seen_objects;
38*b5893f02SDimitry Andric   libgcc_object   *unseen_objects;
39*b5893f02SDimitry Andric   unsigned         spare[2];
40*b5893f02SDimitry Andric };
41*b5893f02SDimitry Andric 
42*b5893f02SDimitry Andric 
43*b5893f02SDimitry Andric // static linker symbols to prevent wrong two level namespace for _Unwind symbols
44*b5893f02SDimitry Andric #if defined(__arm__)
45*b5893f02SDimitry Andric    #define NOT_HERE_BEFORE_5_0(sym)     \
46*b5893f02SDimitry Andric        extern const char sym##_tmp30 __asm("$ld$hide$os3.0$_" #sym ); \
47*b5893f02SDimitry Andric        __attribute__((visibility("default"))) const char sym##_tmp30 = 0; \
48*b5893f02SDimitry Andric        extern const char sym##_tmp31 __asm("$ld$hide$os3.1$_" #sym ); \
49*b5893f02SDimitry Andric           __attribute__((visibility("default"))) const char sym##_tmp31 = 0; \
50*b5893f02SDimitry Andric        extern const char sym##_tmp32 __asm("$ld$hide$os3.2$_" #sym );\
51*b5893f02SDimitry Andric            __attribute__((visibility("default"))) const char sym##_tmp32 = 0; \
52*b5893f02SDimitry Andric        extern const char sym##_tmp40 __asm("$ld$hide$os4.0$_" #sym ); \
53*b5893f02SDimitry Andric           __attribute__((visibility("default"))) const char sym##_tmp40 = 0; \
54*b5893f02SDimitry Andric        extern const char sym##_tmp41 __asm("$ld$hide$os4.1$_" #sym ); \
55*b5893f02SDimitry Andric           __attribute__((visibility("default"))) const char sym##_tmp41 = 0; \
56*b5893f02SDimitry Andric        extern const char sym##_tmp42 __asm("$ld$hide$os4.2$_" #sym ); \
57*b5893f02SDimitry Andric           __attribute__((visibility("default"))) const char sym##_tmp42 = 0; \
58*b5893f02SDimitry Andric        extern const char sym##_tmp43 __asm("$ld$hide$os4.3$_" #sym ); \
59*b5893f02SDimitry Andric           __attribute__((visibility("default"))) const char sym##_tmp43 = 0;
60*b5893f02SDimitry Andric #elif defined(__arm64__)
61*b5893f02SDimitry Andric   #define NOT_HERE_BEFORE_10_6(sym)
62*b5893f02SDimitry Andric   #define NEVER_HERE(sym)
63*b5893f02SDimitry Andric #else
64*b5893f02SDimitry Andric   #define NOT_HERE_BEFORE_10_6(sym) \
65*b5893f02SDimitry Andric     extern const char sym##_tmp4 __asm("$ld$hide$os10.4$_" #sym ); \
66*b5893f02SDimitry Andric           __attribute__((visibility("default"))) const char sym##_tmp4 = 0; \
67*b5893f02SDimitry Andric     extern const char sym##_tmp5 __asm("$ld$hide$os10.5$_" #sym ); \
68*b5893f02SDimitry Andric           __attribute__((visibility("default"))) const char sym##_tmp5 = 0;
69*b5893f02SDimitry Andric   #define NEVER_HERE(sym) \
70*b5893f02SDimitry Andric     extern const char sym##_tmp4 __asm("$ld$hide$os10.4$_" #sym ); \
71*b5893f02SDimitry Andric           __attribute__((visibility("default"))) const char sym##_tmp4 = 0; \
72*b5893f02SDimitry Andric     extern const char sym##_tmp5 __asm("$ld$hide$os10.5$_" #sym ); \
73*b5893f02SDimitry Andric           __attribute__((visibility("default"))) const char sym##_tmp5 = 0; \
74*b5893f02SDimitry Andric     extern const char sym##_tmp6 __asm("$ld$hide$os10.6$_" #sym ); \
75*b5893f02SDimitry Andric           __attribute__((visibility("default"))) const char sym##_tmp6 = 0;
76*b5893f02SDimitry Andric #endif
77*b5893f02SDimitry Andric 
78*b5893f02SDimitry Andric 
79*b5893f02SDimitry Andric #if defined(_LIBUNWIND_BUILD_ZERO_COST_APIS)
80*b5893f02SDimitry Andric 
81*b5893f02SDimitry Andric //
82*b5893f02SDimitry Andric // symbols in libSystem.dylib in 10.6 and later, but are in libgcc_s.dylib in
83*b5893f02SDimitry Andric // earlier versions
84*b5893f02SDimitry Andric //
85*b5893f02SDimitry Andric NOT_HERE_BEFORE_10_6(_Unwind_DeleteException)
86*b5893f02SDimitry Andric NOT_HERE_BEFORE_10_6(_Unwind_Find_FDE)
87*b5893f02SDimitry Andric NOT_HERE_BEFORE_10_6(_Unwind_ForcedUnwind)
88*b5893f02SDimitry Andric NOT_HERE_BEFORE_10_6(_Unwind_GetGR)
89*b5893f02SDimitry Andric NOT_HERE_BEFORE_10_6(_Unwind_GetIP)
90*b5893f02SDimitry Andric NOT_HERE_BEFORE_10_6(_Unwind_GetLanguageSpecificData)
91*b5893f02SDimitry Andric NOT_HERE_BEFORE_10_6(_Unwind_GetRegionStart)
92*b5893f02SDimitry Andric NOT_HERE_BEFORE_10_6(_Unwind_RaiseException)
93*b5893f02SDimitry Andric NOT_HERE_BEFORE_10_6(_Unwind_Resume)
94*b5893f02SDimitry Andric NOT_HERE_BEFORE_10_6(_Unwind_SetGR)
95*b5893f02SDimitry Andric NOT_HERE_BEFORE_10_6(_Unwind_SetIP)
96*b5893f02SDimitry Andric NOT_HERE_BEFORE_10_6(_Unwind_Backtrace)
97*b5893f02SDimitry Andric NOT_HERE_BEFORE_10_6(_Unwind_FindEnclosingFunction)
98*b5893f02SDimitry Andric NOT_HERE_BEFORE_10_6(_Unwind_GetCFA)
99*b5893f02SDimitry Andric NOT_HERE_BEFORE_10_6(_Unwind_GetDataRelBase)
100*b5893f02SDimitry Andric NOT_HERE_BEFORE_10_6(_Unwind_GetTextRelBase)
101*b5893f02SDimitry Andric NOT_HERE_BEFORE_10_6(_Unwind_Resume_or_Rethrow)
102*b5893f02SDimitry Andric NOT_HERE_BEFORE_10_6(_Unwind_GetIPInfo)
103*b5893f02SDimitry Andric NOT_HERE_BEFORE_10_6(__register_frame)
104*b5893f02SDimitry Andric NOT_HERE_BEFORE_10_6(__deregister_frame)
105*b5893f02SDimitry Andric 
106*b5893f02SDimitry Andric //
107*b5893f02SDimitry Andric // symbols in libSystem.dylib for compatibility, but we don't want any new code
108*b5893f02SDimitry Andric // using them
109*b5893f02SDimitry Andric //
110*b5893f02SDimitry Andric NEVER_HERE(__register_frame_info_bases)
111*b5893f02SDimitry Andric NEVER_HERE(__register_frame_info)
112*b5893f02SDimitry Andric NEVER_HERE(__register_frame_info_table_bases)
113*b5893f02SDimitry Andric NEVER_HERE(__register_frame_info_table)
114*b5893f02SDimitry Andric NEVER_HERE(__register_frame_table)
115*b5893f02SDimitry Andric NEVER_HERE(__deregister_frame_info)
116*b5893f02SDimitry Andric NEVER_HERE(__deregister_frame_info_bases)
117*b5893f02SDimitry Andric 
118*b5893f02SDimitry Andric #endif // defined(_LIBUNWIND_BUILD_ZERO_COST_APIS)
119*b5893f02SDimitry Andric 
120*b5893f02SDimitry Andric 
121*b5893f02SDimitry Andric 
122*b5893f02SDimitry Andric 
123*b5893f02SDimitry Andric #if defined(_LIBUNWIND_BUILD_SJLJ_APIS)
124*b5893f02SDimitry Andric //
125*b5893f02SDimitry Andric // symbols in libSystem.dylib in iOS 5.0 and later, but are in libgcc_s.dylib in
126*b5893f02SDimitry Andric // earlier versions
127*b5893f02SDimitry Andric //
128*b5893f02SDimitry Andric NOT_HERE_BEFORE_5_0(_Unwind_GetLanguageSpecificData)
129*b5893f02SDimitry Andric NOT_HERE_BEFORE_5_0(_Unwind_GetRegionStart)
130*b5893f02SDimitry Andric NOT_HERE_BEFORE_5_0(_Unwind_GetIP)
131*b5893f02SDimitry Andric NOT_HERE_BEFORE_5_0(_Unwind_SetGR)
132*b5893f02SDimitry Andric NOT_HERE_BEFORE_5_0(_Unwind_SetIP)
133*b5893f02SDimitry Andric NOT_HERE_BEFORE_5_0(_Unwind_DeleteException)
134*b5893f02SDimitry Andric NOT_HERE_BEFORE_5_0(_Unwind_SjLj_Register)
135*b5893f02SDimitry Andric NOT_HERE_BEFORE_5_0(_Unwind_GetGR)
136*b5893f02SDimitry Andric NOT_HERE_BEFORE_5_0(_Unwind_GetIPInfo)
137*b5893f02SDimitry Andric NOT_HERE_BEFORE_5_0(_Unwind_GetCFA)
138*b5893f02SDimitry Andric NOT_HERE_BEFORE_5_0(_Unwind_SjLj_Resume)
139*b5893f02SDimitry Andric NOT_HERE_BEFORE_5_0(_Unwind_SjLj_RaiseException)
140*b5893f02SDimitry Andric NOT_HERE_BEFORE_5_0(_Unwind_SjLj_Resume_or_Rethrow)
141*b5893f02SDimitry Andric NOT_HERE_BEFORE_5_0(_Unwind_SjLj_Unregister)
142*b5893f02SDimitry Andric 
143*b5893f02SDimitry Andric #endif // defined(_LIBUNWIND_BUILD_SJLJ_APIS)
144*b5893f02SDimitry Andric 
145*b5893f02SDimitry Andric 
146*b5893f02SDimitry Andric namespace libunwind {
147*b5893f02SDimitry Andric 
148*b5893f02SDimitry Andric _LIBUNWIND_HIDDEN
checkKeyMgrRegisteredFDEs(uintptr_t pc,void * & fde)149*b5893f02SDimitry Andric bool checkKeyMgrRegisteredFDEs(uintptr_t pc, void *&fde) {
150*b5893f02SDimitry Andric #if __MAC_OS_X_VERSION_MIN_REQUIRED
151*b5893f02SDimitry Andric   // lastly check for old style keymgr registration of dynamically generated
152*b5893f02SDimitry Andric   // FDEs acquire exclusive access to libgcc_object_info
153*b5893f02SDimitry Andric   libgcc_object_info *head = (libgcc_object_info *)
154*b5893f02SDimitry Andric                 _keymgr_get_and_lock_processwide_ptr(KEYMGR_GCC3_DW2_OBJ_LIST);
155*b5893f02SDimitry Andric   if (head != NULL) {
156*b5893f02SDimitry Andric     // look at each FDE in keymgr
157*b5893f02SDimitry Andric     for (libgcc_object *ob = head->unseen_objects; ob != NULL; ob = ob->next) {
158*b5893f02SDimitry Andric       CFI_Parser<LocalAddressSpace>::FDE_Info fdeInfo;
159*b5893f02SDimitry Andric       CFI_Parser<LocalAddressSpace>::CIE_Info cieInfo;
160*b5893f02SDimitry Andric       const char *msg = CFI_Parser<LocalAddressSpace>::decodeFDE(
161*b5893f02SDimitry Andric                                       LocalAddressSpace::sThisAddressSpace,
162*b5893f02SDimitry Andric                                       (uintptr_t)ob->fde, &fdeInfo, &cieInfo);
163*b5893f02SDimitry Andric       if (msg == NULL) {
164*b5893f02SDimitry Andric         // Check if this FDE is for a function that includes the pc
165*b5893f02SDimitry Andric         if ((fdeInfo.pcStart <= pc) && (pc < fdeInfo.pcEnd)) {
166*b5893f02SDimitry Andric           fde = (void*)fdeInfo.pcStart;
167*b5893f02SDimitry Andric           _keymgr_set_and_unlock_processwide_ptr(KEYMGR_GCC3_DW2_OBJ_LIST,
168*b5893f02SDimitry Andric                                                  head);
169*b5893f02SDimitry Andric           return true;
170*b5893f02SDimitry Andric         }
171*b5893f02SDimitry Andric       }
172*b5893f02SDimitry Andric     }
173*b5893f02SDimitry Andric   }
174*b5893f02SDimitry Andric   // release libgcc_object_info
175*b5893f02SDimitry Andric   _keymgr_set_and_unlock_processwide_ptr(KEYMGR_GCC3_DW2_OBJ_LIST, head);
176*b5893f02SDimitry Andric #else
177*b5893f02SDimitry Andric   (void)pc;
178*b5893f02SDimitry Andric   (void)fde;
179*b5893f02SDimitry Andric #endif
180*b5893f02SDimitry Andric   return false;
181*b5893f02SDimitry Andric }
182*b5893f02SDimitry Andric 
183*b5893f02SDimitry Andric }
184*b5893f02SDimitry Andric 
185