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