1 /* libunwind - a platform-independent unwind library
2    Copyright (C) 2006-2007 IBM
3    Contributed by
4      Corey Ashford <cjashfor@us.ibm.com>
5      Jose Flavio Aguilar Paulino <jflavio@br.ibm.com> <joseflavio@gmail.com>
6 
7 This file is part of libunwind.
8 
9 Permission is hereby granted, free of charge, to any person obtaining
10 a copy of this software and associated documentation files (the
11 "Software"), to deal in the Software without restriction, including
12 without limitation the rights to use, copy, modify, merge, publish,
13 distribute, sublicense, and/or sell copies of the Software, and to
14 permit persons to whom the Software is furnished to do so, subject to
15 the following conditions:
16 
17 The above copyright notice and this permission notice shall be
18 included in all copies or substantial portions of the Software.
19 
20 THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
21 EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
22 MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
23 NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
24 LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
25 OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
26 WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.  */
27 
28 #include <stdlib.h>
29 #include <string.h>
30 
31 #include "ucontext_i.h"
32 #include "unwind_i.h"
33 
34 #ifdef UNW_REMOTE_ONLY
35 
36 /* unw_local_addr_space is a NULL pointer in this case.  */
37 unw_addr_space_t unw_local_addr_space;
38 
39 #else /* !UNW_REMOTE_ONLY */
40 
41 static struct unw_addr_space local_addr_space;
42 
43 unw_addr_space_t unw_local_addr_space = &local_addr_space;
44 
45 static void *
uc_addr(ucontext_t * uc,int reg)46 uc_addr (ucontext_t *uc, int reg)
47 {
48   void *addr;
49 
50   if ((unsigned) (reg - UNW_PPC32_R0) < 32)
51     addr = &uc->uc_mcontext.uc_regs->gregs[reg - UNW_PPC32_R0];
52 
53   else
54   if ( ((unsigned) (reg - UNW_PPC32_F0) < 32) &&
55        ((unsigned) (reg - UNW_PPC32_F0) >= 0) )
56     addr = &uc->uc_mcontext.uc_regs->fpregs.fpregs[reg - UNW_PPC32_F0];
57 
58   else
59     {
60       unsigned gregs_idx;
61 
62       switch (reg)
63         {
64         case UNW_PPC32_CTR:
65           gregs_idx = CTR_IDX;
66           break;
67         case UNW_PPC32_LR:
68           gregs_idx = LINK_IDX;
69           break;
70         case UNW_PPC32_XER:
71           gregs_idx = XER_IDX;
72           break;
73         case UNW_PPC32_CCR:
74           gregs_idx = CCR_IDX;
75           break;
76         default:
77           return NULL;
78         }
79       addr = &uc->uc_mcontext.uc_regs->gregs[gregs_idx];
80     }
81   return addr;
82 }
83 
84 # ifdef UNW_LOCAL_ONLY
85 
86 HIDDEN void *
tdep_uc_addr(ucontext_t * uc,int reg)87 tdep_uc_addr (ucontext_t *uc, int reg)
88 {
89   return uc_addr (uc, reg);
90 }
91 
92 # endif /* UNW_LOCAL_ONLY */
93 
94 static void
put_unwind_info(unw_addr_space_t as,unw_proc_info_t * proc_info,void * arg)95 put_unwind_info (unw_addr_space_t as, unw_proc_info_t *proc_info, void *arg)
96 {
97   /* it's a no-op */
98 }
99 
100 static int
get_dyn_info_list_addr(unw_addr_space_t as,unw_word_t * dyn_info_list_addr,void * arg)101 get_dyn_info_list_addr (unw_addr_space_t as, unw_word_t *dyn_info_list_addr,
102                         void *arg)
103 {
104 #ifndef UNW_LOCAL_ONLY
105 # pragma weak _U_dyn_info_list_addr
106   if (!_U_dyn_info_list_addr)
107     return -UNW_ENOINFO;
108 #endif
109   // Access the `_U_dyn_info_list` from `LOCAL_ONLY` library, i.e. libunwind.so.
110   *dyn_info_list_addr = _U_dyn_info_list_addr ();
111   return 0;
112 }
113 
114 static int
access_mem(unw_addr_space_t as,unw_word_t addr,unw_word_t * val,int write,void * arg)115 access_mem (unw_addr_space_t as, unw_word_t addr, unw_word_t *val, int write,
116             void *arg)
117 {
118   if (write)
119     {
120       Debug (12, "mem[%lx] <- %lx\n", addr, *val);
121       *(unw_word_t *) addr = *val;
122     }
123   else
124     {
125       *val = *(unw_word_t *) addr;
126       Debug (12, "mem[%lx] -> %lx\n", addr, *val);
127     }
128   return 0;
129 }
130 
131 static int
access_reg(unw_addr_space_t as,unw_regnum_t reg,unw_word_t * val,int write,void * arg)132 access_reg (unw_addr_space_t as, unw_regnum_t reg, unw_word_t *val,
133             int write, void *arg)
134 {
135   unw_word_t *addr;
136   ucontext_t *uc = arg;
137 
138   if ( ((unsigned int) (reg - UNW_PPC32_F0) < 32) &&
139        ((unsigned int) (reg - UNW_PPC32_F0) >= 0))
140     goto badreg;
141 
142   addr = uc_addr (uc, reg);
143   if (!addr)
144     goto badreg;
145 
146   if (write)
147     {
148       *(unw_word_t *) addr = *val;
149       Debug (12, "%s <- %lx\n", unw_regname (reg), *val);
150     }
151   else
152     {
153       *val = *(unw_word_t *) addr;
154       Debug (12, "%s -> %lx\n", unw_regname (reg), *val);
155     }
156   return 0;
157 
158 badreg:
159   Debug (1, "bad register number %u\n", reg);
160   return -UNW_EBADREG;
161 }
162 
163 static int
access_fpreg(unw_addr_space_t as,unw_regnum_t reg,unw_fpreg_t * val,int write,void * arg)164 access_fpreg (unw_addr_space_t as, unw_regnum_t reg, unw_fpreg_t *val,
165               int write, void *arg)
166 {
167   ucontext_t *uc = arg;
168   unw_fpreg_t *addr;
169 
170   if ((unsigned) (reg - UNW_PPC32_F0) < 0)
171     goto badreg;
172 
173   addr = uc_addr (uc, reg);
174   if (!addr)
175     goto badreg;
176 
177   if (write)
178     {
179       Debug (12, "%s <- %016Lf\n", unw_regname (reg), *val);
180       *(unw_fpreg_t *) addr = *val;
181     }
182   else
183     {
184       *val = *(unw_fpreg_t *) addr;
185       Debug (12, "%s -> %016Lf\n", unw_regname (reg), *val);
186     }
187   return 0;
188 
189 badreg:
190   Debug (1, "bad register number %u\n", reg);
191   /* attempt to access a non-preserved register */
192   return -UNW_EBADREG;
193 }
194 
195 static int
get_static_proc_name(unw_addr_space_t as,unw_word_t ip,char * buf,size_t buf_len,unw_word_t * offp,void * arg)196 get_static_proc_name (unw_addr_space_t as, unw_word_t ip,
197                       char *buf, size_t buf_len, unw_word_t *offp,
198                       void *arg)
199 {
200   return _Uelf32_get_proc_name (as, getpid (), ip, buf, buf_len, offp);
201 }
202 
203 HIDDEN void
ppc32_local_addr_space_init(void)204 ppc32_local_addr_space_init (void)
205 {
206   memset (&local_addr_space, 0, sizeof (local_addr_space));
207   local_addr_space.caching_policy = UNWI_DEFAULT_CACHING_POLICY;
208   local_addr_space.acc.find_proc_info = dwarf_find_proc_info;
209   local_addr_space.acc.put_unwind_info = put_unwind_info;
210   local_addr_space.acc.get_dyn_info_list_addr = get_dyn_info_list_addr;
211   local_addr_space.acc.access_mem = access_mem;
212   local_addr_space.acc.access_reg = access_reg;
213   local_addr_space.acc.access_fpreg = access_fpreg;
214   local_addr_space.acc.resume = ppc32_local_resume;
215   local_addr_space.acc.get_proc_name = get_static_proc_name;
216   unw_flush_cache (&local_addr_space, 0, 0);
217 }
218 
219 #endif /* !UNW_REMOTE_ONLY */
220