• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /* libunwind - a platform-independent unwind library
2    Copyright (C) 2002 Hewlett-Packard Co
3    Copyright (C) 2007 David Mosberger-Tang
4         Contributed by David Mosberger-Tang <dmosberger@gmail.com>
5 
6    Modified for x86_64 by Max Asbock <masbock@us.ibm.com>
7 
8 This file is part of libunwind.
9 
10 Permission is hereby granted, free of charge, to any person obtaining
11 a copy of this software and associated documentation files (the
12 "Software"), to deal in the Software without restriction, including
13 without limitation the rights to use, copy, modify, merge, publish,
14 distribute, sublicense, and/or sell copies of the Software, and to
15 permit persons to whom the Software is furnished to do so, subject to
16 the following conditions:
17 
18 The above copyright notice and this permission notice shall be
19 included in all copies or substantial portions of the Software.
20 
21 THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
22 EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
23 MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
24 NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
25 LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
26 OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
27 WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.  */
28 
29 #include "libunwind_i.h"
30 #ifdef HAVE_CONFIG_H
31 #include <config.h>
32 #endif
33 
34 #include <errno.h>
35 #include <stdlib.h>
36 #include <string.h>
37 #include <sys/mman.h>
38 #if defined(HAVE_SYS_SYSCALL_H)
39 # include <sys/syscall.h>
40 #endif
41 #include <stdatomic.h>
42 
43 #include "unwind_i.h"
44 
45 #ifdef UNW_REMOTE_ONLY
46 
47 /* unw_local_addr_space is a NULL pointer in this case.  */
48 unw_addr_space_t unw_local_addr_space;
49 
50 #else /* !UNW_REMOTE_ONLY */
51 
52 static struct unw_addr_space local_addr_space;
53 
54 unw_addr_space_t unw_local_addr_space = &local_addr_space;
55 
56 static void
put_unwind_info(unw_addr_space_t as,unw_proc_info_t * proc_info,void * arg)57 put_unwind_info (unw_addr_space_t as, unw_proc_info_t *proc_info, void *arg)
58 {
59   /* it's a no-op */
60 }
61 
62 static int
get_dyn_info_list_addr(unw_addr_space_t as,unw_word_t * dyn_info_list_addr,void * arg)63 get_dyn_info_list_addr (unw_addr_space_t as, unw_word_t *dyn_info_list_addr,
64                         void *arg)
65 {
66 #ifndef UNW_LOCAL_ONLY
67 # pragma weak _U_dyn_info_list_addr
68   if (!_U_dyn_info_list_addr)
69     return -UNW_ENOINFO;
70 #endif
71   // Access the `_U_dyn_info_list` from `LOCAL_ONLY` library, i.e. libunwind.so.
72   *dyn_info_list_addr = _U_dyn_info_list_addr ();
73   return 0;
74 }
75 
76 #define PAGE_SIZE 4096
77 #define PAGE_START(a)   ((a) & ~(PAGE_SIZE-1))
78 
79 static int mem_validate_pipe[2] = {-1, -1};
80 
81 #ifdef HAVE_PIPE2
82 static inline void
do_pipe2(int pipefd[2])83 do_pipe2 (int pipefd[2])
84 {
85   int result UNUSED = pipe2 (pipefd, O_CLOEXEC | O_NONBLOCK);
86 }
87 #else
88 static inline void
set_pipe_flags(int fd)89 set_pipe_flags (int fd)
90 {
91   int fd_flags = fcntl (fd, F_GETFD, 0);
92   int status_flags = fcntl (fd, F_GETFL, 0);
93 
94   fd_flags |= FD_CLOEXEC;
95   fcntl (fd, F_SETFD, fd_flags);
96 
97   status_flags |= O_NONBLOCK;
98   fcntl (fd, F_SETFL, status_flags);
99 }
100 
101 static inline void
do_pipe2(int pipefd[2])102 do_pipe2 (int pipefd[2])
103 {
104   pipe (pipefd);
105   set_pipe_flags(pipefd[0]);
106   set_pipe_flags(pipefd[1]);
107 }
108 #endif
109 
110 static inline void
open_pipe(void)111 open_pipe (void)
112 {
113   if (mem_validate_pipe[0] != -1)
114     close (mem_validate_pipe[0]);
115   if (mem_validate_pipe[1] != -1)
116     close (mem_validate_pipe[1]);
117 
118   do_pipe2 (mem_validate_pipe);
119 }
120 
121 ALWAYS_INLINE
122 static int
write_validate(void * addr)123 write_validate (void *addr)
124 {
125   int ret = -1;
126   ssize_t bytes = 0;
127 
128   do
129     {
130       char buf;
131       bytes = read (mem_validate_pipe[0], &buf, 1);
132     }
133   while ( errno == EINTR );
134 
135   int valid_read = (bytes > 0 || errno == EAGAIN || errno == EWOULDBLOCK);
136   if (!valid_read)
137     {
138       // re-open closed pipe
139       open_pipe ();
140     }
141 
142   do
143     {
144 #ifdef HAVE_SYS_SYSCALL_H
145        /* use syscall insteadof write() so that ASAN does not complain */
146        ret = syscall (SYS_write, mem_validate_pipe[1], addr, 1);
147 #else
148 	  ret = write (mem_validate_pipe[1], addr, 1);
149 #endif
150     }
151   while ( errno == EINTR );
152 
153   return ret;
154 }
155 
156 static int (*mem_validate_func) (void *addr, size_t len);
msync_validate(void * addr,size_t len)157 static int msync_validate (void *addr, size_t len)
158 {
159   if (msync (addr, len, MS_ASYNC) != 0)
160     {
161       return -1;
162     }
163 
164   return write_validate (addr);
165 }
166 
167 #ifdef HAVE_MINCORE
mincore_validate(void * addr,size_t len)168 static int mincore_validate (void *addr, size_t len)
169 {
170   unsigned char mvec[2]; /* Unaligned access may cross page boundary */
171 
172   /* mincore could fail with EAGAIN but we conservatively return -1
173      instead of looping. */
174   if (mincore (addr, len, (unsigned char *)mvec) != 0)
175     {
176       return -1;
177     }
178 
179   return write_validate (addr);
180 }
181 #endif
182 
183 /* Initialise memory validation method. On linux kernels <2.6.21,
184    mincore() returns incorrect value for MAP_PRIVATE mappings,
185    such as stacks. If mincore() was available at compile time,
186    check if we can actually use it. If not, use msync() instead. */
187 HIDDEN void
tdep_init_mem_validate(void)188 tdep_init_mem_validate (void)
189 {
190   open_pipe ();
191 
192 #ifdef HAVE_MINCORE
193   unsigned char present = 1;
194   unw_word_t addr = PAGE_START((unw_word_t)&present);
195   unsigned char mvec[1];
196   int ret;
197   while ((ret = mincore ((void*)addr, PAGE_SIZE, (unsigned char *)mvec)) == -1 &&
198          errno == EAGAIN) {}
199   if (ret == 0)
200     {
201       Debug(1, "using mincore to validate memory\n");
202       mem_validate_func = mincore_validate;
203     }
204   else
205 #endif
206     {
207       Debug(1, "using msync to validate memory\n");
208       mem_validate_func = msync_validate;
209     }
210 }
211 
212 /* Cache of already validated addresses */
213 #define NLGA 4
214 #if defined(HAVE___CACHE_PER_THREAD) && HAVE___CACHE_PER_THREAD
215 // thread-local variant
216 static _Thread_local unw_word_t last_good_addr[NLGA];
217 static _Thread_local int lga_victim;
218 
219 static int
is_cached_valid_mem(unw_word_t addr)220 is_cached_valid_mem(unw_word_t addr)
221 {
222   int i;
223   for (i = 0; i < NLGA; i++)
224     {
225       if (addr == last_good_addr[i])
226         return 1;
227     }
228   return 0;
229 }
230 
231 static void
cache_valid_mem(unw_word_t addr)232 cache_valid_mem(unw_word_t addr)
233 {
234   int i, victim;
235   victim = lga_victim;
236   for (i = 0; i < NLGA; i++) {
237     if (last_good_addr[victim] == 0) {
238       last_good_addr[victim] = addr;
239       return;
240     }
241     victim = (victim + 1) % NLGA;
242   }
243 
244   /* All slots full. Evict the victim. */
245   last_good_addr[victim] = addr;
246   victim = (victim + 1) % NLGA;
247   lga_victim = victim;
248 }
249 
250 #else
251 // global, thread safe variant
252 static _Atomic unw_word_t last_good_addr[NLGA];
253 static _Atomic int lga_victim;
254 
255 static int
is_cached_valid_mem(unw_word_t addr)256 is_cached_valid_mem(unw_word_t addr)
257 {
258   int i;
259   for (i = 0; i < NLGA; i++)
260     {
261       if (addr == atomic_load(&last_good_addr[i]))
262         return 1;
263     }
264   return 0;
265 }
266 
267 static void
cache_valid_mem(unw_word_t addr)268 cache_valid_mem(unw_word_t addr)
269 {
270   int i, victim;
271   victim = atomic_load(&lga_victim);
272   unw_word_t zero = 0;
273   for (i = 0; i < NLGA; i++) {
274     if (atomic_compare_exchange_strong(&last_good_addr[victim], &zero, addr)) {
275       return;
276     }
277     victim = (victim + 1) % NLGA;
278   }
279 
280   /* All slots full. Evict the victim. */
281   atomic_store(&last_good_addr[victim], addr);
282   victim = (victim + 1) % NLGA;
283   atomic_store(&lga_victim, victim);
284 }
285 #endif
286 
287 static int
validate_mem(unw_word_t addr)288 validate_mem (unw_word_t addr)
289 {
290   size_t len;
291 
292   if (PAGE_START(addr + sizeof (unw_word_t) - 1) == PAGE_START(addr))
293     len = PAGE_SIZE;
294   else
295     len = PAGE_SIZE * 2;
296 
297   addr = PAGE_START(addr);
298 
299   if (addr == 0)
300     return -1;
301 
302   if (is_cached_valid_mem(addr))
303     return 0;
304 
305   if (mem_validate_func ((void *) addr, len) == -1)
306     return -1;
307 
308   cache_valid_mem(addr);
309 
310   return 0;
311 }
312 
313 static int
access_mem(unw_addr_space_t as,unw_word_t addr,unw_word_t * val,int write,void * arg)314 access_mem (unw_addr_space_t as, unw_word_t addr, unw_word_t *val, int write,
315             void *arg)
316 {
317   if (unlikely (write))
318     {
319       Debug (16, "mem[%016lx] <- %lx\n", addr, *val);
320       *(unw_word_t *) addr = *val;
321     }
322   else
323     {
324       /* validate address */
325       if (unlikely (AS_ARG_GET_VALIDATE(arg))
326           && unlikely (validate_mem (addr))) {
327         Debug (16, "mem[%016lx] -> invalid\n", addr);
328         return -1;
329       }
330       *val = *(unw_word_t *) addr;
331       Debug (16, "mem[%016lx] -> %lx\n", addr, *val);
332     }
333   return 0;
334 }
335 
336 static int
access_reg(unw_addr_space_t as,unw_regnum_t reg,unw_word_t * val,int write,void * arg)337 access_reg (unw_addr_space_t as, unw_regnum_t reg, unw_word_t *val, int write,
338             void *arg)
339 {
340   unw_word_t *addr;
341   ucontext_t *uc = AS_ARG_GET_UC_PTR(arg);
342 
343   if (unw_is_fpreg (reg))
344     goto badreg;
345 
346   if (!(addr = x86_64_r_uc_addr (uc, reg)))
347     goto badreg;
348 
349   if (write)
350     {
351       *(unw_word_t *) addr = *val;
352       Debug (12, "%s <- 0x%016lx\n", unw_regname (reg), *val);
353     }
354   else
355     {
356       *val = *(unw_word_t *) addr;
357       Debug (12, "%s -> 0x%016lx\n", unw_regname (reg), *val);
358     }
359   return 0;
360 
361  badreg:
362   Debug (1, "bad register number %u\n", reg);
363   return -UNW_EBADREG;
364 }
365 
366 static int
access_fpreg(unw_addr_space_t as,unw_regnum_t reg,unw_fpreg_t * val,int write,void * arg)367 access_fpreg (unw_addr_space_t as, unw_regnum_t reg, unw_fpreg_t *val,
368               int write, void *arg)
369 {
370   ucontext_t *uc = AS_ARG_GET_UC_PTR(arg);
371   unw_fpreg_t *addr;
372 
373   if (!unw_is_fpreg (reg))
374     goto badreg;
375 
376   if (!(addr = x86_64_r_uc_addr (uc, reg)))
377     goto badreg;
378 
379   if (write)
380     {
381       Debug (12, "%s <- %08lx.%08lx.%08lx\n", unw_regname (reg),
382              ((long *)val)[0], ((long *)val)[1], ((long *)val)[2]);
383       *(unw_fpreg_t *) addr = *val;
384     }
385   else
386     {
387       *val = *(unw_fpreg_t *) addr;
388       Debug (12, "%s -> %08lx.%08lx.%08lx\n", unw_regname (reg),
389              ((long *)val)[0], ((long *)val)[1], ((long *)val)[2]);
390     }
391   return 0;
392 
393  badreg:
394   Debug (1, "bad register number %u\n", reg);
395   /* attempt to access a non-preserved register */
396   return -UNW_EBADREG;
397 }
398 
399 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)400 get_static_proc_name (unw_addr_space_t as, unw_word_t ip,
401                       char *buf, size_t buf_len, unw_word_t *offp,
402                       void *arg)
403 {
404   return _Uelf64_get_proc_name (as, getpid (), ip, buf, buf_len, offp);
405 }
406 
407 HIDDEN void
x86_64_local_addr_space_init(void)408 x86_64_local_addr_space_init (void)
409 {
410   memset (&local_addr_space, 0, sizeof (local_addr_space));
411   local_addr_space.caching_policy = UNWI_DEFAULT_CACHING_POLICY;
412   local_addr_space.acc.find_proc_info = dwarf_find_proc_info;
413   local_addr_space.acc.put_unwind_info = put_unwind_info;
414   local_addr_space.acc.get_dyn_info_list_addr = get_dyn_info_list_addr;
415   local_addr_space.acc.access_mem = access_mem;
416   local_addr_space.acc.access_reg = access_reg;
417   local_addr_space.acc.access_fpreg = access_fpreg;
418   local_addr_space.acc.resume = x86_64_local_resume;
419   local_addr_space.acc.get_proc_name = get_static_proc_name;
420   unw_flush_cache (&local_addr_space, 0, 0);
421 }
422 
423 HIDDEN void
init_local_addr_space(unw_addr_space_t as)424 init_local_addr_space (unw_addr_space_t as)
425 {
426   memset (as, 0, sizeof (struct unw_addr_space));
427   as->caching_policy = UNW_CACHE_GLOBAL;
428   as->acc.find_proc_info = dwarf_find_proc_info;
429   as->acc.put_unwind_info = put_unwind_info;
430   as->acc.get_dyn_info_list_addr = get_dyn_info_list_addr;
431   as->acc.access_mem = access_mem;
432   as->acc.access_reg = access_reg;
433   as->acc.access_fpreg = access_fpreg;
434   as->acc.resume = x86_64_local_resume;
435   as->acc.get_proc_name = get_static_proc_name;
436   unw_flush_cache (as, 0, 0);
437 }
438 
439 #endif /* !UNW_REMOTE_ONLY */
440