1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
|
/* Interfaces for libdwfl.
Copyright (C) 2005-2010 Red Hat, Inc.
This file is part of elfutils.
This file is free software; you can redistribute it and/or modify
it under the terms of either
* the GNU Lesser General Public License as published by the Free
Software Foundation; either version 3 of the License, or (at
your option) any later version
or
* the GNU General Public License as published by the Free
Software Foundation; either version 2 of the License, or (at
your option) any later version
or both in parallel, as here.
elfutils is distributed in the hope that it will be useful, but
WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
General Public License for more details.
You should have received copies of the GNU General Public License and
the GNU Lesser General Public License along with this program. If
not, see <http://www.gnu.org/licenses/>. */
#ifndef _LIBDWFL_H
#define _LIBDWFL_H 1
#include "libdw.h"
#include <stdio.h>
/* Handle for a session using the library. */
typedef struct Dwfl Dwfl;
/* Handle for a module. */
typedef struct Dwfl_Module Dwfl_Module;
/* Handle describing a line record. */
typedef struct Dwfl_Line Dwfl_Line;
/* This holds everything we know about the state of the frame at a particular
PC location described by an FDE. */
typedef struct Dwfl_Frame_State Dwfl_Frame_State;
/* Callbacks. */
typedef struct
{
int (*find_elf) (Dwfl_Module *mod, void **userdata,
const char *modname, Dwarf_Addr base,
char **file_name, Elf **elfp);
int (*find_debuginfo) (Dwfl_Module *mod, void **userdata,
const char *modname, Dwarf_Addr base,
const char *file_name,
const char *debuglink_file, GElf_Word debuglink_crc,
char **debuginfo_file_name);
/* Fill *ADDR with the loaded address of the section called SECNAME in
the given module. Use (Dwarf_Addr) -1 if this section is omitted from
accessible memory. This is called exactly once for each SHF_ALLOC
section that relocations affecting DWARF data refer to, so it can
easily be used to collect state about the sections referenced. */
int (*section_address) (Dwfl_Module *mod, void **userdata,
const char *modname, Dwarf_Addr base,
const char *secname,
GElf_Word shndx, const GElf_Shdr *shdr,
Dwarf_Addr *addr);
char **debuginfo_path; /* See dwfl_standard_find_debuginfo. */
} Dwfl_Callbacks;
#ifdef __cplusplus
extern "C" {
#endif
/* Start a new session with the library. */
extern Dwfl *dwfl_begin (const Dwfl_Callbacks *callbacks)
__nonnull_attribute__ (1);
/* End a session. */
extern void dwfl_end (Dwfl *);
/* Return implementation's version string suitable for printing. */
extern const char *dwfl_version (Dwfl *);
/* Return error code of last failing function call. This value is kept
separately for each thread. */
extern int dwfl_errno (void);
/* Return error string for ERROR. If ERROR is zero, return error string
for most recent error or NULL if none occurred. If ERROR is -1 the
behaviour is similar to the last case except that not NULL but a legal
string is returned. */
extern const char *dwfl_errmsg (int err);
/* Start reporting the current set of segments and modules to the library.
All existing segments are wiped. Existing modules are marked to be
deleted, and will not be found via dwfl_addrmodule et al if they are not
re-reported before dwfl_report_end is called. */
extern void dwfl_report_begin (Dwfl *dwfl);
/* Report that segment NDX begins at PHDR->p_vaddr + BIAS.
If NDX is < 0, the value succeeding the last call's NDX
is used instead (zero on the first call).
If nonzero, the smallest PHDR->p_align value seen sets the
effective page size for the address space DWFL describes.
This is the granularity at which reported module boundary
addresses will be considered to fall in or out of a segment.
Returns -1 for errors, or NDX (or its assigned replacement) on success.
When NDX is the value succeeding the last call's NDX (or is implicitly
so as above), IDENT is nonnull and matches the value in the last call,
and the PHDR and BIAS values reflect a segment that would be contiguous,
in both memory and file, with the last segment reported, then this
segment may be coalesced internally with preceding segments. When given
an address inside this segment, dwfl_addrsegment may return the NDX of a
preceding contiguous segment. To prevent coalesced segments, always
pass a null pointer for IDENT.
The values passed are not stored (except to track coalescence).
The only information that can be extracted from DWFL later is the
mapping of an address to a segment index that starts at or below
it. Reporting segments at all is optional. Its only benefit to
the caller is to offer this quick lookup via dwfl_addrsegment,
or use other segment-based calls. */
extern int dwfl_report_segment (Dwfl *dwfl, int ndx,
const GElf_Phdr *phdr, GElf_Addr bias,
const void *ident);
/* Report that a module called NAME spans addresses [START, END).
Returns the module handle, either existing or newly allocated,
or returns a null pointer for an allocation error. */
extern Dwfl_Module *dwfl_report_module (Dwfl *dwfl, const char *name,
Dwarf_Addr start, Dwarf_Addr end);
/* Report a module with start and end addresses computed from the ELF
program headers in the given file, plus BASE. For an ET_REL file,
does a simple absolute section layout starting at BASE.
FD may be -1 to open FILE_NAME. On success, FD is consumed by the
library, and the `find_elf' callback will not be used for this module. */
extern Dwfl_Module *dwfl_report_elf (Dwfl *dwfl, const char *name,
const char *file_name, int fd,
GElf_Addr base);
/* See dwfl_report_elf except that BASE is the page-aligned absolute VMA
address where the ELF file should start. Any possible file prelinking of
the disk file is compensated. */
extern Dwfl_Module *dwfl_report_elf_baseaddr (Dwfl *dwfl, const char *name,
const char *file_name, int fd,
GElf_Addr base);
/* Similar, but report the module for offline use. All ET_EXEC files
being reported must be reported before any relocatable objects.
If this is used, dwfl_report_module and dwfl_report_elf may not be
used in the same reporting session. */
extern Dwfl_Module *dwfl_report_offline (Dwfl *dwfl, const char *name,
const char *file_name, int fd);
/* Finish reporting the current set of modules to the library.
If REMOVED is not null, it's called for each module that
existed before but was not included in the current report.
Returns a nonzero return value from the callback.
The callback may call dwfl_report_module; doing so with the
details of the module being removed prevents its removal.
DWFL cannot be used until this function has returned zero. */
extern int dwfl_report_end (Dwfl *dwfl,
int (*removed) (Dwfl_Module *, void *,
const char *, Dwarf_Addr,
void *arg),
void *arg);
/* Start reporting additional modules to the library. No calls but
dwfl_report_* can be made on DWFL until dwfl_report_end is called.
This is like dwfl_report_begin, but all the old modules are kept on.
More dwfl_report_* calls can follow to add more modules.
When dwfl_report_end is called, no old modules will be removed. */
extern void dwfl_report_begin_add (Dwfl *dwfl);
/* Return the name of the module, and for each non-null argument store
interesting details: *USERDATA is a location for storing your own
pointer, **USERDATA is initially null; *START and *END give the address
range covered by the module; *DWBIAS is the address bias for debugging
information, and *SYMBIAS for symbol table entries (either is -1 if not
yet accessed); *MAINFILE is the name of the ELF file, and *DEBUGFILE the
name of the debuginfo file (might be equal to *MAINFILE; either is null
if not yet accessed). */
extern const char *dwfl_module_info (Dwfl_Module *mod, void ***userdata,
Dwarf_Addr *start, Dwarf_Addr *end,
Dwarf_Addr *dwbias, Dwarf_Addr *symbias,
const char **mainfile,
const char **debugfile);
/* Iterate through the modules, starting the walk with OFFSET == 0.
Calls *CALLBACK for each module as long as it returns DWARF_CB_OK.
When *CALLBACK returns another value, the walk stops and the
return value can be passed as OFFSET to resume it. Returns 0 when
there are no more modules, or -1 for errors. */
extern ptrdiff_t dwfl_getmodules (Dwfl *dwfl,
int (*callback) (Dwfl_Module *, void **,
const char *, Dwarf_Addr,
void *arg),
void *arg,
ptrdiff_t offset);
/* Find the module containing the given address. */
extern Dwfl_Module *dwfl_addrmodule (Dwfl *dwfl, Dwarf_Addr address);
/* Find the segment, if any, and module, if any, containing ADDRESS.
Returns a segment index returned by dwfl_report_segment, or -1
if no segment matches the address. Regardless of the return value,
*MOD is always set to the module containing ADDRESS, or to null. */
extern int dwfl_addrsegment (Dwfl *dwfl, Dwarf_Addr address, Dwfl_Module **mod);
/* Report the known build ID bits associated with a module.
If VADDR is nonzero, it gives the absolute address where those
bits are found within the module. This can be called at any
time, but is usually used immediately after dwfl_report_module.
Once the module's main ELF file is opened, the ID note found
there takes precedence and cannot be changed. */
extern int dwfl_module_report_build_id (Dwfl_Module *mod,
const unsigned char *bits, size_t len,
GElf_Addr vaddr)
__nonnull_attribute__ (2);
/* Extract the build ID bits associated with a module.
Returns -1 for errors, 0 if no ID is known, or the number of ID bytes.
When an ID is found, *BITS points to it; *VADDR is the absolute address
at which the ID bits are found within the module, or 0 if unknown.
This returns 0 when the module's main ELF file has not yet been loaded
and its build ID bits were not reported. To ensure the ID is always
returned when determinable, call dwfl_module_getelf first. */
extern int dwfl_module_build_id (Dwfl_Module *mod,
const unsigned char **bits, GElf_Addr *vaddr)
__nonnull_attribute__ (2, 3);
/*** Standard callbacks ***/
/* These standard find_elf and find_debuginfo callbacks are
controlled by a string specifying directories to look in.
If `debuginfo_path' is set in the Dwfl_Callbacks structure
and the char * it points to is not null, that supplies the
string. Otherwise a default path is used.
If the first character of the string is + or - that enables or
disables CRC32 checksum validation when it's necessary. The
remainder of the string is composed of elements separated by
colons. Each element can start with + or - to override the
global checksum behavior. This flag is never relevant when
working with build IDs, but it's always parsed in the path
string. The remainder of the element indicates a directory.
Searches by build ID consult only the elements naming absolute
directory paths. They look under those directories for a link
named ".build-id/xx/yy" or ".build-id/xx/yy.debug", where "xxyy"
is the lower-case hexadecimal representation of the ID bytes.
In searches for debuginfo by name, if the remainder of the
element is empty, the directory containing the main file is
tried; if it's an absolute path name, the absolute directory path
containing the main file is taken as a subdirectory of this path;
a relative path name is taken as a subdirectory of the directory
containing the main file. Hence for /bin/ls, the default string
":.debug:/usr/lib/debug" says to look in /bin, then /bin/.debug,
then /usr/lib/debug/bin, for the file name in the .gnu_debuglink
section (or "ls.debug" if none was found). */
/* Standard find_elf callback function working solely on build ID.
This can be tried first by any find_elf callback, to use the
bits passed to dwfl_module_report_build_id, if any. */
extern int dwfl_build_id_find_elf (Dwfl_Module *, void **,
const char *, Dwarf_Addr,
char **, Elf **);
/* Standard find_debuginfo callback function working solely on build ID.
This can be tried first by any find_debuginfo callback,
to use the build ID bits from the main file when present. */
extern int dwfl_build_id_find_debuginfo (Dwfl_Module *, void **,
const char *, Dwarf_Addr,
const char *, const char *,
GElf_Word, char **);
/* Standard find_debuginfo callback function.
If a build ID is available, this tries first to use that.
If there is no build ID or no valid debuginfo found by ID,
it searches the debuginfo path by name, as described above.
Any file found in the path is validated by build ID if possible,
or else by CRC32 checksum if enabled, and skipped if it does not match. */
extern int dwfl_standard_find_debuginfo (Dwfl_Module *, void **,
const char *, Dwarf_Addr,
const char *, const char *,
GElf_Word, char **);
/* This callback must be used when using dwfl_offline_* to report modules,
if ET_REL is to be supported. */
extern int dwfl_offline_section_address (Dwfl_Module *, void **,
const char *, Dwarf_Addr,
const char *, GElf_Word,
const GElf_Shdr *,
Dwarf_Addr *addr);
/* Callbacks for working with kernel modules in the running Linux kernel. */
extern int dwfl_linux_kernel_find_elf (Dwfl_Module *, void **,
const char *, Dwarf_Addr,
char **, Elf **);
extern int dwfl_linux_kernel_module_section_address (Dwfl_Module *, void **,
const char *, Dwarf_Addr,
const char *, GElf_Word,
const GElf_Shdr *,
Dwarf_Addr *addr);
/* Call dwfl_report_elf for the running Linux kernel.
Returns zero on success, -1 if dwfl_report_module failed,
or an errno code if opening the kernel binary failed. */
extern int dwfl_linux_kernel_report_kernel (Dwfl *dwfl);
/* Call dwfl_report_module for each kernel module in the running Linux kernel.
Returns zero on success, -1 if dwfl_report_module failed,
or an errno code if reading the list of modules failed. */
extern int dwfl_linux_kernel_report_modules (Dwfl *dwfl);
/* Report a kernel and its modules found on disk, for offline use.
If RELEASE starts with '/', it names a directory to look in;
if not, it names a directory to find under /lib/modules/;
if null, /lib/modules/`uname -r` is used.
Returns zero on success, -1 if dwfl_report_module failed,
or an errno code if finding the files on disk failed.
If PREDICATE is not null, it is called with each module to be reported;
its arguments are the module name, and the ELF file name or null if unknown,
and its return value should be zero to skip the module, one to report it,
or -1 to cause the call to fail and return errno. */
extern int dwfl_linux_kernel_report_offline (Dwfl *dwfl, const char *release,
int (*predicate) (const char *,
const char *));
/* Examine an ET_CORE file and report modules based on its contents.
This can follow a dwfl_report_offline call to bootstrap the
DT_DEBUG method of following the dynamic linker link_map chain, in
case the core file does not contain enough of the executable's text
segment to locate its PT_DYNAMIC in the dump. This might call
dwfl_report_elf on file names found in the dump if reading some
link_map files is the only way to ascertain those modules' addresses.
Returns the number of modules reported, or -1 for errors. */
extern int dwfl_core_file_report (Dwfl *dwfl, Elf *elf);
/* Call dwfl_core_file_report but providing FILENAME instead. Function returns
opened core file Elf * and its *FDP file descriptor which must be held valid
until dwfl_end is called for DWFL. FDP can be passed as NULL (but the
caller cannot close the file descriptor then). Function returns NULL for
errors, check dwfl_errmsg in such case. */
extern Elf *dwfl_core_filename_report (Dwfl *dwfl, int *fdp,
const char *filename);
/* Call dwfl_report_module for each file mapped into the address space of PID.
Returns zero on success, -1 if dwfl_report_module failed,
or an errno code if opening the kernel binary failed. */
extern int dwfl_linux_proc_report (Dwfl *dwfl, pid_t pid);
/* Similar, but reads an input stream in the format of Linux /proc/PID/maps
files giving module layout, not the file for a live process. */
extern int dwfl_linux_proc_maps_report (Dwfl *dwfl, FILE *);
/* Trivial find_elf callback for use with dwfl_linux_proc_report.
This uses the module name as a file name directly and tries to open it
if it begin with a slash, or handles the magic string "[vdso]". */
extern int dwfl_linux_proc_find_elf (Dwfl_Module *mod, void **userdata,
const char *module_name, Dwarf_Addr base,
char **file_name, Elf **);
/* Standard argument parsing for using a standard callback set.
Field Dwfl_Module->USERDATA is reserved for "[exe]" or "[pie]" modules. */
struct argp;
extern const struct argp *dwfl_standard_argp (void) __attribute__ ((const));
/*** Relocation of addresses from Dwfl ***/
/* Return the number of relocatable bases associated with the module,
which is zero for ET_EXEC and one for ET_DYN. Returns -1 for errors. */
extern int dwfl_module_relocations (Dwfl_Module *mod);
/* Return the relocation base index associated with the *ADDRESS location,
and adjust *ADDRESS to be an offset relative to that base.
Returns -1 for errors. */
extern int dwfl_module_relocate_address (Dwfl_Module *mod,
Dwarf_Addr *address);
/* Return the ELF section name for the given relocation base index;
if SHNDXP is not null, set *SHNDXP to the ELF section index.
For ET_DYN, returns "" and sets *SHNDXP to SHN_ABS; the relocation
base is the runtime start address reported for the module.
Returns null for errors. */
extern const char *dwfl_module_relocation_info (Dwfl_Module *mod,
unsigned int idx,
GElf_Word *shndxp);
/* Validate that ADDRESS and ADDRESS+OFFSET lie in a known module
and both within the same contiguous region for relocation purposes.
Returns zero for success and -1 for errors. */
extern int dwfl_validate_address (Dwfl *dwfl,
Dwarf_Addr address, Dwarf_Sword offset);
/*** ELF access functions ***/
/* Fetch the module main ELF file (where the allocated sections
are found) for use with libelf. If successful, fills in *BIAS
with the difference between addresses within the loaded module
and those in symbol tables or Dwarf information referring to it. */
extern Elf *dwfl_module_getelf (Dwfl_Module *, GElf_Addr *bias)
__nonnull_attribute__ (1, 2);
/* Return the number of symbols in the module's symbol table,
or -1 for errors. */
extern int dwfl_module_getsymtab (Dwfl_Module *mod);
/* Fetch one entry from the module's symbol table. On errors, returns
NULL. If successful, fills in *SYM and returns the string for st_name.
This works like gelf_getsym except that st_value is always adjusted to
an absolute value based on the module's location, when the symbol is in
an SHF_ALLOC section. If SHNDXP is non-null, it's set with the section
index (whether from st_shndx or extended index table); in case of a
symbol in a non-allocated section, *SHNDXP is instead set to -1. */
extern const char *dwfl_module_getsym (Dwfl_Module *mod, int ndx,
GElf_Sym *sym, GElf_Word *shndxp)
__nonnull_attribute__ (3);
/* Find the symbol that ADDRESS lies inside, and return its name. */
extern const char *dwfl_module_addrname (Dwfl_Module *mod, GElf_Addr address);
/* Find the symbol that ADDRESS lies inside, and return detailed
information as for dwfl_module_getsym (above). */
extern const char *dwfl_module_addrsym (Dwfl_Module *mod, GElf_Addr address,
GElf_Sym *sym, GElf_Word *shndxp)
__nonnull_attribute__ (3);
/* Find the ELF section that *ADDRESS lies inside and return it.
On success, adjusts *ADDRESS to be relative to the section,
and sets *BIAS to the difference between addresses used in
the returned section's headers and run-time addresses. */
extern Elf_Scn *dwfl_module_address_section (Dwfl_Module *mod,
Dwarf_Addr *address,
Dwarf_Addr *bias)
__nonnull_attribute__ (2, 3);
/*** Dwarf access functions ***/
/* Fetch the module's debug information for use with libdw.
If successful, fills in *BIAS with the difference between
addresses within the loaded module and those to use with libdw. */
extern Dwarf *dwfl_module_getdwarf (Dwfl_Module *, Dwarf_Addr *bias)
__nonnull_attribute__ (2);
/* Get the libdw handle for each module. */
extern ptrdiff_t dwfl_getdwarf (Dwfl *,
int (*callback) (Dwfl_Module *, void **,
const char *, Dwarf_Addr,
Dwarf *, Dwarf_Addr, void *),
void *arg, ptrdiff_t offset);
/* Look up the module containing ADDR and return its debugging information,
loading it if necessary. */
extern Dwarf *dwfl_addrdwarf (Dwfl *dwfl, Dwarf_Addr addr, Dwarf_Addr *bias)
__nonnull_attribute__ (3);
/* Find the CU containing ADDR and return its DIE. */
extern Dwarf_Die *dwfl_addrdie (Dwfl *dwfl, Dwarf_Addr addr, Dwarf_Addr *bias)
__nonnull_attribute__ (3);
extern Dwarf_Die *dwfl_module_addrdie (Dwfl_Module *mod,
Dwarf_Addr addr, Dwarf_Addr *bias)
__nonnull_attribute__ (3);
/* Iterate through the CUs, start with null for LASTCU. */
extern Dwarf_Die *dwfl_nextcu (Dwfl *dwfl, Dwarf_Die *lastcu, Dwarf_Addr *bias)
__nonnull_attribute__ (3);
extern Dwarf_Die *dwfl_module_nextcu (Dwfl_Module *mod,
Dwarf_Die *lastcu, Dwarf_Addr *bias)
__nonnull_attribute__ (3);
/* Return the module containing the CU DIE. */
extern Dwfl_Module *dwfl_cumodule (Dwarf_Die *cudie);
/* Cache the source line information fo the CU and return the
number of Dwfl_Line entries it has. */
extern int dwfl_getsrclines (Dwarf_Die *cudie, size_t *nlines);
/* Access one line number entry within the CU. */
extern Dwfl_Line *dwfl_onesrcline (Dwarf_Die *cudie, size_t idx);
/* Get source for address. */
extern Dwfl_Line *dwfl_module_getsrc (Dwfl_Module *mod, Dwarf_Addr addr);
extern Dwfl_Line *dwfl_getsrc (Dwfl *dwfl, Dwarf_Addr addr);
/* Get address for source. */
extern int dwfl_module_getsrc_file (Dwfl_Module *mod,
const char *fname, int lineno, int column,
Dwfl_Line ***srcsp, size_t *nsrcs);
/* Return the module containing this line record. */
extern Dwfl_Module *dwfl_linemodule (Dwfl_Line *line);
/* Return the CU containing this line record. */
extern Dwarf_Die *dwfl_linecu (Dwfl_Line *line);
/* Return the source file name and fill in other information.
Arguments may be null for unneeded fields. */
extern const char *dwfl_lineinfo (Dwfl_Line *line, Dwarf_Addr *addr,
int *linep, int *colp,
Dwarf_Word *mtime, Dwarf_Word *length);
/* Return the equivalent Dwarf_Line and the bias to apply to its address. */
extern Dwarf_Line *dwfl_dwarf_line (Dwfl_Line *line, Dwarf_Addr *bias);
/* Return the compilation directory (AT_comp_dir) from this line's CU. */
extern const char *dwfl_line_comp_dir (Dwfl_Line *line);
/*** Machine backend access functions ***/
/* Return location expression to find return value given a
DW_TAG_subprogram, DW_TAG_subroutine_type, or similar DIE describing
function itself (whose DW_AT_type attribute describes its return type).
The given DIE must come from the given module. Returns -1 for errors.
Returns zero if the function has no return value (e.g. "void" in C).
Otherwise, *LOCOPS gets a location expression to find the return value,
and returns the number of operations in the expression. The pointer is
permanently allocated at least as long as the module is live. */
extern int dwfl_module_return_value_location (Dwfl_Module *mod,
Dwarf_Die *functypedie,
const Dwarf_Op **locops);
/* Enumerate the DWARF register numbers and their names.
For each register, CALLBACK gets its DWARF number, a string describing
the register set (such as "integer" or "FPU"), a prefix used in
assembler syntax (such as "%" or "$", may be ""), and the name for the
register (contains identifier characters only, possibly all digits).
The REGNAME string is valid only during the callback. */
extern int dwfl_module_register_names (Dwfl_Module *mod,
int (*callback) (void *arg,
int regno,
const char *setname,
const char *prefix,
const char *regname,
int bits, int type),
void *arg);
/* Find the CFI for this module. Returns NULL if there is no CFI.
On success, fills in *BIAS with the difference between addresses
within the loaded module and those in the CFI referring to it.
The pointer returned can be used until the module is cleaned up.
Calling these more than once returns the same pointers.
dwfl_module_dwarf_cfi gets the '.debug_frame' information found with the
rest of the DWARF information. dwfl_module_eh_cfi gets the '.eh_frame'
information found linked into the text. A module might have either or
both. */
extern Dwarf_CFI *dwfl_module_dwarf_cfi (Dwfl_Module *mod, Dwarf_Addr *bias);
extern Dwarf_CFI *dwfl_module_eh_cfi (Dwfl_Module *mod, Dwarf_Addr *bias);
/* Get innermost frame of first thread of live process PID. Returns NULL on
failure. */
extern Dwfl_Frame_State *dwfl_frame_state_pid (Dwfl *dwfl, pid_t pid);
/* Get innermost frame of first thread of core file COREFILE. Returns NULL on
failure. */
extern Dwfl_Frame_State *dwfl_frame_state_core (Dwfl *dwfl,
const char *corefile);
/* Fetch inferior registers from a caller supplied storage. */
typedef bool dwfl_frame_memory_read_t (Dwarf_Addr addr, Dwarf_Addr *result,
void *user_data);
extern Dwfl_Frame_State *dwfl_frame_state_data (Dwfl *dwfl, bool pc_set,
Dwarf_Addr pc, unsigned nregs,
const uint64_t *regs_set,
const Dwarf_Addr *regs,
dwfl_frame_memory_read_t
*memory_read,
void *memory_read_user_data);
/* Return TRUE and update *STATEP for the unwound frame for successful unwind.
Return TRUE and set *STATEP to NULL for the outermost frame. Return FALSE
(and call __libdwfl_seterrno) otherwise. */
extern bool dwfl_frame_unwind (Dwfl_Frame_State **statep);
/* Get return address register value for frame. Return TRUE if *PC set and
optionally *MINUSONE is also set, if MINUSONE is not NULL. Return FALSE
(and call __libdw_seterrno) otherwise. *MINUSONE is TRUE for normal calls
where *PC should be decremented by one to get the call instruction, it is
FALSE if this frame was interrupted by a signal handler. */
extern bool dwfl_frame_state_pc (Dwfl_Frame_State *state, Dwarf_Addr *pc,
bool *minusone);
/* Get innermost frame of the next thread from STATE. STATE can be any frame
of (the previous) thread. */
extern Dwfl_Frame_State *dwfl_frame_thread_next (Dwfl_Frame_State *state);
/* Get Task ID of the thread of STATE. This is PID for the thread started by
function main and gettid () for the other threads. */
extern pid_t dwfl_frame_tid_get (Dwfl_Frame_State *state);
#ifdef __cplusplus
}
#endif
#endif /* libdwfl.h */
|