1 /* Test custom provided Dwfl_Thread_Callbacks vector.
2 Copyright (C) 2013 Red Hat, Inc.
3 This file is part of elfutils.
4
5 This file is free software; you can redistribute it and/or modify
6 it under the terms of the GNU General Public License as published by
7 the Free Software Foundation; either version 3 of the License, or
8 (at your option) any later version.
9
10 elfutils is distributed in the hope that it will be useful, but
11 WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 GNU General Public License for more details.
14
15 You should have received a copy of the GNU General Public License
16 along with this program. If not, see <http://www.gnu.org/licenses/>. */
17
18 /* Test custom provided Dwfl_Thread_Callbacks vector. Test mimics what
19 a ptrace based vector would do. */
20
21 #include <config.h>
22 #include <assert.h>
23 #include <inttypes.h>
24 #include <stdio.h>
25 #include <stdio_ext.h>
26 #include <locale.h>
27 #include <dirent.h>
28 #include <stdlib.h>
29 #include <errno.h>
30 #include <unistd.h>
31 #include <dwarf.h>
32 #if defined(__x86_64__) && defined(__linux__)
33 #include <sys/resource.h>
34 #include <sys/ptrace.h>
35 #include <signal.h>
36 #include <sys/types.h>
37 #include <sys/wait.h>
38 #include <sys/user.h>
39 #include <fcntl.h>
40 #include <string.h>
41 #include ELFUTILS_HEADER(dwfl)
42 #endif
43 #include "system.h"
44
45 #if !defined(__x86_64__) || !defined(__linux__)
46
47 int
main(int argc,char ** argv)48 main (int argc __attribute__ ((unused)), char **argv)
49 {
50 fprintf (stderr, "%s: x86_64 linux only test\n",
51 argv[0]);
52 return 77;
53 }
54
55 #else /* __x86_64__ && __linux__ */
56
57 /* The only arch specific code is set_initial_registers. */
58
59 static int
find_elf(Dwfl_Module * mod,void ** userdata,const char * modname,Dwarf_Addr base,char ** file_name,Elf ** elfp)60 find_elf (Dwfl_Module *mod __attribute__ ((unused)),
61 void **userdata __attribute__ ((unused)),
62 const char *modname __attribute__ ((unused)),
63 Dwarf_Addr base __attribute__ ((unused)),
64 char **file_name __attribute__ ((unused)),
65 Elf **elfp __attribute__ ((unused)))
66 {
67 /* Not used as modules are reported explicitly. */
68 assert (0);
69 }
70
71 static bool
memory_read(Dwfl * dwfl,Dwarf_Addr addr,Dwarf_Word * result,void * dwfl_arg)72 memory_read (Dwfl *dwfl, Dwarf_Addr addr, Dwarf_Word *result,
73 void *dwfl_arg __attribute__ ((unused)))
74 {
75 pid_t child = dwfl_pid (dwfl);
76
77 errno = 0;
78 long l = ptrace (PTRACE_PEEKDATA, child, (void *) (uintptr_t) addr, NULL);
79
80 // The unwinder can ask for an invalid address.
81 // Don't assert on that but just politely refuse.
82 if (errno != 0) {
83 errno = 0;
84 return false;
85 }
86 *result = l;
87
88 return true;
89 }
90
91 /* Return filename and VMA address *BASEP where its mapping starts which
92 contains ADDR. */
93
94 static char *
maps_lookup(pid_t pid,Dwarf_Addr addr,GElf_Addr * basep)95 maps_lookup (pid_t pid, Dwarf_Addr addr, GElf_Addr *basep)
96 {
97 char *fname;
98 int i = asprintf (&fname, "/proc/%ld/maps", (long) pid);
99 assert (i > 0);
100 FILE *f = fopen (fname, "r");
101 assert (f);
102 free (fname);
103 for (;;)
104 {
105 // 37e3c22000-37e3c23000 rw-p 00022000 00:11 49532 /lib64/ld-2.14.90.so */
106 unsigned long start, end, offset;
107 i = fscanf (f, "%lx-%lx %*s %lx %*x:%*x %*u", &start, &end, &offset);
108 if (i != 3)
109 break;
110 char *filename = strdup ("");
111 assert (filename);
112 size_t filename_len = 0;
113 for (;;)
114 {
115 int c = fgetc (f);
116 assert (c != EOF);
117 if (c == '\n')
118 break;
119 if (c == ' ' && *filename == '\0')
120 continue;
121 filename = realloc (filename, filename_len + 2);
122 assert (filename);
123 filename[filename_len++] = c;
124 filename[filename_len] = '\0';
125 }
126 if (start <= addr && addr < end)
127 {
128 i = fclose (f);
129 assert (i == 0);
130
131 *basep = start - offset;
132 return filename;
133 }
134 free (filename);
135 }
136 *basep = 0;
137 return NULL;
138 }
139
140 /* Add module containing ADDR to the DWFL address space.
141
142 dwfl_report_elf call here violates Dwfl manipulation as one should call
143 dwfl_report only between dwfl_report_begin_add and dwfl_report_end.
144 Current elfutils implementation does not mind as dwfl_report_begin_add is
145 empty. */
146
147 static Dwfl_Module *
report_module(Dwfl * dwfl,pid_t child,Dwarf_Addr addr)148 report_module (Dwfl *dwfl, pid_t child, Dwarf_Addr addr)
149 {
150 GElf_Addr base;
151 char *long_name = maps_lookup (child, addr, &base);
152 if (!long_name)
153 return NULL; // not found
154 Dwfl_Module *mod = dwfl_report_elf (dwfl, long_name, long_name, -1,
155 base, false /* add_p_vaddr */);
156 assert (mod);
157 free (long_name);
158 assert (dwfl_addrmodule (dwfl, addr) == mod);
159 return mod;
160 }
161
162 static pid_t
next_thread(Dwfl * dwfl,void * dwfl_arg,void ** thread_argp)163 next_thread (Dwfl *dwfl, void *dwfl_arg __attribute__ ((unused)),
164 void **thread_argp)
165 {
166 if (*thread_argp != NULL)
167 return 0;
168 /* Put arbitrary non-NULL value into *THREAD_ARGP as a marker so that this
169 function returns non-zero PID only once. */
170 *thread_argp = thread_argp;
171 return dwfl_pid (dwfl);
172 }
173
174 static bool
set_initial_registers(Dwfl_Thread * thread,void * thread_arg)175 set_initial_registers (Dwfl_Thread *thread,
176 void *thread_arg __attribute__ ((unused)))
177 {
178 pid_t child = dwfl_pid (dwfl_thread_dwfl (thread));
179
180 struct user_regs_struct user_regs;
181 long l = ptrace (PTRACE_GETREGS, child, NULL, &user_regs);
182 assert (l == 0);
183
184 Dwarf_Word dwarf_regs[17];
185 dwarf_regs[0] = user_regs.rax;
186 dwarf_regs[1] = user_regs.rdx;
187 dwarf_regs[2] = user_regs.rcx;
188 dwarf_regs[3] = user_regs.rbx;
189 dwarf_regs[4] = user_regs.rsi;
190 dwarf_regs[5] = user_regs.rdi;
191 dwarf_regs[6] = user_regs.rbp;
192 dwarf_regs[7] = user_regs.rsp;
193 dwarf_regs[8] = user_regs.r8;
194 dwarf_regs[9] = user_regs.r9;
195 dwarf_regs[10] = user_regs.r10;
196 dwarf_regs[11] = user_regs.r11;
197 dwarf_regs[12] = user_regs.r12;
198 dwarf_regs[13] = user_regs.r13;
199 dwarf_regs[14] = user_regs.r14;
200 dwarf_regs[15] = user_regs.r15;
201 dwarf_regs[16] = user_regs.rip;
202 bool ok = dwfl_thread_state_registers (thread, 0, 17, dwarf_regs);
203 assert (ok);
204
205 /* x86_64 has PC contained in its CFI subset of DWARF register set so
206 elfutils will figure out the real PC value from REGS.
207 So no need to explicitly call dwfl_thread_state_register_pc. */
208
209 return true;
210 }
211
212 static const Dwfl_Thread_Callbacks callbacks =
213 {
214 next_thread,
215 NULL, /* get_thread */
216 memory_read,
217 set_initial_registers,
218 NULL, /* detach */
219 NULL, /* thread_detach */
220 };
221
222 static int
frame_callback(Dwfl_Frame * state,void * arg)223 frame_callback (Dwfl_Frame *state, void *arg)
224 {
225 unsigned *framenop = arg;
226 Dwarf_Addr pc;
227 bool isactivation;
228 if (! dwfl_frame_pc (state, &pc, &isactivation))
229 {
230 error (1, 0, "%s", dwfl_errmsg (-1));
231 return 1;
232 }
233 Dwarf_Addr pc_adjusted = pc - (isactivation ? 0 : 1);
234
235 /* Get PC->SYMNAME. */
236 Dwfl *dwfl = dwfl_thread_dwfl (dwfl_frame_thread (state));
237 Dwfl_Module *mod = dwfl_addrmodule (dwfl, pc_adjusted);
238 if (mod == NULL)
239 mod = report_module (dwfl, dwfl_pid (dwfl), pc_adjusted);
240 const char *symname = NULL;
241 symname = dwfl_module_addrname (mod, pc_adjusted);
242
243 printf ("#%2u %#" PRIx64 "%4s\t%s\n", (*framenop)++, (uint64_t) pc,
244 ! isactivation ? "- 1" : "", symname);
245 return DWARF_CB_OK;
246 }
247
248 static int
thread_callback(Dwfl_Thread * thread,void * thread_arg)249 thread_callback (Dwfl_Thread *thread, void *thread_arg __attribute__ ((unused)))
250 {
251 unsigned frameno = 0;
252 switch (dwfl_thread_getframes (thread, frame_callback, &frameno))
253 {
254 case 0:
255 break;
256 case -1:
257 error (1, 0, "dwfl_thread_getframes: %s", dwfl_errmsg (-1));
258 break;
259 default:
260 abort ();
261 }
262 return DWARF_CB_OK;
263 }
264
265 int
main(int argc,char ** argv)266 main (int argc __attribute__ ((unused)), char **argv __attribute__ ((unused)))
267 {
268 /* We use no threads here which can interfere with handling a stream. */
269 __fsetlocking (stdin, FSETLOCKING_BYCALLER);
270 __fsetlocking (stdout, FSETLOCKING_BYCALLER);
271 __fsetlocking (stderr, FSETLOCKING_BYCALLER);
272
273 /* Set locale. */
274 (void) setlocale (LC_ALL, "");
275
276 elf_version (EV_CURRENT);
277
278 pid_t child = fork ();
279 switch (child)
280 {
281 case -1:
282 assert (0);
283 case 0:;
284 long l = ptrace (PTRACE_TRACEME, 0, NULL, NULL);
285 assert (l == 0);
286 raise (SIGUSR1);
287 return 0;
288 default:
289 break;
290 }
291
292 int status;
293 pid_t pid = waitpid (child, &status, 0);
294 assert (pid == child);
295 assert (WIFSTOPPED (status));
296 assert (WSTOPSIG (status) == SIGUSR1);
297
298 static char *debuginfo_path;
299 static const Dwfl_Callbacks offline_callbacks =
300 {
301 .find_debuginfo = dwfl_standard_find_debuginfo,
302 .debuginfo_path = &debuginfo_path,
303 .section_address = dwfl_offline_section_address,
304 .find_elf = find_elf,
305 };
306 Dwfl *dwfl = dwfl_begin (&offline_callbacks);
307 assert (dwfl);
308
309 struct user_regs_struct user_regs;
310 long l = ptrace (PTRACE_GETREGS, child, NULL, &user_regs);
311 assert (l == 0);
312 report_module (dwfl, child, user_regs.rip);
313
314 bool ok = dwfl_attach_state (dwfl, EM_NONE, child, &callbacks, NULL);
315 assert (ok);
316
317 /* Multiple threads are not handled here. */
318 int err = dwfl_getthreads (dwfl, thread_callback, NULL);
319 assert (! err);
320
321 dwfl_end (dwfl);
322 kill (child, SIGKILL);
323 pid = waitpid (child, &status, 0);
324 assert (pid == child);
325 assert (WIFSIGNALED (status));
326 assert (WTERMSIG (status) == SIGKILL);
327
328 return EXIT_SUCCESS;
329 }
330
331 #endif /* x86_64 */
332