VirtualBox

source: vbox/trunk/src/VBox/Runtime/common/dbg/dbgmoddwarf.cpp@ 75235

Last change on this file since 75235 was 75235, checked in by vboxsync, 6 years ago

IPRT: Starting to get somewhere with DWARF unwinding. bugref:3897

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 253.1 KB
Line 
1/* $Id: dbgmoddwarf.cpp 75235 2018-11-02 21:01:36Z vboxsync $ */
2/** @file
3 * IPRT - Debug Info Reader For DWARF.
4 */
5
6/*
7 * Copyright (C) 2011-2017 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 *
17 * The contents of this file may alternatively be used under the terms
18 * of the Common Development and Distribution License Version 1.0
19 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the
20 * VirtualBox OSE distribution, in which case the provisions of the
21 * CDDL are applicable instead of those of the GPL.
22 *
23 * You may elect to license modified versions of this file under the
24 * terms and conditions of either the GPL or the CDDL or both.
25 */
26
27
28/*********************************************************************************************************************************
29* Header Files *
30*********************************************************************************************************************************/
31#define LOG_GROUP RTLOGGROUP_DBG_DWARF
32#include <iprt/dbg.h>
33#include "internal/iprt.h"
34
35#include <iprt/asm.h>
36#include <iprt/ctype.h>
37#include <iprt/err.h>
38#include <iprt/list.h>
39#include <iprt/log.h>
40#include <iprt/mem.h>
41#define RTDBGMODDWARF_WITH_MEM_CACHE
42#ifdef RTDBGMODDWARF_WITH_MEM_CACHE
43# include <iprt/memcache.h>
44#endif
45#include <iprt/path.h>
46#include <iprt/string.h>
47#include <iprt/strcache.h>
48#include <iprt/x86.h>
49#include "internal/dbgmod.h"
50
51
52/*********************************************************************************************************************************
53* Defined Constants And Macros *
54*********************************************************************************************************************************/
55/** @name Standard DWARF Line Number Opcodes
56 * @{ */
57#define DW_LNS_extended UINT8_C(0x00)
58#define DW_LNS_copy UINT8_C(0x01)
59#define DW_LNS_advance_pc UINT8_C(0x02)
60#define DW_LNS_advance_line UINT8_C(0x03)
61#define DW_LNS_set_file UINT8_C(0x04)
62#define DW_LNS_set_column UINT8_C(0x05)
63#define DW_LNS_negate_stmt UINT8_C(0x06)
64#define DW_LNS_set_basic_block UINT8_C(0x07)
65#define DW_LNS_const_add_pc UINT8_C(0x08)
66#define DW_LNS_fixed_advance_pc UINT8_C(0x09)
67#define DW_LNS_set_prologue_end UINT8_C(0x0a)
68#define DW_LNS_set_epilogue_begin UINT8_C(0x0b)
69#define DW_LNS_set_isa UINT8_C(0x0c)
70#define DW_LNS_what_question_mark UINT8_C(0x0d)
71/** @} */
72
73
74/** @name Extended DWARF Line Number Opcodes
75 * @{ */
76#define DW_LNE_end_sequence UINT8_C(1)
77#define DW_LNE_set_address UINT8_C(2)
78#define DW_LNE_define_file UINT8_C(3)
79#define DW_LNE_set_descriminator UINT8_C(4)
80/** @} */
81
82/** @name DIE Tags.
83 * @{ */
84#define DW_TAG_array_type UINT16_C(0x0001)
85#define DW_TAG_class_type UINT16_C(0x0002)
86#define DW_TAG_entry_point UINT16_C(0x0003)
87#define DW_TAG_enumeration_type UINT16_C(0x0004)
88#define DW_TAG_formal_parameter UINT16_C(0x0005)
89#define DW_TAG_imported_declaration UINT16_C(0x0008)
90#define DW_TAG_label UINT16_C(0x000a)
91#define DW_TAG_lexical_block UINT16_C(0x000b)
92#define DW_TAG_member UINT16_C(0x000d)
93#define DW_TAG_pointer_type UINT16_C(0x000f)
94#define DW_TAG_reference_type UINT16_C(0x0010)
95#define DW_TAG_compile_unit UINT16_C(0x0011)
96#define DW_TAG_string_type UINT16_C(0x0012)
97#define DW_TAG_structure_type UINT16_C(0x0013)
98#define DW_TAG_subroutine_type UINT16_C(0x0015)
99#define DW_TAG_typedef UINT16_C(0x0016)
100#define DW_TAG_union_type UINT16_C(0x0017)
101#define DW_TAG_unspecified_parameters UINT16_C(0x0018)
102#define DW_TAG_variant UINT16_C(0x0019)
103#define DW_TAG_common_block UINT16_C(0x001a)
104#define DW_TAG_common_inclusion UINT16_C(0x001b)
105#define DW_TAG_inheritance UINT16_C(0x001c)
106#define DW_TAG_inlined_subroutine UINT16_C(0x001d)
107#define DW_TAG_module UINT16_C(0x001e)
108#define DW_TAG_ptr_to_member_type UINT16_C(0x001f)
109#define DW_TAG_set_type UINT16_C(0x0020)
110#define DW_TAG_subrange_type UINT16_C(0x0021)
111#define DW_TAG_with_stmt UINT16_C(0x0022)
112#define DW_TAG_access_declaration UINT16_C(0x0023)
113#define DW_TAG_base_type UINT16_C(0x0024)
114#define DW_TAG_catch_block UINT16_C(0x0025)
115#define DW_TAG_const_type UINT16_C(0x0026)
116#define DW_TAG_constant UINT16_C(0x0027)
117#define DW_TAG_enumerator UINT16_C(0x0028)
118#define DW_TAG_file_type UINT16_C(0x0029)
119#define DW_TAG_friend UINT16_C(0x002a)
120#define DW_TAG_namelist UINT16_C(0x002b)
121#define DW_TAG_namelist_item UINT16_C(0x002c)
122#define DW_TAG_packed_type UINT16_C(0x002d)
123#define DW_TAG_subprogram UINT16_C(0x002e)
124#define DW_TAG_template_type_parameter UINT16_C(0x002f)
125#define DW_TAG_template_value_parameter UINT16_C(0x0030)
126#define DW_TAG_thrown_type UINT16_C(0x0031)
127#define DW_TAG_try_block UINT16_C(0x0032)
128#define DW_TAG_variant_part UINT16_C(0x0033)
129#define DW_TAG_variable UINT16_C(0x0034)
130#define DW_TAG_volatile_type UINT16_C(0x0035)
131#define DW_TAG_dwarf_procedure UINT16_C(0x0036)
132#define DW_TAG_restrict_type UINT16_C(0x0037)
133#define DW_TAG_interface_type UINT16_C(0x0038)
134#define DW_TAG_namespace UINT16_C(0x0039)
135#define DW_TAG_imported_module UINT16_C(0x003a)
136#define DW_TAG_unspecified_type UINT16_C(0x003b)
137#define DW_TAG_partial_unit UINT16_C(0x003c)
138#define DW_TAG_imported_unit UINT16_C(0x003d)
139#define DW_TAG_condition UINT16_C(0x003f)
140#define DW_TAG_shared_type UINT16_C(0x0040)
141#define DW_TAG_type_unit UINT16_C(0x0041)
142#define DW_TAG_rvalue_reference_type UINT16_C(0x0042)
143#define DW_TAG_template_alias UINT16_C(0x0043)
144#define DW_TAG_lo_user UINT16_C(0x4080)
145#define DW_TAG_GNU_call_site UINT16_C(0x4109)
146#define DW_TAG_GNU_call_site_parameter UINT16_C(0x410a)
147#define DW_TAG_WATCOM_address_class_type UINT16_C(0x4100) /**< Watcom extension. */
148#define DW_TAG_WATCOM_namespace UINT16_C(0x4101) /**< Watcom extension. */
149#define DW_TAG_hi_user UINT16_C(0xffff)
150/** @} */
151
152
153/** @name DIE Attributes.
154 * @{ */
155#define DW_AT_sibling UINT16_C(0x0001)
156#define DW_AT_location UINT16_C(0x0002)
157#define DW_AT_name UINT16_C(0x0003)
158#define DW_AT_ordering UINT16_C(0x0009)
159#define DW_AT_byte_size UINT16_C(0x000b)
160#define DW_AT_bit_offset UINT16_C(0x000c)
161#define DW_AT_bit_size UINT16_C(0x000d)
162#define DW_AT_stmt_list UINT16_C(0x0010)
163#define DW_AT_low_pc UINT16_C(0x0011)
164#define DW_AT_high_pc UINT16_C(0x0012)
165#define DW_AT_language UINT16_C(0x0013)
166#define DW_AT_discr UINT16_C(0x0015)
167#define DW_AT_discr_value UINT16_C(0x0016)
168#define DW_AT_visibility UINT16_C(0x0017)
169#define DW_AT_import UINT16_C(0x0018)
170#define DW_AT_string_length UINT16_C(0x0019)
171#define DW_AT_common_reference UINT16_C(0x001a)
172#define DW_AT_comp_dir UINT16_C(0x001b)
173#define DW_AT_const_value UINT16_C(0x001c)
174#define DW_AT_containing_type UINT16_C(0x001d)
175#define DW_AT_default_value UINT16_C(0x001e)
176#define DW_AT_inline UINT16_C(0x0020)
177#define DW_AT_is_optional UINT16_C(0x0021)
178#define DW_AT_lower_bound UINT16_C(0x0022)
179#define DW_AT_producer UINT16_C(0x0025)
180#define DW_AT_prototyped UINT16_C(0x0027)
181#define DW_AT_return_addr UINT16_C(0x002a)
182#define DW_AT_start_scope UINT16_C(0x002c)
183#define DW_AT_bit_stride UINT16_C(0x002e)
184#define DW_AT_upper_bound UINT16_C(0x002f)
185#define DW_AT_abstract_origin UINT16_C(0x0031)
186#define DW_AT_accessibility UINT16_C(0x0032)
187#define DW_AT_address_class UINT16_C(0x0033)
188#define DW_AT_artificial UINT16_C(0x0034)
189#define DW_AT_base_types UINT16_C(0x0035)
190#define DW_AT_calling_convention UINT16_C(0x0036)
191#define DW_AT_count UINT16_C(0x0037)
192#define DW_AT_data_member_location UINT16_C(0x0038)
193#define DW_AT_decl_column UINT16_C(0x0039)
194#define DW_AT_decl_file UINT16_C(0x003a)
195#define DW_AT_decl_line UINT16_C(0x003b)
196#define DW_AT_declaration UINT16_C(0x003c)
197#define DW_AT_discr_list UINT16_C(0x003d)
198#define DW_AT_encoding UINT16_C(0x003e)
199#define DW_AT_external UINT16_C(0x003f)
200#define DW_AT_frame_base UINT16_C(0x0040)
201#define DW_AT_friend UINT16_C(0x0041)
202#define DW_AT_identifier_case UINT16_C(0x0042)
203#define DW_AT_macro_info UINT16_C(0x0043)
204#define DW_AT_namelist_item UINT16_C(0x0044)
205#define DW_AT_priority UINT16_C(0x0045)
206#define DW_AT_segment UINT16_C(0x0046)
207#define DW_AT_specification UINT16_C(0x0047)
208#define DW_AT_static_link UINT16_C(0x0048)
209#define DW_AT_type UINT16_C(0x0049)
210#define DW_AT_use_location UINT16_C(0x004a)
211#define DW_AT_variable_parameter UINT16_C(0x004b)
212#define DW_AT_virtuality UINT16_C(0x004c)
213#define DW_AT_vtable_elem_location UINT16_C(0x004d)
214#define DW_AT_allocated UINT16_C(0x004e)
215#define DW_AT_associated UINT16_C(0x004f)
216#define DW_AT_data_location UINT16_C(0x0050)
217#define DW_AT_byte_stride UINT16_C(0x0051)
218#define DW_AT_entry_pc UINT16_C(0x0052)
219#define DW_AT_use_UTF8 UINT16_C(0x0053)
220#define DW_AT_extension UINT16_C(0x0054)
221#define DW_AT_ranges UINT16_C(0x0055)
222#define DW_AT_trampoline UINT16_C(0x0056)
223#define DW_AT_call_column UINT16_C(0x0057)
224#define DW_AT_call_file UINT16_C(0x0058)
225#define DW_AT_call_line UINT16_C(0x0059)
226#define DW_AT_description UINT16_C(0x005a)
227#define DW_AT_binary_scale UINT16_C(0x005b)
228#define DW_AT_decimal_scale UINT16_C(0x005c)
229#define DW_AT_small UINT16_C(0x005d)
230#define DW_AT_decimal_sign UINT16_C(0x005e)
231#define DW_AT_digit_count UINT16_C(0x005f)
232#define DW_AT_picture_string UINT16_C(0x0060)
233#define DW_AT_mutable UINT16_C(0x0061)
234#define DW_AT_threads_scaled UINT16_C(0x0062)
235#define DW_AT_explicit UINT16_C(0x0063)
236#define DW_AT_object_pointer UINT16_C(0x0064)
237#define DW_AT_endianity UINT16_C(0x0065)
238#define DW_AT_elemental UINT16_C(0x0066)
239#define DW_AT_pure UINT16_C(0x0067)
240#define DW_AT_recursive UINT16_C(0x0068)
241#define DW_AT_signature UINT16_C(0x0069)
242#define DW_AT_main_subprogram UINT16_C(0x006a)
243#define DW_AT_data_bit_offset UINT16_C(0x006b)
244#define DW_AT_const_expr UINT16_C(0x006c)
245#define DW_AT_enum_class UINT16_C(0x006d)
246#define DW_AT_linkage_name UINT16_C(0x006e)
247#define DW_AT_lo_user UINT16_C(0x2000)
248/** Used by GCC and others, same as DW_AT_linkage_name. See http://wiki.dwarfstd.org/index.php?title=DW_AT_linkage_name*/
249#define DW_AT_MIPS_linkage_name UINT16_C(0x2007)
250#define DW_AT_WATCOM_memory_model UINT16_C(0x2082) /**< Watcom extension. */
251#define DW_AT_WATCOM_references_start UINT16_C(0x2083) /**< Watcom extension. */
252#define DW_AT_WATCOM_parm_entry UINT16_C(0x2084) /**< Watcom extension. */
253#define DW_AT_hi_user UINT16_C(0x3fff)
254/** @} */
255
256/** @name DIE Forms.
257 * @{ */
258#define DW_FORM_addr UINT16_C(0x01)
259/* 0x02 was FORM_REF in DWARF v1, obsolete now. */
260#define DW_FORM_block2 UINT16_C(0x03)
261#define DW_FORM_block4 UINT16_C(0x04)
262#define DW_FORM_data2 UINT16_C(0x05)
263#define DW_FORM_data4 UINT16_C(0x06)
264#define DW_FORM_data8 UINT16_C(0x07)
265#define DW_FORM_string UINT16_C(0x08)
266#define DW_FORM_block UINT16_C(0x09)
267#define DW_FORM_block1 UINT16_C(0x0a)
268#define DW_FORM_data1 UINT16_C(0x0b)
269#define DW_FORM_flag UINT16_C(0x0c)
270#define DW_FORM_sdata UINT16_C(0x0d)
271#define DW_FORM_strp UINT16_C(0x0e)
272#define DW_FORM_udata UINT16_C(0x0f)
273#define DW_FORM_ref_addr UINT16_C(0x10)
274#define DW_FORM_ref1 UINT16_C(0x11)
275#define DW_FORM_ref2 UINT16_C(0x12)
276#define DW_FORM_ref4 UINT16_C(0x13)
277#define DW_FORM_ref8 UINT16_C(0x14)
278#define DW_FORM_ref_udata UINT16_C(0x15)
279#define DW_FORM_indirect UINT16_C(0x16)
280#define DW_FORM_sec_offset UINT16_C(0x17)
281#define DW_FORM_exprloc UINT16_C(0x18)
282#define DW_FORM_flag_present UINT16_C(0x19)
283#define DW_FORM_ref_sig8 UINT16_C(0x20)
284/** @} */
285
286/** @name Address classes.
287 * @{ */
288#define DW_ADDR_none UINT8_C(0)
289#define DW_ADDR_i386_near16 UINT8_C(1)
290#define DW_ADDR_i386_far16 UINT8_C(2)
291#define DW_ADDR_i386_huge16 UINT8_C(3)
292#define DW_ADDR_i386_near32 UINT8_C(4)
293#define DW_ADDR_i386_far32 UINT8_C(5)
294/** @} */
295
296
297/** @name Location Expression Opcodes
298 * @{ */
299#define DW_OP_addr UINT8_C(0x03) /**< 1 operand, a constant address (size target specific). */
300#define DW_OP_deref UINT8_C(0x06) /**< 0 operands. */
301#define DW_OP_const1u UINT8_C(0x08) /**< 1 operand, a 1-byte constant. */
302#define DW_OP_const1s UINT8_C(0x09) /**< 1 operand, a 1-byte constant. */
303#define DW_OP_const2u UINT8_C(0x0a) /**< 1 operand, a 2-byte constant. */
304#define DW_OP_const2s UINT8_C(0x0b) /**< 1 operand, a 2-byte constant. */
305#define DW_OP_const4u UINT8_C(0x0c) /**< 1 operand, a 4-byte constant. */
306#define DW_OP_const4s UINT8_C(0x0d) /**< 1 operand, a 4-byte constant. */
307#define DW_OP_const8u UINT8_C(0x0e) /**< 1 operand, a 8-byte constant. */
308#define DW_OP_const8s UINT8_C(0x0f) /**< 1 operand, a 8-byte constant. */
309#define DW_OP_constu UINT8_C(0x10) /**< 1 operand, a ULEB128 constant. */
310#define DW_OP_consts UINT8_C(0x11) /**< 1 operand, a SLEB128 constant. */
311#define DW_OP_dup UINT8_C(0x12) /**< 0 operands. */
312#define DW_OP_drop UINT8_C(0x13) /**< 0 operands. */
313#define DW_OP_over UINT8_C(0x14) /**< 0 operands. */
314#define DW_OP_pick UINT8_C(0x15) /**< 1 operands, a 1-byte stack index. */
315#define DW_OP_swap UINT8_C(0x16) /**< 0 operands. */
316#define DW_OP_rot UINT8_C(0x17) /**< 0 operands. */
317#define DW_OP_xderef UINT8_C(0x18) /**< 0 operands. */
318#define DW_OP_abs UINT8_C(0x19) /**< 0 operands. */
319#define DW_OP_and UINT8_C(0x1a) /**< 0 operands. */
320#define DW_OP_div UINT8_C(0x1b) /**< 0 operands. */
321#define DW_OP_minus UINT8_C(0x1c) /**< 0 operands. */
322#define DW_OP_mod UINT8_C(0x1d) /**< 0 operands. */
323#define DW_OP_mul UINT8_C(0x1e) /**< 0 operands. */
324#define DW_OP_neg UINT8_C(0x1f) /**< 0 operands. */
325#define DW_OP_not UINT8_C(0x20) /**< 0 operands. */
326#define DW_OP_or UINT8_C(0x21) /**< 0 operands. */
327#define DW_OP_plus UINT8_C(0x22) /**< 0 operands. */
328#define DW_OP_plus_uconst UINT8_C(0x23) /**< 1 operands, a ULEB128 addend. */
329#define DW_OP_shl UINT8_C(0x24) /**< 0 operands. */
330#define DW_OP_shr UINT8_C(0x25) /**< 0 operands. */
331#define DW_OP_shra UINT8_C(0x26) /**< 0 operands. */
332#define DW_OP_xor UINT8_C(0x27) /**< 0 operands. */
333#define DW_OP_skip UINT8_C(0x2f) /**< 1 signed 2-byte constant. */
334#define DW_OP_bra UINT8_C(0x28) /**< 1 signed 2-byte constant. */
335#define DW_OP_eq UINT8_C(0x29) /**< 0 operands. */
336#define DW_OP_ge UINT8_C(0x2a) /**< 0 operands. */
337#define DW_OP_gt UINT8_C(0x2b) /**< 0 operands. */
338#define DW_OP_le UINT8_C(0x2c) /**< 0 operands. */
339#define DW_OP_lt UINT8_C(0x2d) /**< 0 operands. */
340#define DW_OP_ne UINT8_C(0x2e) /**< 0 operands. */
341#define DW_OP_lit0 UINT8_C(0x30) /**< 0 operands - literals 0..31 */
342#define DW_OP_lit31 UINT8_C(0x4f) /**< last litteral. */
343#define DW_OP_reg0 UINT8_C(0x50) /**< 0 operands - reg 0..31. */
344#define DW_OP_reg31 UINT8_C(0x6f) /**< last register. */
345#define DW_OP_breg0 UINT8_C(0x70) /**< 1 operand, a SLEB128 offset. */
346#define DW_OP_breg31 UINT8_C(0x8f) /**< last branch register. */
347#define DW_OP_regx UINT8_C(0x90) /**< 1 operand, a ULEB128 register. */
348#define DW_OP_fbreg UINT8_C(0x91) /**< 1 operand, a SLEB128 offset. */
349#define DW_OP_bregx UINT8_C(0x92) /**< 2 operands, a ULEB128 register followed by a SLEB128 offset. */
350#define DW_OP_piece UINT8_C(0x93) /**< 1 operand, a ULEB128 size of piece addressed. */
351#define DW_OP_deref_size UINT8_C(0x94) /**< 1 operand, a 1-byte size of data retrieved. */
352#define DW_OP_xderef_size UINT8_C(0x95) /**< 1 operand, a 1-byte size of data retrieved. */
353#define DW_OP_nop UINT8_C(0x96) /**< 0 operands. */
354#define DW_OP_lo_user UINT8_C(0xe0) /**< First user opcode */
355#define DW_OP_hi_user UINT8_C(0xff) /**< Last user opcode. */
356/** @} */
357
358/** @name Exception Handler Pointer Encodings (GCC/LSB).
359 * @{ */
360#define DW_EH_PE_FORMAT_MASK UINT8_C(0x0f) /**< Format mask. */
361#define DW_EH_PE_APPL_MASK UINT8_C(0x70) /**< Application mask. */
362#define DW_EH_PE_indirect UINT8_C(0x80) /**< Flag: Indirect pointer. */
363#define DW_EH_PE_omit UINT8_C(0xff) /**< Special value: Omitted. */
364#define DW_EH_PE_ptr UINT8_C(0x00) /**< Format: pointer sized, unsigned. */
365#define DW_EH_PE_uleb128 UINT8_C(0x01) /**< Format: unsigned LEB128. */
366#define DW_EH_PE_udata2 UINT8_C(0x02) /**< Format: unsigned 16-bit. */
367#define DW_EH_PE_udata4 UINT8_C(0x03) /**< Format: unsigned 32-bit. */
368#define DW_EH_PE_udata8 UINT8_C(0x04) /**< Format: unsigned 64-bit. */
369#define DW_EH_PE_sleb128 UINT8_C(0x09) /**< Format: signed LEB128. */
370#define DW_EH_PE_sdata2 UINT8_C(0x0a) /**< Format: signed 16-bit. */
371#define DW_EH_PE_sdata4 UINT8_C(0x0b) /**< Format: signed 32-bit. */
372#define DW_EH_PE_sdata8 UINT8_C(0x0c) /**< Format: signed 64-bit. */
373#define DW_EH_PE_absptr UINT8_C(0x00) /**< Application: Absolute */
374#define DW_EH_PE_pcrel UINT8_C(0x10) /**< Application: PC relative, i.e. relative pointer address. */
375#define DW_EH_PE_textrel UINT8_C(0x20) /**< Application: text section relative. */
376#define DW_EH_PE_datarel UINT8_C(0x30) /**< Application: data section relative. */
377#define DW_EH_PE_funcrel UINT8_C(0x40) /**< Application: relative to start of function. */
378#define DW_EH_PE_aligned UINT8_C(0x50) /**< Application: aligned pointer. */
379/** @} */
380
381/** @name Call frame instructions.
382 * @{ */
383/** Mask to use to identify DW_CFA_advance_loc, DW_CFA_offset and DW_CFA_restore. */
384#define DW_CFA_high_bit_mask UINT8_C(0xc0)
385
386#define DW_CFA_nop UINT8_C(0x00) /**< No operands. */
387
388#define DW_CFA_advance_loc UINT8_C(0x40) /**< low 6 bits: delta to advance. */
389#define DW_CFA_set_loc UINT8_C(0x01) /**< op1: address. */
390#define DW_CFA_advance_loc1 UINT8_C(0x02) /**< op1: 1-byte delta. */
391#define DW_CFA_advance_loc2 UINT8_C(0x03) /**< op1: 2-byte delta. */
392#define DW_CFA_advance_loc4 UINT8_C(0x04) /**< op1: 4-byte delta. */
393
394#define DW_CFA_offset UINT8_C(0x80) /**< low 6 bits: register; op1: ULEB128 offset. */
395#define DW_CFA_offset_extended UINT8_C(0x05) /**< op1: ULEB128 register; op2: ULEB128 offset. */
396#define DW_CFA_offset_extended_sf UINT8_C(0x11) /**< op1: ULEB128 register; op2: SLEB128 offset. */
397#define DW_CFA_restore UINT8_C(0xc0) /**< low 6 bits: register. */
398#define DW_CFA_restore_extended UINT8_C(0x06) /**< op1: ULEB128 register. */
399#define DW_CFA_undefined UINT8_C(0x07) /**< op1: ULEB128 register. */
400#define DW_CFA_same_value UINT8_C(0x08) /**< op1: ULEB128 register. */
401#define DW_CFA_register UINT8_C(0x09) /**< op1: ULEB128 destination register; op2: ULEB128 source register. */
402#define DW_CFA_expression UINT8_C(0x10) /**< op1: ULEB128 register; op2: BLOCK. */
403
404#define DW_CFA_val_offset UINT8_C(0x14) /**< op1: ULEB128 register; op2: ULEB128. */
405#define DW_CFA_val_offset_sf UINT8_C(0x15) /**< op1: ULEB128 register; op2: SLEB128. */
406#define DW_CFA_val_expression UINT8_C(0x16) /**< op1: ULEB128 register; op2: BLOCK. */
407
408#define DW_CFA_remember_state UINT8_C(0x0a) /**< No operands. */
409#define DW_CFA_restore_state UINT8_C(0x0b) /**< No operands. */
410
411#define DW_CFA_def_cfa UINT8_C(0x0c) /**< op1: ULEB128 register; op2: ULEB128 offset. */
412#define DW_CFA_def_cfa_register UINT8_C(0x0d) /**< op1: ULEB128 register. */
413#define DW_CFA_def_cfa_offset UINT8_C(0x0e) /**< op1: ULEB128 offset. */
414#define DW_CFA_def_cfa_expression UINT8_C(0x0f) /**< op1: BLOCK. */
415#define DW_CFA_def_cfa_sf UINT8_C(0x12) /**< op1: ULEB128 register; op2: SLEB128 offset. */
416#define DW_CFA_def_cfa_offset_sf UINT8_C(0x13) /**< op1: SLEB128 offset. */
417
418#define DW_CFA_lo_user UINT8_C(0x1c) /**< User defined operands. */
419#define DW_CFA_MIPS_advance_loc8 UINT8_C(0x1d) /**< op1: 8-byte delta? */
420#define DW_CFA_GNU_window_save UINT8_C(0x2d) /**< op1: ??; op2: ?? */
421#define DW_CFA_GNU_args_size UINT8_C(0x2e) /**< op1: ??; op2: ?? */
422#define DW_CFA_GNU_negative_offset_extended UINT8_C(0x2f) /**< op1: ??; op2: ?? */
423#define DW_CFA_hi_user UINT8_C(0x3f) /**< User defined operands. */
424/** @} */
425
426
427/** @name DWREG_X86_XXX - 386+ register number mappings.
428 * @{ */
429#define DWREG_X86_EAX 0
430#define DWREG_X86_ECX 1
431#define DWREG_X86_EDX 2
432#define DWREG_X86_EBX 3
433#define DWREG_X86_ESP 4
434#define DWREG_X86_EBP 5
435#define DWREG_X86_ESI 6
436#define DWREG_X86_EDI 7
437#define DWREG_X86_RA 8 /* return address (=EIP) */
438#define DWREG_X86_EFLAGS 9
439#define DWREG_X86_ST1 11
440#define DWREG_X86_ST2 12
441#define DWREG_X86_ST3 13
442#define DWREG_X86_ST4 14
443#define DWREG_X86_ST5 15
444#define DWREG_X86_ST6 16
445#define DWREG_X86_ST7 17
446#define DWREG_X86_XMM0 21
447#define DWREG_X86_XMM1 22
448#define DWREG_X86_XMM2 23
449#define DWREG_X86_XMM3 24
450#define DWREG_X86_XMM4 25
451#define DWREG_X86_XMM5 26
452#define DWREG_X86_XMM6 27
453#define DWREG_X86_XMM7 28
454#define DWREG_X86_MM0 29
455#define DWREG_X86_MM1 30
456#define DWREG_X86_MM2 31
457#define DWREG_X86_MM3 32
458#define DWREG_X86_MM4 33
459#define DWREG_X86_MM5 34
460#define DWREG_X86_MM6 35
461#define DWREG_X86_MM7 36
462#define DWREG_X86_MXCSR 39
463#define DWREG_X86_ES 40
464#define DWREG_X86_CS 41
465#define DWREG_X86_SS 42
466#define DWREG_X86_DS 43
467#define DWREG_X86_FS 44
468#define DWREG_X86_GS 45
469#define DWREG_X86_TR 48
470#define DWREG_X86_LDTR 49
471/** @} */
472
473
474/** @name DWREG_AMD64_XXX - AMD64 register number mappings.
475 * @note This for some braindead reason the first 8 GPR are in intel encoding
476 * order, unlike the DWREG_X86_XXX variant. Utter stupidity.
477 * @{ */
478#define DWREG_AMD64_RAX 0
479#define DWREG_AMD64_RDX 1
480#define DWREG_AMD64_RCX 2
481#define DWREG_AMD64_RBX 3
482#define DWREG_AMD64_RSI 4
483#define DWREG_AMD64_RDI 5
484#define DWREG_AMD64_RBP 6
485#define DWREG_AMD64_RSP 7
486#define DWREG_AMD64_R8 8
487#define DWREG_AMD64_R9 9
488#define DWREG_AMD64_R10 10
489#define DWREG_AMD64_R11 11
490#define DWREG_AMD64_R12 12
491#define DWREG_AMD64_R13 13
492#define DWREG_AMD64_R14 14
493#define DWREG_AMD64_R15 15
494#define DWREG_AMD64_RA 16 /* return address (=RIP) */
495#define DWREG_AMD64_XMM0 17
496#define DWREG_AMD64_XMM1 18
497#define DWREG_AMD64_XMM2 19
498#define DWREG_AMD64_XMM3 20
499#define DWREG_AMD64_XMM4 21
500#define DWREG_AMD64_XMM5 22
501#define DWREG_AMD64_XMM6 23
502#define DWREG_AMD64_XMM7 24
503#define DWREG_AMD64_XMM8 25
504#define DWREG_AMD64_XMM9 26
505#define DWREG_AMD64_XMM10 27
506#define DWREG_AMD64_XMM11 28
507#define DWREG_AMD64_XMM12 29
508#define DWREG_AMD64_XMM13 30
509#define DWREG_AMD64_XMM14 31
510#define DWREG_AMD64_XMM15 32
511#define DWREG_AMD64_ST0 33
512#define DWREG_AMD64_ST1 34
513#define DWREG_AMD64_ST2 35
514#define DWREG_AMD64_ST3 36
515#define DWREG_AMD64_ST4 37
516#define DWREG_AMD64_ST5 38
517#define DWREG_AMD64_ST6 39
518#define DWREG_AMD64_ST7 40
519#define DWREG_AMD64_MM0 41
520#define DWREG_AMD64_MM1 42
521#define DWREG_AMD64_MM2 43
522#define DWREG_AMD64_MM3 44
523#define DWREG_AMD64_MM4 45
524#define DWREG_AMD64_MM5 46
525#define DWREG_AMD64_MM6 47
526#define DWREG_AMD64_MM7 48
527#define DWREG_AMD64_RFLAGS 49
528#define DWREG_AMD64_ES 50
529#define DWREG_AMD64_CS 51
530#define DWREG_AMD64_SS 52
531#define DWREG_AMD64_DS 53
532#define DWREG_AMD64_FS 54
533#define DWREG_AMD64_GS 55
534#define DWREG_AMD64_FS_BASE 58
535#define DWREG_AMD64_GS_BASE 59
536#define DWREG_AMD64_TR 62
537#define DWREG_AMD64_LDTR 63
538#define DWREG_AMD64_MXCSR 64
539#define DWREG_AMD64_FCW 65
540#define DWREG_AMD64_FSW 66
541/** @} */
542
543
544/*********************************************************************************************************************************
545* Structures and Typedefs *
546*********************************************************************************************************************************/
547/** Pointer to a DWARF section reader. */
548typedef struct RTDWARFCURSOR *PRTDWARFCURSOR;
549/** Pointer to an attribute descriptor. */
550typedef struct RTDWARFATTRDESC const *PCRTDWARFATTRDESC;
551/** Pointer to a DIE. */
552typedef struct RTDWARFDIE *PRTDWARFDIE;
553/** Pointer to a const DIE. */
554typedef struct RTDWARFDIE const *PCRTDWARFDIE;
555
556/**
557 * DWARF sections.
558 */
559typedef enum krtDbgModDwarfSect
560{
561 krtDbgModDwarfSect_abbrev = 0,
562 krtDbgModDwarfSect_aranges,
563 krtDbgModDwarfSect_frame,
564 krtDbgModDwarfSect_info,
565 krtDbgModDwarfSect_inlined,
566 krtDbgModDwarfSect_line,
567 krtDbgModDwarfSect_loc,
568 krtDbgModDwarfSect_macinfo,
569 krtDbgModDwarfSect_pubnames,
570 krtDbgModDwarfSect_pubtypes,
571 krtDbgModDwarfSect_ranges,
572 krtDbgModDwarfSect_str,
573 krtDbgModDwarfSect_types,
574 /** End of valid parts (exclusive). */
575 krtDbgModDwarfSect_End
576} krtDbgModDwarfSect;
577
578/**
579 * Abbreviation cache entry.
580 */
581typedef struct RTDWARFABBREV
582{
583 /** Whether there are children or not. */
584 bool fChildren;
585#ifdef LOG_ENABLED
586 uint8_t cbHdr; /**< For calcing ABGOFF matching dwarfdump. */
587#endif
588 /** The tag. */
589 uint16_t uTag;
590 /** Offset into the abbrev section of the specification pairs. */
591 uint32_t offSpec;
592 /** The abbreviation table offset this is entry is valid for.
593 * UINT32_MAX if not valid. */
594 uint32_t offAbbrev;
595} RTDWARFABBREV;
596/** Pointer to an abbreviation cache entry. */
597typedef RTDWARFABBREV *PRTDWARFABBREV;
598/** Pointer to a const abbreviation cache entry. */
599typedef RTDWARFABBREV const *PCRTDWARFABBREV;
600
601/**
602 * Structure for gathering segment info.
603 */
604typedef struct RTDBGDWARFSEG
605{
606 /** The highest offset in the segment. */
607 uint64_t offHighest;
608 /** Calculated base address. */
609 uint64_t uBaseAddr;
610 /** Estimated The segment size. */
611 uint64_t cbSegment;
612 /** Segment number (RTLDRSEG::Sel16bit). */
613 RTSEL uSegment;
614} RTDBGDWARFSEG;
615/** Pointer to segment info. */
616typedef RTDBGDWARFSEG *PRTDBGDWARFSEG;
617
618
619/**
620 * The instance data of the DWARF reader.
621 */
622typedef struct RTDBGMODDWARF
623{
624 /** The debug container containing doing the real work. */
625 RTDBGMOD hCnt;
626 /** The image module (no reference). */
627 PRTDBGMODINT pImgMod;
628 /** The debug info module (no reference). */
629 PRTDBGMODINT pDbgInfoMod;
630 /** Nested image module (with reference ofc). */
631 PRTDBGMODINT pNestedMod;
632
633 /** DWARF debug info sections. */
634 struct
635 {
636 /** The file offset of the part. */
637 RTFOFF offFile;
638 /** The size of the part. */
639 size_t cb;
640 /** The memory mapping of the part. */
641 void const *pv;
642 /** Set if present. */
643 bool fPresent;
644 /** The debug info ordinal number in the image file. */
645 uint32_t iDbgInfo;
646 } aSections[krtDbgModDwarfSect_End];
647
648 /** The offset into the abbreviation section of the current cache. */
649 uint32_t offCachedAbbrev;
650 /** The number of cached abbreviations we've allocated space for. */
651 uint32_t cCachedAbbrevsAlloced;
652 /** Array of cached abbreviations, indexed by code. */
653 PRTDWARFABBREV paCachedAbbrevs;
654 /** Used by rtDwarfAbbrev_Lookup when the result is uncachable. */
655 RTDWARFABBREV LookupAbbrev;
656
657 /** The list of compilation units (RTDWARFDIE). */
658 RTLISTANCHOR CompileUnitList;
659
660 /** Set if we have to use link addresses because the module does not have
661 * fixups (mach_kernel). */
662 bool fUseLinkAddress;
663 /** This is set to -1 if we're doing everything in one pass.
664 * Otherwise it's 1 or 2:
665 * - In pass 1, we collect segment info.
666 * - In pass 2, we add debug info to the container.
667 * The two pass parsing is necessary for watcom generated symbol files as
668 * these contains no information about the code and data segments in the
669 * image. So we have to figure out some approximate stuff based on the
670 * segments and offsets we encounter in the debug info. */
671 int8_t iWatcomPass;
672 /** Segment index hint. */
673 uint16_t iSegHint;
674 /** The number of segments in paSegs.
675 * (During segment copying, this is abused to count useful segments.) */
676 uint32_t cSegs;
677 /** Pointer to segments if iWatcomPass isn't -1. */
678 PRTDBGDWARFSEG paSegs;
679#ifdef RTDBGMODDWARF_WITH_MEM_CACHE
680 /** DIE allocators. */
681 struct
682 {
683 RTMEMCACHE hMemCache;
684 uint32_t cbMax;
685 } aDieAllocators[2];
686#endif
687} RTDBGMODDWARF;
688/** Pointer to instance data of the DWARF reader. */
689typedef RTDBGMODDWARF *PRTDBGMODDWARF;
690
691/**
692 * DWARF cursor for reading byte data.
693 */
694typedef struct RTDWARFCURSOR
695{
696 /** The current position. */
697 uint8_t const *pb;
698 /** The number of bytes left to read. */
699 size_t cbLeft;
700 /** The number of bytes left to read in the current unit. */
701 size_t cbUnitLeft;
702 /** The DWARF debug info reader instance. (Can be NULL for eh_frame.) */
703 PRTDBGMODDWARF pDwarfMod;
704 /** Set if this is 64-bit DWARF, clear if 32-bit. */
705 bool f64bitDwarf;
706 /** Set if the format endian is native, clear if endian needs to be
707 * inverted. */
708 bool fNativEndian;
709 /** The size of a native address. */
710 uint8_t cbNativeAddr;
711 /** The cursor status code. This is VINF_SUCCESS until some error
712 * occurs. */
713 int rc;
714 /** The start of the area covered by the cursor.
715 * Used for repositioning the cursor relative to the start of a section. */
716 uint8_t const *pbStart;
717 /** The section. */
718 krtDbgModDwarfSect enmSect;
719} RTDWARFCURSOR;
720
721
722/**
723 * DWARF line number program state.
724 */
725typedef struct RTDWARFLINESTATE
726{
727 /** Virtual Line Number Machine Registers. */
728 struct
729 {
730 uint64_t uAddress;
731 uint64_t idxOp;
732 uint32_t iFile;
733 uint32_t uLine;
734 uint32_t uColumn;
735 bool fIsStatement;
736 bool fBasicBlock;
737 bool fEndSequence;
738 bool fPrologueEnd;
739 bool fEpilogueBegin;
740 uint32_t uIsa;
741 uint32_t uDiscriminator;
742 RTSEL uSegment;
743 } Regs;
744 /** @} */
745
746 /** Header. */
747 struct
748 {
749 uint32_t uVer;
750 uint64_t offFirstOpcode;
751 uint8_t cbMinInstr;
752 uint8_t cMaxOpsPerInstr;
753 uint8_t u8DefIsStmt;
754 int8_t s8LineBase;
755 uint8_t u8LineRange;
756 uint8_t u8OpcodeBase;
757 uint8_t const *pacStdOperands;
758 } Hdr;
759
760 /** @name Include Path Table (0-based)
761 * @{ */
762 const char **papszIncPaths;
763 uint32_t cIncPaths;
764 /** @} */
765
766 /** @name File Name Table (0-based, dummy zero entry)
767 * @{ */
768 char **papszFileNames;
769 uint32_t cFileNames;
770 /** @} */
771
772 /** The DWARF debug info reader instance. */
773 PRTDBGMODDWARF pDwarfMod;
774} RTDWARFLINESTATE;
775/** Pointer to a DWARF line number program state. */
776typedef RTDWARFLINESTATE *PRTDWARFLINESTATE;
777
778
779/**
780 * Decodes an attribute and stores it in the specified DIE member field.
781 *
782 * @returns IPRT status code.
783 * @param pDie Pointer to the DIE structure.
784 * @param pbMember Pointer to the first byte in the member.
785 * @param pDesc The attribute descriptor.
786 * @param uForm The data form.
787 * @param pCursor The cursor to read data from.
788 */
789typedef DECLCALLBACK(int) FNRTDWARFATTRDECODER(PRTDWARFDIE pDie, uint8_t *pbMember, PCRTDWARFATTRDESC pDesc,
790 uint32_t uForm, PRTDWARFCURSOR pCursor);
791/** Pointer to an attribute decoder callback. */
792typedef FNRTDWARFATTRDECODER *PFNRTDWARFATTRDECODER;
793
794/**
795 * Attribute descriptor.
796 */
797typedef struct RTDWARFATTRDESC
798{
799 /** The attribute. */
800 uint16_t uAttr;
801 /** The data member offset. */
802 uint16_t off;
803 /** The data member size and initialization method. */
804 uint8_t cbInit;
805 uint8_t bPadding[3]; /**< Alignment padding. */
806 /** The decoder function. */
807 PFNRTDWARFATTRDECODER pfnDecoder;
808} RTDWARFATTRDESC;
809
810/** Define a attribute entry. */
811#define ATTR_ENTRY(a_uAttr, a_Struct, a_Member, a_Init, a_pfnDecoder) \
812 { \
813 a_uAttr, \
814 (uint16_t)RT_OFFSETOF(a_Struct, a_Member), \
815 a_Init | ((uint8_t)RT_SIZEOFMEMB(a_Struct, a_Member) & ATTR_SIZE_MASK), \
816 { 0, 0, 0 }, \
817 a_pfnDecoder\
818 }
819
820/** @name Attribute size and init methods.
821 * @{ */
822#define ATTR_INIT_ZERO UINT8_C(0x00)
823#define ATTR_INIT_FFFS UINT8_C(0x80)
824#define ATTR_INIT_MASK UINT8_C(0x80)
825#define ATTR_SIZE_MASK UINT8_C(0x3f)
826#define ATTR_GET_SIZE(a_pAttrDesc) ((a_pAttrDesc)->cbInit & ATTR_SIZE_MASK)
827/** @} */
828
829
830/**
831 * DIE descriptor.
832 */
833typedef struct RTDWARFDIEDESC
834{
835 /** The size of the DIE. */
836 size_t cbDie;
837 /** The number of attributes. */
838 size_t cAttributes;
839 /** Pointer to the array of attributes. */
840 PCRTDWARFATTRDESC paAttributes;
841} RTDWARFDIEDESC;
842typedef struct RTDWARFDIEDESC const *PCRTDWARFDIEDESC;
843/** DIE descriptor initializer. */
844#define DIE_DESC_INIT(a_Type, a_aAttrs) { sizeof(a_Type), RT_ELEMENTS(a_aAttrs), &a_aAttrs[0] }
845
846
847/**
848 * DIE core structure, all inherits (starts with) this.
849 */
850typedef struct RTDWARFDIE
851{
852 /** Pointer to the parent node. NULL if root unit. */
853 struct RTDWARFDIE *pParent;
854 /** Our node in the sibling list. */
855 RTLISTNODE SiblingNode;
856 /** List of children. */
857 RTLISTNODE ChildList;
858 /** The number of attributes successfully decoded. */
859 uint8_t cDecodedAttrs;
860 /** The number of unknown or otherwise unhandled attributes. */
861 uint8_t cUnhandledAttrs;
862#ifdef RTDBGMODDWARF_WITH_MEM_CACHE
863 /** The allocator index. */
864 uint8_t iAllocator;
865#endif
866 /** The die tag, indicating which union structure to use. */
867 uint16_t uTag;
868 /** Offset of the abbreviation specification (within debug_abbrev). */
869 uint32_t offSpec;
870} RTDWARFDIE;
871
872
873/**
874 * DWARF address structure.
875 */
876typedef struct RTDWARFADDR
877{
878 /** The address. */
879 uint64_t uAddress;
880} RTDWARFADDR;
881typedef RTDWARFADDR *PRTDWARFADDR;
882typedef RTDWARFADDR const *PCRTDWARFADDR;
883
884
885/**
886 * DWARF address range.
887 */
888typedef struct RTDWARFADDRRANGE
889{
890 uint64_t uLowAddress;
891 uint64_t uHighAddress;
892 uint8_t const *pbRanges; /* ?? */
893 uint8_t cAttrs : 2;
894 uint8_t fHaveLowAddress : 1;
895 uint8_t fHaveHighAddress : 1;
896 uint8_t fHaveHighIsAddress : 1;
897 uint8_t fHaveRanges : 1;
898} RTDWARFADDRRANGE;
899typedef RTDWARFADDRRANGE *PRTDWARFADDRRANGE;
900typedef RTDWARFADDRRANGE const *PCRTDWARFADDRRANGE;
901
902/** What a RTDWARFREF is relative to. */
903typedef enum krtDwarfRef
904{
905 krtDwarfRef_NotSet,
906 krtDwarfRef_LineSection,
907 krtDwarfRef_LocSection,
908 krtDwarfRef_RangesSection,
909 krtDwarfRef_InfoSection,
910 krtDwarfRef_SameUnit,
911 krtDwarfRef_TypeId64
912} krtDwarfRef;
913
914/**
915 * DWARF reference.
916 */
917typedef struct RTDWARFREF
918{
919 /** The offset. */
920 uint64_t off;
921 /** What the offset is relative to. */
922 krtDwarfRef enmWrt;
923} RTDWARFREF;
924typedef RTDWARFREF *PRTDWARFREF;
925typedef RTDWARFREF const *PCRTDWARFREF;
926
927
928/**
929 * DWARF Location state.
930 */
931typedef struct RTDWARFLOCST
932{
933 /** The input cursor. */
934 RTDWARFCURSOR Cursor;
935 /** Points to the current top of the stack. Initial value -1. */
936 int32_t iTop;
937 /** The value stack. */
938 uint64_t auStack[64];
939} RTDWARFLOCST;
940/** Pointer to location state. */
941typedef RTDWARFLOCST *PRTDWARFLOCST;
942
943
944
945/*********************************************************************************************************************************
946* Internal Functions *
947*********************************************************************************************************************************/
948static FNRTDWARFATTRDECODER rtDwarfDecode_Address;
949static FNRTDWARFATTRDECODER rtDwarfDecode_Bool;
950static FNRTDWARFATTRDECODER rtDwarfDecode_LowHighPc;
951static FNRTDWARFATTRDECODER rtDwarfDecode_Ranges;
952static FNRTDWARFATTRDECODER rtDwarfDecode_Reference;
953static FNRTDWARFATTRDECODER rtDwarfDecode_SectOff;
954static FNRTDWARFATTRDECODER rtDwarfDecode_String;
955static FNRTDWARFATTRDECODER rtDwarfDecode_UnsignedInt;
956static FNRTDWARFATTRDECODER rtDwarfDecode_SegmentLoc;
957
958
959/*********************************************************************************************************************************
960* Global Variables *
961*********************************************************************************************************************************/
962/** RTDWARFDIE description. */
963static const RTDWARFDIEDESC g_CoreDieDesc = { sizeof(RTDWARFDIE), 0, NULL };
964
965
966/**
967 * DW_TAG_compile_unit & DW_TAG_partial_unit.
968 */
969typedef struct RTDWARFDIECOMPILEUNIT
970{
971 /** The DIE core structure. */
972 RTDWARFDIE Core;
973 /** The unit name. */
974 const char *pszName;
975 /** The address range of the code belonging to this unit. */
976 RTDWARFADDRRANGE PcRange;
977 /** The language name. */
978 uint16_t uLanguage;
979 /** The identifier case. */
980 uint8_t uIdentifierCase;
981 /** String are UTF-8 encoded. If not set, the encoding is
982 * unknown. */
983 bool fUseUtf8;
984 /** The unit contains main() or equivalent. */
985 bool fMainFunction;
986 /** The line numbers for this unit. */
987 RTDWARFREF StmtListRef;
988 /** The macro information for this unit. */
989 RTDWARFREF MacroInfoRef;
990 /** Reference to the base types. */
991 RTDWARFREF BaseTypesRef;
992 /** Working directory for the unit. */
993 const char *pszCurDir;
994 /** The name of the compiler or whatever that produced this unit. */
995 const char *pszProducer;
996
997 /** @name From the unit header.
998 * @{ */
999 /** The offset into debug_info of this unit (for references). */
1000 uint64_t offUnit;
1001 /** The length of this unit. */
1002 uint64_t cbUnit;
1003 /** The offset into debug_abbrev of the abbreviation for this unit. */
1004 uint64_t offAbbrev;
1005 /** The native address size. */
1006 uint8_t cbNativeAddr;
1007 /** The DWARF version. */
1008 uint8_t uDwarfVer;
1009 /** @} */
1010} RTDWARFDIECOMPILEUNIT;
1011typedef RTDWARFDIECOMPILEUNIT *PRTDWARFDIECOMPILEUNIT;
1012
1013
1014/** RTDWARFDIECOMPILEUNIT attributes. */
1015static const RTDWARFATTRDESC g_aCompileUnitAttrs[] =
1016{
1017 ATTR_ENTRY(DW_AT_name, RTDWARFDIECOMPILEUNIT, pszName, ATTR_INIT_ZERO, rtDwarfDecode_String),
1018 ATTR_ENTRY(DW_AT_low_pc, RTDWARFDIECOMPILEUNIT, PcRange, ATTR_INIT_ZERO, rtDwarfDecode_LowHighPc),
1019 ATTR_ENTRY(DW_AT_high_pc, RTDWARFDIECOMPILEUNIT, PcRange, ATTR_INIT_ZERO, rtDwarfDecode_LowHighPc),
1020 ATTR_ENTRY(DW_AT_ranges, RTDWARFDIECOMPILEUNIT, PcRange, ATTR_INIT_ZERO, rtDwarfDecode_Ranges),
1021 ATTR_ENTRY(DW_AT_language, RTDWARFDIECOMPILEUNIT, uLanguage, ATTR_INIT_ZERO, rtDwarfDecode_UnsignedInt),
1022 ATTR_ENTRY(DW_AT_macro_info, RTDWARFDIECOMPILEUNIT, MacroInfoRef, ATTR_INIT_ZERO, rtDwarfDecode_SectOff),
1023 ATTR_ENTRY(DW_AT_stmt_list, RTDWARFDIECOMPILEUNIT, StmtListRef, ATTR_INIT_ZERO, rtDwarfDecode_SectOff),
1024 ATTR_ENTRY(DW_AT_comp_dir, RTDWARFDIECOMPILEUNIT, pszCurDir, ATTR_INIT_ZERO, rtDwarfDecode_String),
1025 ATTR_ENTRY(DW_AT_producer, RTDWARFDIECOMPILEUNIT, pszProducer, ATTR_INIT_ZERO, rtDwarfDecode_String),
1026 ATTR_ENTRY(DW_AT_identifier_case, RTDWARFDIECOMPILEUNIT, uIdentifierCase,ATTR_INIT_ZERO, rtDwarfDecode_UnsignedInt),
1027 ATTR_ENTRY(DW_AT_base_types, RTDWARFDIECOMPILEUNIT, BaseTypesRef, ATTR_INIT_ZERO, rtDwarfDecode_Reference),
1028 ATTR_ENTRY(DW_AT_use_UTF8, RTDWARFDIECOMPILEUNIT, fUseUtf8, ATTR_INIT_ZERO, rtDwarfDecode_Bool),
1029 ATTR_ENTRY(DW_AT_main_subprogram, RTDWARFDIECOMPILEUNIT, fMainFunction, ATTR_INIT_ZERO, rtDwarfDecode_Bool)
1030};
1031
1032/** RTDWARFDIECOMPILEUNIT description. */
1033static const RTDWARFDIEDESC g_CompileUnitDesc = DIE_DESC_INIT(RTDWARFDIECOMPILEUNIT, g_aCompileUnitAttrs);
1034
1035
1036/**
1037 * DW_TAG_subprogram.
1038 */
1039typedef struct RTDWARFDIESUBPROGRAM
1040{
1041 /** The DIE core structure. */
1042 RTDWARFDIE Core;
1043 /** The name. */
1044 const char *pszName;
1045 /** The linkage name. */
1046 const char *pszLinkageName;
1047 /** The address range of the code belonging to this unit. */
1048 RTDWARFADDRRANGE PcRange;
1049 /** The first instruction in the function. */
1050 RTDWARFADDR EntryPc;
1051 /** Segment number (watcom). */
1052 RTSEL uSegment;
1053 /** Reference to the specification. */
1054 RTDWARFREF SpecRef;
1055} RTDWARFDIESUBPROGRAM;
1056/** Pointer to a DW_TAG_subprogram DIE. */
1057typedef RTDWARFDIESUBPROGRAM *PRTDWARFDIESUBPROGRAM;
1058/** Pointer to a const DW_TAG_subprogram DIE. */
1059typedef RTDWARFDIESUBPROGRAM const *PCRTDWARFDIESUBPROGRAM;
1060
1061
1062/** RTDWARFDIESUBPROGRAM attributes. */
1063static const RTDWARFATTRDESC g_aSubProgramAttrs[] =
1064{
1065 ATTR_ENTRY(DW_AT_name, RTDWARFDIESUBPROGRAM, pszName, ATTR_INIT_ZERO, rtDwarfDecode_String),
1066 ATTR_ENTRY(DW_AT_linkage_name, RTDWARFDIESUBPROGRAM, pszLinkageName, ATTR_INIT_ZERO, rtDwarfDecode_String),
1067 ATTR_ENTRY(DW_AT_MIPS_linkage_name, RTDWARFDIESUBPROGRAM, pszLinkageName, ATTR_INIT_ZERO, rtDwarfDecode_String),
1068 ATTR_ENTRY(DW_AT_low_pc, RTDWARFDIESUBPROGRAM, PcRange, ATTR_INIT_ZERO, rtDwarfDecode_LowHighPc),
1069 ATTR_ENTRY(DW_AT_high_pc, RTDWARFDIESUBPROGRAM, PcRange, ATTR_INIT_ZERO, rtDwarfDecode_LowHighPc),
1070 ATTR_ENTRY(DW_AT_ranges, RTDWARFDIESUBPROGRAM, PcRange, ATTR_INIT_ZERO, rtDwarfDecode_Ranges),
1071 ATTR_ENTRY(DW_AT_entry_pc, RTDWARFDIESUBPROGRAM, EntryPc, ATTR_INIT_ZERO, rtDwarfDecode_Address),
1072 ATTR_ENTRY(DW_AT_segment, RTDWARFDIESUBPROGRAM, uSegment, ATTR_INIT_ZERO, rtDwarfDecode_SegmentLoc),
1073 ATTR_ENTRY(DW_AT_specification, RTDWARFDIESUBPROGRAM, SpecRef, ATTR_INIT_ZERO, rtDwarfDecode_Reference)
1074};
1075
1076/** RTDWARFDIESUBPROGRAM description. */
1077static const RTDWARFDIEDESC g_SubProgramDesc = DIE_DESC_INIT(RTDWARFDIESUBPROGRAM, g_aSubProgramAttrs);
1078
1079
1080/** RTDWARFDIESUBPROGRAM attributes for the specification hack. */
1081static const RTDWARFATTRDESC g_aSubProgramSpecHackAttrs[] =
1082{
1083 ATTR_ENTRY(DW_AT_name, RTDWARFDIESUBPROGRAM, pszName, ATTR_INIT_ZERO, rtDwarfDecode_String),
1084 ATTR_ENTRY(DW_AT_linkage_name, RTDWARFDIESUBPROGRAM, pszLinkageName, ATTR_INIT_ZERO, rtDwarfDecode_String),
1085 ATTR_ENTRY(DW_AT_MIPS_linkage_name, RTDWARFDIESUBPROGRAM, pszLinkageName, ATTR_INIT_ZERO, rtDwarfDecode_String),
1086};
1087
1088/** RTDWARFDIESUBPROGRAM description for the specification hack. */
1089static const RTDWARFDIEDESC g_SubProgramSpecHackDesc = DIE_DESC_INIT(RTDWARFDIESUBPROGRAM, g_aSubProgramSpecHackAttrs);
1090
1091
1092/**
1093 * DW_TAG_label.
1094 */
1095typedef struct RTDWARFDIELABEL
1096{
1097 /** The DIE core structure. */
1098 RTDWARFDIE Core;
1099 /** The name. */
1100 const char *pszName;
1101 /** The address of the first instruction. */
1102 RTDWARFADDR Address;
1103 /** Segment number (watcom). */
1104 RTSEL uSegment;
1105 /** Externally visible? */
1106 bool fExternal;
1107} RTDWARFDIELABEL;
1108/** Pointer to a DW_TAG_label DIE. */
1109typedef RTDWARFDIELABEL *PRTDWARFDIELABEL;
1110/** Pointer to a const DW_TAG_label DIE. */
1111typedef RTDWARFDIELABEL const *PCRTDWARFDIELABEL;
1112
1113
1114/** RTDWARFDIESUBPROGRAM attributes. */
1115static const RTDWARFATTRDESC g_aLabelAttrs[] =
1116{
1117 ATTR_ENTRY(DW_AT_name, RTDWARFDIELABEL, pszName, ATTR_INIT_ZERO, rtDwarfDecode_String),
1118 ATTR_ENTRY(DW_AT_low_pc, RTDWARFDIELABEL, Address, ATTR_INIT_ZERO, rtDwarfDecode_Address),
1119 ATTR_ENTRY(DW_AT_segment, RTDWARFDIELABEL, uSegment, ATTR_INIT_ZERO, rtDwarfDecode_SegmentLoc),
1120 ATTR_ENTRY(DW_AT_external, RTDWARFDIELABEL, fExternal, ATTR_INIT_ZERO, rtDwarfDecode_Bool)
1121};
1122
1123/** RTDWARFDIESUBPROGRAM description. */
1124static const RTDWARFDIEDESC g_LabelDesc = DIE_DESC_INIT(RTDWARFDIELABEL, g_aLabelAttrs);
1125
1126
1127/**
1128 * Tag names and descriptors.
1129 */
1130static const struct RTDWARFTAGDESC
1131{
1132 /** The tag value. */
1133 uint16_t uTag;
1134 /** The tag name as string. */
1135 const char *pszName;
1136 /** The DIE descriptor to use. */
1137 PCRTDWARFDIEDESC pDesc;
1138} g_aTagDescs[] =
1139{
1140#define TAGDESC(a_Name, a_pDesc) { DW_ ## a_Name, #a_Name, a_pDesc }
1141#define TAGDESC_EMPTY() { 0, NULL, NULL }
1142#define TAGDESC_CORE(a_Name) TAGDESC(a_Name, &g_CoreDieDesc)
1143 TAGDESC_EMPTY(), /* 0x00 */
1144 TAGDESC_CORE(TAG_array_type),
1145 TAGDESC_CORE(TAG_class_type),
1146 TAGDESC_CORE(TAG_entry_point),
1147 TAGDESC_CORE(TAG_enumeration_type), /* 0x04 */
1148 TAGDESC_CORE(TAG_formal_parameter),
1149 TAGDESC_EMPTY(),
1150 TAGDESC_EMPTY(),
1151 TAGDESC_CORE(TAG_imported_declaration), /* 0x08 */
1152 TAGDESC_EMPTY(),
1153 TAGDESC(TAG_label, &g_LabelDesc),
1154 TAGDESC_CORE(TAG_lexical_block),
1155 TAGDESC_EMPTY(), /* 0x0c */
1156 TAGDESC_CORE(TAG_member),
1157 TAGDESC_EMPTY(),
1158 TAGDESC_CORE(TAG_pointer_type),
1159 TAGDESC_CORE(TAG_reference_type), /* 0x10 */
1160 TAGDESC_CORE(TAG_compile_unit),
1161 TAGDESC_CORE(TAG_string_type),
1162 TAGDESC_CORE(TAG_structure_type),
1163 TAGDESC_EMPTY(), /* 0x14 */
1164 TAGDESC_CORE(TAG_subroutine_type),
1165 TAGDESC_CORE(TAG_typedef),
1166 TAGDESC_CORE(TAG_union_type),
1167 TAGDESC_CORE(TAG_unspecified_parameters), /* 0x18 */
1168 TAGDESC_CORE(TAG_variant),
1169 TAGDESC_CORE(TAG_common_block),
1170 TAGDESC_CORE(TAG_common_inclusion),
1171 TAGDESC_CORE(TAG_inheritance), /* 0x1c */
1172 TAGDESC_CORE(TAG_inlined_subroutine),
1173 TAGDESC_CORE(TAG_module),
1174 TAGDESC_CORE(TAG_ptr_to_member_type),
1175 TAGDESC_CORE(TAG_set_type), /* 0x20 */
1176 TAGDESC_CORE(TAG_subrange_type),
1177 TAGDESC_CORE(TAG_with_stmt),
1178 TAGDESC_CORE(TAG_access_declaration),
1179 TAGDESC_CORE(TAG_base_type), /* 0x24 */
1180 TAGDESC_CORE(TAG_catch_block),
1181 TAGDESC_CORE(TAG_const_type),
1182 TAGDESC_CORE(TAG_constant),
1183 TAGDESC_CORE(TAG_enumerator), /* 0x28 */
1184 TAGDESC_CORE(TAG_file_type),
1185 TAGDESC_CORE(TAG_friend),
1186 TAGDESC_CORE(TAG_namelist),
1187 TAGDESC_CORE(TAG_namelist_item), /* 0x2c */
1188 TAGDESC_CORE(TAG_packed_type),
1189 TAGDESC(TAG_subprogram, &g_SubProgramDesc),
1190 TAGDESC_CORE(TAG_template_type_parameter),
1191 TAGDESC_CORE(TAG_template_value_parameter), /* 0x30 */
1192 TAGDESC_CORE(TAG_thrown_type),
1193 TAGDESC_CORE(TAG_try_block),
1194 TAGDESC_CORE(TAG_variant_part),
1195 TAGDESC_CORE(TAG_variable), /* 0x34 */
1196 TAGDESC_CORE(TAG_volatile_type),
1197 TAGDESC_CORE(TAG_dwarf_procedure),
1198 TAGDESC_CORE(TAG_restrict_type),
1199 TAGDESC_CORE(TAG_interface_type), /* 0x38 */
1200 TAGDESC_CORE(TAG_namespace),
1201 TAGDESC_CORE(TAG_imported_module),
1202 TAGDESC_CORE(TAG_unspecified_type),
1203 TAGDESC_CORE(TAG_partial_unit), /* 0x3c */
1204 TAGDESC_CORE(TAG_imported_unit),
1205 TAGDESC_EMPTY(),
1206 TAGDESC_CORE(TAG_condition),
1207 TAGDESC_CORE(TAG_shared_type), /* 0x40 */
1208 TAGDESC_CORE(TAG_type_unit),
1209 TAGDESC_CORE(TAG_rvalue_reference_type),
1210 TAGDESC_CORE(TAG_template_alias)
1211#undef TAGDESC
1212#undef TAGDESC_EMPTY
1213#undef TAGDESC_CORE
1214};
1215
1216
1217/*********************************************************************************************************************************
1218* Internal Functions *
1219*********************************************************************************************************************************/
1220static int rtDwarfInfo_ParseDie(PRTDBGMODDWARF pThis, PRTDWARFDIE pDie, PCRTDWARFDIEDESC pDieDesc,
1221 PRTDWARFCURSOR pCursor, PCRTDWARFABBREV pAbbrev, bool fInitDie);
1222
1223
1224
1225#if defined(LOG_ENABLED) || defined(RT_STRICT)
1226
1227# if 0 /* unused */
1228/**
1229 * Turns a tag value into a string for logging purposes.
1230 *
1231 * @returns String name.
1232 * @param uTag The tag.
1233 */
1234static const char *rtDwarfLog_GetTagName(uint32_t uTag)
1235{
1236 if (uTag < RT_ELEMENTS(g_aTagDescs))
1237 {
1238 const char *pszTag = g_aTagDescs[uTag].pszName;
1239 if (pszTag)
1240 return pszTag;
1241 }
1242
1243 static char s_szStatic[32];
1244 RTStrPrintf(s_szStatic, sizeof(s_szStatic),"DW_TAG_%#x", uTag);
1245 return s_szStatic;
1246}
1247# endif
1248
1249
1250/**
1251 * Turns an attributevalue into a string for logging purposes.
1252 *
1253 * @returns String name.
1254 * @param uAttr The attribute.
1255 */
1256static const char *rtDwarfLog_AttrName(uint32_t uAttr)
1257{
1258 switch (uAttr)
1259 {
1260 RT_CASE_RET_STR(DW_AT_sibling);
1261 RT_CASE_RET_STR(DW_AT_location);
1262 RT_CASE_RET_STR(DW_AT_name);
1263 RT_CASE_RET_STR(DW_AT_ordering);
1264 RT_CASE_RET_STR(DW_AT_byte_size);
1265 RT_CASE_RET_STR(DW_AT_bit_offset);
1266 RT_CASE_RET_STR(DW_AT_bit_size);
1267 RT_CASE_RET_STR(DW_AT_stmt_list);
1268 RT_CASE_RET_STR(DW_AT_low_pc);
1269 RT_CASE_RET_STR(DW_AT_high_pc);
1270 RT_CASE_RET_STR(DW_AT_language);
1271 RT_CASE_RET_STR(DW_AT_discr);
1272 RT_CASE_RET_STR(DW_AT_discr_value);
1273 RT_CASE_RET_STR(DW_AT_visibility);
1274 RT_CASE_RET_STR(DW_AT_import);
1275 RT_CASE_RET_STR(DW_AT_string_length);
1276 RT_CASE_RET_STR(DW_AT_common_reference);
1277 RT_CASE_RET_STR(DW_AT_comp_dir);
1278 RT_CASE_RET_STR(DW_AT_const_value);
1279 RT_CASE_RET_STR(DW_AT_containing_type);
1280 RT_CASE_RET_STR(DW_AT_default_value);
1281 RT_CASE_RET_STR(DW_AT_inline);
1282 RT_CASE_RET_STR(DW_AT_is_optional);
1283 RT_CASE_RET_STR(DW_AT_lower_bound);
1284 RT_CASE_RET_STR(DW_AT_producer);
1285 RT_CASE_RET_STR(DW_AT_prototyped);
1286 RT_CASE_RET_STR(DW_AT_return_addr);
1287 RT_CASE_RET_STR(DW_AT_start_scope);
1288 RT_CASE_RET_STR(DW_AT_bit_stride);
1289 RT_CASE_RET_STR(DW_AT_upper_bound);
1290 RT_CASE_RET_STR(DW_AT_abstract_origin);
1291 RT_CASE_RET_STR(DW_AT_accessibility);
1292 RT_CASE_RET_STR(DW_AT_address_class);
1293 RT_CASE_RET_STR(DW_AT_artificial);
1294 RT_CASE_RET_STR(DW_AT_base_types);
1295 RT_CASE_RET_STR(DW_AT_calling_convention);
1296 RT_CASE_RET_STR(DW_AT_count);
1297 RT_CASE_RET_STR(DW_AT_data_member_location);
1298 RT_CASE_RET_STR(DW_AT_decl_column);
1299 RT_CASE_RET_STR(DW_AT_decl_file);
1300 RT_CASE_RET_STR(DW_AT_decl_line);
1301 RT_CASE_RET_STR(DW_AT_declaration);
1302 RT_CASE_RET_STR(DW_AT_discr_list);
1303 RT_CASE_RET_STR(DW_AT_encoding);
1304 RT_CASE_RET_STR(DW_AT_external);
1305 RT_CASE_RET_STR(DW_AT_frame_base);
1306 RT_CASE_RET_STR(DW_AT_friend);
1307 RT_CASE_RET_STR(DW_AT_identifier_case);
1308 RT_CASE_RET_STR(DW_AT_macro_info);
1309 RT_CASE_RET_STR(DW_AT_namelist_item);
1310 RT_CASE_RET_STR(DW_AT_priority);
1311 RT_CASE_RET_STR(DW_AT_segment);
1312 RT_CASE_RET_STR(DW_AT_specification);
1313 RT_CASE_RET_STR(DW_AT_static_link);
1314 RT_CASE_RET_STR(DW_AT_type);
1315 RT_CASE_RET_STR(DW_AT_use_location);
1316 RT_CASE_RET_STR(DW_AT_variable_parameter);
1317 RT_CASE_RET_STR(DW_AT_virtuality);
1318 RT_CASE_RET_STR(DW_AT_vtable_elem_location);
1319 RT_CASE_RET_STR(DW_AT_allocated);
1320 RT_CASE_RET_STR(DW_AT_associated);
1321 RT_CASE_RET_STR(DW_AT_data_location);
1322 RT_CASE_RET_STR(DW_AT_byte_stride);
1323 RT_CASE_RET_STR(DW_AT_entry_pc);
1324 RT_CASE_RET_STR(DW_AT_use_UTF8);
1325 RT_CASE_RET_STR(DW_AT_extension);
1326 RT_CASE_RET_STR(DW_AT_ranges);
1327 RT_CASE_RET_STR(DW_AT_trampoline);
1328 RT_CASE_RET_STR(DW_AT_call_column);
1329 RT_CASE_RET_STR(DW_AT_call_file);
1330 RT_CASE_RET_STR(DW_AT_call_line);
1331 RT_CASE_RET_STR(DW_AT_description);
1332 RT_CASE_RET_STR(DW_AT_binary_scale);
1333 RT_CASE_RET_STR(DW_AT_decimal_scale);
1334 RT_CASE_RET_STR(DW_AT_small);
1335 RT_CASE_RET_STR(DW_AT_decimal_sign);
1336 RT_CASE_RET_STR(DW_AT_digit_count);
1337 RT_CASE_RET_STR(DW_AT_picture_string);
1338 RT_CASE_RET_STR(DW_AT_mutable);
1339 RT_CASE_RET_STR(DW_AT_threads_scaled);
1340 RT_CASE_RET_STR(DW_AT_explicit);
1341 RT_CASE_RET_STR(DW_AT_object_pointer);
1342 RT_CASE_RET_STR(DW_AT_endianity);
1343 RT_CASE_RET_STR(DW_AT_elemental);
1344 RT_CASE_RET_STR(DW_AT_pure);
1345 RT_CASE_RET_STR(DW_AT_recursive);
1346 RT_CASE_RET_STR(DW_AT_signature);
1347 RT_CASE_RET_STR(DW_AT_main_subprogram);
1348 RT_CASE_RET_STR(DW_AT_data_bit_offset);
1349 RT_CASE_RET_STR(DW_AT_const_expr);
1350 RT_CASE_RET_STR(DW_AT_enum_class);
1351 RT_CASE_RET_STR(DW_AT_linkage_name);
1352 RT_CASE_RET_STR(DW_AT_MIPS_linkage_name);
1353 RT_CASE_RET_STR(DW_AT_WATCOM_memory_model);
1354 RT_CASE_RET_STR(DW_AT_WATCOM_references_start);
1355 RT_CASE_RET_STR(DW_AT_WATCOM_parm_entry);
1356 }
1357 static char s_szStatic[32];
1358 RTStrPrintf(s_szStatic, sizeof(s_szStatic),"DW_AT_%#x", uAttr);
1359 return s_szStatic;
1360}
1361
1362
1363/**
1364 * Turns a form value into a string for logging purposes.
1365 *
1366 * @returns String name.
1367 * @param uForm The form.
1368 */
1369static const char *rtDwarfLog_FormName(uint32_t uForm)
1370{
1371 switch (uForm)
1372 {
1373 RT_CASE_RET_STR(DW_FORM_addr);
1374 RT_CASE_RET_STR(DW_FORM_block2);
1375 RT_CASE_RET_STR(DW_FORM_block4);
1376 RT_CASE_RET_STR(DW_FORM_data2);
1377 RT_CASE_RET_STR(DW_FORM_data4);
1378 RT_CASE_RET_STR(DW_FORM_data8);
1379 RT_CASE_RET_STR(DW_FORM_string);
1380 RT_CASE_RET_STR(DW_FORM_block);
1381 RT_CASE_RET_STR(DW_FORM_block1);
1382 RT_CASE_RET_STR(DW_FORM_data1);
1383 RT_CASE_RET_STR(DW_FORM_flag);
1384 RT_CASE_RET_STR(DW_FORM_sdata);
1385 RT_CASE_RET_STR(DW_FORM_strp);
1386 RT_CASE_RET_STR(DW_FORM_udata);
1387 RT_CASE_RET_STR(DW_FORM_ref_addr);
1388 RT_CASE_RET_STR(DW_FORM_ref1);
1389 RT_CASE_RET_STR(DW_FORM_ref2);
1390 RT_CASE_RET_STR(DW_FORM_ref4);
1391 RT_CASE_RET_STR(DW_FORM_ref8);
1392 RT_CASE_RET_STR(DW_FORM_ref_udata);
1393 RT_CASE_RET_STR(DW_FORM_indirect);
1394 RT_CASE_RET_STR(DW_FORM_sec_offset);
1395 RT_CASE_RET_STR(DW_FORM_exprloc);
1396 RT_CASE_RET_STR(DW_FORM_flag_present);
1397 RT_CASE_RET_STR(DW_FORM_ref_sig8);
1398 }
1399 static char s_szStatic[32];
1400 RTStrPrintf(s_szStatic, sizeof(s_szStatic),"DW_FORM_%#x", uForm);
1401 return s_szStatic;
1402}
1403
1404#endif /* LOG_ENABLED || RT_STRICT */
1405
1406
1407
1408/** @callback_method_impl{FNRTLDRENUMSEGS} */
1409static DECLCALLBACK(int) rtDbgModDwarfScanSegmentsCallback(RTLDRMOD hLdrMod, PCRTLDRSEG pSeg, void *pvUser)
1410{
1411 PRTDBGMODDWARF pThis = (PRTDBGMODDWARF)pvUser;
1412 Log(("Segment %.*s: LinkAddress=%#llx RVA=%#llx cb=%#llx\n",
1413 pSeg->cchName, pSeg->pszName, (uint64_t)pSeg->LinkAddress, (uint64_t)pSeg->RVA, pSeg->cb));
1414 NOREF(hLdrMod);
1415
1416 /* Count relevant segments. */
1417 if (pSeg->RVA != NIL_RTLDRADDR)
1418 pThis->cSegs++;
1419
1420 return VINF_SUCCESS;
1421}
1422
1423
1424/** @callback_method_impl{FNRTLDRENUMSEGS} */
1425static DECLCALLBACK(int) rtDbgModDwarfAddSegmentsCallback(RTLDRMOD hLdrMod, PCRTLDRSEG pSeg, void *pvUser)
1426{
1427 PRTDBGMODDWARF pThis = (PRTDBGMODDWARF)pvUser;
1428 Log(("Segment %.*s: LinkAddress=%#llx RVA=%#llx cb=%#llx cbMapped=%#llx\n",
1429 pSeg->cchName, pSeg->pszName, (uint64_t)pSeg->LinkAddress, (uint64_t)pSeg->RVA, pSeg->cb, pSeg->cbMapped));
1430 NOREF(hLdrMod);
1431 Assert(pSeg->cchName > 0);
1432 Assert(!pSeg->pszName[pSeg->cchName]);
1433
1434 /* If the segment doesn't have a mapping, just add a dummy so the indexing
1435 works out correctly (same as for the image). */
1436 if (pSeg->RVA == NIL_RTLDRADDR)
1437 return RTDbgModSegmentAdd(pThis->hCnt, 0, 0, pSeg->pszName, 0 /*fFlags*/, NULL);
1438
1439 /* The link address is 0 for all segments in a relocatable ELF image. */
1440 RTLDRADDR cb = pSeg->cb;
1441 if ( cb < pSeg->cbMapped
1442 && RTLdrGetFormat(hLdrMod) != RTLDRFMT_LX /* for debugging our drivers; 64KB section align by linker, 4KB by loader. */
1443 )
1444 cb = pSeg->cbMapped;
1445 return RTDbgModSegmentAdd(pThis->hCnt, pSeg->RVA, cb, pSeg->pszName, 0 /*fFlags*/, NULL);
1446}
1447
1448
1449/**
1450 * Calls rtDbgModDwarfAddSegmentsCallback for each segment in the executable
1451 * image.
1452 *
1453 * @returns IPRT status code.
1454 * @param pThis The DWARF instance.
1455 */
1456static int rtDbgModDwarfAddSegmentsFromImage(PRTDBGMODDWARF pThis)
1457{
1458 AssertReturn(pThis->pImgMod && pThis->pImgMod->pImgVt, VERR_INTERNAL_ERROR_2);
1459 Assert(!pThis->cSegs);
1460 int rc = pThis->pImgMod->pImgVt->pfnEnumSegments(pThis->pImgMod, rtDbgModDwarfScanSegmentsCallback, pThis);
1461 if (RT_SUCCESS(rc))
1462 {
1463 if (pThis->cSegs == 0)
1464 pThis->iWatcomPass = 1;
1465 else
1466 {
1467 pThis->cSegs = 0;
1468 pThis->iWatcomPass = -1;
1469 rc = pThis->pImgMod->pImgVt->pfnEnumSegments(pThis->pImgMod, rtDbgModDwarfAddSegmentsCallback, pThis);
1470 }
1471 }
1472
1473 return rc;
1474}
1475
1476
1477/**
1478 * Looks up a segment.
1479 *
1480 * @returns Pointer to the segment on success, NULL if not found.
1481 * @param pThis The DWARF instance.
1482 * @param uSeg The segment number / selector.
1483 */
1484static PRTDBGDWARFSEG rtDbgModDwarfFindSegment(PRTDBGMODDWARF pThis, RTSEL uSeg)
1485{
1486 uint32_t cSegs = pThis->cSegs;
1487 uint32_t iSeg = pThis->iSegHint;
1488 PRTDBGDWARFSEG paSegs = pThis->paSegs;
1489 if ( iSeg < cSegs
1490 && paSegs[iSeg].uSegment == uSeg)
1491 return &paSegs[iSeg];
1492
1493 for (iSeg = 0; iSeg < cSegs; iSeg++)
1494 if (uSeg == paSegs[iSeg].uSegment)
1495 {
1496 pThis->iSegHint = iSeg;
1497 return &paSegs[iSeg];
1498 }
1499
1500 AssertFailed();
1501 return NULL;
1502}
1503
1504
1505/**
1506 * Record a segment:offset during pass 1.
1507 *
1508 * @returns IPRT status code.
1509 * @param pThis The DWARF instance.
1510 * @param uSeg The segment number / selector.
1511 * @param offSeg The segment offset.
1512 */
1513static int rtDbgModDwarfRecordSegOffset(PRTDBGMODDWARF pThis, RTSEL uSeg, uint64_t offSeg)
1514{
1515 /* Look up the segment. */
1516 uint32_t cSegs = pThis->cSegs;
1517 uint32_t iSeg = pThis->iSegHint;
1518 PRTDBGDWARFSEG paSegs = pThis->paSegs;
1519 if ( iSeg >= cSegs
1520 || paSegs[iSeg].uSegment != uSeg)
1521 {
1522 for (iSeg = 0; iSeg < cSegs; iSeg++)
1523 if (uSeg <= paSegs[iSeg].uSegment)
1524 break;
1525 if ( iSeg >= cSegs
1526 || paSegs[iSeg].uSegment != uSeg)
1527 {
1528 /* Add */
1529 void *pvNew = RTMemRealloc(paSegs, (pThis->cSegs + 1) * sizeof(paSegs[0]));
1530 if (!pvNew)
1531 return VERR_NO_MEMORY;
1532 pThis->paSegs = paSegs = (PRTDBGDWARFSEG)pvNew;
1533 if (iSeg != cSegs)
1534 memmove(&paSegs[iSeg + 1], &paSegs[iSeg], (cSegs - iSeg) * sizeof(paSegs[0]));
1535 paSegs[iSeg].offHighest = offSeg;
1536 paSegs[iSeg].uBaseAddr = 0;
1537 paSegs[iSeg].cbSegment = 0;
1538 paSegs[iSeg].uSegment = uSeg;
1539 pThis->cSegs++;
1540 }
1541
1542 pThis->iSegHint = iSeg;
1543 }
1544
1545 /* Increase it's range? */
1546 if (paSegs[iSeg].offHighest < offSeg)
1547 {
1548 Log3(("rtDbgModDwarfRecordSegOffset: iSeg=%d uSeg=%#06x offSeg=%#llx\n", iSeg, uSeg, offSeg));
1549 paSegs[iSeg].offHighest = offSeg;
1550 }
1551
1552 return VINF_SUCCESS;
1553}
1554
1555
1556/**
1557 * Calls pfnSegmentAdd for each segment in the executable image.
1558 *
1559 * @returns IPRT status code.
1560 * @param pThis The DWARF instance.
1561 */
1562static int rtDbgModDwarfAddSegmentsFromPass1(PRTDBGMODDWARF pThis)
1563{
1564 AssertReturn(pThis->cSegs, VERR_DWARF_BAD_INFO);
1565 uint32_t const cSegs = pThis->cSegs;
1566 PRTDBGDWARFSEG paSegs = pThis->paSegs;
1567
1568 /*
1569 * Are the segments assigned more or less in numerical order?
1570 */
1571 if ( paSegs[0].uSegment < 16U
1572 && paSegs[cSegs - 1].uSegment - paSegs[0].uSegment + 1U <= cSegs + 16U)
1573 {
1574 /** @todo heuristics, plase. */
1575 AssertFailedReturn(VERR_DWARF_TODO);
1576
1577 }
1578 /*
1579 * Assume DOS segmentation.
1580 */
1581 else
1582 {
1583 for (uint32_t iSeg = 0; iSeg < cSegs; iSeg++)
1584 paSegs[iSeg].uBaseAddr = (uint32_t)paSegs[iSeg].uSegment << 16;
1585 for (uint32_t iSeg = 0; iSeg < cSegs; iSeg++)
1586 paSegs[iSeg].cbSegment = paSegs[iSeg].offHighest;
1587 }
1588
1589 /*
1590 * Add them.
1591 */
1592 for (uint32_t iSeg = 0; iSeg < cSegs; iSeg++)
1593 {
1594 Log3(("rtDbgModDwarfAddSegmentsFromPass1: Seg#%u: %#010llx LB %#llx uSegment=%#x\n",
1595 iSeg, paSegs[iSeg].uBaseAddr, paSegs[iSeg].cbSegment, paSegs[iSeg].uSegment));
1596 char szName[32];
1597 RTStrPrintf(szName, sizeof(szName), "seg-%#04xh", paSegs[iSeg].uSegment);
1598 int rc = RTDbgModSegmentAdd(pThis->hCnt, paSegs[iSeg].uBaseAddr, paSegs[iSeg].cbSegment,
1599 szName, 0 /*fFlags*/, NULL);
1600 if (RT_FAILURE(rc))
1601 return rc;
1602 }
1603
1604 return VINF_SUCCESS;
1605}
1606
1607
1608/**
1609 * Loads a DWARF section from the image file.
1610 *
1611 * @returns IPRT status code.
1612 * @param pThis The DWARF instance.
1613 * @param enmSect The section to load.
1614 */
1615static int rtDbgModDwarfLoadSection(PRTDBGMODDWARF pThis, krtDbgModDwarfSect enmSect)
1616{
1617 /*
1618 * Don't load stuff twice.
1619 */
1620 if (pThis->aSections[enmSect].pv)
1621 return VINF_SUCCESS;
1622
1623 /*
1624 * Sections that are not present cannot be loaded, treat them like they
1625 * are empty
1626 */
1627 if (!pThis->aSections[enmSect].fPresent)
1628 {
1629 Assert(pThis->aSections[enmSect].cb);
1630 return VINF_SUCCESS;
1631 }
1632 if (!pThis->aSections[enmSect].cb)
1633 return VINF_SUCCESS;
1634
1635 /*
1636 * Sections must be readable with the current image interface.
1637 */
1638 if (pThis->aSections[enmSect].offFile < 0)
1639 return VERR_OUT_OF_RANGE;
1640
1641 /*
1642 * Do the job.
1643 */
1644 return pThis->pDbgInfoMod->pImgVt->pfnMapPart(pThis->pDbgInfoMod,
1645 pThis->aSections[enmSect].iDbgInfo,
1646 pThis->aSections[enmSect].offFile,
1647 pThis->aSections[enmSect].cb,
1648 &pThis->aSections[enmSect].pv);
1649}
1650
1651
1652#ifdef SOME_UNUSED_FUNCTION
1653/**
1654 * Unloads a DWARF section previously mapped by rtDbgModDwarfLoadSection.
1655 *
1656 * @returns IPRT status code.
1657 * @param pThis The DWARF instance.
1658 * @param enmSect The section to unload.
1659 */
1660static int rtDbgModDwarfUnloadSection(PRTDBGMODDWARF pThis, krtDbgModDwarfSect enmSect)
1661{
1662 if (!pThis->aSections[enmSect].pv)
1663 return VINF_SUCCESS;
1664
1665 int rc = pThis->pDbgInfoMod->pImgVt->pfnUnmapPart(pThis->pDbgInfoMod, pThis->aSections[enmSect].cb, &pThis->aSections[enmSect].pv);
1666 AssertRC(rc);
1667 return rc;
1668}
1669#endif
1670
1671
1672/**
1673 * Converts to UTF-8 or otherwise makes sure it's valid UTF-8.
1674 *
1675 * @returns IPRT status code.
1676 * @param pThis The DWARF instance.
1677 * @param ppsz Pointer to the string pointer. May be
1678 * reallocated (RTStr*).
1679 */
1680static int rtDbgModDwarfStringToUtf8(PRTDBGMODDWARF pThis, char **ppsz)
1681{
1682 /** @todo DWARF & UTF-8. */
1683 NOREF(pThis);
1684 RTStrPurgeEncoding(*ppsz);
1685 return VINF_SUCCESS;
1686}
1687
1688
1689/**
1690 * Convers a link address into a segment+offset or RVA.
1691 *
1692 * @returns IPRT status code.
1693 * @param pThis The DWARF instance.
1694 * @param uSegment The segment, 0 if not applicable.
1695 * @param LinkAddress The address to convert..
1696 * @param piSeg The segment index.
1697 * @param poffSeg Where to return the segment offset.
1698 */
1699static int rtDbgModDwarfLinkAddressToSegOffset(PRTDBGMODDWARF pThis, RTSEL uSegment, uint64_t LinkAddress,
1700 PRTDBGSEGIDX piSeg, PRTLDRADDR poffSeg)
1701{
1702 if (pThis->paSegs)
1703 {
1704 PRTDBGDWARFSEG pSeg = rtDbgModDwarfFindSegment(pThis, uSegment);
1705 if (pSeg)
1706 {
1707 *piSeg = pSeg - pThis->paSegs;
1708 *poffSeg = LinkAddress;
1709 return VINF_SUCCESS;
1710 }
1711 }
1712
1713 if (pThis->fUseLinkAddress)
1714 return pThis->pImgMod->pImgVt->pfnLinkAddressToSegOffset(pThis->pImgMod, LinkAddress, piSeg, poffSeg);
1715
1716 /* If we have a non-zero segment number, assume it's correct for now.
1717 This helps loading watcom linked LX drivers. */
1718 if (uSegment > 0)
1719 {
1720 *piSeg = uSegment - 1;
1721 *poffSeg = LinkAddress;
1722 return VINF_SUCCESS;
1723 }
1724
1725 return pThis->pImgMod->pImgVt->pfnRvaToSegOffset(pThis->pImgMod, LinkAddress, piSeg, poffSeg);
1726}
1727
1728
1729/**
1730 * Converts a segment+offset address into an RVA.
1731 *
1732 * @returns IPRT status code.
1733 * @param pThis The DWARF instance.
1734 * @param idxSegment The segment index.
1735 * @param offSegment The segment offset.
1736 * @param puRva Where to return the calculated RVA.
1737 */
1738static int rtDbgModDwarfSegOffsetToRva(PRTDBGMODDWARF pThis, RTDBGSEGIDX idxSegment, uint64_t offSegment, PRTUINTPTR puRva)
1739{
1740 if (pThis->paSegs)
1741 {
1742 PRTDBGDWARFSEG pSeg = rtDbgModDwarfFindSegment(pThis, idxSegment);
1743 if (pSeg)
1744 {
1745 *puRva = pSeg->uBaseAddr + offSegment;
1746 return VINF_SUCCESS;
1747 }
1748 }
1749
1750 RTUINTPTR uRva = RTDbgModSegmentRva(pThis->pImgMod, idxSegment);
1751 if (uRva != RTUINTPTR_MAX)
1752 {
1753 *puRva = uRva + offSegment;
1754 return VINF_SUCCESS;
1755 }
1756 return VERR_INVALID_POINTER;
1757}
1758
1759/**
1760 * Converts a segment+offset address into an RVA.
1761 *
1762 * @returns IPRT status code.
1763 * @param pThis The DWARF instance.
1764 * @param uRva The RVA to convert.
1765 * @param pidxSegment Where to return the segment index.
1766 * @param poffSegment Where to return the segment offset.
1767 */
1768static int rtDbgModDwarfRvaToSegOffset(PRTDBGMODDWARF pThis, RTUINTPTR uRva, RTDBGSEGIDX *pidxSegment, uint64_t *poffSegment)
1769{
1770 RTUINTPTR offSeg = 0;
1771 RTDBGSEGIDX idxSeg = RTDbgModRvaToSegOff(pThis->pImgMod, uRva, &offSeg);
1772 if (idxSeg != NIL_RTDBGSEGIDX)
1773 {
1774 *pidxSegment = idxSeg;
1775 *poffSegment = offSeg;
1776 return VINF_SUCCESS;
1777 }
1778 return VERR_INVALID_POINTER;
1779}
1780
1781
1782
1783/*
1784 *
1785 * DWARF Cursor.
1786 * DWARF Cursor.
1787 * DWARF Cursor.
1788 *
1789 */
1790
1791
1792/**
1793 * Reads a 8-bit unsigned integer and advances the cursor.
1794 *
1795 * @returns 8-bit unsigned integer. On error RTDWARFCURSOR::rc is set and @a
1796 * uErrValue is returned.
1797 * @param pCursor The cursor.
1798 * @param uErrValue What to return on read error.
1799 */
1800static uint8_t rtDwarfCursor_GetU8(PRTDWARFCURSOR pCursor, uint8_t uErrValue)
1801{
1802 if (pCursor->cbUnitLeft < 1)
1803 {
1804 pCursor->rc = VERR_DWARF_UNEXPECTED_END;
1805 return uErrValue;
1806 }
1807
1808 uint8_t u8 = pCursor->pb[0];
1809 pCursor->pb += 1;
1810 pCursor->cbUnitLeft -= 1;
1811 pCursor->cbLeft -= 1;
1812 return u8;
1813}
1814
1815
1816/**
1817 * Reads a 16-bit unsigned integer and advances the cursor.
1818 *
1819 * @returns 16-bit unsigned integer. On error RTDWARFCURSOR::rc is set and @a
1820 * uErrValue is returned.
1821 * @param pCursor The cursor.
1822 * @param uErrValue What to return on read error.
1823 */
1824static uint16_t rtDwarfCursor_GetU16(PRTDWARFCURSOR pCursor, uint16_t uErrValue)
1825{
1826 if (pCursor->cbUnitLeft < 2)
1827 {
1828 pCursor->pb += pCursor->cbUnitLeft;
1829 pCursor->cbLeft -= pCursor->cbUnitLeft;
1830 pCursor->cbUnitLeft = 0;
1831 pCursor->rc = VERR_DWARF_UNEXPECTED_END;
1832 return uErrValue;
1833 }
1834
1835 uint16_t u16 = RT_MAKE_U16(pCursor->pb[0], pCursor->pb[1]);
1836 pCursor->pb += 2;
1837 pCursor->cbUnitLeft -= 2;
1838 pCursor->cbLeft -= 2;
1839 if (!pCursor->fNativEndian)
1840 u16 = RT_BSWAP_U16(u16);
1841 return u16;
1842}
1843
1844
1845/**
1846 * Reads a 32-bit unsigned integer and advances the cursor.
1847 *
1848 * @returns 32-bit unsigned integer. On error RTDWARFCURSOR::rc is set and @a
1849 * uErrValue is returned.
1850 * @param pCursor The cursor.
1851 * @param uErrValue What to return on read error.
1852 */
1853static uint32_t rtDwarfCursor_GetU32(PRTDWARFCURSOR pCursor, uint32_t uErrValue)
1854{
1855 if (pCursor->cbUnitLeft < 4)
1856 {
1857 pCursor->pb += pCursor->cbUnitLeft;
1858 pCursor->cbLeft -= pCursor->cbUnitLeft;
1859 pCursor->cbUnitLeft = 0;
1860 pCursor->rc = VERR_DWARF_UNEXPECTED_END;
1861 return uErrValue;
1862 }
1863
1864 uint32_t u32 = RT_MAKE_U32_FROM_U8(pCursor->pb[0], pCursor->pb[1], pCursor->pb[2], pCursor->pb[3]);
1865 pCursor->pb += 4;
1866 pCursor->cbUnitLeft -= 4;
1867 pCursor->cbLeft -= 4;
1868 if (!pCursor->fNativEndian)
1869 u32 = RT_BSWAP_U32(u32);
1870 return u32;
1871}
1872
1873
1874/**
1875 * Reads a 64-bit unsigned integer and advances the cursor.
1876 *
1877 * @returns 64-bit unsigned integer. On error RTDWARFCURSOR::rc is set and @a
1878 * uErrValue is returned.
1879 * @param pCursor The cursor.
1880 * @param uErrValue What to return on read error.
1881 */
1882static uint64_t rtDwarfCursor_GetU64(PRTDWARFCURSOR pCursor, uint64_t uErrValue)
1883{
1884 if (pCursor->cbUnitLeft < 8)
1885 {
1886 pCursor->pb += pCursor->cbUnitLeft;
1887 pCursor->cbLeft -= pCursor->cbUnitLeft;
1888 pCursor->cbUnitLeft = 0;
1889 pCursor->rc = VERR_DWARF_UNEXPECTED_END;
1890 return uErrValue;
1891 }
1892
1893 uint64_t u64 = RT_MAKE_U64_FROM_U8(pCursor->pb[0], pCursor->pb[1], pCursor->pb[2], pCursor->pb[3],
1894 pCursor->pb[4], pCursor->pb[5], pCursor->pb[6], pCursor->pb[7]);
1895 pCursor->pb += 8;
1896 pCursor->cbUnitLeft -= 8;
1897 pCursor->cbLeft -= 8;
1898 if (!pCursor->fNativEndian)
1899 u64 = RT_BSWAP_U64(u64);
1900 return u64;
1901}
1902
1903
1904/**
1905 * Reads an unsigned LEB128 encoded number.
1906 *
1907 * @returns unsigned 64-bit number. On error RTDWARFCURSOR::rc is set and @a
1908 * uErrValue is returned.
1909 * @param pCursor The cursor.
1910 * @param uErrValue The value to return on error.
1911 */
1912static uint64_t rtDwarfCursor_GetULeb128(PRTDWARFCURSOR pCursor, uint64_t uErrValue)
1913{
1914 if (pCursor->cbUnitLeft < 1)
1915 {
1916 pCursor->rc = VERR_DWARF_UNEXPECTED_END;
1917 return uErrValue;
1918 }
1919
1920 /*
1921 * Special case - single byte.
1922 */
1923 uint8_t b = pCursor->pb[0];
1924 if (!(b & 0x80))
1925 {
1926 pCursor->pb += 1;
1927 pCursor->cbUnitLeft -= 1;
1928 pCursor->cbLeft -= 1;
1929 return b;
1930 }
1931
1932 /*
1933 * Generic case.
1934 */
1935 /* Decode. */
1936 uint32_t off = 1;
1937 uint64_t u64Ret = b & 0x7f;
1938 do
1939 {
1940 if (off == pCursor->cbUnitLeft)
1941 {
1942 pCursor->rc = VERR_DWARF_UNEXPECTED_END;
1943 u64Ret = uErrValue;
1944 break;
1945 }
1946 b = pCursor->pb[off];
1947 u64Ret |= (b & 0x7f) << off * 7;
1948 off++;
1949 } while (b & 0x80);
1950
1951 /* Update the cursor. */
1952 pCursor->pb += off;
1953 pCursor->cbUnitLeft -= off;
1954 pCursor->cbLeft -= off;
1955
1956 /* Check the range. */
1957 uint32_t cBits = off * 7;
1958 if (cBits > 64)
1959 {
1960 pCursor->rc = VERR_DWARF_LEB_OVERFLOW;
1961 u64Ret = uErrValue;
1962 }
1963
1964 return u64Ret;
1965}
1966
1967
1968/**
1969 * Reads a signed LEB128 encoded number.
1970 *
1971 * @returns signed 64-bit number. On error RTDWARFCURSOR::rc is set and @a
1972 * uErrValue is returned.
1973 * @param pCursor The cursor.
1974 * @param sErrValue The value to return on error.
1975 */
1976static int64_t rtDwarfCursor_GetSLeb128(PRTDWARFCURSOR pCursor, int64_t sErrValue)
1977{
1978 if (pCursor->cbUnitLeft < 1)
1979 {
1980 pCursor->rc = VERR_DWARF_UNEXPECTED_END;
1981 return sErrValue;
1982 }
1983
1984 /*
1985 * Special case - single byte.
1986 */
1987 uint8_t b = pCursor->pb[0];
1988 if (!(b & 0x80))
1989 {
1990 pCursor->pb += 1;
1991 pCursor->cbUnitLeft -= 1;
1992 pCursor->cbLeft -= 1;
1993 if (b & 0x40)
1994 b |= 0x80;
1995 return (int8_t)b;
1996 }
1997
1998 /*
1999 * Generic case.
2000 */
2001 /* Decode it. */
2002 uint32_t off = 1;
2003 uint64_t u64Ret = b & 0x7f;
2004 do
2005 {
2006 if (off == pCursor->cbUnitLeft)
2007 {
2008 pCursor->rc = VERR_DWARF_UNEXPECTED_END;
2009 u64Ret = (uint64_t)sErrValue;
2010 break;
2011 }
2012 b = pCursor->pb[off];
2013 u64Ret |= (b & 0x7f) << off * 7;
2014 off++;
2015 } while (b & 0x80);
2016
2017 /* Update cursor. */
2018 pCursor->pb += off;
2019 pCursor->cbUnitLeft -= off;
2020 pCursor->cbLeft -= off;
2021
2022 /* Check the range. */
2023 uint32_t cBits = off * 7;
2024 if (cBits > 64)
2025 {
2026 pCursor->rc = VERR_DWARF_LEB_OVERFLOW;
2027 u64Ret = (uint64_t)sErrValue;
2028 }
2029 /* Sign extend the value. */
2030 else if (u64Ret & RT_BIT_64(cBits - 1))
2031 u64Ret |= ~(RT_BIT_64(cBits - 1) - 1);
2032
2033 return (int64_t)u64Ret;
2034}
2035
2036
2037/**
2038 * Reads an unsigned LEB128 encoded number, max 32-bit width.
2039 *
2040 * @returns unsigned 32-bit number. On error RTDWARFCURSOR::rc is set and @a
2041 * uErrValue is returned.
2042 * @param pCursor The cursor.
2043 * @param uErrValue The value to return on error.
2044 */
2045static uint32_t rtDwarfCursor_GetULeb128AsU32(PRTDWARFCURSOR pCursor, uint32_t uErrValue)
2046{
2047 uint64_t u64 = rtDwarfCursor_GetULeb128(pCursor, uErrValue);
2048 if (u64 > UINT32_MAX)
2049 {
2050 pCursor->rc = VERR_DWARF_LEB_OVERFLOW;
2051 return uErrValue;
2052 }
2053 return (uint32_t)u64;
2054}
2055
2056
2057/**
2058 * Reads a signed LEB128 encoded number, max 32-bit width.
2059 *
2060 * @returns signed 32-bit number. On error RTDWARFCURSOR::rc is set and @a
2061 * uErrValue is returned.
2062 * @param pCursor The cursor.
2063 * @param sErrValue The value to return on error.
2064 */
2065static int32_t rtDwarfCursor_GetSLeb128AsS32(PRTDWARFCURSOR pCursor, int32_t sErrValue)
2066{
2067 int64_t s64 = rtDwarfCursor_GetSLeb128(pCursor, sErrValue);
2068 if (s64 > INT32_MAX || s64 < INT32_MIN)
2069 {
2070 pCursor->rc = VERR_DWARF_LEB_OVERFLOW;
2071 return sErrValue;
2072 }
2073 return (int32_t)s64;
2074}
2075
2076
2077/**
2078 * Skips a LEB128 encoded number.
2079 *
2080 * @returns IPRT status code.
2081 * @param pCursor The cursor.
2082 */
2083static int rtDwarfCursor_SkipLeb128(PRTDWARFCURSOR pCursor)
2084{
2085 if (RT_FAILURE(pCursor->rc))
2086 return pCursor->rc;
2087
2088 if (pCursor->cbUnitLeft < 1)
2089 return pCursor->rc = VERR_DWARF_UNEXPECTED_END;
2090
2091 uint32_t offSkip = 1;
2092 if (pCursor->pb[0] & 0x80)
2093 do
2094 {
2095 if (offSkip == pCursor->cbUnitLeft)
2096 {
2097 pCursor->rc = VERR_DWARF_UNEXPECTED_END;
2098 break;
2099 }
2100 } while (pCursor->pb[offSkip++] & 0x80);
2101
2102 pCursor->pb += offSkip;
2103 pCursor->cbUnitLeft -= offSkip;
2104 pCursor->cbLeft -= offSkip;
2105 return pCursor->rc;
2106}
2107
2108
2109/**
2110 * Advances the cursor a given number of bytes.
2111 *
2112 * @returns IPRT status code.
2113 * @param pCursor The cursor.
2114 * @param offSkip The number of bytes to advance.
2115 */
2116static int rtDwarfCursor_SkipBytes(PRTDWARFCURSOR pCursor, uint64_t offSkip)
2117{
2118 if (RT_FAILURE(pCursor->rc))
2119 return pCursor->rc;
2120 if (pCursor->cbUnitLeft < offSkip)
2121 return pCursor->rc = VERR_DWARF_UNEXPECTED_END;
2122
2123 size_t const offSkipSizeT = (size_t)offSkip;
2124 pCursor->cbUnitLeft -= offSkipSizeT;
2125 pCursor->cbLeft -= offSkipSizeT;
2126 pCursor->pb += offSkipSizeT;
2127
2128 return VINF_SUCCESS;
2129}
2130
2131
2132/**
2133 * Reads a zero terminated string, advancing the cursor beyond the terminator.
2134 *
2135 * @returns Pointer to the string.
2136 * @param pCursor The cursor.
2137 * @param pszErrValue What to return if the string isn't terminated
2138 * before the end of the unit.
2139 */
2140static const char *rtDwarfCursor_GetSZ(PRTDWARFCURSOR pCursor, const char *pszErrValue)
2141{
2142 const char *pszRet = (const char *)pCursor->pb;
2143 for (;;)
2144 {
2145 if (!pCursor->cbUnitLeft)
2146 {
2147 pCursor->rc = VERR_DWARF_BAD_STRING;
2148 return pszErrValue;
2149 }
2150 pCursor->cbUnitLeft--;
2151 pCursor->cbLeft--;
2152 if (!*pCursor->pb++)
2153 break;
2154 }
2155 return pszRet;
2156}
2157
2158
2159/**
2160 * Reads a 1, 2, 4 or 8 byte unsigned value.
2161 *
2162 * @returns 64-bit unsigned value.
2163 * @param pCursor The cursor.
2164 * @param cbValue The value size.
2165 * @param uErrValue The error value.
2166 */
2167static uint64_t rtDwarfCursor_GetVarSizedU(PRTDWARFCURSOR pCursor, size_t cbValue, uint64_t uErrValue)
2168{
2169 uint64_t u64Ret;
2170 switch (cbValue)
2171 {
2172 case 1: u64Ret = rtDwarfCursor_GetU8( pCursor, UINT8_MAX); break;
2173 case 2: u64Ret = rtDwarfCursor_GetU16(pCursor, UINT16_MAX); break;
2174 case 4: u64Ret = rtDwarfCursor_GetU32(pCursor, UINT32_MAX); break;
2175 case 8: u64Ret = rtDwarfCursor_GetU64(pCursor, UINT64_MAX); break;
2176 default:
2177 pCursor->rc = VERR_DWARF_BAD_INFO;
2178 return uErrValue;
2179 }
2180 if (RT_FAILURE(pCursor->rc))
2181 return uErrValue;
2182 return u64Ret;
2183}
2184
2185
2186#if 0 /* unused */
2187/**
2188 * Gets the pointer to a variable size block and advances the cursor.
2189 *
2190 * @returns Pointer to the block at the current cursor location. On error
2191 * RTDWARFCURSOR::rc is set and NULL returned.
2192 * @param pCursor The cursor.
2193 * @param cbBlock The block size.
2194 */
2195static const uint8_t *rtDwarfCursor_GetBlock(PRTDWARFCURSOR pCursor, uint32_t cbBlock)
2196{
2197 if (cbBlock > pCursor->cbUnitLeft)
2198 {
2199 pCursor->rc = VERR_DWARF_UNEXPECTED_END;
2200 return NULL;
2201 }
2202
2203 uint8_t const *pb = &pCursor->pb[0];
2204 pCursor->pb += cbBlock;
2205 pCursor->cbUnitLeft -= cbBlock;
2206 pCursor->cbLeft -= cbBlock;
2207 return pb;
2208}
2209#endif
2210
2211
2212/**
2213 * Reads an unsigned DWARF half number.
2214 *
2215 * @returns The number. On error RTDWARFCURSOR::rc is set and @a
2216 * uErrValue is returned.
2217 * @param pCursor The cursor.
2218 * @param uErrValue What to return on error.
2219 */
2220static uint16_t rtDwarfCursor_GetUHalf(PRTDWARFCURSOR pCursor, uint16_t uErrValue)
2221{
2222 return rtDwarfCursor_GetU16(pCursor, uErrValue);
2223}
2224
2225
2226/**
2227 * Reads an unsigned DWARF byte number.
2228 *
2229 * @returns The number. On error RTDWARFCURSOR::rc is set and @a
2230 * uErrValue is returned.
2231 * @param pCursor The cursor.
2232 * @param uErrValue What to return on error.
2233 */
2234static uint8_t rtDwarfCursor_GetUByte(PRTDWARFCURSOR pCursor, uint8_t uErrValue)
2235{
2236 return rtDwarfCursor_GetU8(pCursor, uErrValue);
2237}
2238
2239
2240/**
2241 * Reads a signed DWARF byte number.
2242 *
2243 * @returns The number. On error RTDWARFCURSOR::rc is set and @a
2244 * uErrValue is returned.
2245 * @param pCursor The cursor.
2246 * @param iErrValue What to return on error.
2247 */
2248static int8_t rtDwarfCursor_GetSByte(PRTDWARFCURSOR pCursor, int8_t iErrValue)
2249{
2250 return (int8_t)rtDwarfCursor_GetU8(pCursor, (uint8_t)iErrValue);
2251}
2252
2253
2254/**
2255 * Reads a unsigned DWARF offset value.
2256 *
2257 * @returns The value. On error RTDWARFCURSOR::rc is set and @a
2258 * uErrValue is returned.
2259 * @param pCursor The cursor.
2260 * @param uErrValue What to return on error.
2261 */
2262static uint64_t rtDwarfCursor_GetUOff(PRTDWARFCURSOR pCursor, uint64_t uErrValue)
2263{
2264 if (pCursor->f64bitDwarf)
2265 return rtDwarfCursor_GetU64(pCursor, uErrValue);
2266 return rtDwarfCursor_GetU32(pCursor, (uint32_t)uErrValue);
2267}
2268
2269
2270/**
2271 * Reads a unsigned DWARF native offset value.
2272 *
2273 * @returns The value. On error RTDWARFCURSOR::rc is set and @a
2274 * uErrValue is returned.
2275 * @param pCursor The cursor.
2276 * @param uErrValue What to return on error.
2277 */
2278static uint64_t rtDwarfCursor_GetNativeUOff(PRTDWARFCURSOR pCursor, uint64_t uErrValue)
2279{
2280 switch (pCursor->cbNativeAddr)
2281 {
2282 case 1: return rtDwarfCursor_GetU8(pCursor, (uint8_t )uErrValue);
2283 case 2: return rtDwarfCursor_GetU16(pCursor, (uint16_t)uErrValue);
2284 case 4: return rtDwarfCursor_GetU32(pCursor, (uint32_t)uErrValue);
2285 case 8: return rtDwarfCursor_GetU64(pCursor, uErrValue);
2286 default:
2287 pCursor->rc = VERR_INTERNAL_ERROR_2;
2288 return uErrValue;
2289 }
2290}
2291
2292
2293/**
2294 * Reads a 1, 2, 4 or 8 byte unsigned value.
2295 *
2296 * @returns 64-bit unsigned value.
2297 * @param pCursor The cursor.
2298 * @param bPtrEnc The pointer encoding.
2299 * @param uErrValue The error value.
2300 */
2301static uint64_t rtDwarfCursor_GetPtrEnc(PRTDWARFCURSOR pCursor, uint8_t bPtrEnc, uint64_t uErrValue)
2302{
2303 uint64_t u64Ret;
2304 switch (bPtrEnc & DW_EH_PE_FORMAT_MASK)
2305 {
2306 case DW_EH_PE_ptr:
2307 u64Ret = rtDwarfCursor_GetNativeUOff(pCursor, uErrValue);
2308 break;
2309 case DW_EH_PE_uleb128:
2310 u64Ret = rtDwarfCursor_GetULeb128(pCursor, uErrValue);
2311 break;
2312 case DW_EH_PE_udata2:
2313 u64Ret = rtDwarfCursor_GetU16(pCursor, UINT16_MAX);
2314 break;
2315 case DW_EH_PE_udata4:
2316 u64Ret = rtDwarfCursor_GetU32(pCursor, UINT32_MAX);
2317 break;
2318 case DW_EH_PE_udata8:
2319 u64Ret = rtDwarfCursor_GetU64(pCursor, UINT64_MAX);
2320 break;
2321 case DW_EH_PE_sleb128:
2322 u64Ret = rtDwarfCursor_GetSLeb128(pCursor, uErrValue);
2323 break;
2324 case DW_EH_PE_sdata2:
2325 u64Ret = (int64_t)(int16_t)rtDwarfCursor_GetU16(pCursor, UINT16_MAX);
2326 break;
2327 case DW_EH_PE_sdata4:
2328 u64Ret = (int64_t)(int32_t)rtDwarfCursor_GetU32(pCursor, UINT32_MAX);
2329 break;
2330 case DW_EH_PE_sdata8:
2331 u64Ret = rtDwarfCursor_GetU64(pCursor, UINT64_MAX);
2332 break;
2333 default:
2334 pCursor->rc = VERR_DWARF_BAD_INFO;
2335 return uErrValue;
2336 }
2337 if (RT_FAILURE(pCursor->rc))
2338 return uErrValue;
2339 return u64Ret;
2340}
2341
2342
2343/**
2344 * Gets the unit length, updating the unit length member and DWARF bitness
2345 * members of the cursor.
2346 *
2347 * @returns The unit length.
2348 * @param pCursor The cursor.
2349 */
2350static uint64_t rtDwarfCursor_GetInitialLength(PRTDWARFCURSOR pCursor)
2351{
2352 /*
2353 * Read the initial length.
2354 */
2355 pCursor->cbUnitLeft = pCursor->cbLeft;
2356 uint64_t cbUnit = rtDwarfCursor_GetU32(pCursor, 0);
2357 if (cbUnit != UINT32_C(0xffffffff))
2358 pCursor->f64bitDwarf = false;
2359 else
2360 {
2361 pCursor->f64bitDwarf = true;
2362 cbUnit = rtDwarfCursor_GetU64(pCursor, 0);
2363 }
2364
2365
2366 /*
2367 * Set the unit length, quitely fixing bad lengths.
2368 */
2369 pCursor->cbUnitLeft = (size_t)cbUnit;
2370 if ( pCursor->cbUnitLeft > pCursor->cbLeft
2371 || pCursor->cbUnitLeft != cbUnit)
2372 pCursor->cbUnitLeft = pCursor->cbLeft;
2373
2374 return cbUnit;
2375}
2376
2377
2378/**
2379 * Calculates the section offset corresponding to the current cursor position.
2380 *
2381 * @returns 32-bit section offset. If out of range, RTDWARFCURSOR::rc will be
2382 * set and UINT32_MAX returned.
2383 * @param pCursor The cursor.
2384 */
2385static uint32_t rtDwarfCursor_CalcSectOffsetU32(PRTDWARFCURSOR pCursor)
2386{
2387 size_t off = pCursor->pb - pCursor->pbStart;
2388 uint32_t offRet = (uint32_t)off;
2389 if (offRet != off)
2390 {
2391 AssertFailed();
2392 pCursor->rc = VERR_OUT_OF_RANGE;
2393 offRet = UINT32_MAX;
2394 }
2395 return offRet;
2396}
2397
2398
2399/**
2400 * Calculates an absolute cursor position from one relative to the current
2401 * cursor position.
2402 *
2403 * @returns The absolute cursor position.
2404 * @param pCursor The cursor.
2405 * @param offRelative The relative position. Must be a positive
2406 * offset.
2407 */
2408static uint8_t const *rtDwarfCursor_CalcPos(PRTDWARFCURSOR pCursor, size_t offRelative)
2409{
2410 if (offRelative > pCursor->cbUnitLeft)
2411 {
2412 Log(("rtDwarfCursor_CalcPos: bad position %#zx, cbUnitLeft=%#zu\n", offRelative, pCursor->cbUnitLeft));
2413 pCursor->rc = VERR_DWARF_BAD_POS;
2414 return NULL;
2415 }
2416 return pCursor->pb + offRelative;
2417}
2418
2419
2420/**
2421 * Advances the cursor to the given position.
2422 *
2423 * @returns IPRT status code.
2424 * @param pCursor The cursor.
2425 * @param pbNewPos The new position - returned by
2426 * rtDwarfCursor_CalcPos().
2427 */
2428static int rtDwarfCursor_AdvanceToPos(PRTDWARFCURSOR pCursor, uint8_t const *pbNewPos)
2429{
2430 if (RT_FAILURE(pCursor->rc))
2431 return pCursor->rc;
2432 AssertPtr(pbNewPos);
2433 if ((uintptr_t)pbNewPos < (uintptr_t)pCursor->pb)
2434 {
2435 Log(("rtDwarfCursor_AdvanceToPos: bad position %p, current %p\n", pbNewPos, pCursor->pb));
2436 return pCursor->rc = VERR_DWARF_BAD_POS;
2437 }
2438
2439 uintptr_t cbAdj = (uintptr_t)pbNewPos - (uintptr_t)pCursor->pb;
2440 if (RT_UNLIKELY(cbAdj > pCursor->cbUnitLeft))
2441 {
2442 AssertFailed();
2443 pCursor->rc = VERR_DWARF_BAD_POS;
2444 cbAdj = pCursor->cbUnitLeft;
2445 }
2446
2447 pCursor->cbUnitLeft -= cbAdj;
2448 pCursor->cbLeft -= cbAdj;
2449 pCursor->pb += cbAdj;
2450 return pCursor->rc;
2451}
2452
2453
2454/**
2455 * Check if the cursor is at the end of the current DWARF unit.
2456 *
2457 * @retval true if at the end or a cursor error is pending.
2458 * @retval false if not.
2459 * @param pCursor The cursor.
2460 */
2461static bool rtDwarfCursor_IsAtEndOfUnit(PRTDWARFCURSOR pCursor)
2462{
2463 return !pCursor->cbUnitLeft || RT_FAILURE(pCursor->rc);
2464}
2465
2466
2467/**
2468 * Skips to the end of the current unit.
2469 *
2470 * @returns IPRT status code.
2471 * @param pCursor The cursor.
2472 */
2473static int rtDwarfCursor_SkipUnit(PRTDWARFCURSOR pCursor)
2474{
2475 pCursor->pb += pCursor->cbUnitLeft;
2476 pCursor->cbLeft -= pCursor->cbUnitLeft;
2477 pCursor->cbUnitLeft = 0;
2478 return pCursor->rc;
2479}
2480
2481
2482/**
2483 * Check if the cursor is at the end of the section (or whatever the cursor is
2484 * processing).
2485 *
2486 * @retval true if at the end or a cursor error is pending.
2487 * @retval false if not.
2488 * @param pCursor The cursor.
2489 */
2490static bool rtDwarfCursor_IsAtEnd(PRTDWARFCURSOR pCursor)
2491{
2492 return !pCursor->cbLeft || RT_FAILURE(pCursor->rc);
2493}
2494
2495
2496/**
2497 * Initialize a section reader cursor.
2498 *
2499 * @returns IPRT status code.
2500 * @param pCursor The cursor.
2501 * @param pThis The dwarf module.
2502 * @param enmSect The name of the section to read.
2503 */
2504static int rtDwarfCursor_Init(PRTDWARFCURSOR pCursor, PRTDBGMODDWARF pThis, krtDbgModDwarfSect enmSect)
2505{
2506 int rc = rtDbgModDwarfLoadSection(pThis, enmSect);
2507 if (RT_FAILURE(rc))
2508 return rc;
2509
2510 pCursor->enmSect = enmSect;
2511 pCursor->pbStart = (uint8_t const *)pThis->aSections[enmSect].pv;
2512 pCursor->pb = pCursor->pbStart;
2513 pCursor->cbLeft = pThis->aSections[enmSect].cb;
2514 pCursor->cbUnitLeft = pCursor->cbLeft;
2515 pCursor->pDwarfMod = pThis;
2516 pCursor->f64bitDwarf = false;
2517 /** @todo ask the image about the endian used as well as the address
2518 * width. */
2519 pCursor->fNativEndian = true;
2520 pCursor->cbNativeAddr = 4;
2521 pCursor->rc = VINF_SUCCESS;
2522
2523 return VINF_SUCCESS;
2524}
2525
2526
2527/**
2528 * Initialize a section reader cursor with a skip offset.
2529 *
2530 * @returns IPRT status code.
2531 * @param pCursor The cursor.
2532 * @param pThis The dwarf module.
2533 * @param enmSect The name of the section to read.
2534 * @param offSect The offset to skip into the section.
2535 */
2536static int rtDwarfCursor_InitWithOffset(PRTDWARFCURSOR pCursor, PRTDBGMODDWARF pThis,
2537 krtDbgModDwarfSect enmSect, uint32_t offSect)
2538{
2539 if (offSect > pThis->aSections[enmSect].cb)
2540 {
2541 Log(("rtDwarfCursor_InitWithOffset: offSect=%#x cb=%#x enmSect=%d\n", offSect, pThis->aSections[enmSect].cb, enmSect));
2542 return VERR_DWARF_BAD_POS;
2543 }
2544
2545 int rc = rtDwarfCursor_Init(pCursor, pThis, enmSect);
2546 if (RT_SUCCESS(rc))
2547 {
2548 /* pCursor->pbStart += offSect; - we're skipping, offsets are relative to start of section... */
2549 pCursor->pb += offSect;
2550 pCursor->cbLeft -= offSect;
2551 pCursor->cbUnitLeft -= offSect;
2552 }
2553
2554 return rc;
2555}
2556
2557
2558/**
2559 * Initialize a cursor for a block (subsection) retrieved from the given cursor.
2560 *
2561 * The parent cursor will be advanced past the block.
2562 *
2563 * @returns IPRT status code.
2564 * @param pCursor The cursor.
2565 * @param pParent The parent cursor. Will be moved by @a cbBlock.
2566 * @param cbBlock The size of the block the new cursor should
2567 * cover.
2568 */
2569static int rtDwarfCursor_InitForBlock(PRTDWARFCURSOR pCursor, PRTDWARFCURSOR pParent, uint32_t cbBlock)
2570{
2571 if (RT_FAILURE(pParent->rc))
2572 return pParent->rc;
2573 if (pParent->cbUnitLeft < cbBlock)
2574 {
2575 Log(("rtDwarfCursor_InitForBlock: cbUnitLeft=%#x < cbBlock=%#x \n", pParent->cbUnitLeft, cbBlock));
2576 return VERR_DWARF_BAD_POS;
2577 }
2578
2579 *pCursor = *pParent;
2580 pCursor->cbLeft = cbBlock;
2581 pCursor->cbUnitLeft = cbBlock;
2582
2583 pParent->pb += cbBlock;
2584 pParent->cbLeft -= cbBlock;
2585 pParent->cbUnitLeft -= cbBlock;
2586
2587 return VINF_SUCCESS;
2588}
2589
2590
2591/**
2592 * Initialize a reader cursor for a memory block (eh_frame).
2593 *
2594 * @returns IPRT status code.
2595 * @param pCursor The cursor.
2596 * @param pvMem The memory block.
2597 * @param cbMem The size of the memory block.
2598 */
2599static int rtDwarfCursor_InitForMem(PRTDWARFCURSOR pCursor, void const *pvMem, size_t cbMem)
2600{
2601 pCursor->enmSect = krtDbgModDwarfSect_End;
2602 pCursor->pbStart = (uint8_t const *)pvMem;
2603 pCursor->pb = (uint8_t const *)pvMem;
2604 pCursor->cbLeft = cbMem;
2605 pCursor->cbUnitLeft = cbMem;
2606 pCursor->pDwarfMod = NULL;
2607 pCursor->f64bitDwarf = false;
2608 /** @todo ask the image about the endian used as well as the address
2609 * width. */
2610 pCursor->fNativEndian = true;
2611 pCursor->cbNativeAddr = 4;
2612 pCursor->rc = VINF_SUCCESS;
2613
2614 return VINF_SUCCESS;
2615}
2616
2617
2618/**
2619 * Deletes a section reader initialized by rtDwarfCursor_Init.
2620 *
2621 * @returns @a rcOther or RTDWARCURSOR::rc.
2622 * @param pCursor The section reader.
2623 * @param rcOther Other error code to be returned if it indicates
2624 * error or if the cursor status is OK.
2625 */
2626static int rtDwarfCursor_Delete(PRTDWARFCURSOR pCursor, int rcOther)
2627{
2628 /* ... and a drop of poison. */
2629 pCursor->pb = NULL;
2630 pCursor->cbLeft = ~(size_t)0;
2631 pCursor->cbUnitLeft = ~(size_t)0;
2632 pCursor->pDwarfMod = NULL;
2633 if (RT_FAILURE(pCursor->rc) && RT_SUCCESS(rcOther))
2634 rcOther = pCursor->rc;
2635 pCursor->rc = VERR_INTERNAL_ERROR_4;
2636 return rcOther;
2637}
2638
2639
2640/*
2641 *
2642 * DWARF Frame Unwind Information.
2643 * DWARF Frame Unwind Information.
2644 * DWARF Frame Unwind Information.
2645 *
2646 */
2647
2648/**
2649 * Common information entry (CIE) information.
2650 */
2651typedef struct RTDWARFCIEINFO
2652{
2653 /** The segment location of the CIE. */
2654 uint64_t offCie;
2655 /** The DWARF version. */
2656 uint8_t uDwarfVer;
2657 /** The address pointer encoding. */
2658 uint8_t bAddressPtrEnc;
2659 /** The segment size (v4). */
2660 uint8_t cbSegment;
2661 /** The return register column. UINT8_MAX if default register. */
2662 uint8_t bRetReg;
2663 /** The LSDA pointer encoding. */
2664 uint8_t bLsdaPtrEnc;
2665
2666 /** Set if the EH data field is present ('eh'). */
2667 bool fHasEhData : 1;
2668 /** Set if there is an augmentation data size ('z'). */
2669 bool fHasAugmentationSize : 1;
2670 /** Set if the augmentation data contains a LSDA (pointer size byte in CIE,
2671 * pointer in FDA) ('L'). */
2672 bool fHasLanguageSpecificDataArea : 1;
2673 /** Set if the augmentation data contains a personality routine
2674 * (pointer size + pointer) ('P'). */
2675 bool fHasPersonalityRoutine : 1;
2676 /** Set if the augmentation data contains the address encoding . */
2677 bool fHasAddressEnc : 1;
2678 /** Set if signal frame. */
2679 bool fIsSignalFrame : 1;
2680 /** Set if we've encountered unknown augmentation data. This
2681 * means the CIE is incomplete and cannot be used. */
2682 bool fHasUnknowAugmentation : 1;
2683
2684 /** Copy of the augmentation string. */
2685 const char *pszAugmentation;
2686
2687 /** Code alignment factor for the instruction. */
2688 uint64_t uCodeAlignFactor;
2689 /** Data alignment factor for the instructions. */
2690 int64_t iDataAlignFactor;
2691
2692 /** Pointer to the instruction sequence. */
2693 uint8_t const *pbInstructions;
2694 /** The length of the instruction sequence. */
2695 size_t cbInstructions;
2696} RTDWARFCIEINFO;
2697/** Pointer to CIE info. */
2698typedef RTDWARFCIEINFO *PRTDWARFCIEINFO;
2699/** Pointer to const CIE info. */
2700typedef RTDWARFCIEINFO const *PCRTDWARFCIEINFO;
2701
2702
2703/** Number of registers we care about.
2704 * @note We're currently not expecting to be decoding ppc, arm, ia64 or such,
2705 * only x86 and x86_64. We can easily increase the column count. */
2706#define RTDWARFCF_MAX_REGISTERS 96
2707
2708
2709/**
2710 * Call frame state row.
2711 */
2712typedef struct RTDWARFCFROW
2713{
2714 /** Stack worked by DW_CFA_remember_state and DW_CFA_restore_state. */
2715 struct RTDWARFCFROW *pNextOnStack;
2716
2717 /** @name CFA - Canonical frame address expression.
2718 * Since there are partial CFA instructions, we cannot be lazy like with the
2719 * register but keep register+offset around. For DW_CFA_def_cfa_expression
2720 * we just take down the program location, though.
2721 * @{ */
2722 /** Pointer to DW_CFA_def_cfa_expression instruction, NULL if reg+offset. */
2723 uint8_t const *pbCfaExprInstr;
2724 /** The CFA register offset. */
2725 int64_t offCfaReg;
2726 /** The CFA base register number. */
2727 uint16_t uCfaBaseReg;
2728 /** Set if we've got a valid CFA definition. */
2729 bool fCfaDefined : 1;
2730 /** @} */
2731
2732 /** Set if on the heap and needs freeing. */
2733 bool fOnHeap : 1;
2734 /** Pointer to the instructions bytes defining registers.
2735 * NULL means */
2736 uint8_t const *apbRegInstrs[RTDWARFCF_MAX_REGISTERS];
2737} RTDWARFCFROW;
2738typedef RTDWARFCFROW *PRTDWARFCFROW;
2739typedef RTDWARFCFROW const *PCRTDWARFCFROW;
2740
2741/** Row program execution state. */
2742typedef struct RTDWARFCFEXEC
2743{
2744 PRTDWARFCFROW pRow;
2745 /** Number of PC bytes left to advance before we get a hit. */
2746 uint64_t cbLeftToAdvance;
2747 /** Number of pushed rows. */
2748 uint32_t cPushes;
2749 /** Set if little endian, clear if big endian. */
2750 bool fLittleEndian;
2751 /** The CIE. */
2752 PCRTDWARFCIEINFO pCie;
2753 /** The program counter value for the FDE. Subjected to segment.
2754 * Needed for DW_CFA_set_loc. */
2755 uint64_t uPcBegin;
2756 /** The offset relative to uPcBegin for which we're searching for a row.
2757 * Needed for DW_CFA_set_loc. */
2758 uint64_t offInRange;
2759} RTDWARFCFEXEC;
2760typedef RTDWARFCFEXEC *PRTDWARFCFEXEC;
2761
2762
2763/* Set of macros for getting and skipping operands. */
2764#define SKIP_ULEB128_OR_LEB128() \
2765 do \
2766 { \
2767 AssertReturn(offInstr < cbInstr, VERR_DBG_MALFORMED_UNWIND_INFO); \
2768 } while (pbInstr[offInstr++] & 0x80)
2769
2770#define GET_ULEB128_AS_U14(a_uDst) \
2771 do \
2772 { \
2773 AssertReturn(offInstr < cbInstr, VERR_DBG_MALFORMED_UNWIND_INFO); \
2774 uint8_t b = pbInstr[offInstr++]; \
2775 (a_uDst) = b & 0x7f; \
2776 if (b & 0x80) \
2777 { \
2778 AssertReturn(offInstr < cbInstr, VERR_DBG_MALFORMED_UNWIND_INFO); \
2779 b = pbInstr[offInstr++]; \
2780 AssertReturn(!(b & 0x80), VERR_DBG_MALFORMED_UNWIND_INFO); \
2781 (a_uDst) |= (uint16_t)b << 7; \
2782 } \
2783 } while (0)
2784#define GET_ULEB128_AS_U63(a_uDst) \
2785 do \
2786 { \
2787 AssertReturn(offInstr < cbInstr, VERR_DBG_MALFORMED_UNWIND_INFO); \
2788 uint8_t b = pbInstr[offInstr++]; \
2789 (a_uDst) = b & 0x7f; \
2790 if (b & 0x80) \
2791 { \
2792 unsigned cShift = 7; \
2793 do \
2794 { \
2795 AssertReturn(offInstr < cbInstr, VERR_DBG_MALFORMED_UNWIND_INFO); \
2796 AssertReturn(cShift < 63, VERR_DWARF_LEB_OVERFLOW); \
2797 b = pbInstr[offInstr++]; \
2798 (a_uDst) |= (uint16_t)(b & 0x7f) << cShift; \
2799 cShift += 7; \
2800 } while (b & 0x80); \
2801 } \
2802 } while (0)
2803#define GET_LEB128_AS_I63(a_uDst) \
2804 do \
2805 { \
2806 AssertReturn(offInstr < cbInstr, VERR_DBG_MALFORMED_UNWIND_INFO); \
2807 uint8_t b = pbInstr[offInstr++]; \
2808 if (!(b & 0x80)) \
2809 (a_uDst) = !(b & 0x40) ? b : (int64_t)(int8_t)(b | 0x80); \
2810 else \
2811 { \
2812 /* Read value into unsigned variable: */ \
2813 unsigned cShift = 7; \
2814 uint64_t uTmp = b & 0x7f; \
2815 do \
2816 { \
2817 AssertReturn(offInstr < cbInstr, VERR_DBG_MALFORMED_UNWIND_INFO); \
2818 AssertReturn(cShift < 63, VERR_DWARF_LEB_OVERFLOW); \
2819 b = pbInstr[offInstr++]; \
2820 uTmp |= (uint16_t)(b & 0x7f) << cShift; \
2821 cShift += 7; \
2822 } while (b & 0x80); \
2823 /* Sign extend before setting the destination value: */ \
2824 cShift -= 7 + 1; \
2825 if (uTmp & RT_BIT_64(cShift)) \
2826 uTmp |= ~(RT_BIT_64(cShift) - 1); \
2827 (a_uDst) = (int64_t)uTmp; \
2828 } \
2829 } while (0)
2830
2831#define SKIP_BLOCK() \
2832 do \
2833 { \
2834 uint16_t cbBlock; \
2835 GET_ULEB128_AS_U14(cbBlock); \
2836 AssertReturn(offInstr + cbBlock <= cbInstr, VERR_DBG_MALFORMED_UNWIND_INFO); \
2837 offInstr += cbBlock; \
2838 } while (0)
2839
2840
2841static int rtDwarfUnwind_Execute(PRTDWARFCFEXEC pExecState, uint8_t const *pbInstr, uint32_t cbInstr)
2842{
2843 PRTDWARFCFROW pRow = pExecState->pRow;
2844 for (uint32_t offInstr = 0; offInstr < cbInstr;)
2845 {
2846 /*
2847 * Instruction switches.
2848 */
2849 uint8_t const bInstr = pbInstr[offInstr++];
2850 switch (bInstr & DW_CFA_high_bit_mask)
2851 {
2852 case DW_CFA_advance_loc:
2853 {
2854 uint8_t const cbAdvance = bInstr & ~DW_CFA_high_bit_mask;
2855 if (cbAdvance > pExecState->cbLeftToAdvance)
2856 return VINF_SUCCESS;
2857 pExecState->cbLeftToAdvance -= cbAdvance;
2858 break;
2859 }
2860
2861 case DW_CFA_offset:
2862 {
2863 uint8_t iReg = bInstr & ~DW_CFA_high_bit_mask;
2864 if (iReg < RT_ELEMENTS(pRow->apbRegInstrs))
2865 pRow->apbRegInstrs[iReg] = &pbInstr[offInstr - 1];
2866 SKIP_ULEB128_OR_LEB128();
2867 break;
2868 }
2869
2870 case 0:
2871 switch (bInstr)
2872 {
2873 case DW_CFA_nop:
2874 break;
2875
2876 /*
2877 * Register instructions.
2878 */
2879 case DW_CFA_register:
2880 case DW_CFA_offset_extended:
2881 case DW_CFA_offset_extended_sf:
2882 case DW_CFA_val_offset:
2883 case DW_CFA_val_offset_sf:
2884 {
2885 uint8_t const * const pbCurInstr = &pbInstr[offInstr - 1];
2886 uint16_t iReg;
2887 GET_ULEB128_AS_U14(iReg);
2888 if (iReg < RT_ELEMENTS(pRow->apbRegInstrs))
2889 pRow->apbRegInstrs[iReg] = pbCurInstr;
2890 SKIP_ULEB128_OR_LEB128();
2891 break;
2892 }
2893
2894 case DW_CFA_expression:
2895 case DW_CFA_val_expression:
2896 {
2897 uint8_t const * const pbCurInstr = &pbInstr[offInstr - 1];
2898 uint16_t iReg;
2899 GET_ULEB128_AS_U14(iReg);
2900 if (iReg < RT_ELEMENTS(pRow->apbRegInstrs))
2901 pRow->apbRegInstrs[iReg] = pbCurInstr;
2902 SKIP_BLOCK();
2903 break;
2904 }
2905
2906 case DW_CFA_restore_extended:
2907 {
2908 uint8_t const * const pbCurInstr = &pbInstr[offInstr - 1];
2909 uint16_t iReg;
2910 GET_ULEB128_AS_U14(iReg);
2911 if (iReg < RT_ELEMENTS(pRow->apbRegInstrs))
2912 pRow->apbRegInstrs[iReg] = pbCurInstr;
2913 break;
2914 }
2915
2916 case DW_CFA_undefined:
2917 {
2918 uint16_t iReg;
2919 GET_ULEB128_AS_U14(iReg);
2920 if (iReg < RT_ELEMENTS(pRow->apbRegInstrs))
2921 pRow->apbRegInstrs[iReg] = NULL;
2922 break;
2923 }
2924
2925 case DW_CFA_same_value:
2926 {
2927 uint8_t const * const pbCurInstr = &pbInstr[offInstr - 1];
2928 uint16_t iReg;
2929 GET_ULEB128_AS_U14(iReg);
2930 if (iReg < RT_ELEMENTS(pRow->apbRegInstrs))
2931 pRow->apbRegInstrs[iReg] = pbCurInstr;
2932 break;
2933 }
2934
2935
2936 /*
2937 * CFA instructions.
2938 */
2939 case DW_CFA_def_cfa:
2940 {
2941 GET_ULEB128_AS_U14(pRow->uCfaBaseReg);
2942 uint64_t offCfaReg;
2943 GET_ULEB128_AS_U63(offCfaReg);
2944 pRow->offCfaReg = offCfaReg;
2945 pRow->pbCfaExprInstr = NULL;
2946 pRow->fCfaDefined = true;
2947 break;
2948 }
2949
2950 case DW_CFA_def_cfa_register:
2951 {
2952 GET_ULEB128_AS_U14(pRow->uCfaBaseReg);
2953 pRow->pbCfaExprInstr = NULL;
2954 pRow->fCfaDefined = true;
2955 /* Leaves offCfaReg as is. */
2956 break;
2957 }
2958
2959 case DW_CFA_def_cfa_offset:
2960 {
2961 uint64_t offCfaReg;
2962 GET_ULEB128_AS_U63(offCfaReg);
2963 pRow->offCfaReg = offCfaReg;
2964 pRow->pbCfaExprInstr = NULL;
2965 pRow->fCfaDefined = true;
2966 /* Leaves uCfaBaseReg as is. */
2967 break;
2968 }
2969
2970 case DW_CFA_def_cfa_sf:
2971 GET_ULEB128_AS_U14(pRow->uCfaBaseReg);
2972 GET_LEB128_AS_I63(pRow->offCfaReg);
2973 pRow->pbCfaExprInstr = NULL;
2974 pRow->fCfaDefined = true;
2975 break;
2976
2977 case DW_CFA_def_cfa_offset_sf:
2978 GET_LEB128_AS_I63(pRow->offCfaReg);
2979 pRow->pbCfaExprInstr = NULL;
2980 pRow->fCfaDefined = true;
2981 /* Leaves uCfaBaseReg as is. */
2982 break;
2983
2984 case DW_CFA_def_cfa_expression:
2985 pRow->pbCfaExprInstr = &pbInstr[offInstr - 1];
2986 pRow->fCfaDefined = true;
2987 SKIP_BLOCK();
2988 break;
2989
2990 /*
2991 * Less likely instructions:
2992 */
2993 case DW_CFA_advance_loc1:
2994 {
2995 AssertReturn(offInstr < cbInstr, VERR_DBG_MALFORMED_UNWIND_INFO);
2996 uint8_t const cbAdvance = pbInstr[offInstr++];
2997 if (cbAdvance > pExecState->cbLeftToAdvance)
2998 return VINF_SUCCESS;
2999 pExecState->cbLeftToAdvance -= cbAdvance;
3000 break;
3001 }
3002
3003 case DW_CFA_advance_loc2:
3004 {
3005 AssertReturn(offInstr + 1 < cbInstr, VERR_DBG_MALFORMED_UNWIND_INFO);
3006 uint16_t const cbAdvance = pExecState->fLittleEndian
3007 ? RT_MAKE_U16(pbInstr[offInstr], pbInstr[offInstr + 1])
3008 : RT_MAKE_U16(pbInstr[offInstr + 1], pbInstr[offInstr]);
3009 if (cbAdvance > pExecState->cbLeftToAdvance)
3010 return VINF_SUCCESS;
3011 pExecState->cbLeftToAdvance -= cbAdvance;
3012 offInstr += 2;
3013 break;
3014 }
3015
3016 case DW_CFA_advance_loc4:
3017 {
3018 AssertReturn(offInstr + 3 < cbInstr, VERR_DBG_MALFORMED_UNWIND_INFO);
3019 uint32_t const cbAdvance = pExecState->fLittleEndian
3020 ? RT_MAKE_U32_FROM_U8(pbInstr[offInstr + 0], pbInstr[offInstr + 1],
3021 pbInstr[offInstr + 2], pbInstr[offInstr + 3])
3022 : RT_MAKE_U32_FROM_U8(pbInstr[offInstr + 3], pbInstr[offInstr + 2],
3023 pbInstr[offInstr + 1], pbInstr[offInstr + 0]);
3024 if (cbAdvance > pExecState->cbLeftToAdvance)
3025 return VINF_SUCCESS;
3026 pExecState->cbLeftToAdvance -= cbAdvance;
3027 offInstr += 4;
3028 break;
3029 }
3030
3031 /*
3032 * This bugger is really annoying and probably never used.
3033 */
3034 case DW_CFA_set_loc:
3035 {
3036 /* Ignore the segment number. */
3037 if (pExecState->pCie->cbSegment)
3038 {
3039 offInstr += pExecState->pCie->cbSegment;
3040 AssertReturn(offInstr < cbInstr, VERR_DBG_MALFORMED_UNWIND_INFO);
3041 }
3042
3043 /* Retrieve the address. sigh. */
3044 uint64_t uAddress;
3045 switch (pExecState->pCie->bAddressPtrEnc & (DW_EH_PE_FORMAT_MASK | DW_EH_PE_indirect))
3046 {
3047 case DW_EH_PE_udata2:
3048 AssertReturn(offInstr + 1 < cbInstr, VERR_DBG_MALFORMED_UNWIND_INFO);
3049 if (pExecState->fLittleEndian)
3050 uAddress = RT_MAKE_U16(pbInstr[offInstr], pbInstr[offInstr + 1]);
3051 else
3052 uAddress = RT_MAKE_U16(pbInstr[offInstr + 1], pbInstr[offInstr]);
3053 offInstr += 2;
3054 break;
3055 case DW_EH_PE_sdata2:
3056 AssertReturn(offInstr + 1 < cbInstr, VERR_DBG_MALFORMED_UNWIND_INFO);
3057 if (pExecState->fLittleEndian)
3058 uAddress = (int64_t)(int16_t)RT_MAKE_U16(pbInstr[offInstr], pbInstr[offInstr + 1]);
3059 else
3060 uAddress = (int64_t)(int16_t)RT_MAKE_U16(pbInstr[offInstr + 1], pbInstr[offInstr]);
3061 offInstr += 2;
3062 break;
3063 case DW_EH_PE_udata4:
3064 AssertReturn(offInstr + 3 < cbInstr, VERR_DBG_MALFORMED_UNWIND_INFO);
3065 if (pExecState->fLittleEndian)
3066 uAddress = RT_MAKE_U32_FROM_U8(pbInstr[offInstr + 0], pbInstr[offInstr + 1],
3067 pbInstr[offInstr + 2], pbInstr[offInstr + 3]);
3068 else
3069 uAddress = RT_MAKE_U32_FROM_U8(pbInstr[offInstr + 3], pbInstr[offInstr + 2],
3070 pbInstr[offInstr + 1], pbInstr[offInstr + 0]);
3071
3072 offInstr += 4;
3073 break;
3074 case DW_EH_PE_sdata4:
3075 AssertReturn(offInstr + 3 < cbInstr, VERR_DBG_MALFORMED_UNWIND_INFO);
3076 if (pExecState->fLittleEndian)
3077 uAddress = (int64_t)(int32_t)RT_MAKE_U32_FROM_U8(pbInstr[offInstr + 0], pbInstr[offInstr + 1],
3078 pbInstr[offInstr + 2], pbInstr[offInstr + 3]);
3079 else
3080 uAddress = (int64_t)(int32_t)RT_MAKE_U32_FROM_U8(pbInstr[offInstr + 3], pbInstr[offInstr + 2],
3081 pbInstr[offInstr + 1], pbInstr[offInstr + 0]);
3082 offInstr += 4;
3083 break;
3084 case DW_EH_PE_udata8:
3085 case DW_EH_PE_sdata8:
3086 AssertReturn(offInstr + 7 < cbInstr, VERR_DBG_MALFORMED_UNWIND_INFO);
3087 if (pExecState->fLittleEndian)
3088 uAddress = RT_MAKE_U64_FROM_U8(pbInstr[offInstr + 0], pbInstr[offInstr + 1],
3089 pbInstr[offInstr + 2], pbInstr[offInstr + 3],
3090 pbInstr[offInstr + 4], pbInstr[offInstr + 5],
3091 pbInstr[offInstr + 6], pbInstr[offInstr + 7]);
3092 else
3093 uAddress = RT_MAKE_U64_FROM_U8(pbInstr[offInstr + 7], pbInstr[offInstr + 6],
3094 pbInstr[offInstr + 5], pbInstr[offInstr + 4],
3095 pbInstr[offInstr + 3], pbInstr[offInstr + 2],
3096 pbInstr[offInstr + 1], pbInstr[offInstr + 0]);
3097 offInstr += 8;
3098 break;
3099 case DW_EH_PE_sleb128:
3100 case DW_EH_PE_uleb128:
3101 default:
3102 AssertMsgFailedReturn(("%#x\n", pExecState->pCie->bAddressPtrEnc), VERR_DWARF_TODO);
3103 }
3104 AssertReturn(uAddress >= pExecState->uPcBegin, VERR_DBG_MALFORMED_UNWIND_INFO);
3105
3106 /* Did we advance past the desire address already? */
3107 if (uAddress > pExecState->uPcBegin + pExecState->offInRange)
3108 return VINF_SUCCESS;
3109 pExecState->cbLeftToAdvance = pExecState->uPcBegin + pExecState->offInRange - uAddress;
3110 break;
3111
3112
3113 /*
3114 * Row state push/pop instructions.
3115 */
3116
3117 case DW_CFA_remember_state:
3118 {
3119 AssertReturn(pExecState->cPushes < 10, VERR_DBG_MALFORMED_UNWIND_INFO);
3120 PRTDWARFCFROW pNewRow = (PRTDWARFCFROW)RTMemTmpAlloc(sizeof(*pNewRow));
3121 AssertReturn(pNewRow, VERR_NO_TMP_MEMORY);
3122 memcpy(pNewRow, pRow, sizeof(*pNewRow));
3123 pNewRow->pNextOnStack = pRow;
3124 pNewRow->fOnHeap = true;
3125 pExecState->pRow = pNewRow;
3126 pExecState->cPushes += 1;
3127 pRow = pNewRow;
3128 break;
3129 }
3130
3131 case DW_CFA_restore_state:
3132 AssertReturn(pRow->pNextOnStack, VERR_DBG_MALFORMED_UNWIND_INFO);
3133 Assert(pRow->fOnHeap);
3134 Assert(pExecState->cPushes > 0);
3135 pExecState->cPushes -= 1;
3136 pExecState->pRow = pRow->pNextOnStack;
3137 RTMemTmpFree(pRow);
3138 pRow = pExecState->pRow;
3139 break;
3140 }
3141 }
3142 break;
3143
3144 case DW_CFA_restore:
3145 {
3146 uint8_t const * const pbCurInstr = &pbInstr[offInstr - 1];
3147 uint8_t const iReg = bInstr & ~DW_CFA_high_bit_mask;
3148 if (iReg < RT_ELEMENTS(pRow->apbRegInstrs))
3149 pRow->apbRegInstrs[iReg] = pbCurInstr;
3150 break;
3151 }
3152 }
3153 }
3154 return VINF_TRY_AGAIN;
3155}
3156
3157
3158/**
3159 * Register getter for AMD64.
3160 *
3161 * @returns true if found, false if not.
3162 * @param pState The unwind state to get the register from.
3163 * @param iReg The dwarf register number.
3164 * @param puValue Where to store the register value.
3165 */
3166static bool rtDwarfUnwind_Amd64GetRegFromState(PCRTDBGUNWINDSTATE pState, uint16_t iReg, uint64_t *puValue)
3167{
3168 switch (iReg)
3169 {
3170 case DWREG_AMD64_RAX: *puValue = pState->u.x86.auRegs[X86_GREG_xAX]; return true;
3171 case DWREG_AMD64_RDX: *puValue = pState->u.x86.auRegs[X86_GREG_xDX]; return true;
3172 case DWREG_AMD64_RCX: *puValue = pState->u.x86.auRegs[X86_GREG_xCX]; return true;
3173 case DWREG_AMD64_RBX: *puValue = pState->u.x86.auRegs[X86_GREG_xBX]; return true;
3174 case DWREG_AMD64_RSI: *puValue = pState->u.x86.auRegs[X86_GREG_xSI]; return true;
3175 case DWREG_AMD64_RDI: *puValue = pState->u.x86.auRegs[X86_GREG_xDI]; return true;
3176 case DWREG_AMD64_RBP: *puValue = pState->u.x86.auRegs[X86_GREG_xBP]; return true;
3177 case DWREG_AMD64_RSP: *puValue = pState->u.x86.auRegs[X86_GREG_xSP]; return true;
3178 case DWREG_AMD64_R8: *puValue = pState->u.x86.auRegs[X86_GREG_x8]; return true;
3179 case DWREG_AMD64_R9: *puValue = pState->u.x86.auRegs[X86_GREG_x9]; return true;
3180 case DWREG_AMD64_R10: *puValue = pState->u.x86.auRegs[X86_GREG_x10]; return true;
3181 case DWREG_AMD64_R11: *puValue = pState->u.x86.auRegs[X86_GREG_x11]; return true;
3182 case DWREG_AMD64_R12: *puValue = pState->u.x86.auRegs[X86_GREG_x12]; return true;
3183 case DWREG_AMD64_R13: *puValue = pState->u.x86.auRegs[X86_GREG_x13]; return true;
3184 case DWREG_AMD64_R14: *puValue = pState->u.x86.auRegs[X86_GREG_x14]; return true;
3185 case DWREG_AMD64_R15: *puValue = pState->u.x86.auRegs[X86_GREG_x15]; return true;
3186 case DWREG_AMD64_RFLAGS: *puValue = pState->u.x86.uRFlags; return true;
3187 case DWREG_AMD64_ES: *puValue = pState->u.x86.auSegs[X86_SREG_ES]; return true;
3188 case DWREG_AMD64_CS: *puValue = pState->u.x86.auSegs[X86_SREG_CS]; return true;
3189 case DWREG_AMD64_SS: *puValue = pState->u.x86.auSegs[X86_SREG_SS]; return true;
3190 case DWREG_AMD64_DS: *puValue = pState->u.x86.auSegs[X86_SREG_DS]; return true;
3191 case DWREG_AMD64_FS: *puValue = pState->u.x86.auSegs[X86_SREG_FS]; return true;
3192 case DWREG_AMD64_GS: *puValue = pState->u.x86.auSegs[X86_SREG_GS]; return true;
3193 }
3194 return false;
3195}
3196
3197
3198/**
3199 * Register getter for 386+.
3200 *
3201 * @returns true if found, false if not.
3202 * @param pState The unwind state to get the register from.
3203 * @param iReg The dwarf register number.
3204 * @param puValue Where to store the register value.
3205 */
3206static bool rtDwarfUnwind_X86GetRegFromState(PCRTDBGUNWINDSTATE pState, uint16_t iReg, uint64_t *puValue)
3207{
3208 switch (iReg)
3209 {
3210 case DWREG_X86_EAX: *puValue = pState->u.x86.auRegs[X86_GREG_xAX]; return true;
3211 case DWREG_X86_ECX: *puValue = pState->u.x86.auRegs[X86_GREG_xCX]; return true;
3212 case DWREG_X86_EDX: *puValue = pState->u.x86.auRegs[X86_GREG_xDX]; return true;
3213 case DWREG_X86_EBX: *puValue = pState->u.x86.auRegs[X86_GREG_xBX]; return true;
3214 case DWREG_X86_ESP: *puValue = pState->u.x86.auRegs[X86_GREG_xSP]; return true;
3215 case DWREG_X86_EBP: *puValue = pState->u.x86.auRegs[X86_GREG_xBP]; return true;
3216 case DWREG_X86_ESI: *puValue = pState->u.x86.auRegs[X86_GREG_xSI]; return true;
3217 case DWREG_X86_EDI: *puValue = pState->u.x86.auRegs[X86_GREG_xDI]; return true;
3218 case DWREG_X86_EFLAGS: *puValue = pState->u.x86.uRFlags; return true;
3219 case DWREG_X86_ES: *puValue = pState->u.x86.auSegs[X86_SREG_ES]; return true;
3220 case DWREG_X86_CS: *puValue = pState->u.x86.auSegs[X86_SREG_CS]; return true;
3221 case DWREG_X86_SS: *puValue = pState->u.x86.auSegs[X86_SREG_SS]; return true;
3222 case DWREG_X86_DS: *puValue = pState->u.x86.auSegs[X86_SREG_DS]; return true;
3223 case DWREG_X86_FS: *puValue = pState->u.x86.auSegs[X86_SREG_FS]; return true;
3224 case DWREG_X86_GS: *puValue = pState->u.x86.auSegs[X86_SREG_GS]; return true;
3225 }
3226 return false;
3227}
3228
3229/** Register getter. */
3230typedef bool FNDWARFUNWINDGEREGFROMSTATE(PCRTDBGUNWINDSTATE pState, uint16_t iReg, uint64_t *puValue);
3231/** Pointer to a register getter. */
3232typedef FNDWARFUNWINDGEREGFROMSTATE *PFNDWARFUNWINDGEREGFROMSTATE;
3233
3234
3235
3236/**
3237 * Does the heavy work for figuring out the return value of a register.
3238 *
3239 * @returns IPRT status code.
3240 * @retval VERR_NOT_FOUND if register is undefined.
3241 *
3242 * @param pRow The DWARF unwind table "row" to use.
3243 * @param uReg The DWARF register number.
3244 * @param pCie The corresponding CIE.
3245 * @param uCfa The canonical frame address to use.
3246 * @param pState The unwind to use when reading stack.
3247 * @param pOldState The unwind state to get register values from.
3248 * @param pfnGetReg The register value getter.
3249 * @param puValue Where to store the return value.
3250 * @param cbValue The size this register would have on the stack.
3251 */
3252static int rtDwarfUnwind_CalcRegisterValue(PRTDWARFCFROW pRow, unsigned uReg, PCRTDWARFCIEINFO pCie, uint64_t uCfa,
3253 PRTDBGUNWINDSTATE pState, PCRTDBGUNWINDSTATE pOldState,
3254 PFNDWARFUNWINDGEREGFROMSTATE pfnGetReg, uint64_t *puValue, uint8_t cbValue)
3255{
3256 Assert(uReg < RT_ELEMENTS(pRow->apbRegInstrs));
3257 uint8_t const *pbInstr = pRow->apbRegInstrs[uReg];
3258 if (!pbInstr)
3259 return VERR_NOT_FOUND;
3260
3261 uint32_t cbInstr = UINT32_MAX / 2;
3262 uint32_t offInstr = 1;
3263 uint8_t const bInstr = *pbInstr;
3264 switch (bInstr)
3265 {
3266 default:
3267 if ((bInstr & DW_CFA_high_bit_mask) == DW_CFA_offset)
3268 {
3269 uint64_t offCfa;
3270 GET_ULEB128_AS_U63(offCfa);
3271 int rc = pState->pfnReadStack(pState, uCfa + (int64_t)offCfa * pCie->iDataAlignFactor, cbValue, puValue);
3272 Log8(("rtDwarfUnwind_CalcRegisterValue(%#x): DW_CFA_offset %#RX64: %Rrc, %#RX64\n", uReg, uCfa + (int64_t)offCfa * pCie->iDataAlignFactor, rc, *puValue));
3273 return rc;
3274 }
3275 AssertReturn((bInstr & DW_CFA_high_bit_mask) == DW_CFA_restore, VERR_INTERNAL_ERROR);
3276 RT_FALL_THRU();
3277 case DW_CFA_restore_extended:
3278 /* Need to search the CIE for the rule. */
3279 Log8(("rtDwarfUnwind_CalcRegisterValue(%#x): DW_CFA_restore/extended:\n", uReg));
3280 AssertFailedReturn(VERR_DWARF_TODO);
3281
3282 case DW_CFA_offset_extended:
3283 {
3284 SKIP_ULEB128_OR_LEB128();
3285 uint64_t offCfa;
3286 GET_ULEB128_AS_U63(offCfa);
3287 int rc = pState->pfnReadStack(pState, uCfa + (int64_t)offCfa * pCie->iDataAlignFactor, cbValue, puValue);
3288 Log8(("rtDwarfUnwind_CalcRegisterValue(%#x): DW_CFA_offset_extended %#RX64: %Rrc, %#RX64\n", uReg, uCfa + (int64_t)offCfa * pCie->iDataAlignFactor, rc, *puValue));
3289 return rc;
3290 }
3291
3292 case DW_CFA_offset_extended_sf:
3293 {
3294 SKIP_ULEB128_OR_LEB128();
3295 int64_t offCfa;
3296 GET_LEB128_AS_I63(offCfa);
3297 int rc = pState->pfnReadStack(pState, uCfa + offCfa * pCie->iDataAlignFactor, cbValue, puValue);
3298 Log8(("rtDwarfUnwind_CalcRegisterValue(%#x): DW_CFA_offset_extended_sf %#RX64: %Rrc, %#RX64\n", uReg, uCfa + offCfa * pCie->iDataAlignFactor, rc, *puValue));
3299 return rc;
3300 }
3301
3302 case DW_CFA_val_offset:
3303 {
3304 SKIP_ULEB128_OR_LEB128();
3305 uint64_t offCfa;
3306 GET_ULEB128_AS_U63(offCfa);
3307 *puValue = uCfa + (int64_t)offCfa * pCie->iDataAlignFactor;
3308 Log8(("rtDwarfUnwind_CalcRegisterValue(%#x): DW_CFA_val_offset: %#RX64\n", uReg, *puValue));
3309 return VINF_SUCCESS;
3310 }
3311
3312 case DW_CFA_val_offset_sf:
3313 {
3314 SKIP_ULEB128_OR_LEB128();
3315 int64_t offCfa;
3316 GET_LEB128_AS_I63(offCfa);
3317 *puValue = uCfa + offCfa * pCie->iDataAlignFactor;
3318 Log8(("rtDwarfUnwind_CalcRegisterValue(%#x): DW_CFA_val_offset_sf: %#RX64\n", uReg, *puValue));
3319 return VINF_SUCCESS;
3320 }
3321
3322 case DW_CFA_register:
3323 {
3324 SKIP_ULEB128_OR_LEB128();
3325 uint16_t iSrcReg;
3326 GET_ULEB128_AS_U14(iSrcReg);
3327 if (pfnGetReg(pOldState, uReg, puValue))
3328 {
3329 Log8(("rtDwarfUnwind_CalcRegisterValue(%#x): DW_CFA_register: %#RX64\n", uReg, *puValue));
3330 return VINF_SUCCESS;
3331 }
3332 Log8(("rtDwarfUnwind_CalcRegisterValue(%#x): DW_CFA_register: VERR_NOT_FOUND\n", uReg));
3333 return VERR_NOT_FOUND;
3334 }
3335
3336 case DW_CFA_expression:
3337 Log8(("rtDwarfUnwind_CalcRegisterValue(%#x): DW_CFA_expression: TODO\n", uReg));
3338 AssertFailedReturn(VERR_DWARF_TODO);
3339
3340 case DW_CFA_val_expression:
3341 Log8(("rtDwarfUnwind_CalcRegisterValue(%#x): DW_CFA_val_expression: TODO\n", uReg));
3342 AssertFailedReturn(VERR_DWARF_TODO);
3343
3344 case DW_CFA_undefined:
3345 Log8(("rtDwarfUnwind_CalcRegisterValue(%#x): DW_CFA_undefined\n", uReg));
3346 return VERR_NOT_FOUND;
3347
3348 case DW_CFA_same_value:
3349 if (pfnGetReg(pOldState, uReg, puValue))
3350 {
3351 Log8(("rtDwarfUnwind_CalcRegisterValue(%#x): DW_CFA_same_value: %#RX64\n", uReg, *puValue));
3352 return VINF_SUCCESS;
3353 }
3354 Log8(("rtDwarfUnwind_CalcRegisterValue(%#x): DW_CFA_same_value: VERR_NOT_FOUND\n", uReg));
3355 return VERR_NOT_FOUND;
3356 }
3357}
3358
3359
3360DECLINLINE(void) rtDwarfUnwind_UpdateX86GRegFromRow(PRTDBGUNWINDSTATE pState, PCRTDBGUNWINDSTATE pOldState, unsigned idxGReg,
3361 PRTDWARFCFROW pRow, unsigned idxDwReg, PCRTDWARFCIEINFO pCie,
3362 uint64_t uCfa, PFNDWARFUNWINDGEREGFROMSTATE pfnGetReg, uint8_t cbGReg)
3363{
3364 int rc = rtDwarfUnwind_CalcRegisterValue(pRow, idxDwReg, pCie, uCfa, pState, pOldState, pfnGetReg,
3365 &pState->u.x86.auRegs[idxGReg], cbGReg);
3366 if (RT_SUCCESS(rc))
3367 pState->u.x86.Loaded.s.fRegs |= RT_BIT_32(idxGReg);
3368}
3369
3370
3371DECLINLINE(void) rtDwarfUnwind_UpdateX86SRegFromRow(PRTDBGUNWINDSTATE pState, PCRTDBGUNWINDSTATE pOldState, unsigned idxSReg,
3372 PRTDWARFCFROW pRow, unsigned idxDwReg, PCRTDWARFCIEINFO pCie,
3373 uint64_t uCfa, PFNDWARFUNWINDGEREGFROMSTATE pfnGetReg)
3374{
3375 uint64_t uValue = pState->u.x86.auSegs[idxSReg];
3376 int rc = rtDwarfUnwind_CalcRegisterValue(pRow, idxDwReg, pCie, uCfa, pState, pOldState, pfnGetReg, &uValue, sizeof(uint16_t));
3377 if (RT_SUCCESS(rc))
3378 {
3379 pState->u.x86.auSegs[idxSReg] = (uint16_t)uValue;
3380 pState->u.x86.Loaded.s.fSegs |= RT_BIT_32(idxSReg);
3381 }
3382}
3383
3384
3385DECLINLINE(void) rtDwarfUnwind_UpdateX86RFlagsFromRow(PRTDBGUNWINDSTATE pState, PCRTDBGUNWINDSTATE pOldState,
3386 PRTDWARFCFROW pRow, unsigned idxDwReg, PCRTDWARFCIEINFO pCie,
3387 uint64_t uCfa, PFNDWARFUNWINDGEREGFROMSTATE pfnGetReg)
3388{
3389 int rc = rtDwarfUnwind_CalcRegisterValue(pRow, idxDwReg, pCie, uCfa, pState, pOldState, pfnGetReg,
3390 &pState->u.x86.uRFlags, sizeof(uint32_t));
3391 if (RT_SUCCESS(rc))
3392 pState->u.x86.Loaded.s.fRFlags = 1;
3393}
3394
3395
3396DECLINLINE(void) rtDwarfUnwind_UpdatePCFromRow(PRTDBGUNWINDSTATE pState, PCRTDBGUNWINDSTATE pOldState,
3397 PRTDWARFCFROW pRow, unsigned idxDwReg, PCRTDWARFCIEINFO pCie,
3398 uint64_t uCfa, PFNDWARFUNWINDGEREGFROMSTATE pfnGetReg, uint8_t cbPc)
3399{
3400 if (pCie->bRetReg != UINT8_MAX)
3401 idxDwReg = pCie->bRetReg;
3402 int rc = rtDwarfUnwind_CalcRegisterValue(pRow, idxDwReg, pCie, uCfa, pState, pOldState, pfnGetReg, &pState->uPc, cbPc);
3403 if (RT_SUCCESS(rc))
3404 pState->u.x86.Loaded.s.fPc = 1;
3405 else
3406 {
3407 rc = pState->pfnReadStack(pState, uCfa - cbPc, cbPc, &pState->uPc);
3408 if (RT_SUCCESS(rc))
3409 pState->u.x86.Loaded.s.fPc = 1;
3410 }
3411}
3412
3413
3414
3415/**
3416 * Updates @a pState with the rules found in @a pRow.
3417 *
3418 * @returns IPRT status code.
3419 * @param pState The unwind state to update.
3420 * @param pRow The "row" in the dwarf unwind table.
3421 * @param pCie The CIE structure for the row.
3422 */
3423static int rtDwarfUnwind_UpdateStateFromRow(PRTDBGUNWINDSTATE pState, PRTDWARFCFROW pRow, PCRTDWARFCIEINFO pCie)
3424{
3425 /*
3426 * We need to make a copy of the current state so we can get at the
3427 * current register values while calculating the ones of the next frame.
3428 */
3429 RTDBGUNWINDSTATE const Old = *pState;
3430
3431 /*
3432 * Get the register state getter.
3433 */
3434 PFNDWARFUNWINDGEREGFROMSTATE pfnGetReg;
3435 switch (pState->enmArch)
3436 {
3437 case RTLDRARCH_AMD64:
3438 pfnGetReg = rtDwarfUnwind_Amd64GetRegFromState;
3439 break;
3440 case RTLDRARCH_X86_32:
3441 case RTLDRARCH_X86_16:
3442 pfnGetReg = rtDwarfUnwind_X86GetRegFromState;
3443 break;
3444 default:
3445 return VERR_NOT_SUPPORTED;
3446 }
3447
3448 /*
3449 * Calc the canonical frame address for the current row.
3450 */
3451 AssertReturn(pRow->fCfaDefined, VERR_DBG_MALFORMED_UNWIND_INFO);
3452 uint64_t uCfa = 0;
3453 if (!pRow->pbCfaExprInstr)
3454 {
3455 pfnGetReg(&Old, pRow->uCfaBaseReg, &uCfa);
3456 uCfa += pRow->offCfaReg;
3457 }
3458 else
3459 {
3460 AssertFailed();
3461 return VERR_DWARF_TODO;
3462 }
3463 Log8(("rtDwarfUnwind_UpdateStateFromRow: uCfa=%RX64\n", uCfa));
3464
3465 /*
3466 * Do the architecture specific register updating.
3467 */
3468 switch (pState->enmArch)
3469 {
3470 case RTLDRARCH_AMD64:
3471 pState->u.x86.FrameAddr.off = uCfa - 8*2;
3472 pState->u.x86.Loaded.fAll = 0;
3473 pState->u.x86.Loaded.s.fFrameAddr = 1;
3474 rtDwarfUnwind_UpdatePCFromRow(pState, &Old, pRow, DWREG_AMD64_RA, pCie, uCfa, pfnGetReg, sizeof(uint64_t));
3475 rtDwarfUnwind_UpdateX86RFlagsFromRow(pState, &Old, pRow, DWREG_AMD64_RFLAGS, pCie, uCfa, pfnGetReg);
3476 rtDwarfUnwind_UpdateX86GRegFromRow(pState, &Old, X86_GREG_xAX, pRow, DWREG_AMD64_RAX, pCie, uCfa, pfnGetReg, sizeof(uint64_t));
3477 rtDwarfUnwind_UpdateX86GRegFromRow(pState, &Old, X86_GREG_xCX, pRow, DWREG_AMD64_RCX, pCie, uCfa, pfnGetReg, sizeof(uint64_t));
3478 rtDwarfUnwind_UpdateX86GRegFromRow(pState, &Old, X86_GREG_xDX, pRow, DWREG_AMD64_RDX, pCie, uCfa, pfnGetReg, sizeof(uint64_t));
3479 rtDwarfUnwind_UpdateX86GRegFromRow(pState, &Old, X86_GREG_xBX, pRow, DWREG_AMD64_RBX, pCie, uCfa, pfnGetReg, sizeof(uint64_t));
3480 rtDwarfUnwind_UpdateX86GRegFromRow(pState, &Old, X86_GREG_xSP, pRow, DWREG_AMD64_RSP, pCie, uCfa, pfnGetReg, sizeof(uint64_t));
3481 rtDwarfUnwind_UpdateX86GRegFromRow(pState, &Old, X86_GREG_xBP, pRow, DWREG_AMD64_RBP, pCie, uCfa, pfnGetReg, sizeof(uint64_t));
3482 rtDwarfUnwind_UpdateX86GRegFromRow(pState, &Old, X86_GREG_xSI, pRow, DWREG_AMD64_RSI, pCie, uCfa, pfnGetReg, sizeof(uint64_t));
3483 rtDwarfUnwind_UpdateX86GRegFromRow(pState, &Old, X86_GREG_xDI, pRow, DWREG_AMD64_RDI, pCie, uCfa, pfnGetReg, sizeof(uint64_t));
3484 rtDwarfUnwind_UpdateX86GRegFromRow(pState, &Old, X86_GREG_x8, pRow, DWREG_AMD64_R8, pCie, uCfa, pfnGetReg, sizeof(uint64_t));
3485 rtDwarfUnwind_UpdateX86GRegFromRow(pState, &Old, X86_GREG_x9, pRow, DWREG_AMD64_R9, pCie, uCfa, pfnGetReg, sizeof(uint64_t));
3486 rtDwarfUnwind_UpdateX86GRegFromRow(pState, &Old, X86_GREG_x10, pRow, DWREG_AMD64_R10, pCie, uCfa, pfnGetReg, sizeof(uint64_t));
3487 rtDwarfUnwind_UpdateX86GRegFromRow(pState, &Old, X86_GREG_x11, pRow, DWREG_AMD64_R11, pCie, uCfa, pfnGetReg, sizeof(uint64_t));
3488 rtDwarfUnwind_UpdateX86GRegFromRow(pState, &Old, X86_GREG_x12, pRow, DWREG_AMD64_R12, pCie, uCfa, pfnGetReg, sizeof(uint64_t));
3489 rtDwarfUnwind_UpdateX86GRegFromRow(pState, &Old, X86_GREG_x13, pRow, DWREG_AMD64_R13, pCie, uCfa, pfnGetReg, sizeof(uint64_t));
3490 rtDwarfUnwind_UpdateX86GRegFromRow(pState, &Old, X86_GREG_x14, pRow, DWREG_AMD64_R14, pCie, uCfa, pfnGetReg, sizeof(uint64_t));
3491 rtDwarfUnwind_UpdateX86GRegFromRow(pState, &Old, X86_GREG_x15, pRow, DWREG_AMD64_R15, pCie, uCfa, pfnGetReg, sizeof(uint64_t));
3492 rtDwarfUnwind_UpdateX86SRegFromRow(pState, &Old, X86_SREG_ES, pRow, DWREG_AMD64_ES, pCie, uCfa, pfnGetReg);
3493 rtDwarfUnwind_UpdateX86SRegFromRow(pState, &Old, X86_SREG_CS, pRow, DWREG_AMD64_CS, pCie, uCfa, pfnGetReg);
3494 rtDwarfUnwind_UpdateX86SRegFromRow(pState, &Old, X86_SREG_SS, pRow, DWREG_AMD64_SS, pCie, uCfa, pfnGetReg);
3495 rtDwarfUnwind_UpdateX86SRegFromRow(pState, &Old, X86_SREG_DS, pRow, DWREG_AMD64_DS, pCie, uCfa, pfnGetReg);
3496 rtDwarfUnwind_UpdateX86SRegFromRow(pState, &Old, X86_SREG_FS, pRow, DWREG_AMD64_FS, pCie, uCfa, pfnGetReg);
3497 rtDwarfUnwind_UpdateX86SRegFromRow(pState, &Old, X86_SREG_GS, pRow, DWREG_AMD64_GS, pCie, uCfa, pfnGetReg);
3498 break;
3499
3500 case RTLDRARCH_X86_32:
3501 case RTLDRARCH_X86_16:
3502 pState->u.x86.FrameAddr.off = uCfa - 4*2;
3503 pState->u.x86.Loaded.fAll = 0;
3504 pState->u.x86.Loaded.s.fFrameAddr = 1;
3505 rtDwarfUnwind_UpdatePCFromRow(pState, &Old, pRow, DWREG_X86_RA, pCie, uCfa, pfnGetReg, sizeof(uint32_t));
3506 rtDwarfUnwind_UpdateX86RFlagsFromRow(pState, &Old, pRow, DWREG_X86_EFLAGS, pCie, uCfa, pfnGetReg);
3507 rtDwarfUnwind_UpdateX86GRegFromRow(pState, &Old, X86_GREG_xAX, pRow, DWREG_X86_EAX, pCie, uCfa, pfnGetReg, sizeof(uint32_t));
3508 rtDwarfUnwind_UpdateX86GRegFromRow(pState, &Old, X86_GREG_xCX, pRow, DWREG_X86_ECX, pCie, uCfa, pfnGetReg, sizeof(uint32_t));
3509 rtDwarfUnwind_UpdateX86GRegFromRow(pState, &Old, X86_GREG_xDX, pRow, DWREG_X86_EDX, pCie, uCfa, pfnGetReg, sizeof(uint32_t));
3510 rtDwarfUnwind_UpdateX86GRegFromRow(pState, &Old, X86_GREG_xBX, pRow, DWREG_X86_EBX, pCie, uCfa, pfnGetReg, sizeof(uint32_t));
3511 rtDwarfUnwind_UpdateX86GRegFromRow(pState, &Old, X86_GREG_xSP, pRow, DWREG_X86_ESP, pCie, uCfa, pfnGetReg, sizeof(uint32_t));
3512 rtDwarfUnwind_UpdateX86GRegFromRow(pState, &Old, X86_GREG_xBP, pRow, DWREG_X86_EBP, pCie, uCfa, pfnGetReg, sizeof(uint32_t));
3513 rtDwarfUnwind_UpdateX86GRegFromRow(pState, &Old, X86_GREG_xSI, pRow, DWREG_X86_ESI, pCie, uCfa, pfnGetReg, sizeof(uint32_t));
3514 rtDwarfUnwind_UpdateX86GRegFromRow(pState, &Old, X86_GREG_xDI, pRow, DWREG_X86_EDI, pCie, uCfa, pfnGetReg, sizeof(uint32_t));
3515 rtDwarfUnwind_UpdateX86SRegFromRow(pState, &Old, X86_SREG_ES, pRow, DWREG_X86_ES, pCie, uCfa, pfnGetReg);
3516 rtDwarfUnwind_UpdateX86SRegFromRow(pState, &Old, X86_SREG_CS, pRow, DWREG_X86_CS, pCie, uCfa, pfnGetReg);
3517 rtDwarfUnwind_UpdateX86SRegFromRow(pState, &Old, X86_SREG_SS, pRow, DWREG_X86_SS, pCie, uCfa, pfnGetReg);
3518 rtDwarfUnwind_UpdateX86SRegFromRow(pState, &Old, X86_SREG_DS, pRow, DWREG_X86_DS, pCie, uCfa, pfnGetReg);
3519 rtDwarfUnwind_UpdateX86SRegFromRow(pState, &Old, X86_SREG_FS, pRow, DWREG_X86_FS, pCie, uCfa, pfnGetReg);
3520 rtDwarfUnwind_UpdateX86SRegFromRow(pState, &Old, X86_SREG_GS, pRow, DWREG_X86_GS, pCie, uCfa, pfnGetReg);
3521 break;
3522
3523 default:
3524 AssertFailedReturn(VERR_NOT_SUPPORTED);
3525 }
3526
3527 return VINF_SUCCESS;
3528}
3529
3530
3531/**
3532 * Processes a FDE, taking over after the PC range field.
3533 *
3534 * @returns IPRT status code.
3535 * @param pCursor The cursor.
3536 * @param pCie Information about the corresponding CIE.
3537 * @param uPcBegin The PC begin field value (sans segment).
3538 * @param cbPcRange The PC range from @a uPcBegin.
3539 * @param offInRange The offset into the range corresponding to
3540 * pState->uPc.
3541 * @param pState The unwind state to work.
3542 */
3543static int rtDwarfUnwind_ProcessFde(PRTDWARFCURSOR pCursor, PCRTDWARFCIEINFO pCie, uint64_t uPcBegin,
3544 uint64_t cbPcRange, uint64_t offInRange, PRTDBGUNWINDSTATE pState)
3545{
3546 /*
3547 * Deal with augmented data fields.
3548 */
3549 /* The size. */
3550 size_t cbInstr = ~(size_t)0;
3551 if (pCie->fHasAugmentationSize)
3552 {
3553 uint64_t cbAugData = rtDwarfCursor_GetULeb128(pCursor, UINT64_MAX);
3554 if (RT_FAILURE(pCursor->rc))
3555 return pCursor->rc;
3556 if (cbAugData > pCursor->cbUnitLeft)
3557 return VERR_DBG_MALFORMED_UNWIND_INFO;
3558 cbInstr = pCursor->cbUnitLeft - cbAugData;
3559 }
3560 else if (pCie->fHasUnknowAugmentation)
3561 return VERR_DBG_MALFORMED_UNWIND_INFO;
3562
3563 /* Parse the string and fetch FDE fields. */
3564 if (!pCie->fHasEhData)
3565 for (const char *pszAug = pCie->pszAugmentation; *pszAug != '\0'; pszAug++)
3566 switch (*pszAug)
3567 {
3568 case 'L':
3569 if (pCie->bLsdaPtrEnc != DW_EH_PE_omit)
3570 rtDwarfCursor_GetPtrEnc(pCursor, pCie->bLsdaPtrEnc, 0);
3571 break;
3572 }
3573
3574 /* Skip unconsumed bytes. */
3575 if ( cbInstr != ~(size_t)0
3576 && pCursor->cbUnitLeft > cbInstr)
3577 rtDwarfCursor_SkipBytes(pCursor, pCursor->cbUnitLeft - cbInstr);
3578 if (RT_FAILURE(pCursor->rc))
3579 return pCursor->rc;
3580
3581 /*
3582 * Now "execute" the programs till we've constructed the desired row.
3583 */
3584 RTDWARFCFROW Row;
3585 RTDWARFCFEXEC ExecState = { &Row, offInRange, 0, true /** @todo byte-order*/, pCie, uPcBegin, offInRange };
3586 RT_ZERO(Row);
3587
3588 int rc = rtDwarfUnwind_Execute(&ExecState, pCie->pbInstructions, (uint32_t)pCie->cbInstructions);
3589 if (rc == VINF_TRY_AGAIN)
3590 rc = rtDwarfUnwind_Execute(&ExecState, pCursor->pb, (uint32_t)pCursor->cbUnitLeft);
3591
3592 /* On success, extract whatever state we've got. */
3593 if (RT_SUCCESS(rc))
3594 rc = rtDwarfUnwind_UpdateStateFromRow(pState, &Row, pCie);
3595
3596 /*
3597 * Clean up allocations in case of pushes.
3598 */
3599 if (ExecState.pRow == &Row)
3600 Assert(!ExecState.pRow->fOnHeap);
3601 else
3602 do
3603 {
3604 PRTDWARFCFROW pPopped = ExecState.pRow;
3605 ExecState.pRow = ExecState.pRow->pNextOnStack;
3606 Assert(pPopped->fOnHeap);
3607 RTMemTmpFree(pPopped);
3608 } while (ExecState.pRow && ExecState.pRow != &Row);
3609
3610 RT_NOREF(pState, uPcBegin, cbPcRange, offInRange);
3611 return rc;
3612}
3613
3614
3615/**
3616 * Load the information we need from a CIE.
3617 *
3618 * This starts after the initial length and CIE_pointer fields has
3619 * been processed.
3620 *
3621 * @returns IPRT status code.
3622 * @param pCursor The cursor.
3623 * @param pNewCie The structure to populate with parsed CIE info.
3624 * @param offUnit The unit offset.
3625 * @param bDefaultPtrEnc The default pointer encoding.
3626 */
3627static int rtDwarfUnwind_LoadCie(PRTDWARFCURSOR pCursor, PRTDWARFCIEINFO pNewCie, uint64_t offUnit, uint8_t bDefaultPtrEnc)
3628{
3629Log8(("%#08RX64: rtDwarfUnwind_LoadCie: %.*Rhxs\n", offUnit, pCursor->cbUnitLeft, pCursor->pb));
3630 /*
3631 * Initialize the CIE record and get the version.
3632 */
3633 RT_ZERO(*pNewCie);
3634 pNewCie->offCie = offUnit;
3635 pNewCie->bLsdaPtrEnc = DW_EH_PE_omit;
3636 pNewCie->bAddressPtrEnc = DW_EH_PE_omit; /* set later */
3637 pNewCie->uDwarfVer = rtDwarfCursor_GetUByte(pCursor, 0);
3638 if ( pNewCie->uDwarfVer >= 1 /* Note! Some GCC versions may emit v1 here. */
3639 && pNewCie->uDwarfVer <= 5)
3640 { /* likely */ }
3641 else
3642 {
3643 Log(("rtDwarfUnwind_LoadCie(%RX64): uDwarfVer=%u: VERR_VERSION_MISMATCH\n", offUnit, pNewCie->uDwarfVer));
3644 return VERR_VERSION_MISMATCH;
3645 }
3646
3647 /*
3648 * The augmentation string.
3649 *
3650 * First deal with special "eh" string from oldish GCC (dwarf2out.c about 1997), specified in LSB:
3651 * https://refspecs.linuxfoundation.org/LSB_3.0.0/LSB-PDA/LSB-PDA/ehframechpt.html
3652 */
3653 pNewCie->pszAugmentation = rtDwarfCursor_GetSZ(pCursor, "");
3654 if ( pNewCie->pszAugmentation[0] == 'e'
3655 && pNewCie->pszAugmentation[1] == 'h'
3656 && pNewCie->pszAugmentation[2] == '\0')
3657 {
3658 pNewCie->fHasEhData = true;
3659 rtDwarfCursor_GetPtrEnc(pCursor, bDefaultPtrEnc, 0);
3660 }
3661 else
3662 {
3663 /* Regular augmentation string. */
3664 for (const char *pszAug = pNewCie->pszAugmentation; *pszAug != '\0'; pszAug++)
3665 switch (*pszAug)
3666 {
3667 case 'z':
3668 pNewCie->fHasAugmentationSize = true;
3669 break;
3670 case 'L':
3671 pNewCie->fHasLanguageSpecificDataArea = true;
3672 break;
3673 case 'P':
3674 pNewCie->fHasPersonalityRoutine = true;
3675 break;
3676 case 'R':
3677 pNewCie->fHasAddressEnc = true;
3678 break;
3679 case 'S':
3680 pNewCie->fIsSignalFrame = true;
3681 break;
3682 default:
3683 pNewCie->fHasUnknowAugmentation = true;
3684 break;
3685 }
3686 }
3687
3688 /*
3689 * More standard fields
3690 */
3691 uint8_t cbAddress = 0;
3692 if (pNewCie->uDwarfVer >= 4)
3693 {
3694 cbAddress = rtDwarfCursor_GetU8(pCursor, bDefaultPtrEnc == DW_EH_PE_udata8 ? 8 : 4);
3695 pNewCie->cbSegment = rtDwarfCursor_GetU8(pCursor, 0);
3696 }
3697 pNewCie->uCodeAlignFactor = rtDwarfCursor_GetULeb128(pCursor, 1);
3698 pNewCie->iDataAlignFactor = rtDwarfCursor_GetSLeb128(pCursor, 1);
3699 pNewCie->bRetReg = rtDwarfCursor_GetU8(pCursor, UINT8_MAX);
3700
3701 /*
3702 * Augmentation data.
3703 */
3704 if (!pNewCie->fHasEhData)
3705 {
3706 /* The size. */
3707 size_t cbInstr = ~(size_t)0;
3708 if (pNewCie->fHasAugmentationSize)
3709 {
3710 uint64_t cbAugData = rtDwarfCursor_GetULeb128(pCursor, UINT64_MAX);
3711 if (RT_FAILURE(pCursor->rc))
3712 {
3713 Log(("rtDwarfUnwind_LoadCie(%#RX64): rtDwarfCursor_GetULeb128 -> %Rrc!\n", offUnit, pCursor->rc));
3714 return pCursor->rc;
3715 }
3716 if (cbAugData > pCursor->cbUnitLeft)
3717 {
3718 Log(("rtDwarfUnwind_LoadCie(%#RX64): cbAugData=%#x pCursor->cbUnitLeft=%#x -> VERR_DBG_MALFORMED_UNWIND_INFO!\n", offUnit, cbAugData, pCursor->cbUnitLeft));
3719 return VERR_DBG_MALFORMED_UNWIND_INFO;
3720 }
3721 cbInstr = pCursor->cbUnitLeft - cbAugData;
3722 }
3723 else if (pNewCie->fHasUnknowAugmentation)
3724 {
3725 Log(("rtDwarfUnwind_LoadCie(%#RX64): fHasUnknowAugmentation=%1 -> VERR_DBG_MALFORMED_UNWIND_INFO!\n", offUnit));
3726 return VERR_DBG_MALFORMED_UNWIND_INFO;
3727 }
3728
3729 /* Parse the string. */
3730 for (const char *pszAug = pNewCie->pszAugmentation; *pszAug != '\0'; pszAug++)
3731 switch (*pszAug)
3732 {
3733 case 'L':
3734 pNewCie->bLsdaPtrEnc = rtDwarfCursor_GetU8(pCursor, DW_EH_PE_omit);
3735 break;
3736 case 'P':
3737 rtDwarfCursor_GetPtrEnc(pCursor, rtDwarfCursor_GetU8(pCursor, DW_EH_PE_omit), 0);
3738 break;
3739 case 'R':
3740 pNewCie->bAddressPtrEnc = rtDwarfCursor_GetU8(pCursor, DW_EH_PE_omit);
3741 break;
3742 }
3743
3744 /* Skip unconsumed bytes. */
3745 if ( cbInstr != ~(size_t)0
3746 && pCursor->cbUnitLeft > cbInstr)
3747 rtDwarfCursor_SkipBytes(pCursor, pCursor->cbUnitLeft - cbInstr);
3748 }
3749
3750 /*
3751 * Note down where the instructions are.
3752 */
3753 pNewCie->pbInstructions = pCursor->pb;
3754 pNewCie->cbInstructions = pCursor->cbUnitLeft;
3755
3756 /*
3757 * Determine the target address encoding. Make sure we resolve DW_EH_PE_ptr.
3758 */
3759 if (pNewCie->bAddressPtrEnc == DW_EH_PE_omit)
3760 switch (cbAddress)
3761 {
3762 case 2: pNewCie->bAddressPtrEnc = DW_EH_PE_udata2; break;
3763 case 4: pNewCie->bAddressPtrEnc = DW_EH_PE_udata4; break;
3764 case 8: pNewCie->bAddressPtrEnc = DW_EH_PE_udata8; break;
3765 default: pNewCie->bAddressPtrEnc = bDefaultPtrEnc; break;
3766 }
3767 else if ((pNewCie->bAddressPtrEnc & DW_EH_PE_FORMAT_MASK) == DW_EH_PE_ptr)
3768 pNewCie->bAddressPtrEnc = bDefaultPtrEnc;
3769
3770 return VINF_SUCCESS;
3771}
3772
3773
3774/**
3775 * Does a slow unwind of a '.debug_frame' or '.eh_frame' section.
3776 *
3777 * @returns IPRT status code.
3778 * @param pCursor The cursor.
3779 * @param uRvaCursor The RVA corrsponding to the cursor start location.
3780 * @param idxSeg The segment of the PC location.
3781 * @param offSeg The segment offset of the PC location.
3782 * @param uRva The RVA of the PC location.
3783 * @param pState The unwind state to work.
3784 * @param bDefaultPtrEnc The default pointer encoding.
3785 * @param fIsEhFrame Set if this is a '.eh_frame'. GCC generate these
3786 * with different CIE_pointer values.
3787 */
3788DECLHIDDEN(int) rtDwarfUnwind_Slow(PRTDWARFCURSOR pCursor, RTUINTPTR uRvaCursor,
3789 RTDBGSEGIDX idxSeg, RTUINTPTR offSeg, RTUINTPTR uRva,
3790 PRTDBGUNWINDSTATE pState, uint8_t bDefaultPtrEnc, bool fIsEhFrame)
3791{
3792 Log8(("rtDwarfUnwind_Slow: idxSeg=%#x offSeg=%RTptr uRva=%RTptr enmArch=%d PC=%#RX64\n", idxSeg, offSeg, uRva, pState->enmArch, pState->uPc));
3793
3794 /*
3795 * CIE info we collect.
3796 */
3797 PRTDWARFCIEINFO paCies = NULL;
3798 uint32_t cCies = 0;
3799 PRTDWARFCIEINFO pCieHint = NULL;
3800
3801 /*
3802 * Do the scanning.
3803 */
3804 int rc = VERR_DBG_UNWIND_INFO_NOT_FOUND;
3805 while (!rtDwarfCursor_IsAtEnd(pCursor))
3806 {
3807 uint64_t const offUnit = rtDwarfCursor_CalcSectOffsetU32(pCursor);
3808 if (rtDwarfCursor_GetInitialLength(pCursor) == 0)
3809 break;
3810Log8(("%#08RX64: rtDwarfCursor_GetInitialLength -> %#x\n", offUnit, pCursor->cbUnitLeft));
3811
3812 uint64_t const offRelCie = rtDwarfCursor_GetUOff(pCursor, 0);
3813 if ( offRelCie != 0
3814 && offRelCie != (pCursor->f64bitDwarf ? UINT64_MAX : UINT32_MAX))
3815 {
3816 /*
3817 * Frame descriptor entry (FDE).
3818 */
3819 /* Locate the corresponding CIE. The CIE pointer is self relative
3820 in .eh_frame and section relative in .debug_frame. */
3821 PRTDWARFCIEINFO pCieForFde;
3822 uint64_t offCie = fIsEhFrame ? offUnit + 4 - offRelCie : offRelCie;
3823 if (pCieHint && pCieHint->offCie == offCie)
3824 pCieForFde = pCieHint;
3825 else
3826 {
3827 pCieForFde = NULL;
3828 uint32_t i = cCies;
3829 while (i-- > 0)
3830 if (paCies[i].offCie == offCie)
3831 {
3832 pCieHint = pCieForFde = &paCies[i];
3833 break;
3834 }
3835 }
3836 if (pCieForFde)
3837 {
3838 /* Read the PC range covered by this FDE (the fields are also known as initial_location). */
3839 RTDBGSEGIDX idxFdeSeg = RTDBGSEGIDX_RVA;
3840 if (pCieForFde->cbSegment)
3841 idxFdeSeg = rtDwarfCursor_GetVarSizedU(pCursor, pCieForFde->cbSegment, RTDBGSEGIDX_RVA);
3842 uint64_t uPcBegin;
3843 switch (pCieForFde->bAddressPtrEnc & DW_EH_PE_APPL_MASK)
3844 {
3845 default: AssertFailed();
3846 RT_FALL_THRU();
3847 case DW_EH_PE_absptr:
3848 uPcBegin = rtDwarfCursor_GetPtrEnc(pCursor, pCieForFde->bAddressPtrEnc, 0);
3849 break;
3850 case DW_EH_PE_pcrel:
3851 {
3852 uPcBegin = rtDwarfCursor_CalcSectOffsetU32(pCursor) + uRvaCursor;
3853 uPcBegin += rtDwarfCursor_GetPtrEnc(pCursor, pCieForFde->bAddressPtrEnc, 0);
3854 break;
3855 }
3856 }
3857 uint64_t cbPcRange = rtDwarfCursor_GetPtrEnc(pCursor, pCieForFde->bAddressPtrEnc, 0);
3858
3859 /* Match it with what we're looking for. */
3860 bool fMatch = idxFdeSeg == RTDBGSEGIDX_RVA
3861 ? uRva - uPcBegin < cbPcRange
3862 : idxSeg == idxFdeSeg && offSeg - uPcBegin < cbPcRange;
3863 Log8(("%#08RX64: FDE pCie=%p idxFdeSeg=%#x uPcBegin=%#RX64 cbPcRange=%#x fMatch=%d\n",
3864 offUnit, pCieForFde, idxFdeSeg, uPcBegin, cbPcRange, fMatch));
3865 if (fMatch)
3866 {
3867 rc = rtDwarfUnwind_ProcessFde(pCursor, pCieForFde, uPcBegin, cbPcRange,
3868 idxFdeSeg == RTDBGSEGIDX_RVA ? uRva - uPcBegin : offSeg - uPcBegin,
3869 pState);
3870 break;
3871 }
3872 }
3873 else
3874 Log8(("%#08RX64: FDE - pCie=NULL!! offCie=%#RX64 offRelCie=%#RX64 fIsEhFrame=%d\n", offUnit, offCie, offRelCie, fIsEhFrame));
3875 }
3876 else
3877 {
3878 /*
3879 * Common information entry (CIE). Record the info we need about it.
3880 */
3881 if ((cCies & 8) == 0)
3882 {
3883 void *pvNew = RTMemRealloc(paCies, sizeof(paCies[0]) * (cCies + 8));
3884 if (pvNew)
3885 paCies = (PRTDWARFCIEINFO)pvNew;
3886 else
3887 {
3888 rc = VERR_NO_MEMORY;
3889 break;
3890 }
3891 }
3892 Log8(("%#08RX64: CIE\n", offUnit));
3893 int rc2 = rtDwarfUnwind_LoadCie(pCursor, &paCies[cCies], offUnit, bDefaultPtrEnc);
3894 if (RT_SUCCESS(rc2))
3895 {
3896 Log8(("%#08RX64: CIE #%u: offCie=%#RX64\n", offUnit, cCies, paCies[cCies].offCie));
3897 cCies++;
3898 }
3899 }
3900 rtDwarfCursor_SkipUnit(pCursor);
3901 }
3902
3903 /*
3904 * Cleanup.
3905 */
3906 if (paCies)
3907 RTMemFree(paCies);
3908 Log8(("rtDwarfUnwind_Slow: returns %Rrc PC=%#RX64\n", rc, pState->uPc));
3909 return rc;
3910}
3911
3912
3913/**
3914 * Helper for translating a loader architecture value to a pointe encoding.
3915 *
3916 * @returns Pointer encoding.
3917 * @param enmLdrArch The loader architecture value to convert.
3918 */
3919static uint8_t rtDwarfUnwind_ArchToPtrEnc(RTLDRARCH enmLdrArch)
3920{
3921 switch (enmLdrArch)
3922 {
3923 case RTLDRARCH_AMD64:
3924 case RTLDRARCH_ARM64:
3925 return DW_EH_PE_udata8;
3926 case RTLDRARCH_X86_16:
3927 case RTLDRARCH_X86_32:
3928 case RTLDRARCH_ARM32:
3929 return DW_EH_PE_udata4;
3930 case RTLDRARCH_HOST:
3931 case RTLDRARCH_WHATEVER:
3932 case RTLDRARCH_INVALID:
3933 case RTLDRARCH_END:
3934 case RTLDRARCH_32BIT_HACK:
3935 break;
3936 }
3937 AssertFailed();
3938 return DW_EH_PE_udata4;
3939}
3940
3941
3942/**
3943 * Interface for the loader code.
3944 *
3945 * @returns IPRT status.
3946 * @param pvSection The '.eh_frame' section data.
3947 * @param cbSection The size of the '.eh_frame' section data.
3948 * @param uRvaSection The RVA of the '.eh_frame' section.
3949 * @param idxSeg The segment of the PC location.
3950 * @param offSeg The segment offset of the PC location.
3951 * @param uRva The RVA of the PC location.
3952 * @param pState The unwind state to work.
3953 * @param enmArch The image architecture.
3954 */
3955DECLHIDDEN(int) rtDwarfUnwind_EhData(void const *pvSection, size_t cbSection, RTUINTPTR uRvaSection,
3956 RTDBGSEGIDX idxSeg, RTUINTPTR offSeg, RTUINTPTR uRva,
3957 PRTDBGUNWINDSTATE pState, RTLDRARCH enmArch)
3958{
3959 RTDWARFCURSOR Cursor;
3960 rtDwarfCursor_InitForMem(&Cursor, pvSection, cbSection);
3961 int rc = rtDwarfUnwind_Slow(&Cursor, uRvaSection, idxSeg, offSeg, uRva, pState,
3962 rtDwarfUnwind_ArchToPtrEnc(enmArch), true /*fIsEhFrame*/);
3963 LogFlow(("rtDwarfUnwind_EhData: rtDwarfUnwind_Slow -> %Rrc\n", rc));
3964 rc = rtDwarfCursor_Delete(&Cursor, rc);
3965 LogFlow(("rtDwarfUnwind_EhData: returns %Rrc\n", rc));
3966 return rc;
3967}
3968
3969
3970/*
3971 *
3972 * DWARF Line Numbers.
3973 * DWARF Line Numbers.
3974 * DWARF Line Numbers.
3975 *
3976 */
3977
3978
3979/**
3980 * Defines a file name.
3981 *
3982 * @returns IPRT status code.
3983 * @param pLnState The line number program state.
3984 * @param pszFilename The name of the file.
3985 * @param idxInc The include path index.
3986 */
3987static int rtDwarfLine_DefineFileName(PRTDWARFLINESTATE pLnState, const char *pszFilename, uint64_t idxInc)
3988{
3989 /*
3990 * Resize the array if necessary.
3991 */
3992 uint32_t iFileName = pLnState->cFileNames;
3993 if ((iFileName % 2) == 0)
3994 {
3995 void *pv = RTMemRealloc(pLnState->papszFileNames, sizeof(pLnState->papszFileNames[0]) * (iFileName + 2));
3996 if (!pv)
3997 return VERR_NO_MEMORY;
3998 pLnState->papszFileNames = (char **)pv;
3999 }
4000
4001 /*
4002 * Add the file name.
4003 */
4004 if ( pszFilename[0] == '/'
4005 || pszFilename[0] == '\\'
4006 || (RT_C_IS_ALPHA(pszFilename[0]) && pszFilename[1] == ':') )
4007 pLnState->papszFileNames[iFileName] = RTStrDup(pszFilename);
4008 else if (idxInc < pLnState->cIncPaths)
4009 pLnState->papszFileNames[iFileName] = RTPathJoinA(pLnState->papszIncPaths[idxInc], pszFilename);
4010 else
4011 return VERR_DWARF_BAD_LINE_NUMBER_HEADER;
4012 if (!pLnState->papszFileNames[iFileName])
4013 return VERR_NO_STR_MEMORY;
4014 pLnState->cFileNames = iFileName + 1;
4015
4016 /*
4017 * Sanitize the name.
4018 */
4019 int rc = rtDbgModDwarfStringToUtf8(pLnState->pDwarfMod, &pLnState->papszFileNames[iFileName]);
4020 Log((" File #%02u = '%s'\n", iFileName, pLnState->papszFileNames[iFileName]));
4021 return rc;
4022}
4023
4024
4025/**
4026 * Adds a line to the table and resets parts of the state (DW_LNS_copy).
4027 *
4028 * @returns IPRT status code
4029 * @param pLnState The line number program state.
4030 * @param offOpCode The opcode offset (for logging
4031 * purposes).
4032 */
4033static int rtDwarfLine_AddLine(PRTDWARFLINESTATE pLnState, uint32_t offOpCode)
4034{
4035 PRTDBGMODDWARF pThis = pLnState->pDwarfMod;
4036 int rc;
4037 if (pThis->iWatcomPass == 1)
4038 rc = rtDbgModDwarfRecordSegOffset(pThis, pLnState->Regs.uSegment, pLnState->Regs.uAddress + 1);
4039 else
4040 {
4041 const char *pszFile = pLnState->Regs.iFile < pLnState->cFileNames
4042 ? pLnState->papszFileNames[pLnState->Regs.iFile]
4043 : "<bad file name index>";
4044 NOREF(offOpCode);
4045
4046 RTDBGSEGIDX iSeg;
4047 RTUINTPTR offSeg;
4048 rc = rtDbgModDwarfLinkAddressToSegOffset(pLnState->pDwarfMod, pLnState->Regs.uSegment, pLnState->Regs.uAddress,
4049 &iSeg, &offSeg); /*AssertRC(rc);*/
4050 if (RT_SUCCESS(rc))
4051 {
4052 Log2(("rtDwarfLine_AddLine: %x:%08llx (%#llx) %s(%d) [offOpCode=%08x]\n", iSeg, offSeg, pLnState->Regs.uAddress, pszFile, pLnState->Regs.uLine, offOpCode));
4053 rc = RTDbgModLineAdd(pLnState->pDwarfMod->hCnt, pszFile, pLnState->Regs.uLine, iSeg, offSeg, NULL);
4054
4055 /* Ignore address conflicts for now. */
4056 if (rc == VERR_DBG_ADDRESS_CONFLICT)
4057 rc = VINF_SUCCESS;
4058 }
4059 else
4060 rc = VINF_SUCCESS; /* ignore failure */
4061 }
4062
4063 pLnState->Regs.fBasicBlock = false;
4064 pLnState->Regs.fPrologueEnd = false;
4065 pLnState->Regs.fEpilogueBegin = false;
4066 pLnState->Regs.uDiscriminator = 0;
4067 return rc;
4068}
4069
4070
4071/**
4072 * Reset the program to the start-of-sequence state.
4073 *
4074 * @param pLnState The line number program state.
4075 */
4076static void rtDwarfLine_ResetState(PRTDWARFLINESTATE pLnState)
4077{
4078 pLnState->Regs.uAddress = 0;
4079 pLnState->Regs.idxOp = 0;
4080 pLnState->Regs.iFile = 1;
4081 pLnState->Regs.uLine = 1;
4082 pLnState->Regs.uColumn = 0;
4083 pLnState->Regs.fIsStatement = RT_BOOL(pLnState->Hdr.u8DefIsStmt);
4084 pLnState->Regs.fBasicBlock = false;
4085 pLnState->Regs.fEndSequence = false;
4086 pLnState->Regs.fPrologueEnd = false;
4087 pLnState->Regs.fEpilogueBegin = false;
4088 pLnState->Regs.uIsa = 0;
4089 pLnState->Regs.uDiscriminator = 0;
4090 pLnState->Regs.uSegment = 0;
4091}
4092
4093
4094/**
4095 * Runs the line number program.
4096 *
4097 * @returns IPRT status code.
4098 * @param pLnState The line number program state.
4099 * @param pCursor The cursor.
4100 */
4101static int rtDwarfLine_RunProgram(PRTDWARFLINESTATE pLnState, PRTDWARFCURSOR pCursor)
4102{
4103 LogFlow(("rtDwarfLine_RunProgram: cbUnitLeft=%zu\n", pCursor->cbUnitLeft));
4104
4105 int rc = VINF_SUCCESS;
4106 rtDwarfLine_ResetState(pLnState);
4107
4108 while (!rtDwarfCursor_IsAtEndOfUnit(pCursor))
4109 {
4110#ifdef LOG_ENABLED
4111 uint32_t const offOpCode = rtDwarfCursor_CalcSectOffsetU32(pCursor);
4112#else
4113 uint32_t const offOpCode = 0;
4114#endif
4115 uint8_t bOpCode = rtDwarfCursor_GetUByte(pCursor, DW_LNS_extended);
4116 if (bOpCode >= pLnState->Hdr.u8OpcodeBase)
4117 {
4118 /*
4119 * Special opcode.
4120 */
4121 uint8_t const bLogOpCode = bOpCode; NOREF(bLogOpCode);
4122 bOpCode -= pLnState->Hdr.u8OpcodeBase;
4123
4124 int32_t const cLineDelta = bOpCode % pLnState->Hdr.u8LineRange + (int32_t)pLnState->Hdr.s8LineBase;
4125 bOpCode /= pLnState->Hdr.u8LineRange;
4126
4127 uint64_t uTmp = bOpCode + pLnState->Regs.idxOp;
4128 uint64_t const cAddressDelta = uTmp / pLnState->Hdr.cMaxOpsPerInstr * pLnState->Hdr.cbMinInstr;
4129 uint64_t const cOpIndexDelta = uTmp % pLnState->Hdr.cMaxOpsPerInstr;
4130
4131 pLnState->Regs.uLine += cLineDelta;
4132 pLnState->Regs.uAddress += cAddressDelta;
4133 pLnState->Regs.idxOp += cOpIndexDelta;
4134 Log2(("%08x: DW Special Opcode %#04x: uLine + %d => %u; uAddress + %#llx => %#llx; idxOp + %#llx => %#llx\n",
4135 offOpCode, bLogOpCode, cLineDelta, pLnState->Regs.uLine, cAddressDelta, pLnState->Regs.uAddress,
4136 cOpIndexDelta, pLnState->Regs.idxOp));
4137
4138 /*
4139 * LLVM emits debug info for global constructors (_GLOBAL__I_a) which are not part of source
4140 * code but are inserted by the compiler: The resulting line number will be 0
4141 * because they are not part of the source file obviously (see https://reviews.llvm.org/rL205999),
4142 * so skip adding them when they are encountered.
4143 */
4144 if (pLnState->Regs.uLine)
4145 rc = rtDwarfLine_AddLine(pLnState, offOpCode);
4146 }
4147 else
4148 {
4149 switch (bOpCode)
4150 {
4151 /*
4152 * Standard opcode.
4153 */
4154 case DW_LNS_copy:
4155 Log2(("%08x: DW_LNS_copy\n", offOpCode));
4156 /* See the comment about LLVM above. */
4157 if (pLnState->Regs.uLine)
4158 rc = rtDwarfLine_AddLine(pLnState, offOpCode);
4159 break;
4160
4161 case DW_LNS_advance_pc:
4162 {
4163 uint64_t u64Adv = rtDwarfCursor_GetULeb128(pCursor, 0);
4164 pLnState->Regs.uAddress += (pLnState->Regs.idxOp + u64Adv) / pLnState->Hdr.cMaxOpsPerInstr
4165 * pLnState->Hdr.cbMinInstr;
4166 pLnState->Regs.idxOp += (pLnState->Regs.idxOp + u64Adv) % pLnState->Hdr.cMaxOpsPerInstr;
4167 Log2(("%08x: DW_LNS_advance_pc: u64Adv=%#llx (%lld) )\n", offOpCode, u64Adv, u64Adv));
4168 break;
4169 }
4170
4171 case DW_LNS_advance_line:
4172 {
4173 int32_t cLineDelta = rtDwarfCursor_GetSLeb128AsS32(pCursor, 0);
4174 pLnState->Regs.uLine += cLineDelta;
4175 Log2(("%08x: DW_LNS_advance_line: uLine + %d => %u\n", offOpCode, cLineDelta, pLnState->Regs.uLine));
4176 break;
4177 }
4178
4179 case DW_LNS_set_file:
4180 pLnState->Regs.iFile = rtDwarfCursor_GetULeb128AsU32(pCursor, 0);
4181 Log2(("%08x: DW_LNS_set_file: iFile=%u\n", offOpCode, pLnState->Regs.iFile));
4182 break;
4183
4184 case DW_LNS_set_column:
4185 pLnState->Regs.uColumn = rtDwarfCursor_GetULeb128AsU32(pCursor, 0);
4186 Log2(("%08x: DW_LNS_set_column\n", offOpCode));
4187 break;
4188
4189 case DW_LNS_negate_stmt:
4190 pLnState->Regs.fIsStatement = !pLnState->Regs.fIsStatement;
4191 Log2(("%08x: DW_LNS_negate_stmt\n", offOpCode));
4192 break;
4193
4194 case DW_LNS_set_basic_block:
4195 pLnState->Regs.fBasicBlock = true;
4196 Log2(("%08x: DW_LNS_set_basic_block\n", offOpCode));
4197 break;
4198
4199 case DW_LNS_const_add_pc:
4200 {
4201 uint8_t u8Adv = (255 - pLnState->Hdr.u8OpcodeBase) / pLnState->Hdr.u8LineRange;
4202 if (pLnState->Hdr.cMaxOpsPerInstr <= 1)
4203 pLnState->Regs.uAddress += (uint32_t)pLnState->Hdr.cbMinInstr * u8Adv;
4204 else
4205 {
4206 pLnState->Regs.uAddress += (pLnState->Regs.idxOp + u8Adv) / pLnState->Hdr.cMaxOpsPerInstr
4207 * pLnState->Hdr.cbMinInstr;
4208 pLnState->Regs.idxOp = (pLnState->Regs.idxOp + u8Adv) % pLnState->Hdr.cMaxOpsPerInstr;
4209 }
4210 Log2(("%08x: DW_LNS_const_add_pc\n", offOpCode));
4211 break;
4212 }
4213 case DW_LNS_fixed_advance_pc:
4214 pLnState->Regs.uAddress += rtDwarfCursor_GetUHalf(pCursor, 0);
4215 pLnState->Regs.idxOp = 0;
4216 Log2(("%08x: DW_LNS_fixed_advance_pc\n", offOpCode));
4217 break;
4218
4219 case DW_LNS_set_prologue_end:
4220 pLnState->Regs.fPrologueEnd = true;
4221 Log2(("%08x: DW_LNS_set_prologue_end\n", offOpCode));
4222 break;
4223
4224 case DW_LNS_set_epilogue_begin:
4225 pLnState->Regs.fEpilogueBegin = true;
4226 Log2(("%08x: DW_LNS_set_epilogue_begin\n", offOpCode));
4227 break;
4228
4229 case DW_LNS_set_isa:
4230 pLnState->Regs.uIsa = rtDwarfCursor_GetULeb128AsU32(pCursor, 0);
4231 Log2(("%08x: DW_LNS_set_isa %#x\n", offOpCode, pLnState->Regs.uIsa));
4232 break;
4233
4234 default:
4235 {
4236 unsigned cOpsToSkip = pLnState->Hdr.pacStdOperands[bOpCode - 1];
4237 Log(("rtDwarfLine_RunProgram: Unknown standard opcode %#x, %#x operands, at %08x.\n", bOpCode, cOpsToSkip, offOpCode));
4238 while (cOpsToSkip-- > 0)
4239 rc = rtDwarfCursor_SkipLeb128(pCursor);
4240 break;
4241 }
4242
4243 /*
4244 * Extended opcode.
4245 */
4246 case DW_LNS_extended:
4247 {
4248 /* The instruction has a length prefix. */
4249 uint64_t cbInstr = rtDwarfCursor_GetULeb128(pCursor, UINT64_MAX);
4250 if (RT_FAILURE(pCursor->rc))
4251 return pCursor->rc;
4252 if (cbInstr > pCursor->cbUnitLeft)
4253 return VERR_DWARF_BAD_LNE;
4254 uint8_t const * const pbEndOfInstr = rtDwarfCursor_CalcPos(pCursor, cbInstr);
4255
4256 /* Get the opcode and deal with it if we know it. */
4257 bOpCode = rtDwarfCursor_GetUByte(pCursor, 0);
4258 switch (bOpCode)
4259 {
4260 case DW_LNE_end_sequence:
4261#if 0 /* No need for this, I think. */
4262 pLnState->Regs.fEndSequence = true;
4263 rc = rtDwarfLine_AddLine(pLnState, offOpCode);
4264#endif
4265 rtDwarfLine_ResetState(pLnState);
4266 Log2(("%08x: DW_LNE_end_sequence\n", offOpCode));
4267 break;
4268
4269 case DW_LNE_set_address:
4270 pLnState->Regs.uAddress = rtDwarfCursor_GetVarSizedU(pCursor, cbInstr - 1, UINT64_MAX);
4271 pLnState->Regs.idxOp = 0;
4272 Log2(("%08x: DW_LNE_set_address: %#llx\n", offOpCode, pLnState->Regs.uAddress));
4273 break;
4274
4275 case DW_LNE_define_file:
4276 {
4277 const char *pszFilename = rtDwarfCursor_GetSZ(pCursor, NULL);
4278 uint32_t idxInc = rtDwarfCursor_GetULeb128AsU32(pCursor, UINT32_MAX);
4279 rtDwarfCursor_SkipLeb128(pCursor); /* st_mtime */
4280 rtDwarfCursor_SkipLeb128(pCursor); /* st_size */
4281 Log2(("%08x: DW_LNE_define_file: {%d}/%s\n", offOpCode, idxInc, pszFilename));
4282
4283 rc = rtDwarfCursor_AdvanceToPos(pCursor, pbEndOfInstr);
4284 if (RT_SUCCESS(rc))
4285 rc = rtDwarfLine_DefineFileName(pLnState, pszFilename, idxInc);
4286 break;
4287 }
4288
4289 /*
4290 * Note! Was defined in DWARF 4. But... Watcom used it
4291 * for setting the segment in DWARF 2, creating
4292 * an incompatibility with the newer standard.
4293 */
4294 case DW_LNE_set_descriminator:
4295 if (pLnState->Hdr.uVer != 2)
4296 {
4297 Assert(pLnState->Hdr.uVer >= 4);
4298 pLnState->Regs.uDiscriminator = rtDwarfCursor_GetULeb128AsU32(pCursor, UINT32_MAX);
4299 Log2(("%08x: DW_LNE_set_descriminator: %u\n", offOpCode, pLnState->Regs.uDiscriminator));
4300 }
4301 else
4302 {
4303 uint64_t uSeg = rtDwarfCursor_GetVarSizedU(pCursor, cbInstr - 1, UINT64_MAX);
4304 Log2(("%08x: DW_LNE_set_segment: %#llx, cbInstr=%#x - Watcom Extension\n", offOpCode, uSeg, cbInstr));
4305 pLnState->Regs.uSegment = (RTSEL)uSeg;
4306 AssertStmt(pLnState->Regs.uSegment == uSeg, rc = VERR_DWARF_BAD_INFO);
4307 }
4308 break;
4309
4310 default:
4311 Log(("rtDwarfLine_RunProgram: Unknown extended opcode %#x, length %#x at %08x\n", bOpCode, cbInstr, offOpCode));
4312 break;
4313 }
4314
4315 /* Advance the cursor to the end of the instruction . */
4316 rtDwarfCursor_AdvanceToPos(pCursor, pbEndOfInstr);
4317 break;
4318 }
4319 }
4320 }
4321
4322 /*
4323 * Check the status before looping.
4324 */
4325 if (RT_FAILURE(rc))
4326 return rc;
4327 if (RT_FAILURE(pCursor->rc))
4328 return pCursor->rc;
4329 }
4330 return rc;
4331}
4332
4333
4334/**
4335 * Reads the include directories for a line number unit.
4336 *
4337 * @returns IPRT status code
4338 * @param pLnState The line number program state.
4339 * @param pCursor The cursor.
4340 */
4341static int rtDwarfLine_ReadFileNames(PRTDWARFLINESTATE pLnState, PRTDWARFCURSOR pCursor)
4342{
4343 int rc = rtDwarfLine_DefineFileName(pLnState, "/<bad-zero-file-name-entry>", 0);
4344 if (RT_FAILURE(rc))
4345 return rc;
4346
4347 for (;;)
4348 {
4349 const char *psz = rtDwarfCursor_GetSZ(pCursor, NULL);
4350 if (!*psz)
4351 break;
4352
4353 uint64_t idxInc = rtDwarfCursor_GetULeb128(pCursor, UINT64_MAX);
4354 rtDwarfCursor_SkipLeb128(pCursor); /* st_mtime */
4355 rtDwarfCursor_SkipLeb128(pCursor); /* st_size */
4356
4357 rc = rtDwarfLine_DefineFileName(pLnState, psz, idxInc);
4358 if (RT_FAILURE(rc))
4359 return rc;
4360 }
4361 return pCursor->rc;
4362}
4363
4364
4365/**
4366 * Reads the include directories for a line number unit.
4367 *
4368 * @returns IPRT status code
4369 * @param pLnState The line number program state.
4370 * @param pCursor The cursor.
4371 */
4372static int rtDwarfLine_ReadIncludePaths(PRTDWARFLINESTATE pLnState, PRTDWARFCURSOR pCursor)
4373{
4374 const char *psz = ""; /* The zeroth is the unit dir. */
4375 for (;;)
4376 {
4377 if ((pLnState->cIncPaths % 2) == 0)
4378 {
4379 void *pv = RTMemRealloc(pLnState->papszIncPaths, sizeof(pLnState->papszIncPaths[0]) * (pLnState->cIncPaths + 2));
4380 if (!pv)
4381 return VERR_NO_MEMORY;
4382 pLnState->papszIncPaths = (const char **)pv;
4383 }
4384 Log((" Path #%02u = '%s'\n", pLnState->cIncPaths, psz));
4385 pLnState->papszIncPaths[pLnState->cIncPaths] = psz;
4386 pLnState->cIncPaths++;
4387
4388 psz = rtDwarfCursor_GetSZ(pCursor, NULL);
4389 if (!*psz)
4390 break;
4391 }
4392
4393 return pCursor->rc;
4394}
4395
4396
4397/**
4398 * Explodes the line number table for a compilation unit.
4399 *
4400 * @returns IPRT status code
4401 * @param pThis The DWARF instance.
4402 * @param pCursor The cursor to read the line number information
4403 * via.
4404 */
4405static int rtDwarfLine_ExplodeUnit(PRTDBGMODDWARF pThis, PRTDWARFCURSOR pCursor)
4406{
4407 RTDWARFLINESTATE LnState;
4408 RT_ZERO(LnState);
4409 LnState.pDwarfMod = pThis;
4410
4411 /*
4412 * Parse the header.
4413 */
4414 rtDwarfCursor_GetInitialLength(pCursor);
4415 LnState.Hdr.uVer = rtDwarfCursor_GetUHalf(pCursor, 0);
4416 if ( LnState.Hdr.uVer < 2
4417 || LnState.Hdr.uVer > 4)
4418 return rtDwarfCursor_SkipUnit(pCursor);
4419
4420 LnState.Hdr.offFirstOpcode = rtDwarfCursor_GetUOff(pCursor, 0);
4421 uint8_t const * const pbFirstOpcode = rtDwarfCursor_CalcPos(pCursor, LnState.Hdr.offFirstOpcode);
4422
4423 LnState.Hdr.cbMinInstr = rtDwarfCursor_GetUByte(pCursor, 0);
4424 if (LnState.Hdr.uVer >= 4)
4425 LnState.Hdr.cMaxOpsPerInstr = rtDwarfCursor_GetUByte(pCursor, 0);
4426 else
4427 LnState.Hdr.cMaxOpsPerInstr = 1;
4428 LnState.Hdr.u8DefIsStmt = rtDwarfCursor_GetUByte(pCursor, 0);
4429 LnState.Hdr.s8LineBase = rtDwarfCursor_GetSByte(pCursor, 0);
4430 LnState.Hdr.u8LineRange = rtDwarfCursor_GetUByte(pCursor, 0);
4431 LnState.Hdr.u8OpcodeBase = rtDwarfCursor_GetUByte(pCursor, 0);
4432
4433 if ( !LnState.Hdr.u8OpcodeBase
4434 || !LnState.Hdr.cMaxOpsPerInstr
4435 || !LnState.Hdr.u8LineRange
4436 || LnState.Hdr.u8DefIsStmt > 1)
4437 return VERR_DWARF_BAD_LINE_NUMBER_HEADER;
4438 Log2(("DWARF Line number header:\n"
4439 " uVer %d\n"
4440 " offFirstOpcode %#llx\n"
4441 " cbMinInstr %u\n"
4442 " cMaxOpsPerInstr %u\n"
4443 " u8DefIsStmt %u\n"
4444 " s8LineBase %d\n"
4445 " u8LineRange %u\n"
4446 " u8OpcodeBase %u\n",
4447 LnState.Hdr.uVer, LnState.Hdr.offFirstOpcode, LnState.Hdr.cbMinInstr, LnState.Hdr.cMaxOpsPerInstr,
4448 LnState.Hdr.u8DefIsStmt, LnState.Hdr.s8LineBase, LnState.Hdr.u8LineRange, LnState.Hdr.u8OpcodeBase));
4449
4450 LnState.Hdr.pacStdOperands = pCursor->pb;
4451 for (uint8_t iStdOpcode = 1; iStdOpcode < LnState.Hdr.u8OpcodeBase; iStdOpcode++)
4452 rtDwarfCursor_GetUByte(pCursor, 0);
4453
4454 int rc = pCursor->rc;
4455 if (RT_SUCCESS(rc))
4456 rc = rtDwarfLine_ReadIncludePaths(&LnState, pCursor);
4457 if (RT_SUCCESS(rc))
4458 rc = rtDwarfLine_ReadFileNames(&LnState, pCursor);
4459
4460 /*
4461 * Run the program....
4462 */
4463 if (RT_SUCCESS(rc))
4464 rc = rtDwarfCursor_AdvanceToPos(pCursor, pbFirstOpcode);
4465 if (RT_SUCCESS(rc))
4466 rc = rtDwarfLine_RunProgram(&LnState, pCursor);
4467
4468 /*
4469 * Clean up.
4470 */
4471 size_t i = LnState.cFileNames;
4472 while (i-- > 0)
4473 RTStrFree(LnState.papszFileNames[i]);
4474 RTMemFree(LnState.papszFileNames);
4475 RTMemFree(LnState.papszIncPaths);
4476
4477 Assert(rtDwarfCursor_IsAtEndOfUnit(pCursor) || RT_FAILURE(rc));
4478 return rc;
4479}
4480
4481
4482/**
4483 * Explodes the line number table.
4484 *
4485 * The line numbers are insered into the debug info container.
4486 *
4487 * @returns IPRT status code
4488 * @param pThis The DWARF instance.
4489 */
4490static int rtDwarfLine_ExplodeAll(PRTDBGMODDWARF pThis)
4491{
4492 if (!pThis->aSections[krtDbgModDwarfSect_line].fPresent)
4493 return VINF_SUCCESS;
4494
4495 RTDWARFCURSOR Cursor;
4496 int rc = rtDwarfCursor_Init(&Cursor, pThis, krtDbgModDwarfSect_line);
4497 if (RT_FAILURE(rc))
4498 return rc;
4499
4500 while ( !rtDwarfCursor_IsAtEnd(&Cursor)
4501 && RT_SUCCESS(rc))
4502 rc = rtDwarfLine_ExplodeUnit(pThis, &Cursor);
4503
4504 return rtDwarfCursor_Delete(&Cursor, rc);
4505}
4506
4507
4508/*
4509 *
4510 * DWARF Abbreviations.
4511 * DWARF Abbreviations.
4512 * DWARF Abbreviations.
4513 *
4514 */
4515
4516/**
4517 * Deals with a cache miss in rtDwarfAbbrev_Lookup.
4518 *
4519 * @returns Pointer to abbreviation cache entry (read only). May be rendered
4520 * invalid by subsequent calls to this function.
4521 * @param pThis The DWARF instance.
4522 * @param uCode The abbreviation code to lookup.
4523 */
4524static PCRTDWARFABBREV rtDwarfAbbrev_LookupMiss(PRTDBGMODDWARF pThis, uint32_t uCode)
4525{
4526 /*
4527 * There is no entry with code zero.
4528 */
4529 if (!uCode)
4530 return NULL;
4531
4532 /*
4533 * Resize the cache array if the code is considered cachable.
4534 */
4535 bool fFillCache = true;
4536 if (pThis->cCachedAbbrevsAlloced < uCode)
4537 {
4538 if (uCode >= _64K)
4539 fFillCache = false;
4540 else
4541 {
4542 uint32_t cNew = RT_ALIGN(uCode, 64);
4543 void *pv = RTMemRealloc(pThis->paCachedAbbrevs, sizeof(pThis->paCachedAbbrevs[0]) * cNew);
4544 if (!pv)
4545 fFillCache = false;
4546 else
4547 {
4548 Log(("rtDwarfAbbrev_LookupMiss: Growing from %u to %u...\n", pThis->cCachedAbbrevsAlloced, cNew));
4549 pThis->paCachedAbbrevs = (PRTDWARFABBREV)pv;
4550 for (uint32_t i = pThis->cCachedAbbrevsAlloced; i < cNew; i++)
4551 pThis->paCachedAbbrevs[i].offAbbrev = UINT32_MAX;
4552 pThis->cCachedAbbrevsAlloced = cNew;
4553 }
4554 }
4555 }
4556
4557 /*
4558 * Walk the abbreviations till we find the desired code.
4559 */
4560 RTDWARFCURSOR Cursor;
4561 int rc = rtDwarfCursor_InitWithOffset(&Cursor, pThis, krtDbgModDwarfSect_abbrev, pThis->offCachedAbbrev);
4562 if (RT_FAILURE(rc))
4563 return NULL;
4564
4565 PRTDWARFABBREV pRet = NULL;
4566 if (fFillCache)
4567 {
4568 /*
4569 * Search for the entry and fill the cache while doing so.
4570 * We assume that abbreviation codes for a unit will stop when we see
4571 * zero code or when the code value drops.
4572 */
4573 uint32_t uPrevCode = 0;
4574 for (;;)
4575 {
4576 /* Read the 'header'. Skipping zero code bytes. */
4577#ifdef LOG_ENABLED
4578 uint32_t const offStart = rtDwarfCursor_CalcSectOffsetU32(&Cursor);
4579#endif
4580 uint32_t const uCurCode = rtDwarfCursor_GetULeb128AsU32(&Cursor, 0);
4581 if (pRet && (uCurCode == 0 || uCurCode < uPrevCode))
4582 break; /* probably end of unit. */
4583 if (uCurCode != 0)
4584 {
4585 uint32_t const uCurTag = rtDwarfCursor_GetULeb128AsU32(&Cursor, 0);
4586 uint8_t const uChildren = rtDwarfCursor_GetU8(&Cursor, 0);
4587 if (RT_FAILURE(Cursor.rc))
4588 break;
4589 if ( uCurTag > 0xffff
4590 || uChildren > 1)
4591 {
4592 Cursor.rc = VERR_DWARF_BAD_ABBREV;
4593 break;
4594 }
4595
4596 /* Cache it? */
4597 if (uCurCode <= pThis->cCachedAbbrevsAlloced)
4598 {
4599 PRTDWARFABBREV pEntry = &pThis->paCachedAbbrevs[uCurCode - 1];
4600 if (pEntry->offAbbrev != pThis->offCachedAbbrev)
4601 {
4602 pEntry->offAbbrev = pThis->offCachedAbbrev;
4603 pEntry->fChildren = RT_BOOL(uChildren);
4604 pEntry->uTag = uCurTag;
4605 pEntry->offSpec = rtDwarfCursor_CalcSectOffsetU32(&Cursor);
4606#ifdef LOG_ENABLED
4607 pEntry->cbHdr = (uint8_t)(pEntry->offSpec - offStart);
4608 Log7(("rtDwarfAbbrev_LookupMiss(%#x): fill: %#x: uTag=%#x offAbbrev=%#x%s\n",
4609 uCode, offStart, pEntry->uTag, pEntry->offAbbrev, pEntry->fChildren ? " has-children" : ""));
4610#endif
4611 if (uCurCode == uCode)
4612 {
4613 Assert(!pRet);
4614 pRet = pEntry;
4615 if (uCurCode == pThis->cCachedAbbrevsAlloced)
4616 break;
4617 }
4618 }
4619 else if (pRet)
4620 break; /* Next unit, don't cache more. */
4621 /* else: We're growing the cache and re-reading old data. */
4622 }
4623
4624 /* Skip the specification. */
4625 uint32_t uAttr, uForm;
4626 do
4627 {
4628 uAttr = rtDwarfCursor_GetULeb128AsU32(&Cursor, 0);
4629 uForm = rtDwarfCursor_GetULeb128AsU32(&Cursor, 0);
4630 } while (uAttr != 0);
4631 }
4632 if (RT_FAILURE(Cursor.rc))
4633 break;
4634
4635 /* Done? (Maximize cache filling.) */
4636 if ( pRet != NULL
4637 && uCurCode >= pThis->cCachedAbbrevsAlloced)
4638 break;
4639 uPrevCode = uCurCode;
4640 }
4641 if (pRet)
4642 Log6(("rtDwarfAbbrev_LookupMiss(%#x): uTag=%#x offSpec=%#x offAbbrev=%#x [fill]\n",
4643 uCode, pRet->uTag, pRet->offSpec, pRet->offAbbrev));
4644 else
4645 Log6(("rtDwarfAbbrev_LookupMiss(%#x): failed [fill]\n", uCode));
4646 }
4647 else
4648 {
4649 /*
4650 * Search for the entry with the desired code, no cache filling.
4651 */
4652 for (;;)
4653 {
4654 /* Read the 'header'. */
4655#ifdef LOG_ENABLED
4656 uint32_t const offStart = rtDwarfCursor_CalcSectOffsetU32(&Cursor);
4657#endif
4658 uint32_t const uCurCode = rtDwarfCursor_GetULeb128AsU32(&Cursor, 0);
4659 uint32_t const uCurTag = rtDwarfCursor_GetULeb128AsU32(&Cursor, 0);
4660 uint8_t const uChildren = rtDwarfCursor_GetU8(&Cursor, 0);
4661 if (RT_FAILURE(Cursor.rc))
4662 break;
4663 if ( uCurTag > 0xffff
4664 || uChildren > 1)
4665 {
4666 Cursor.rc = VERR_DWARF_BAD_ABBREV;
4667 break;
4668 }
4669
4670 /* Do we have a match? */
4671 if (uCurCode == uCode)
4672 {
4673 pRet = &pThis->LookupAbbrev;
4674 pRet->fChildren = RT_BOOL(uChildren);
4675 pRet->uTag = uCurTag;
4676 pRet->offSpec = rtDwarfCursor_CalcSectOffsetU32(&Cursor);
4677 pRet->offAbbrev = pThis->offCachedAbbrev;
4678#ifdef LOG_ENABLED
4679 pRet->cbHdr = (uint8_t)(pRet->offSpec - offStart);
4680#endif
4681 break;
4682 }
4683
4684 /* Skip the specification. */
4685 uint32_t uAttr, uForm;
4686 do
4687 {
4688 uAttr = rtDwarfCursor_GetULeb128AsU32(&Cursor, 0);
4689 uForm = rtDwarfCursor_GetULeb128AsU32(&Cursor, 0);
4690 } while (uAttr != 0);
4691 if (RT_FAILURE(Cursor.rc))
4692 break;
4693 }
4694 if (pRet)
4695 Log6(("rtDwarfAbbrev_LookupMiss(%#x): uTag=%#x offSpec=%#x offAbbrev=%#x [no-fill]\n",
4696 uCode, pRet->uTag, pRet->offSpec, pRet->offAbbrev));
4697 else
4698 Log6(("rtDwarfAbbrev_LookupMiss(%#x): failed [no-fill]\n", uCode));
4699 }
4700
4701 rtDwarfCursor_Delete(&Cursor, VINF_SUCCESS);
4702 return pRet;
4703}
4704
4705
4706/**
4707 * Looks up an abbreviation.
4708 *
4709 * @returns Pointer to abbreviation cache entry (read only). May be rendered
4710 * invalid by subsequent calls to this function.
4711 * @param pThis The DWARF instance.
4712 * @param uCode The abbreviation code to lookup.
4713 */
4714static PCRTDWARFABBREV rtDwarfAbbrev_Lookup(PRTDBGMODDWARF pThis, uint32_t uCode)
4715{
4716 uCode -= 1;
4717 if (uCode < pThis->cCachedAbbrevsAlloced)
4718 {
4719 if (pThis->paCachedAbbrevs[uCode].offAbbrev == pThis->offCachedAbbrev)
4720 return &pThis->paCachedAbbrevs[uCode];
4721 }
4722 return rtDwarfAbbrev_LookupMiss(pThis, uCode + 1);
4723}
4724
4725
4726/**
4727 * Sets the abbreviation offset of the current unit.
4728 *
4729 * @param pThis The DWARF instance.
4730 * @param offAbbrev The offset into the abbreviation section.
4731 */
4732static void rtDwarfAbbrev_SetUnitOffset(PRTDBGMODDWARF pThis, uint32_t offAbbrev)
4733{
4734 pThis->offCachedAbbrev = offAbbrev;
4735}
4736
4737
4738
4739/*
4740 *
4741 * DIE Attribute Parsers.
4742 * DIE Attribute Parsers.
4743 * DIE Attribute Parsers.
4744 *
4745 */
4746
4747/**
4748 * Gets the compilation unit a DIE belongs to.
4749 *
4750 * @returns The compilation unit DIE.
4751 * @param pDie Some DIE in the unit.
4752 */
4753static PRTDWARFDIECOMPILEUNIT rtDwarfDie_GetCompileUnit(PRTDWARFDIE pDie)
4754{
4755 while (pDie->pParent)
4756 pDie = pDie->pParent;
4757 AssertReturn( pDie->uTag == DW_TAG_compile_unit
4758 || pDie->uTag == DW_TAG_partial_unit,
4759 NULL);
4760 return (PRTDWARFDIECOMPILEUNIT)pDie;
4761}
4762
4763
4764/**
4765 * Resolves a string section (debug_str) reference.
4766 *
4767 * @returns Pointer to the string (inside the string section).
4768 * @param pThis The DWARF instance.
4769 * @param pCursor The cursor.
4770 * @param pszErrValue What to return on failure (@a
4771 * pCursor->rc is set).
4772 */
4773static const char *rtDwarfDecodeHlp_GetStrp(PRTDBGMODDWARF pThis, PRTDWARFCURSOR pCursor, const char *pszErrValue)
4774{
4775 uint64_t offDebugStr = rtDwarfCursor_GetUOff(pCursor, UINT64_MAX);
4776 if (RT_FAILURE(pCursor->rc))
4777 return pszErrValue;
4778
4779 if (offDebugStr >= pThis->aSections[krtDbgModDwarfSect_str].cb)
4780 {
4781 /* Ugly: Exploit the cursor status field for reporting errors. */
4782 pCursor->rc = VERR_DWARF_BAD_INFO;
4783 return pszErrValue;
4784 }
4785
4786 if (!pThis->aSections[krtDbgModDwarfSect_str].pv)
4787 {
4788 int rc = rtDbgModDwarfLoadSection(pThis, krtDbgModDwarfSect_str);
4789 if (RT_FAILURE(rc))
4790 {
4791 /* Ugly: Exploit the cursor status field for reporting errors. */
4792 pCursor->rc = rc;
4793 return pszErrValue;
4794 }
4795 }
4796
4797 return (const char *)pThis->aSections[krtDbgModDwarfSect_str].pv + (size_t)offDebugStr;
4798}
4799
4800
4801/** @callback_method_impl{FNRTDWARFATTRDECODER} */
4802static DECLCALLBACK(int) rtDwarfDecode_Address(PRTDWARFDIE pDie, uint8_t *pbMember, PCRTDWARFATTRDESC pDesc,
4803 uint32_t uForm, PRTDWARFCURSOR pCursor)
4804{
4805 AssertReturn(ATTR_GET_SIZE(pDesc) == sizeof(RTDWARFADDR), VERR_INTERNAL_ERROR_3);
4806 NOREF(pDie);
4807
4808 uint64_t uAddr;
4809 switch (uForm)
4810 {
4811 case DW_FORM_addr: uAddr = rtDwarfCursor_GetNativeUOff(pCursor, 0); break;
4812 case DW_FORM_data1: uAddr = rtDwarfCursor_GetU8(pCursor, 0); break;
4813 case DW_FORM_data2: uAddr = rtDwarfCursor_GetU16(pCursor, 0); break;
4814 case DW_FORM_data4: uAddr = rtDwarfCursor_GetU32(pCursor, 0); break;
4815 case DW_FORM_data8: uAddr = rtDwarfCursor_GetU64(pCursor, 0); break;
4816 case DW_FORM_udata: uAddr = rtDwarfCursor_GetULeb128(pCursor, 0); break;
4817 default:
4818 AssertMsgFailedReturn(("%#x (%s)\n", uForm, rtDwarfLog_FormName(uForm)), VERR_DWARF_UNEXPECTED_FORM);
4819 }
4820 if (RT_FAILURE(pCursor->rc))
4821 return pCursor->rc;
4822
4823 PRTDWARFADDR pAddr = (PRTDWARFADDR)pbMember;
4824 pAddr->uAddress = uAddr;
4825
4826 Log4((" %-20s %#010llx [%s]\n", rtDwarfLog_AttrName(pDesc->uAttr), uAddr, rtDwarfLog_FormName(uForm)));
4827 return VINF_SUCCESS;
4828}
4829
4830
4831/** @callback_method_impl{FNRTDWARFATTRDECODER} */
4832static DECLCALLBACK(int) rtDwarfDecode_Bool(PRTDWARFDIE pDie, uint8_t *pbMember, PCRTDWARFATTRDESC pDesc,
4833 uint32_t uForm, PRTDWARFCURSOR pCursor)
4834{
4835 AssertReturn(ATTR_GET_SIZE(pDesc) == sizeof(bool), VERR_INTERNAL_ERROR_3);
4836 NOREF(pDie);
4837
4838 bool *pfMember = (bool *)pbMember;
4839 switch (uForm)
4840 {
4841 case DW_FORM_flag:
4842 {
4843 uint8_t b = rtDwarfCursor_GetU8(pCursor, UINT8_MAX);
4844 if (b > 1)
4845 {
4846 Log(("Unexpected boolean value %#x\n", b));
4847 return RT_FAILURE(pCursor->rc) ? pCursor->rc : pCursor->rc = VERR_DWARF_BAD_INFO;
4848 }
4849 *pfMember = RT_BOOL(b);
4850 break;
4851 }
4852
4853 case DW_FORM_flag_present:
4854 *pfMember = true;
4855 break;
4856
4857 default:
4858 AssertMsgFailedReturn(("%#x\n", uForm), VERR_DWARF_UNEXPECTED_FORM);
4859 }
4860
4861 Log4((" %-20s %RTbool [%s]\n", rtDwarfLog_AttrName(pDesc->uAttr), *pfMember, rtDwarfLog_FormName(uForm)));
4862 return VINF_SUCCESS;
4863}
4864
4865
4866/** @callback_method_impl{FNRTDWARFATTRDECODER} */
4867static DECLCALLBACK(int) rtDwarfDecode_LowHighPc(PRTDWARFDIE pDie, uint8_t *pbMember, PCRTDWARFATTRDESC pDesc,
4868 uint32_t uForm, PRTDWARFCURSOR pCursor)
4869{
4870 AssertReturn(ATTR_GET_SIZE(pDesc) == sizeof(RTDWARFADDRRANGE), VERR_INTERNAL_ERROR_3);
4871 AssertReturn(pDesc->uAttr == DW_AT_low_pc || pDesc->uAttr == DW_AT_high_pc, VERR_INTERNAL_ERROR_3);
4872 NOREF(pDie);
4873
4874 uint64_t uAddr;
4875 switch (uForm)
4876 {
4877 case DW_FORM_addr: uAddr = rtDwarfCursor_GetNativeUOff(pCursor, 0); break;
4878 case DW_FORM_data1: uAddr = rtDwarfCursor_GetU8(pCursor, 0); break;
4879 case DW_FORM_data2: uAddr = rtDwarfCursor_GetU16(pCursor, 0); break;
4880 case DW_FORM_data4: uAddr = rtDwarfCursor_GetU32(pCursor, 0); break;
4881 case DW_FORM_data8: uAddr = rtDwarfCursor_GetU64(pCursor, 0); break;
4882 case DW_FORM_udata: uAddr = rtDwarfCursor_GetULeb128(pCursor, 0); break;
4883 default:
4884 AssertMsgFailedReturn(("%#x\n", uForm), VERR_DWARF_UNEXPECTED_FORM);
4885 }
4886 if (RT_FAILURE(pCursor->rc))
4887 return pCursor->rc;
4888
4889 PRTDWARFADDRRANGE pRange = (PRTDWARFADDRRANGE)pbMember;
4890 if (pDesc->uAttr == DW_AT_low_pc)
4891 {
4892 if (pRange->fHaveLowAddress)
4893 {
4894 Log(("rtDwarfDecode_LowHighPc: Duplicate DW_AT_low_pc\n"));
4895 return pCursor->rc = VERR_DWARF_BAD_INFO;
4896 }
4897 pRange->fHaveLowAddress = true;
4898 pRange->uLowAddress = uAddr;
4899 }
4900 else
4901 {
4902 if (pRange->fHaveHighAddress)
4903 {
4904 Log(("rtDwarfDecode_LowHighPc: Duplicate DW_AT_high_pc\n"));
4905 return pCursor->rc = VERR_DWARF_BAD_INFO;
4906 }
4907 pRange->fHaveHighAddress = true;
4908 pRange->fHaveHighIsAddress = uForm == DW_FORM_addr;
4909 if (!pRange->fHaveHighIsAddress && pRange->fHaveLowAddress)
4910 {
4911 pRange->fHaveHighIsAddress = true;
4912 pRange->uHighAddress = uAddr + pRange->uLowAddress;
4913 }
4914 else
4915 pRange->uHighAddress = uAddr;
4916
4917 }
4918 pRange->cAttrs++;
4919
4920 Log4((" %-20s %#010llx [%s]\n", rtDwarfLog_AttrName(pDesc->uAttr), uAddr, rtDwarfLog_FormName(uForm)));
4921 return VINF_SUCCESS;
4922}
4923
4924
4925/** @callback_method_impl{FNRTDWARFATTRDECODER} */
4926static DECLCALLBACK(int) rtDwarfDecode_Ranges(PRTDWARFDIE pDie, uint8_t *pbMember, PCRTDWARFATTRDESC pDesc,
4927 uint32_t uForm, PRTDWARFCURSOR pCursor)
4928{
4929 AssertReturn(ATTR_GET_SIZE(pDesc) == sizeof(RTDWARFADDRRANGE), VERR_INTERNAL_ERROR_3);
4930 AssertReturn(pDesc->uAttr == DW_AT_ranges, VERR_INTERNAL_ERROR_3);
4931 NOREF(pDie);
4932
4933 /* Decode it. */
4934 uint64_t off;
4935 switch (uForm)
4936 {
4937 case DW_FORM_addr: off = rtDwarfCursor_GetNativeUOff(pCursor, 0); break;
4938 case DW_FORM_data4: off = rtDwarfCursor_GetU32(pCursor, 0); break;
4939 case DW_FORM_data8: off = rtDwarfCursor_GetU64(pCursor, 0); break;
4940 case DW_FORM_sec_offset: off = rtDwarfCursor_GetUOff(pCursor, 0); break;
4941 default:
4942 AssertMsgFailedReturn(("%#x\n", uForm), VERR_DWARF_UNEXPECTED_FORM);
4943 }
4944 if (RT_FAILURE(pCursor->rc))
4945 return pCursor->rc;
4946
4947 /* Validate the offset and load the ranges. */
4948 PRTDBGMODDWARF pThis = pCursor->pDwarfMod;
4949 if (off >= pThis->aSections[krtDbgModDwarfSect_ranges].cb)
4950 {
4951 Log(("rtDwarfDecode_Ranges: bad ranges off=%#llx\n", off));
4952 return pCursor->rc = VERR_DWARF_BAD_POS;
4953 }
4954
4955 if (!pThis->aSections[krtDbgModDwarfSect_ranges].pv)
4956 {
4957 int rc = rtDbgModDwarfLoadSection(pThis, krtDbgModDwarfSect_ranges);
4958 if (RT_FAILURE(rc))
4959 return pCursor->rc = rc;
4960 }
4961
4962 /* Store the result. */
4963 PRTDWARFADDRRANGE pRange = (PRTDWARFADDRRANGE)pbMember;
4964 if (pRange->fHaveRanges)
4965 {
4966 Log(("rtDwarfDecode_Ranges: Duplicate DW_AT_ranges\n"));
4967 return pCursor->rc = VERR_DWARF_BAD_INFO;
4968 }
4969 pRange->fHaveRanges = true;
4970 pRange->cAttrs++;
4971 pRange->pbRanges = (uint8_t const *)pThis->aSections[krtDbgModDwarfSect_ranges].pv + (size_t)off;
4972
4973 Log4((" %-20s TODO [%s]\n", rtDwarfLog_AttrName(pDesc->uAttr), rtDwarfLog_FormName(uForm)));
4974 return VINF_SUCCESS;
4975}
4976
4977
4978/** @callback_method_impl{FNRTDWARFATTRDECODER} */
4979static DECLCALLBACK(int) rtDwarfDecode_Reference(PRTDWARFDIE pDie, uint8_t *pbMember, PCRTDWARFATTRDESC pDesc,
4980 uint32_t uForm, PRTDWARFCURSOR pCursor)
4981{
4982 AssertReturn(ATTR_GET_SIZE(pDesc) == sizeof(RTDWARFREF), VERR_INTERNAL_ERROR_3);
4983
4984 /* Decode it. */
4985 uint64_t off;
4986 krtDwarfRef enmWrt = krtDwarfRef_SameUnit;
4987 switch (uForm)
4988 {
4989 case DW_FORM_ref1: off = rtDwarfCursor_GetU8(pCursor, 0); break;
4990 case DW_FORM_ref2: off = rtDwarfCursor_GetU16(pCursor, 0); break;
4991 case DW_FORM_ref4: off = rtDwarfCursor_GetU32(pCursor, 0); break;
4992 case DW_FORM_ref8: off = rtDwarfCursor_GetU64(pCursor, 0); break;
4993 case DW_FORM_ref_udata: off = rtDwarfCursor_GetULeb128(pCursor, 0); break;
4994
4995 case DW_FORM_ref_addr:
4996 enmWrt = krtDwarfRef_InfoSection;
4997 off = rtDwarfCursor_GetUOff(pCursor, 0);
4998 break;
4999
5000 case DW_FORM_ref_sig8:
5001 enmWrt = krtDwarfRef_TypeId64;
5002 off = rtDwarfCursor_GetU64(pCursor, 0);
5003 break;
5004
5005 default:
5006 AssertMsgFailedReturn(("%#x\n", uForm), VERR_DWARF_UNEXPECTED_FORM);
5007 }
5008 if (RT_FAILURE(pCursor->rc))
5009 return pCursor->rc;
5010
5011 /* Validate the offset and convert to debug_info relative offsets. */
5012 if (enmWrt == krtDwarfRef_InfoSection)
5013 {
5014 if (off >= pCursor->pDwarfMod->aSections[krtDbgModDwarfSect_info].cb)
5015 {
5016 Log(("rtDwarfDecode_Reference: bad info off=%#llx\n", off));
5017 return pCursor->rc = VERR_DWARF_BAD_POS;
5018 }
5019 }
5020 else if (enmWrt == krtDwarfRef_SameUnit)
5021 {
5022 PRTDWARFDIECOMPILEUNIT pUnit = rtDwarfDie_GetCompileUnit(pDie);
5023 if (off >= pUnit->cbUnit)
5024 {
5025 Log(("rtDwarfDecode_Reference: bad unit off=%#llx\n", off));
5026 return pCursor->rc = VERR_DWARF_BAD_POS;
5027 }
5028 off += pUnit->offUnit;
5029 enmWrt = krtDwarfRef_InfoSection;
5030 }
5031 /* else: not bother verifying/resolving the indirect type reference yet. */
5032
5033 /* Store it */
5034 PRTDWARFREF pRef = (PRTDWARFREF)pbMember;
5035 pRef->enmWrt = enmWrt;
5036 pRef->off = off;
5037
5038 Log4((" %-20s %d:%#010llx [%s]\n", rtDwarfLog_AttrName(pDesc->uAttr), enmWrt, off, rtDwarfLog_FormName(uForm)));
5039 return VINF_SUCCESS;
5040}
5041
5042
5043/** @callback_method_impl{FNRTDWARFATTRDECODER} */
5044static DECLCALLBACK(int) rtDwarfDecode_SectOff(PRTDWARFDIE pDie, uint8_t *pbMember, PCRTDWARFATTRDESC pDesc,
5045 uint32_t uForm, PRTDWARFCURSOR pCursor)
5046{
5047 AssertReturn(ATTR_GET_SIZE(pDesc) == sizeof(RTDWARFREF), VERR_INTERNAL_ERROR_3);
5048 NOREF(pDie);
5049
5050 uint64_t off;
5051 switch (uForm)
5052 {
5053 case DW_FORM_data4: off = rtDwarfCursor_GetU32(pCursor, 0); break;
5054 case DW_FORM_data8: off = rtDwarfCursor_GetU64(pCursor, 0); break;
5055 case DW_FORM_sec_offset: off = rtDwarfCursor_GetUOff(pCursor, 0); break;
5056 default:
5057 AssertMsgFailedReturn(("%#x (%s)\n", uForm, rtDwarfLog_FormName(uForm)), VERR_DWARF_UNEXPECTED_FORM);
5058 }
5059 if (RT_FAILURE(pCursor->rc))
5060 return pCursor->rc;
5061
5062 krtDbgModDwarfSect enmSect;
5063 krtDwarfRef enmWrt;
5064 switch (pDesc->uAttr)
5065 {
5066 case DW_AT_stmt_list: enmSect = krtDbgModDwarfSect_line; enmWrt = krtDwarfRef_LineSection; break;
5067 case DW_AT_macro_info: enmSect = krtDbgModDwarfSect_loc; enmWrt = krtDwarfRef_LocSection; break;
5068 case DW_AT_ranges: enmSect = krtDbgModDwarfSect_ranges; enmWrt = krtDwarfRef_RangesSection; break;
5069 default:
5070 AssertMsgFailedReturn(("%u (%s)\n", pDesc->uAttr, rtDwarfLog_AttrName(pDesc->uAttr)), VERR_INTERNAL_ERROR_4);
5071 }
5072 size_t cbSect = pCursor->pDwarfMod->aSections[enmSect].cb;
5073 if (off >= cbSect)
5074 {
5075 /* Watcom generates offset past the end of the section, increasing the
5076 offset by one for each compile unit. So, just fudge it. */
5077 Log(("rtDwarfDecode_SectOff: bad off=%#llx, attr %#x (%s), enmSect=%d cb=%#llx; Assuming watcom/gcc.\n", off,
5078 pDesc->uAttr, rtDwarfLog_AttrName(pDesc->uAttr), enmSect, cbSect));
5079 off = cbSect;
5080 }
5081
5082 PRTDWARFREF pRef = (PRTDWARFREF)pbMember;
5083 pRef->enmWrt = enmWrt;
5084 pRef->off = off;
5085
5086 Log4((" %-20s %d:%#010llx [%s]\n", rtDwarfLog_AttrName(pDesc->uAttr), enmWrt, off, rtDwarfLog_FormName(uForm)));
5087 return VINF_SUCCESS;
5088}
5089
5090
5091/** @callback_method_impl{FNRTDWARFATTRDECODER} */
5092static DECLCALLBACK(int) rtDwarfDecode_String(PRTDWARFDIE pDie, uint8_t *pbMember, PCRTDWARFATTRDESC pDesc,
5093 uint32_t uForm, PRTDWARFCURSOR pCursor)
5094{
5095 AssertReturn(ATTR_GET_SIZE(pDesc) == sizeof(const char *), VERR_INTERNAL_ERROR_3);
5096 NOREF(pDie);
5097
5098 const char *psz;
5099 switch (uForm)
5100 {
5101 case DW_FORM_string:
5102 psz = rtDwarfCursor_GetSZ(pCursor, NULL);
5103 break;
5104
5105 case DW_FORM_strp:
5106 psz = rtDwarfDecodeHlp_GetStrp(pCursor->pDwarfMod, pCursor, NULL);
5107 break;
5108
5109 default:
5110 AssertMsgFailedReturn(("%#x\n", uForm), VERR_DWARF_UNEXPECTED_FORM);
5111 }
5112
5113 *(const char **)pbMember = psz;
5114 Log4((" %-20s '%s' [%s]\n", rtDwarfLog_AttrName(pDesc->uAttr), psz, rtDwarfLog_FormName(uForm)));
5115 return pCursor->rc;
5116}
5117
5118
5119/** @callback_method_impl{FNRTDWARFATTRDECODER} */
5120static DECLCALLBACK(int) rtDwarfDecode_UnsignedInt(PRTDWARFDIE pDie, uint8_t *pbMember, PCRTDWARFATTRDESC pDesc,
5121 uint32_t uForm, PRTDWARFCURSOR pCursor)
5122{
5123 NOREF(pDie);
5124 uint64_t u64Val;
5125 switch (uForm)
5126 {
5127 case DW_FORM_udata: u64Val = rtDwarfCursor_GetULeb128(pCursor, 0); break;
5128 case DW_FORM_data1: u64Val = rtDwarfCursor_GetU8(pCursor, 0); break;
5129 case DW_FORM_data2: u64Val = rtDwarfCursor_GetU16(pCursor, 0); break;
5130 case DW_FORM_data4: u64Val = rtDwarfCursor_GetU32(pCursor, 0); break;
5131 case DW_FORM_data8: u64Val = rtDwarfCursor_GetU64(pCursor, 0); break;
5132 default:
5133 AssertMsgFailedReturn(("%#x\n", uForm), VERR_DWARF_UNEXPECTED_FORM);
5134 }
5135 if (RT_FAILURE(pCursor->rc))
5136 return pCursor->rc;
5137
5138 switch (ATTR_GET_SIZE(pDesc))
5139 {
5140 case 1:
5141 *pbMember = (uint8_t)u64Val;
5142 if (*pbMember != u64Val)
5143 {
5144 AssertFailed();
5145 return VERR_OUT_OF_RANGE;
5146 }
5147 break;
5148
5149 case 2:
5150 *(uint16_t *)pbMember = (uint16_t)u64Val;
5151 if (*(uint16_t *)pbMember != u64Val)
5152 {
5153 AssertFailed();
5154 return VERR_OUT_OF_RANGE;
5155 }
5156 break;
5157
5158 case 4:
5159 *(uint32_t *)pbMember = (uint32_t)u64Val;
5160 if (*(uint32_t *)pbMember != u64Val)
5161 {
5162 AssertFailed();
5163 return VERR_OUT_OF_RANGE;
5164 }
5165 break;
5166
5167 case 8:
5168 *(uint64_t *)pbMember = (uint64_t)u64Val;
5169 if (*(uint64_t *)pbMember != u64Val)
5170 {
5171 AssertFailed();
5172 return VERR_OUT_OF_RANGE;
5173 }
5174 break;
5175
5176 default:
5177 AssertMsgFailedReturn(("%#x\n", ATTR_GET_SIZE(pDesc)), VERR_INTERNAL_ERROR_2);
5178 }
5179 return VINF_SUCCESS;
5180}
5181
5182
5183/**
5184 * Initialize location interpreter state from cursor & form.
5185 *
5186 * @returns IPRT status code.
5187 * @retval VERR_NOT_FOUND if no location information (i.e. there is source but
5188 * it resulted in no byte code).
5189 * @param pLoc The location state structure to initialize.
5190 * @param pCursor The cursor to read from.
5191 * @param uForm The attribute form.
5192 */
5193static int rtDwarfLoc_Init(PRTDWARFLOCST pLoc, PRTDWARFCURSOR pCursor, uint32_t uForm)
5194{
5195 uint32_t cbBlock;
5196 switch (uForm)
5197 {
5198 case DW_FORM_block1:
5199 cbBlock = rtDwarfCursor_GetU8(pCursor, 0);
5200 break;
5201
5202 case DW_FORM_block2:
5203 cbBlock = rtDwarfCursor_GetU16(pCursor, 0);
5204 break;
5205
5206 case DW_FORM_block4:
5207 cbBlock = rtDwarfCursor_GetU32(pCursor, 0);
5208 break;
5209
5210 case DW_FORM_block:
5211 cbBlock = rtDwarfCursor_GetULeb128(pCursor, 0);
5212 break;
5213
5214 default:
5215 AssertMsgFailedReturn(("uForm=%#x\n", uForm), VERR_DWARF_UNEXPECTED_FORM);
5216 }
5217 if (!cbBlock)
5218 return VERR_NOT_FOUND;
5219
5220 int rc = rtDwarfCursor_InitForBlock(&pLoc->Cursor, pCursor, cbBlock);
5221 if (RT_FAILURE(rc))
5222 return rc;
5223 pLoc->iTop = -1;
5224 return VINF_SUCCESS;
5225}
5226
5227
5228/**
5229 * Pushes a value onto the stack.
5230 *
5231 * @returns VINF_SUCCESS or VERR_DWARF_STACK_OVERFLOW.
5232 * @param pLoc The state.
5233 * @param uValue The value to push.
5234 */
5235static int rtDwarfLoc_Push(PRTDWARFLOCST pLoc, uint64_t uValue)
5236{
5237 int iTop = pLoc->iTop + 1;
5238 AssertReturn((unsigned)iTop < RT_ELEMENTS(pLoc->auStack), VERR_DWARF_STACK_OVERFLOW);
5239 pLoc->auStack[iTop] = uValue;
5240 pLoc->iTop = iTop;
5241 return VINF_SUCCESS;
5242}
5243
5244
5245static int rtDwarfLoc_Evaluate(PRTDWARFLOCST pLoc, void *pvLater, void *pvUser)
5246{
5247 RT_NOREF_PV(pvLater); RT_NOREF_PV(pvUser);
5248
5249 while (!rtDwarfCursor_IsAtEndOfUnit(&pLoc->Cursor))
5250 {
5251 /* Read the next opcode.*/
5252 uint8_t const bOpcode = rtDwarfCursor_GetU8(&pLoc->Cursor, 0);
5253
5254 /* Get its operands. */
5255 uint64_t uOperand1 = 0;
5256 uint64_t uOperand2 = 0;
5257 switch (bOpcode)
5258 {
5259 case DW_OP_addr:
5260 uOperand1 = rtDwarfCursor_GetNativeUOff(&pLoc->Cursor, 0);
5261 break;
5262 case DW_OP_pick:
5263 case DW_OP_const1u:
5264 case DW_OP_deref_size:
5265 case DW_OP_xderef_size:
5266 uOperand1 = rtDwarfCursor_GetU8(&pLoc->Cursor, 0);
5267 break;
5268 case DW_OP_const1s:
5269 uOperand1 = (int8_t)rtDwarfCursor_GetU8(&pLoc->Cursor, 0);
5270 break;
5271 case DW_OP_const2u:
5272 uOperand1 = rtDwarfCursor_GetU16(&pLoc->Cursor, 0);
5273 break;
5274 case DW_OP_skip:
5275 case DW_OP_bra:
5276 case DW_OP_const2s:
5277 uOperand1 = (int16_t)rtDwarfCursor_GetU16(&pLoc->Cursor, 0);
5278 break;
5279 case DW_OP_const4u:
5280 uOperand1 = rtDwarfCursor_GetU32(&pLoc->Cursor, 0);
5281 break;
5282 case DW_OP_const4s:
5283 uOperand1 = (int32_t)rtDwarfCursor_GetU32(&pLoc->Cursor, 0);
5284 break;
5285 case DW_OP_const8u:
5286 uOperand1 = rtDwarfCursor_GetU64(&pLoc->Cursor, 0);
5287 break;
5288 case DW_OP_const8s:
5289 uOperand1 = rtDwarfCursor_GetU64(&pLoc->Cursor, 0);
5290 break;
5291 case DW_OP_regx:
5292 case DW_OP_piece:
5293 case DW_OP_plus_uconst:
5294 case DW_OP_constu:
5295 uOperand1 = rtDwarfCursor_GetULeb128(&pLoc->Cursor, 0);
5296 break;
5297 case DW_OP_consts:
5298 case DW_OP_fbreg:
5299 case DW_OP_breg0+0: case DW_OP_breg0+1: case DW_OP_breg0+2: case DW_OP_breg0+3:
5300 case DW_OP_breg0+4: case DW_OP_breg0+5: case DW_OP_breg0+6: case DW_OP_breg0+7:
5301 case DW_OP_breg0+8: case DW_OP_breg0+9: case DW_OP_breg0+10: case DW_OP_breg0+11:
5302 case DW_OP_breg0+12: case DW_OP_breg0+13: case DW_OP_breg0+14: case DW_OP_breg0+15:
5303 case DW_OP_breg0+16: case DW_OP_breg0+17: case DW_OP_breg0+18: case DW_OP_breg0+19:
5304 case DW_OP_breg0+20: case DW_OP_breg0+21: case DW_OP_breg0+22: case DW_OP_breg0+23:
5305 case DW_OP_breg0+24: case DW_OP_breg0+25: case DW_OP_breg0+26: case DW_OP_breg0+27:
5306 case DW_OP_breg0+28: case DW_OP_breg0+29: case DW_OP_breg0+30: case DW_OP_breg0+31:
5307 uOperand1 = rtDwarfCursor_GetSLeb128(&pLoc->Cursor, 0);
5308 break;
5309 case DW_OP_bregx:
5310 uOperand1 = rtDwarfCursor_GetULeb128(&pLoc->Cursor, 0);
5311 uOperand2 = rtDwarfCursor_GetSLeb128(&pLoc->Cursor, 0);
5312 break;
5313 }
5314 if (RT_FAILURE(pLoc->Cursor.rc))
5315 break;
5316
5317 /* Interpret the opcode. */
5318 int rc;
5319 switch (bOpcode)
5320 {
5321 case DW_OP_const1u:
5322 case DW_OP_const1s:
5323 case DW_OP_const2u:
5324 case DW_OP_const2s:
5325 case DW_OP_const4u:
5326 case DW_OP_const4s:
5327 case DW_OP_const8u:
5328 case DW_OP_const8s:
5329 case DW_OP_constu:
5330 case DW_OP_consts:
5331 case DW_OP_addr:
5332 rc = rtDwarfLoc_Push(pLoc, uOperand1);
5333 break;
5334 case DW_OP_lit0 + 0: case DW_OP_lit0 + 1: case DW_OP_lit0 + 2: case DW_OP_lit0 + 3:
5335 case DW_OP_lit0 + 4: case DW_OP_lit0 + 5: case DW_OP_lit0 + 6: case DW_OP_lit0 + 7:
5336 case DW_OP_lit0 + 8: case DW_OP_lit0 + 9: case DW_OP_lit0 + 10: case DW_OP_lit0 + 11:
5337 case DW_OP_lit0 + 12: case DW_OP_lit0 + 13: case DW_OP_lit0 + 14: case DW_OP_lit0 + 15:
5338 case DW_OP_lit0 + 16: case DW_OP_lit0 + 17: case DW_OP_lit0 + 18: case DW_OP_lit0 + 19:
5339 case DW_OP_lit0 + 20: case DW_OP_lit0 + 21: case DW_OP_lit0 + 22: case DW_OP_lit0 + 23:
5340 case DW_OP_lit0 + 24: case DW_OP_lit0 + 25: case DW_OP_lit0 + 26: case DW_OP_lit0 + 27:
5341 case DW_OP_lit0 + 28: case DW_OP_lit0 + 29: case DW_OP_lit0 + 30: case DW_OP_lit0 + 31:
5342 rc = rtDwarfLoc_Push(pLoc, bOpcode - DW_OP_lit0);
5343 break;
5344 case DW_OP_nop:
5345 break;
5346 case DW_OP_dup: /** @todo 0 operands. */
5347 case DW_OP_drop: /** @todo 0 operands. */
5348 case DW_OP_over: /** @todo 0 operands. */
5349 case DW_OP_pick: /** @todo 1 operands, a 1-byte stack index. */
5350 case DW_OP_swap: /** @todo 0 operands. */
5351 case DW_OP_rot: /** @todo 0 operands. */
5352 case DW_OP_abs: /** @todo 0 operands. */
5353 case DW_OP_and: /** @todo 0 operands. */
5354 case DW_OP_div: /** @todo 0 operands. */
5355 case DW_OP_minus: /** @todo 0 operands. */
5356 case DW_OP_mod: /** @todo 0 operands. */
5357 case DW_OP_mul: /** @todo 0 operands. */
5358 case DW_OP_neg: /** @todo 0 operands. */
5359 case DW_OP_not: /** @todo 0 operands. */
5360 case DW_OP_or: /** @todo 0 operands. */
5361 case DW_OP_plus: /** @todo 0 operands. */
5362 case DW_OP_plus_uconst: /** @todo 1 operands, a ULEB128 addend. */
5363 case DW_OP_shl: /** @todo 0 operands. */
5364 case DW_OP_shr: /** @todo 0 operands. */
5365 case DW_OP_shra: /** @todo 0 operands. */
5366 case DW_OP_xor: /** @todo 0 operands. */
5367 case DW_OP_skip: /** @todo 1 signed 2-byte constant. */
5368 case DW_OP_bra: /** @todo 1 signed 2-byte constant. */
5369 case DW_OP_eq: /** @todo 0 operands. */
5370 case DW_OP_ge: /** @todo 0 operands. */
5371 case DW_OP_gt: /** @todo 0 operands. */
5372 case DW_OP_le: /** @todo 0 operands. */
5373 case DW_OP_lt: /** @todo 0 operands. */
5374 case DW_OP_ne: /** @todo 0 operands. */
5375 case DW_OP_reg0 + 0: case DW_OP_reg0 + 1: case DW_OP_reg0 + 2: case DW_OP_reg0 + 3: /** @todo 0 operands - reg 0..31. */
5376 case DW_OP_reg0 + 4: case DW_OP_reg0 + 5: case DW_OP_reg0 + 6: case DW_OP_reg0 + 7:
5377 case DW_OP_reg0 + 8: case DW_OP_reg0 + 9: case DW_OP_reg0 + 10: case DW_OP_reg0 + 11:
5378 case DW_OP_reg0 + 12: case DW_OP_reg0 + 13: case DW_OP_reg0 + 14: case DW_OP_reg0 + 15:
5379 case DW_OP_reg0 + 16: case DW_OP_reg0 + 17: case DW_OP_reg0 + 18: case DW_OP_reg0 + 19:
5380 case DW_OP_reg0 + 20: case DW_OP_reg0 + 21: case DW_OP_reg0 + 22: case DW_OP_reg0 + 23:
5381 case DW_OP_reg0 + 24: case DW_OP_reg0 + 25: case DW_OP_reg0 + 26: case DW_OP_reg0 + 27:
5382 case DW_OP_reg0 + 28: case DW_OP_reg0 + 29: case DW_OP_reg0 + 30: case DW_OP_reg0 + 31:
5383 case DW_OP_breg0+ 0: case DW_OP_breg0+ 1: case DW_OP_breg0+ 2: case DW_OP_breg0+ 3: /** @todo 1 operand, a SLEB128 offset. */
5384 case DW_OP_breg0+ 4: case DW_OP_breg0+ 5: case DW_OP_breg0+ 6: case DW_OP_breg0+ 7:
5385 case DW_OP_breg0+ 8: case DW_OP_breg0+ 9: case DW_OP_breg0+ 10: case DW_OP_breg0+ 11:
5386 case DW_OP_breg0+ 12: case DW_OP_breg0+ 13: case DW_OP_breg0+ 14: case DW_OP_breg0+ 15:
5387 case DW_OP_breg0+ 16: case DW_OP_breg0+ 17: case DW_OP_breg0+ 18: case DW_OP_breg0+ 19:
5388 case DW_OP_breg0+ 20: case DW_OP_breg0+ 21: case DW_OP_breg0+ 22: case DW_OP_breg0+ 23:
5389 case DW_OP_breg0+ 24: case DW_OP_breg0+ 25: case DW_OP_breg0+ 26: case DW_OP_breg0+ 27:
5390 case DW_OP_breg0+ 28: case DW_OP_breg0+ 29: case DW_OP_breg0+ 30: case DW_OP_breg0+ 31:
5391 case DW_OP_piece: /** @todo 1 operand, a ULEB128 size of piece addressed. */
5392 case DW_OP_regx: /** @todo 1 operand, a ULEB128 register. */
5393 case DW_OP_fbreg: /** @todo 1 operand, a SLEB128 offset. */
5394 case DW_OP_bregx: /** @todo 2 operands, a ULEB128 register followed by a SLEB128 offset. */
5395 case DW_OP_deref: /** @todo 0 operands. */
5396 case DW_OP_deref_size: /** @todo 1 operand, a 1-byte size of data retrieved. */
5397 case DW_OP_xderef: /** @todo 0 operands. */
5398 case DW_OP_xderef_size: /** @todo 1 operand, a 1-byte size of data retrieved. */
5399 AssertMsgFailedReturn(("bOpcode=%#x\n", bOpcode), VERR_DWARF_TODO);
5400 default:
5401 AssertMsgFailedReturn(("bOpcode=%#x\n", bOpcode), VERR_DWARF_UNKNOWN_LOC_OPCODE);
5402 }
5403 }
5404
5405 return pLoc->Cursor.rc;
5406}
5407
5408
5409/** @callback_method_impl{FNRTDWARFATTRDECODER} */
5410static DECLCALLBACK(int) rtDwarfDecode_SegmentLoc(PRTDWARFDIE pDie, uint8_t *pbMember, PCRTDWARFATTRDESC pDesc,
5411 uint32_t uForm, PRTDWARFCURSOR pCursor)
5412{
5413 NOREF(pDie);
5414 AssertReturn(ATTR_GET_SIZE(pDesc) == 2, VERR_DWARF_IPE);
5415
5416 RTDWARFLOCST LocSt;
5417 int rc = rtDwarfLoc_Init(&LocSt, pCursor, uForm);
5418 if (RT_SUCCESS(rc))
5419 {
5420 rc = rtDwarfLoc_Evaluate(&LocSt, NULL, NULL);
5421 if (RT_SUCCESS(rc))
5422 {
5423 if (LocSt.iTop >= 0)
5424 {
5425 *(uint16_t *)pbMember = LocSt.auStack[LocSt.iTop];
5426 Log4((" %-20s %#06llx [%s]\n", rtDwarfLog_AttrName(pDesc->uAttr),
5427 LocSt.auStack[LocSt.iTop], rtDwarfLog_FormName(uForm)));
5428 return VINF_SUCCESS;
5429 }
5430 rc = VERR_DWARF_STACK_UNDERFLOW;
5431 }
5432 }
5433 return rc;
5434}
5435
5436/*
5437 *
5438 * DWARF debug_info parser
5439 * DWARF debug_info parser
5440 * DWARF debug_info parser
5441 *
5442 */
5443
5444
5445/**
5446 * Special hack to get the name and/or linkage name for a subprogram via a
5447 * specification reference.
5448 *
5449 * Since this is a hack, we ignore failure.
5450 *
5451 * If we want to really make use of DWARF info, we'll have to create some kind
5452 * of lookup tree for handling this. But currently we don't, so a hack will
5453 * suffice.
5454 *
5455 * @param pThis The DWARF instance.
5456 * @param pSubProgram The subprogram which is short on names.
5457 */
5458static void rtDwarfInfo_TryGetSubProgramNameFromSpecRef(PRTDBGMODDWARF pThis, PRTDWARFDIESUBPROGRAM pSubProgram)
5459{
5460 /*
5461 * Must have a spec ref, and it must be in the info section.
5462 */
5463 if (pSubProgram->SpecRef.enmWrt != krtDwarfRef_InfoSection)
5464 return;
5465
5466 /*
5467 * Create a cursor for reading the info and then the abbrivation code
5468 * starting the off the DIE.
5469 */
5470 RTDWARFCURSOR InfoCursor;
5471 int rc = rtDwarfCursor_InitWithOffset(&InfoCursor, pThis, krtDbgModDwarfSect_info, pSubProgram->SpecRef.off);
5472 if (RT_FAILURE(rc))
5473 return;
5474
5475 uint32_t uAbbrCode = rtDwarfCursor_GetULeb128AsU32(&InfoCursor, UINT32_MAX);
5476 if (uAbbrCode)
5477 {
5478 /* Only references to subprogram tags are interesting here. */
5479 PCRTDWARFABBREV pAbbrev = rtDwarfAbbrev_Lookup(pThis, uAbbrCode);
5480 if ( pAbbrev
5481 && pAbbrev->uTag == DW_TAG_subprogram)
5482 {
5483 /*
5484 * Use rtDwarfInfo_ParseDie to do the parsing, but with a different
5485 * attribute spec than usual.
5486 */
5487 rtDwarfInfo_ParseDie(pThis, &pSubProgram->Core, &g_SubProgramSpecHackDesc, &InfoCursor,
5488 pAbbrev, false /*fInitDie*/);
5489 }
5490 }
5491
5492 rtDwarfCursor_Delete(&InfoCursor, VINF_SUCCESS);
5493}
5494
5495
5496/**
5497 * Select which name to use.
5498 *
5499 * @returns One of the names.
5500 * @param pszName The DWARF name, may exclude namespace and class.
5501 * Can also be NULL.
5502 * @param pszLinkageName The linkage name. Can be NULL.
5503 */
5504static const char *rtDwarfInfo_SelectName(const char *pszName, const char *pszLinkageName)
5505{
5506 if (!pszName || !pszLinkageName)
5507 return pszName ? pszName : pszLinkageName;
5508
5509 /*
5510 * Some heuristics for selecting the link name if the normal name is missing
5511 * namespace or class prefixes.
5512 */
5513 size_t cchName = strlen(pszName);
5514 size_t cchLinkageName = strlen(pszLinkageName);
5515 if (cchLinkageName <= cchName + 1)
5516 return pszName;
5517
5518 const char *psz = strstr(pszLinkageName, pszName);
5519 if (!psz || psz - pszLinkageName < 4)
5520 return pszName;
5521
5522 return pszLinkageName;
5523}
5524
5525
5526/**
5527 * Parse the attributes of a DIE.
5528 *
5529 * @returns IPRT status code.
5530 * @param pThis The DWARF instance.
5531 * @param pDie The internal DIE structure to fill.
5532 */
5533static int rtDwarfInfo_SnoopSymbols(PRTDBGMODDWARF pThis, PRTDWARFDIE pDie)
5534{
5535 int rc = VINF_SUCCESS;
5536 switch (pDie->uTag)
5537 {
5538 case DW_TAG_subprogram:
5539 {
5540 PRTDWARFDIESUBPROGRAM pSubProgram = (PRTDWARFDIESUBPROGRAM)pDie;
5541
5542 /* Obtain referenced specification there is only partial info. */
5543 if ( pSubProgram->PcRange.cAttrs
5544 && !pSubProgram->pszName)
5545 rtDwarfInfo_TryGetSubProgramNameFromSpecRef(pThis, pSubProgram);
5546
5547 if (pSubProgram->PcRange.cAttrs)
5548 {
5549 if (pSubProgram->PcRange.fHaveRanges)
5550 Log5(("subprogram %s (%s) <implement ranges>\n", pSubProgram->pszName, pSubProgram->pszLinkageName));
5551 else
5552 {
5553 Log5(("subprogram %s (%s) %#llx-%#llx%s\n", pSubProgram->pszName, pSubProgram->pszLinkageName,
5554 pSubProgram->PcRange.uLowAddress, pSubProgram->PcRange.uHighAddress,
5555 pSubProgram->PcRange.cAttrs == 2 ? "" : " !bad!"));
5556 if ( ( pSubProgram->pszName || pSubProgram->pszLinkageName)
5557 && pSubProgram->PcRange.cAttrs == 2)
5558 {
5559 if (pThis->iWatcomPass == 1)
5560 rc = rtDbgModDwarfRecordSegOffset(pThis, pSubProgram->uSegment, pSubProgram->PcRange.uHighAddress);
5561 else
5562 {
5563 RTDBGSEGIDX iSeg;
5564 RTUINTPTR offSeg;
5565 rc = rtDbgModDwarfLinkAddressToSegOffset(pThis, pSubProgram->uSegment,
5566 pSubProgram->PcRange.uLowAddress,
5567 &iSeg, &offSeg);
5568 if (RT_SUCCESS(rc))
5569 {
5570 uint64_t cb;
5571 if (pSubProgram->PcRange.uHighAddress >= pSubProgram->PcRange.uLowAddress)
5572 cb = pSubProgram->PcRange.uHighAddress - pSubProgram->PcRange.uLowAddress;
5573 else
5574 cb = 1;
5575 rc = RTDbgModSymbolAdd(pThis->hCnt,
5576 rtDwarfInfo_SelectName(pSubProgram->pszName, pSubProgram->pszLinkageName),
5577 iSeg, offSeg, cb, 0 /*fFlags*/, NULL /*piOrdinal*/);
5578 if (RT_FAILURE(rc))
5579 {
5580 if ( rc == VERR_DBG_DUPLICATE_SYMBOL
5581 || rc == VERR_DBG_ADDRESS_CONFLICT /** @todo figure why this happens with 10.6.8 mach_kernel, 32-bit. */
5582 )
5583 rc = VINF_SUCCESS;
5584 else
5585 AssertMsgFailed(("%Rrc\n", rc));
5586 }
5587 }
5588 else if ( pSubProgram->PcRange.uLowAddress == 0 /* see with vmlinux */
5589 && pSubProgram->PcRange.uHighAddress == 0)
5590 {
5591 Log5(("rtDbgModDwarfLinkAddressToSegOffset: Ignoring empty range.\n"));
5592 rc = VINF_SUCCESS; /* ignore */
5593 }
5594 else
5595 {
5596 AssertRC(rc);
5597 Log5(("rtDbgModDwarfLinkAddressToSegOffset failed: %Rrc\n", rc));
5598 }
5599 }
5600 }
5601 }
5602 }
5603 else
5604 Log5(("subprogram %s (%s) external\n", pSubProgram->pszName, pSubProgram->pszLinkageName));
5605 break;
5606 }
5607
5608 case DW_TAG_label:
5609 {
5610 PCRTDWARFDIELABEL pLabel = (PCRTDWARFDIELABEL)pDie;
5611 if (pLabel->fExternal)
5612 {
5613 Log5(("label %s %#x:%#llx\n", pLabel->pszName, pLabel->uSegment, pLabel->Address.uAddress));
5614 if (pThis->iWatcomPass == 1)
5615 rc = rtDbgModDwarfRecordSegOffset(pThis, pLabel->uSegment, pLabel->Address.uAddress);
5616 else
5617 {
5618 RTDBGSEGIDX iSeg;
5619 RTUINTPTR offSeg;
5620 rc = rtDbgModDwarfLinkAddressToSegOffset(pThis, pLabel->uSegment, pLabel->Address.uAddress,
5621 &iSeg, &offSeg);
5622 AssertRC(rc);
5623 if (RT_SUCCESS(rc))
5624 {
5625 rc = RTDbgModSymbolAdd(pThis->hCnt, pLabel->pszName, iSeg, offSeg, 0 /*cb*/,
5626 0 /*fFlags*/, NULL /*piOrdinal*/);
5627 AssertMsg(RT_SUCCESS(rc) || rc == VERR_DBG_ADDRESS_CONFLICT,
5628 ("%Rrc %s %x:%x\n", rc, pLabel->pszName, iSeg, offSeg));
5629 }
5630 else
5631 Log5(("rtDbgModDwarfLinkAddressToSegOffset failed: %Rrc\n", rc));
5632 }
5633
5634 }
5635 break;
5636 }
5637
5638 }
5639 return rc;
5640}
5641
5642
5643/**
5644 * Initializes the non-core fields of an internal DIE structure.
5645 *
5646 * @param pDie The DIE structure.
5647 * @param pDieDesc The DIE descriptor.
5648 */
5649static void rtDwarfInfo_InitDie(PRTDWARFDIE pDie, PCRTDWARFDIEDESC pDieDesc)
5650{
5651 size_t i = pDieDesc->cAttributes;
5652 while (i-- > 0)
5653 {
5654 switch (pDieDesc->paAttributes[i].cbInit & ATTR_INIT_MASK)
5655 {
5656 case ATTR_INIT_ZERO:
5657 /* Nothing to do (RTMemAllocZ). */
5658 break;
5659
5660 case ATTR_INIT_FFFS:
5661 switch (pDieDesc->paAttributes[i].cbInit & ATTR_SIZE_MASK)
5662 {
5663 case 1:
5664 *(uint8_t *)((uintptr_t)pDie + pDieDesc->paAttributes[i].off) = UINT8_MAX;
5665 break;
5666 case 2:
5667 *(uint16_t *)((uintptr_t)pDie + pDieDesc->paAttributes[i].off) = UINT16_MAX;
5668 break;
5669 case 4:
5670 *(uint32_t *)((uintptr_t)pDie + pDieDesc->paAttributes[i].off) = UINT32_MAX;
5671 break;
5672 case 8:
5673 *(uint64_t *)((uintptr_t)pDie + pDieDesc->paAttributes[i].off) = UINT64_MAX;
5674 break;
5675 default:
5676 AssertFailed();
5677 memset((uint8_t *)pDie + pDieDesc->paAttributes[i].off, 0xff,
5678 pDieDesc->paAttributes[i].cbInit & ATTR_SIZE_MASK);
5679 break;
5680 }
5681 break;
5682
5683 default:
5684 AssertFailed();
5685 }
5686 }
5687}
5688
5689
5690/**
5691 * Creates a new internal DIE structure and links it up.
5692 *
5693 * @returns Pointer to the new DIE structure.
5694 * @param pThis The DWARF instance.
5695 * @param pDieDesc The DIE descriptor (for size and init).
5696 * @param pAbbrev The abbreviation cache entry.
5697 * @param pParent The parent DIE (NULL if unit).
5698 */
5699static PRTDWARFDIE rtDwarfInfo_NewDie(PRTDBGMODDWARF pThis, PCRTDWARFDIEDESC pDieDesc,
5700 PCRTDWARFABBREV pAbbrev, PRTDWARFDIE pParent)
5701{
5702 NOREF(pThis);
5703 Assert(pDieDesc->cbDie >= sizeof(RTDWARFDIE));
5704#ifdef RTDBGMODDWARF_WITH_MEM_CACHE
5705 uint32_t iAllocator = pDieDesc->cbDie > pThis->aDieAllocators[0].cbMax;
5706 Assert(pDieDesc->cbDie <= pThis->aDieAllocators[iAllocator].cbMax);
5707 PRTDWARFDIE pDie = (PRTDWARFDIE)RTMemCacheAlloc(pThis->aDieAllocators[iAllocator].hMemCache);
5708#else
5709 PRTDWARFDIE pDie = (PRTDWARFDIE)RTMemAllocZ(pDieDesc->cbDie);
5710#endif
5711 if (pDie)
5712 {
5713#ifdef RTDBGMODDWARF_WITH_MEM_CACHE
5714 RT_BZERO(pDie, pDieDesc->cbDie);
5715 pDie->iAllocator = iAllocator;
5716#endif
5717 rtDwarfInfo_InitDie(pDie, pDieDesc);
5718
5719 pDie->uTag = pAbbrev->uTag;
5720 pDie->offSpec = pAbbrev->offSpec;
5721 pDie->pParent = pParent;
5722 if (pParent)
5723 RTListAppend(&pParent->ChildList, &pDie->SiblingNode);
5724 else
5725 RTListInit(&pDie->SiblingNode);
5726 RTListInit(&pDie->ChildList);
5727
5728 }
5729 return pDie;
5730}
5731
5732
5733/**
5734 * Free all children of a DIE.
5735 *
5736 * @param pThis The DWARF instance.
5737 * @param pParentDie The parent DIE.
5738 */
5739static void rtDwarfInfo_FreeChildren(PRTDBGMODDWARF pThis, PRTDWARFDIE pParentDie)
5740{
5741 PRTDWARFDIE pChild, pNextChild;
5742 RTListForEachSafe(&pParentDie->ChildList, pChild, pNextChild, RTDWARFDIE, SiblingNode)
5743 {
5744 if (!RTListIsEmpty(&pChild->ChildList))
5745 rtDwarfInfo_FreeChildren(pThis, pChild);
5746 RTListNodeRemove(&pChild->SiblingNode);
5747#ifdef RTDBGMODDWARF_WITH_MEM_CACHE
5748 RTMemCacheFree(pThis->aDieAllocators[pChild->iAllocator].hMemCache, pChild);
5749#else
5750 RTMemFree(pChild);
5751#endif
5752 }
5753}
5754
5755
5756/**
5757 * Free a DIE an all its children.
5758 *
5759 * @param pThis The DWARF instance.
5760 * @param pDie The DIE to free.
5761 */
5762static void rtDwarfInfo_FreeDie(PRTDBGMODDWARF pThis, PRTDWARFDIE pDie)
5763{
5764 rtDwarfInfo_FreeChildren(pThis, pDie);
5765 RTListNodeRemove(&pDie->SiblingNode);
5766#ifdef RTDBGMODDWARF_WITH_MEM_CACHE
5767 RTMemCacheFree(pThis->aDieAllocators[pDie->iAllocator].hMemCache, pDie);
5768#else
5769 RTMemFree(pChild);
5770#endif
5771}
5772
5773
5774/**
5775 * Skips a form.
5776 * @returns IPRT status code
5777 * @param pCursor The cursor.
5778 * @param uForm The form to skip.
5779 */
5780static int rtDwarfInfo_SkipForm(PRTDWARFCURSOR pCursor, uint32_t uForm)
5781{
5782 switch (uForm)
5783 {
5784 case DW_FORM_addr:
5785 return rtDwarfCursor_SkipBytes(pCursor, pCursor->cbNativeAddr);
5786
5787 case DW_FORM_block:
5788 case DW_FORM_exprloc:
5789 return rtDwarfCursor_SkipBytes(pCursor, rtDwarfCursor_GetULeb128(pCursor, 0));
5790
5791 case DW_FORM_block1:
5792 return rtDwarfCursor_SkipBytes(pCursor, rtDwarfCursor_GetU8(pCursor, 0));
5793
5794 case DW_FORM_block2:
5795 return rtDwarfCursor_SkipBytes(pCursor, rtDwarfCursor_GetU16(pCursor, 0));
5796
5797 case DW_FORM_block4:
5798 return rtDwarfCursor_SkipBytes(pCursor, rtDwarfCursor_GetU32(pCursor, 0));
5799
5800 case DW_FORM_data1:
5801 case DW_FORM_ref1:
5802 case DW_FORM_flag:
5803 return rtDwarfCursor_SkipBytes(pCursor, 1);
5804
5805 case DW_FORM_data2:
5806 case DW_FORM_ref2:
5807 return rtDwarfCursor_SkipBytes(pCursor, 2);
5808
5809 case DW_FORM_data4:
5810 case DW_FORM_ref4:
5811 return rtDwarfCursor_SkipBytes(pCursor, 4);
5812
5813 case DW_FORM_data8:
5814 case DW_FORM_ref8:
5815 case DW_FORM_ref_sig8:
5816 return rtDwarfCursor_SkipBytes(pCursor, 8);
5817
5818 case DW_FORM_udata:
5819 case DW_FORM_sdata:
5820 case DW_FORM_ref_udata:
5821 return rtDwarfCursor_SkipLeb128(pCursor);
5822
5823 case DW_FORM_string:
5824 rtDwarfCursor_GetSZ(pCursor, NULL);
5825 return pCursor->rc;
5826
5827 case DW_FORM_indirect:
5828 return rtDwarfInfo_SkipForm(pCursor, rtDwarfCursor_GetULeb128AsU32(pCursor, UINT32_MAX));
5829
5830 case DW_FORM_strp:
5831 case DW_FORM_ref_addr:
5832 case DW_FORM_sec_offset:
5833 return rtDwarfCursor_SkipBytes(pCursor, pCursor->f64bitDwarf ? 8 : 4);
5834
5835 case DW_FORM_flag_present:
5836 return pCursor->rc; /* no data */
5837
5838 default:
5839 Log(("rtDwarfInfo_SkipForm: Unknown form %#x\n", uForm));
5840 return VERR_DWARF_UNKNOWN_FORM;
5841 }
5842}
5843
5844
5845
5846#ifdef SOME_UNUSED_FUNCTION
5847/**
5848 * Skips a DIE.
5849 *
5850 * @returns IPRT status code.
5851 * @param pCursor The cursor.
5852 * @param pAbbrevCursor The abbreviation cursor.
5853 */
5854static int rtDwarfInfo_SkipDie(PRTDWARFCURSOR pCursor, PRTDWARFCURSOR pAbbrevCursor)
5855{
5856 for (;;)
5857 {
5858 uint32_t uAttr = rtDwarfCursor_GetULeb128AsU32(pAbbrevCursor, 0);
5859 uint32_t uForm = rtDwarfCursor_GetULeb128AsU32(pAbbrevCursor, 0);
5860 if (uAttr == 0 && uForm == 0)
5861 break;
5862
5863 int rc = rtDwarfInfo_SkipForm(pCursor, uForm);
5864 if (RT_FAILURE(rc))
5865 return rc;
5866 }
5867 return RT_FAILURE(pCursor->rc) ? pCursor->rc : pAbbrevCursor->rc;
5868}
5869#endif
5870
5871
5872/**
5873 * Parse the attributes of a DIE.
5874 *
5875 * @returns IPRT status code.
5876 * @param pThis The DWARF instance.
5877 * @param pDie The internal DIE structure to fill.
5878 * @param pDieDesc The DIE descriptor.
5879 * @param pCursor The debug_info cursor.
5880 * @param pAbbrev The abbreviation cache entry.
5881 * @param fInitDie Whether to initialize the DIE first. If not (@c
5882 * false) it's safe to assume we're following a
5883 * DW_AT_specification or DW_AT_abstract_origin,
5884 * and that we shouldn't be snooping any symbols.
5885 */
5886static int rtDwarfInfo_ParseDie(PRTDBGMODDWARF pThis, PRTDWARFDIE pDie, PCRTDWARFDIEDESC pDieDesc,
5887 PRTDWARFCURSOR pCursor, PCRTDWARFABBREV pAbbrev, bool fInitDie)
5888{
5889 RTDWARFCURSOR AbbrevCursor;
5890 int rc = rtDwarfCursor_InitWithOffset(&AbbrevCursor, pThis, krtDbgModDwarfSect_abbrev, pAbbrev->offSpec);
5891 if (RT_FAILURE(rc))
5892 return rc;
5893
5894 if (fInitDie)
5895 rtDwarfInfo_InitDie(pDie, pDieDesc);
5896 for (;;)
5897 {
5898#ifdef LOG_ENABLED
5899 uint32_t const off = (uint32_t)(AbbrevCursor.pb - AbbrevCursor.pbStart);
5900#endif
5901 uint32_t uAttr = rtDwarfCursor_GetULeb128AsU32(&AbbrevCursor, 0);
5902 uint32_t uForm = rtDwarfCursor_GetULeb128AsU32(&AbbrevCursor, 0);
5903 Log4((" %04x: %-23s [%s]\n", off, rtDwarfLog_AttrName(uAttr), rtDwarfLog_FormName(uForm)));
5904 if (uAttr == 0)
5905 break;
5906 if (uForm == DW_FORM_indirect)
5907 uForm = rtDwarfCursor_GetULeb128AsU32(pCursor, 0);
5908
5909 /* Look up the attribute in the descriptor and invoke the decoder. */
5910 PCRTDWARFATTRDESC pAttr = NULL;
5911 size_t i = pDieDesc->cAttributes;
5912 while (i-- > 0)
5913 if (pDieDesc->paAttributes[i].uAttr == uAttr)
5914 {
5915 pAttr = &pDieDesc->paAttributes[i];
5916 rc = pAttr->pfnDecoder(pDie, (uint8_t *)pDie + pAttr->off, pAttr, uForm, pCursor);
5917 break;
5918 }
5919
5920 /* Some house keeping. */
5921 if (pAttr)
5922 pDie->cDecodedAttrs++;
5923 else
5924 {
5925 pDie->cUnhandledAttrs++;
5926 rc = rtDwarfInfo_SkipForm(pCursor, uForm);
5927 }
5928 if (RT_FAILURE(rc))
5929 break;
5930 }
5931
5932 rc = rtDwarfCursor_Delete(&AbbrevCursor, rc);
5933 if (RT_SUCCESS(rc))
5934 rc = pCursor->rc;
5935
5936 /*
5937 * Snoop up symbols on the way out.
5938 */
5939 if (RT_SUCCESS(rc) && fInitDie)
5940 {
5941 rc = rtDwarfInfo_SnoopSymbols(pThis, pDie);
5942 /* Ignore duplicates, get work done instead. */
5943 /** @todo clean up global/static symbol mess. */
5944 if (rc == VERR_DBG_DUPLICATE_SYMBOL || rc == VERR_DBG_ADDRESS_CONFLICT)
5945 rc = VINF_SUCCESS;
5946 }
5947
5948 return rc;
5949}
5950
5951
5952/**
5953 * Load the debug information of a unit.
5954 *
5955 * @returns IPRT status code.
5956 * @param pThis The DWARF instance.
5957 * @param pCursor The debug_info cursor.
5958 * @param fKeepDies Whether to keep the DIEs or discard them as soon
5959 * as possible.
5960 */
5961static int rtDwarfInfo_LoadUnit(PRTDBGMODDWARF pThis, PRTDWARFCURSOR pCursor, bool fKeepDies)
5962{
5963 Log(("rtDwarfInfo_LoadUnit: %#x\n", rtDwarfCursor_CalcSectOffsetU32(pCursor)));
5964
5965 /*
5966 * Read the compilation unit header.
5967 */
5968 uint64_t offUnit = rtDwarfCursor_CalcSectOffsetU32(pCursor);
5969 uint64_t cbUnit = rtDwarfCursor_GetInitialLength(pCursor);
5970 cbUnit += rtDwarfCursor_CalcSectOffsetU32(pCursor) - offUnit;
5971 uint16_t const uVer = rtDwarfCursor_GetUHalf(pCursor, 0);
5972 if ( uVer < 2
5973 || uVer > 4)
5974 return rtDwarfCursor_SkipUnit(pCursor);
5975 uint64_t const offAbbrev = rtDwarfCursor_GetUOff(pCursor, UINT64_MAX);
5976 uint8_t const cbNativeAddr = rtDwarfCursor_GetU8(pCursor, UINT8_MAX);
5977 if (RT_FAILURE(pCursor->rc))
5978 return pCursor->rc;
5979 Log((" uVer=%d offAbbrev=%#llx cbNativeAddr=%d\n", uVer, offAbbrev, cbNativeAddr));
5980
5981 /*
5982 * Set up the abbreviation cache and store the native address size in the cursor.
5983 */
5984 if (offAbbrev > UINT32_MAX)
5985 {
5986 Log(("Unexpected abbrviation code offset of %#llx\n", offAbbrev));
5987 return VERR_DWARF_BAD_INFO;
5988 }
5989 rtDwarfAbbrev_SetUnitOffset(pThis, (uint32_t)offAbbrev);
5990 pCursor->cbNativeAddr = cbNativeAddr;
5991
5992 /*
5993 * The first DIE is a compile or partial unit, parse it here.
5994 */
5995 uint32_t uAbbrCode = rtDwarfCursor_GetULeb128AsU32(pCursor, UINT32_MAX);
5996 if (!uAbbrCode)
5997 {
5998 Log(("Unexpected abbrviation code of zero\n"));
5999 return VERR_DWARF_BAD_INFO;
6000 }
6001 PCRTDWARFABBREV pAbbrev = rtDwarfAbbrev_Lookup(pThis, uAbbrCode);
6002 if (!pAbbrev)
6003 return VERR_DWARF_ABBREV_NOT_FOUND;
6004 if ( pAbbrev->uTag != DW_TAG_compile_unit
6005 && pAbbrev->uTag != DW_TAG_partial_unit)
6006 {
6007 Log(("Unexpected compile/partial unit tag %#x\n", pAbbrev->uTag));
6008 return VERR_DWARF_BAD_INFO;
6009 }
6010
6011 PRTDWARFDIECOMPILEUNIT pUnit;
6012 pUnit = (PRTDWARFDIECOMPILEUNIT)rtDwarfInfo_NewDie(pThis, &g_CompileUnitDesc, pAbbrev, NULL /*pParent*/);
6013 if (!pUnit)
6014 return VERR_NO_MEMORY;
6015 pUnit->offUnit = offUnit;
6016 pUnit->cbUnit = cbUnit;
6017 pUnit->offAbbrev = offAbbrev;
6018 pUnit->cbNativeAddr = cbNativeAddr;
6019 pUnit->uDwarfVer = (uint8_t)uVer;
6020 RTListAppend(&pThis->CompileUnitList, &pUnit->Core.SiblingNode);
6021
6022 int rc = rtDwarfInfo_ParseDie(pThis, &pUnit->Core, &g_CompileUnitDesc, pCursor, pAbbrev, true /*fInitDie*/);
6023 if (RT_FAILURE(rc))
6024 return rc;
6025
6026 /*
6027 * Parse DIEs.
6028 */
6029 uint32_t cDepth = 0;
6030 PRTDWARFDIE pParentDie = &pUnit->Core;
6031 while (!rtDwarfCursor_IsAtEndOfUnit(pCursor))
6032 {
6033#ifdef LOG_ENABLED
6034 uint32_t offLog = rtDwarfCursor_CalcSectOffsetU32(pCursor);
6035#endif
6036 uAbbrCode = rtDwarfCursor_GetULeb128AsU32(pCursor, UINT32_MAX);
6037 if (!uAbbrCode)
6038 {
6039 /* End of siblings, up one level. (Is this correct?) */
6040 if (pParentDie->pParent)
6041 {
6042 pParentDie = pParentDie->pParent;
6043 cDepth--;
6044 if (!fKeepDies && pParentDie->pParent)
6045 rtDwarfInfo_FreeChildren(pThis, pParentDie);
6046 }
6047 }
6048 else
6049 {
6050 /*
6051 * Look up the abbreviation and match the tag up with a descriptor.
6052 */
6053 pAbbrev = rtDwarfAbbrev_Lookup(pThis, uAbbrCode);
6054 if (!pAbbrev)
6055 return VERR_DWARF_ABBREV_NOT_FOUND;
6056
6057 PCRTDWARFDIEDESC pDieDesc;
6058 const char *pszName;
6059 if (pAbbrev->uTag < RT_ELEMENTS(g_aTagDescs))
6060 {
6061 Assert(g_aTagDescs[pAbbrev->uTag].uTag == pAbbrev->uTag || g_aTagDescs[pAbbrev->uTag].uTag == 0);
6062 pszName = g_aTagDescs[pAbbrev->uTag].pszName;
6063 pDieDesc = g_aTagDescs[pAbbrev->uTag].pDesc;
6064 }
6065 else
6066 {
6067 pszName = "<unknown>";
6068 pDieDesc = &g_CoreDieDesc;
6069 }
6070 Log4(("%08x: %*stag=%s (%#x, abbrev %u @ %#x)%s\n", offLog, cDepth * 2, "", pszName,
6071 pAbbrev->uTag, uAbbrCode, pAbbrev->offSpec - pAbbrev->cbHdr, pAbbrev->fChildren ? " has children" : ""));
6072
6073 /*
6074 * Create a new internal DIE structure and parse the
6075 * attributes.
6076 */
6077 PRTDWARFDIE pNewDie = rtDwarfInfo_NewDie(pThis, pDieDesc, pAbbrev, pParentDie);
6078 if (!pNewDie)
6079 return VERR_NO_MEMORY;
6080
6081 if (pAbbrev->fChildren)
6082 {
6083 pParentDie = pNewDie;
6084 cDepth++;
6085 }
6086
6087 rc = rtDwarfInfo_ParseDie(pThis, pNewDie, pDieDesc, pCursor, pAbbrev, true /*fInitDie*/);
6088 if (RT_FAILURE(rc))
6089 return rc;
6090
6091 if (!fKeepDies && !pAbbrev->fChildren)
6092 rtDwarfInfo_FreeDie(pThis, pNewDie);
6093 }
6094 } /* while more DIEs */
6095
6096
6097 /* Unlink and free child DIEs if told to do so. */
6098 if (!fKeepDies)
6099 rtDwarfInfo_FreeChildren(pThis, &pUnit->Core);
6100
6101 return RT_SUCCESS(rc) ? pCursor->rc : rc;
6102}
6103
6104
6105/**
6106 * Extracts the symbols.
6107 *
6108 * The symbols are insered into the debug info container.
6109 *
6110 * @returns IPRT status code
6111 * @param pThis The DWARF instance.
6112 */
6113static int rtDwarfInfo_LoadAll(PRTDBGMODDWARF pThis)
6114{
6115 RTDWARFCURSOR Cursor;
6116 int rc = rtDwarfCursor_Init(&Cursor, pThis, krtDbgModDwarfSect_info);
6117 if (RT_SUCCESS(rc))
6118 {
6119 while ( !rtDwarfCursor_IsAtEnd(&Cursor)
6120 && RT_SUCCESS(rc))
6121 rc = rtDwarfInfo_LoadUnit(pThis, &Cursor, false /* fKeepDies */);
6122
6123 rc = rtDwarfCursor_Delete(&Cursor, rc);
6124 }
6125 return rc;
6126}
6127
6128
6129
6130/*
6131 *
6132 * Public and image level symbol handling.
6133 * Public and image level symbol handling.
6134 * Public and image level symbol handling.
6135 * Public and image level symbol handling.
6136 *
6137 *
6138 */
6139
6140#define RTDBGDWARF_SYM_ENUM_BASE_ADDRESS UINT32_C(0x200000)
6141
6142/** @callback_method_impl{FNRTLDRENUMSYMS,
6143 * Adds missing symbols from the image symbol table.} */
6144static DECLCALLBACK(int) rtDwarfSyms_EnumSymbolsCallback(RTLDRMOD hLdrMod, const char *pszSymbol, unsigned uSymbol,
6145 RTLDRADDR Value, void *pvUser)
6146{
6147 PRTDBGMODDWARF pThis = (PRTDBGMODDWARF)pvUser;
6148 RT_NOREF_PV(hLdrMod); RT_NOREF_PV(uSymbol);
6149 Assert(pThis->iWatcomPass != 1);
6150
6151 RTLDRADDR uRva = Value - RTDBGDWARF_SYM_ENUM_BASE_ADDRESS;
6152 if ( Value >= RTDBGDWARF_SYM_ENUM_BASE_ADDRESS
6153 && uRva < _1G)
6154 {
6155 RTDBGSYMBOL SymInfo;
6156 RTINTPTR offDisp;
6157 int rc = RTDbgModSymbolByAddr(pThis->hCnt, RTDBGSEGIDX_RVA, uRva, RTDBGSYMADDR_FLAGS_LESS_OR_EQUAL, &offDisp, &SymInfo);
6158 if ( RT_FAILURE(rc)
6159 || offDisp != 0)
6160 {
6161 rc = RTDbgModSymbolAdd(pThis->hCnt, pszSymbol, RTDBGSEGIDX_RVA, uRva, 1, 0 /*fFlags*/, NULL /*piOrdinal*/);
6162 Log(("Dwarf: Symbol #%05u %#018RTptr %s [%Rrc]\n", uSymbol, Value, pszSymbol, rc)); NOREF(rc);
6163 }
6164 }
6165 else
6166 Log(("Dwarf: Symbol #%05u %#018RTptr '%s' [SKIPPED - INVALID ADDRESS]\n", uSymbol, Value, pszSymbol));
6167 return VINF_SUCCESS;
6168}
6169
6170
6171
6172/**
6173 * Loads additional symbols from the pubnames section and the executable image.
6174 *
6175 * The symbols are insered into the debug info container.
6176 *
6177 * @returns IPRT status code
6178 * @param pThis The DWARF instance.
6179 */
6180static int rtDwarfSyms_LoadAll(PRTDBGMODDWARF pThis)
6181{
6182 /*
6183 * pubnames.
6184 */
6185 int rc = VINF_SUCCESS;
6186 if (pThis->aSections[krtDbgModDwarfSect_pubnames].fPresent)
6187 {
6188// RTDWARFCURSOR Cursor;
6189// int rc = rtDwarfCursor_Init(&Cursor, pThis, krtDbgModDwarfSect_info);
6190// if (RT_SUCCESS(rc))
6191// {
6192// while ( !rtDwarfCursor_IsAtEnd(&Cursor)
6193// && RT_SUCCESS(rc))
6194// rc = rtDwarfInfo_LoadUnit(pThis, &Cursor, false /* fKeepDies */);
6195//
6196// rc = rtDwarfCursor_Delete(&Cursor, rc);
6197// }
6198// return rc;
6199 }
6200
6201 /*
6202 * The executable image.
6203 */
6204 if ( pThis->pImgMod
6205 && pThis->pImgMod->pImgVt->pfnEnumSymbols
6206 && pThis->iWatcomPass != 1
6207 && RT_SUCCESS(rc))
6208 {
6209 rc = pThis->pImgMod->pImgVt->pfnEnumSymbols(pThis->pImgMod,
6210 RTLDR_ENUM_SYMBOL_FLAGS_ALL | RTLDR_ENUM_SYMBOL_FLAGS_NO_FWD,
6211 RTDBGDWARF_SYM_ENUM_BASE_ADDRESS,
6212 rtDwarfSyms_EnumSymbolsCallback,
6213 pThis);
6214 }
6215
6216 return rc;
6217}
6218
6219
6220
6221
6222/*
6223 *
6224 * DWARF Debug module implementation.
6225 * DWARF Debug module implementation.
6226 * DWARF Debug module implementation.
6227 *
6228 */
6229
6230
6231/** @interface_method_impl{RTDBGMODVTDBG,pfnUnwindFrame} */
6232static DECLCALLBACK(int) rtDbgModDwarf_UnwindFrame(PRTDBGMODINT pMod, RTDBGSEGIDX iSeg, RTUINTPTR off, PRTDBGUNWINDSTATE pState)
6233{
6234 PRTDBGMODDWARF pThis = (PRTDBGMODDWARF)pMod->pvDbgPriv;
6235
6236 /*
6237 * Unwinding info is stored in the '.debug_frame' section, or altertively
6238 * in the '.eh_frame' one in the image. In the latter case the dbgmodldr.cpp
6239 * part of the operation will take care of it. Since the sections contain the
6240 * same data, we just create a cursor and call a common function to do the job.
6241 */
6242 if (pThis->aSections[krtDbgModDwarfSect_frame].fPresent)
6243 {
6244 RTDWARFCURSOR Cursor;
6245 int rc = rtDwarfCursor_Init(&Cursor, pThis, krtDbgModDwarfSect_frame);
6246 if (RT_SUCCESS(rc))
6247 {
6248 /* Figure default pointer encoding from image arch. */
6249 uint8_t bPtrEnc = rtDwarfUnwind_ArchToPtrEnc(pMod->pImgVt->pfnGetArch(pMod));
6250
6251 /* Make sure we've got both seg:off and rva for the input address. */
6252 RTUINTPTR uRva = off;
6253 if (iSeg == RTDBGSEGIDX_RVA)
6254 rtDbgModDwarfRvaToSegOffset(pThis, uRva, &iSeg, &off);
6255 else
6256 rtDbgModDwarfSegOffsetToRva(pThis, iSeg, off, &uRva);
6257
6258 /* Do the work */
6259 rc = rtDwarfUnwind_Slow(&Cursor, 0 /** @todo .debug_frame RVA*/, iSeg, off, uRva,
6260 pState, bPtrEnc, false /*fIsEhFrame*/);
6261
6262 rc = rtDwarfCursor_Delete(&Cursor, rc);
6263 }
6264 return rc;
6265 }
6266 return VERR_DBG_NO_UNWIND_INFO;
6267}
6268
6269
6270/** @interface_method_impl{RTDBGMODVTDBG,pfnLineByAddr} */
6271static DECLCALLBACK(int) rtDbgModDwarf_LineByAddr(PRTDBGMODINT pMod, RTDBGSEGIDX iSeg, RTUINTPTR off,
6272 PRTINTPTR poffDisp, PRTDBGLINE pLineInfo)
6273{
6274 PRTDBGMODDWARF pThis = (PRTDBGMODDWARF)pMod->pvDbgPriv;
6275 return RTDbgModLineByAddr(pThis->hCnt, iSeg, off, poffDisp, pLineInfo);
6276}
6277
6278
6279/** @interface_method_impl{RTDBGMODVTDBG,pfnLineByOrdinal} */
6280static DECLCALLBACK(int) rtDbgModDwarf_LineByOrdinal(PRTDBGMODINT pMod, uint32_t iOrdinal, PRTDBGLINE pLineInfo)
6281{
6282 PRTDBGMODDWARF pThis = (PRTDBGMODDWARF)pMod->pvDbgPriv;
6283 return RTDbgModLineByOrdinal(pThis->hCnt, iOrdinal, pLineInfo);
6284}
6285
6286
6287/** @interface_method_impl{RTDBGMODVTDBG,pfnLineCount} */
6288static DECLCALLBACK(uint32_t) rtDbgModDwarf_LineCount(PRTDBGMODINT pMod)
6289{
6290 PRTDBGMODDWARF pThis = (PRTDBGMODDWARF)pMod->pvDbgPriv;
6291 return RTDbgModLineCount(pThis->hCnt);
6292}
6293
6294
6295/** @interface_method_impl{RTDBGMODVTDBG,pfnLineAdd} */
6296static DECLCALLBACK(int) rtDbgModDwarf_LineAdd(PRTDBGMODINT pMod, const char *pszFile, size_t cchFile, uint32_t uLineNo,
6297 uint32_t iSeg, RTUINTPTR off, uint32_t *piOrdinal)
6298{
6299 PRTDBGMODDWARF pThis = (PRTDBGMODDWARF)pMod->pvDbgPriv;
6300 Assert(!pszFile[cchFile]); NOREF(cchFile);
6301 return RTDbgModLineAdd(pThis->hCnt, pszFile, uLineNo, iSeg, off, piOrdinal);
6302}
6303
6304
6305/** @interface_method_impl{RTDBGMODVTDBG,pfnSymbolByAddr} */
6306static DECLCALLBACK(int) rtDbgModDwarf_SymbolByAddr(PRTDBGMODINT pMod, RTDBGSEGIDX iSeg, RTUINTPTR off, uint32_t fFlags,
6307 PRTINTPTR poffDisp, PRTDBGSYMBOL pSymInfo)
6308{
6309 PRTDBGMODDWARF pThis = (PRTDBGMODDWARF)pMod->pvDbgPriv;
6310 return RTDbgModSymbolByAddr(pThis->hCnt, iSeg, off, fFlags, poffDisp, pSymInfo);
6311}
6312
6313
6314/** @interface_method_impl{RTDBGMODVTDBG,pfnSymbolByName} */
6315static DECLCALLBACK(int) rtDbgModDwarf_SymbolByName(PRTDBGMODINT pMod, const char *pszSymbol, size_t cchSymbol,
6316 PRTDBGSYMBOL pSymInfo)
6317{
6318 PRTDBGMODDWARF pThis = (PRTDBGMODDWARF)pMod->pvDbgPriv;
6319 Assert(!pszSymbol[cchSymbol]); RT_NOREF_PV(cchSymbol);
6320 return RTDbgModSymbolByName(pThis->hCnt, pszSymbol/*, cchSymbol*/, pSymInfo);
6321}
6322
6323
6324/** @interface_method_impl{RTDBGMODVTDBG,pfnSymbolByOrdinal} */
6325static DECLCALLBACK(int) rtDbgModDwarf_SymbolByOrdinal(PRTDBGMODINT pMod, uint32_t iOrdinal, PRTDBGSYMBOL pSymInfo)
6326{
6327 PRTDBGMODDWARF pThis = (PRTDBGMODDWARF)pMod->pvDbgPriv;
6328 return RTDbgModSymbolByOrdinal(pThis->hCnt, iOrdinal, pSymInfo);
6329}
6330
6331
6332/** @interface_method_impl{RTDBGMODVTDBG,pfnSymbolCount} */
6333static DECLCALLBACK(uint32_t) rtDbgModDwarf_SymbolCount(PRTDBGMODINT pMod)
6334{
6335 PRTDBGMODDWARF pThis = (PRTDBGMODDWARF)pMod->pvDbgPriv;
6336 return RTDbgModSymbolCount(pThis->hCnt);
6337}
6338
6339
6340/** @interface_method_impl{RTDBGMODVTDBG,pfnSymbolAdd} */
6341static DECLCALLBACK(int) rtDbgModDwarf_SymbolAdd(PRTDBGMODINT pMod, const char *pszSymbol, size_t cchSymbol,
6342 RTDBGSEGIDX iSeg, RTUINTPTR off, RTUINTPTR cb, uint32_t fFlags,
6343 uint32_t *piOrdinal)
6344{
6345 PRTDBGMODDWARF pThis = (PRTDBGMODDWARF)pMod->pvDbgPriv;
6346 Assert(!pszSymbol[cchSymbol]); NOREF(cchSymbol);
6347 return RTDbgModSymbolAdd(pThis->hCnt, pszSymbol, iSeg, off, cb, fFlags, piOrdinal);
6348}
6349
6350
6351/** @interface_method_impl{RTDBGMODVTDBG,pfnSegmentByIndex} */
6352static DECLCALLBACK(int) rtDbgModDwarf_SegmentByIndex(PRTDBGMODINT pMod, RTDBGSEGIDX iSeg, PRTDBGSEGMENT pSegInfo)
6353{
6354 PRTDBGMODDWARF pThis = (PRTDBGMODDWARF)pMod->pvDbgPriv;
6355 return RTDbgModSegmentByIndex(pThis->hCnt, iSeg, pSegInfo);
6356}
6357
6358
6359/** @interface_method_impl{RTDBGMODVTDBG,pfnSegmentCount} */
6360static DECLCALLBACK(RTDBGSEGIDX) rtDbgModDwarf_SegmentCount(PRTDBGMODINT pMod)
6361{
6362 PRTDBGMODDWARF pThis = (PRTDBGMODDWARF)pMod->pvDbgPriv;
6363 return RTDbgModSegmentCount(pThis->hCnt);
6364}
6365
6366
6367/** @interface_method_impl{RTDBGMODVTDBG,pfnSegmentAdd} */
6368static DECLCALLBACK(int) rtDbgModDwarf_SegmentAdd(PRTDBGMODINT pMod, RTUINTPTR uRva, RTUINTPTR cb, const char *pszName, size_t cchName,
6369 uint32_t fFlags, PRTDBGSEGIDX piSeg)
6370{
6371 PRTDBGMODDWARF pThis = (PRTDBGMODDWARF)pMod->pvDbgPriv;
6372 Assert(!pszName[cchName]); NOREF(cchName);
6373 return RTDbgModSegmentAdd(pThis->hCnt, uRva, cb, pszName, fFlags, piSeg);
6374}
6375
6376
6377/** @interface_method_impl{RTDBGMODVTDBG,pfnImageSize} */
6378static DECLCALLBACK(RTUINTPTR) rtDbgModDwarf_ImageSize(PRTDBGMODINT pMod)
6379{
6380 PRTDBGMODDWARF pThis = (PRTDBGMODDWARF)pMod->pvDbgPriv;
6381 RTUINTPTR cb1 = RTDbgModImageSize(pThis->hCnt);
6382 RTUINTPTR cb2 = pThis->pImgMod->pImgVt->pfnImageSize(pMod);
6383 return RT_MAX(cb1, cb2);
6384}
6385
6386
6387/** @interface_method_impl{RTDBGMODVTDBG,pfnRvaToSegOff} */
6388static DECLCALLBACK(RTDBGSEGIDX) rtDbgModDwarf_RvaToSegOff(PRTDBGMODINT pMod, RTUINTPTR uRva, PRTUINTPTR poffSeg)
6389{
6390 PRTDBGMODDWARF pThis = (PRTDBGMODDWARF)pMod->pvDbgPriv;
6391 return RTDbgModRvaToSegOff(pThis->hCnt, uRva, poffSeg);
6392}
6393
6394
6395/** @interface_method_impl{RTDBGMODVTDBG,pfnClose} */
6396static DECLCALLBACK(int) rtDbgModDwarf_Close(PRTDBGMODINT pMod)
6397{
6398 PRTDBGMODDWARF pThis = (PRTDBGMODDWARF)pMod->pvDbgPriv;
6399
6400 for (unsigned iSect = 0; iSect < RT_ELEMENTS(pThis->aSections); iSect++)
6401 if (pThis->aSections[iSect].pv)
6402 pThis->pDbgInfoMod->pImgVt->pfnUnmapPart(pThis->pDbgInfoMod, pThis->aSections[iSect].cb, &pThis->aSections[iSect].pv);
6403
6404 RTDbgModRelease(pThis->hCnt);
6405 RTMemFree(pThis->paCachedAbbrevs);
6406 if (pThis->pNestedMod)
6407 {
6408 pThis->pNestedMod->pImgVt->pfnClose(pThis->pNestedMod);
6409 RTStrCacheRelease(g_hDbgModStrCache, pThis->pNestedMod->pszName);
6410 RTStrCacheRelease(g_hDbgModStrCache, pThis->pNestedMod->pszDbgFile);
6411 RTMemFree(pThis->pNestedMod);
6412 pThis->pNestedMod = NULL;
6413 }
6414
6415#ifdef RTDBGMODDWARF_WITH_MEM_CACHE
6416 uint32_t i = RT_ELEMENTS(pThis->aDieAllocators);
6417 while (i-- > 0)
6418 {
6419 RTMemCacheDestroy(pThis->aDieAllocators[i].hMemCache);
6420 pThis->aDieAllocators[i].hMemCache = NIL_RTMEMCACHE;
6421 }
6422#endif
6423
6424 RTMemFree(pThis);
6425
6426 return VINF_SUCCESS;
6427}
6428
6429
6430/** @callback_method_impl{FNRTLDRENUMDBG} */
6431static DECLCALLBACK(int) rtDbgModDwarfEnumCallback(RTLDRMOD hLdrMod, PCRTLDRDBGINFO pDbgInfo, void *pvUser)
6432{
6433 RT_NOREF_PV(hLdrMod);
6434
6435 /*
6436 * Skip stuff we can't handle.
6437 */
6438 if (pDbgInfo->enmType != RTLDRDBGINFOTYPE_DWARF)
6439 return VINF_SUCCESS;
6440 const char *pszSection = pDbgInfo->u.Dwarf.pszSection;
6441 if (!pszSection || !*pszSection)
6442 return VINF_SUCCESS;
6443 Assert(!pDbgInfo->pszExtFile);
6444
6445 /*
6446 * Must have a part name starting with debug_ and possibly prefixed by dots
6447 * or underscores.
6448 */
6449 if (!strncmp(pszSection, RT_STR_TUPLE(".debug_"))) /* ELF */
6450 pszSection += sizeof(".debug_") - 1;
6451 else if (!strncmp(pszSection, RT_STR_TUPLE("__debug_"))) /* Mach-O */
6452 pszSection += sizeof("__debug_") - 1;
6453 else if (!strcmp(pszSection, ".WATCOM_references"))
6454 return VINF_SUCCESS; /* Ignore special watcom section for now.*/
6455 else if ( !strcmp(pszSection, "__apple_types")
6456 || !strcmp(pszSection, "__apple_namespac")
6457 || !strcmp(pszSection, "__apple_objc")
6458 || !strcmp(pszSection, "__apple_names"))
6459 return VINF_SUCCESS; /* Ignore special apple sections for now. */
6460 else
6461 AssertMsgFailedReturn(("%s\n", pszSection), VINF_SUCCESS /*ignore*/);
6462
6463 /*
6464 * Figure out which part we're talking about.
6465 */
6466 krtDbgModDwarfSect enmSect;
6467 if (0) { /* dummy */ }
6468#define ELSE_IF_STRCMP_SET(a_Name) else if (!strcmp(pszSection, #a_Name)) enmSect = krtDbgModDwarfSect_ ## a_Name
6469 ELSE_IF_STRCMP_SET(abbrev);
6470 ELSE_IF_STRCMP_SET(aranges);
6471 ELSE_IF_STRCMP_SET(frame);
6472 ELSE_IF_STRCMP_SET(info);
6473 ELSE_IF_STRCMP_SET(inlined);
6474 ELSE_IF_STRCMP_SET(line);
6475 ELSE_IF_STRCMP_SET(loc);
6476 ELSE_IF_STRCMP_SET(macinfo);
6477 ELSE_IF_STRCMP_SET(pubnames);
6478 ELSE_IF_STRCMP_SET(pubtypes);
6479 ELSE_IF_STRCMP_SET(ranges);
6480 ELSE_IF_STRCMP_SET(str);
6481 ELSE_IF_STRCMP_SET(types);
6482#undef ELSE_IF_STRCMP_SET
6483 else
6484 {
6485 AssertMsgFailed(("%s\n", pszSection));
6486 return VINF_SUCCESS;
6487 }
6488
6489 /*
6490 * Record the section.
6491 */
6492 PRTDBGMODDWARF pThis = (PRTDBGMODDWARF)pvUser;
6493 AssertMsgReturn(!pThis->aSections[enmSect].fPresent, ("duplicate %s\n", pszSection), VINF_SUCCESS /*ignore*/);
6494
6495 pThis->aSections[enmSect].fPresent = true;
6496 pThis->aSections[enmSect].offFile = pDbgInfo->offFile;
6497 pThis->aSections[enmSect].pv = NULL;
6498 pThis->aSections[enmSect].cb = (size_t)pDbgInfo->cb;
6499 pThis->aSections[enmSect].iDbgInfo = pDbgInfo->iDbgInfo;
6500 if (pThis->aSections[enmSect].cb != pDbgInfo->cb)
6501 pThis->aSections[enmSect].cb = ~(size_t)0;
6502
6503 return VINF_SUCCESS;
6504}
6505
6506
6507static int rtDbgModDwarfTryOpenDbgFile(PRTDBGMODINT pDbgMod, PRTDBGMODDWARF pThis, RTLDRARCH enmArch)
6508{
6509 if ( !pDbgMod->pszDbgFile
6510 || RTPathIsSame(pDbgMod->pszDbgFile, pDbgMod->pszImgFile) == (int)true /* returns VERR too */)
6511 return VERR_DBG_NO_MATCHING_INTERPRETER;
6512
6513 /*
6514 * Only open the image.
6515 */
6516 PRTDBGMODINT pDbgInfoMod = (PRTDBGMODINT)RTMemAllocZ(sizeof(*pDbgInfoMod));
6517 if (!pDbgInfoMod)
6518 return VERR_NO_MEMORY;
6519
6520 int rc;
6521 pDbgInfoMod->u32Magic = RTDBGMOD_MAGIC;
6522 pDbgInfoMod->cRefs = 1;
6523 if (RTStrCacheRetain(pDbgMod->pszDbgFile) != UINT32_MAX)
6524 {
6525 pDbgInfoMod->pszImgFile = pDbgMod->pszDbgFile;
6526 if (RTStrCacheRetain(pDbgMod->pszName) != UINT32_MAX)
6527 {
6528 pDbgInfoMod->pszName = pDbgMod->pszName;
6529 pDbgInfoMod->pImgVt = &g_rtDbgModVtImgLdr;
6530 rc = pDbgInfoMod->pImgVt->pfnTryOpen(pDbgInfoMod, enmArch);
6531 if (RT_SUCCESS(rc))
6532 {
6533 pThis->pDbgInfoMod = pDbgInfoMod;
6534 pThis->pNestedMod = pDbgInfoMod;
6535 return VINF_SUCCESS;
6536 }
6537
6538 RTStrCacheRelease(g_hDbgModStrCache, pDbgInfoMod->pszName);
6539 }
6540 else
6541 rc = VERR_NO_STR_MEMORY;
6542 RTStrCacheRelease(g_hDbgModStrCache, pDbgInfoMod->pszImgFile);
6543 }
6544 else
6545 rc = VERR_NO_STR_MEMORY;
6546 RTMemFree(pDbgInfoMod);
6547 return rc;
6548}
6549
6550
6551/** @interface_method_impl{RTDBGMODVTDBG,pfnTryOpen} */
6552static DECLCALLBACK(int) rtDbgModDwarf_TryOpen(PRTDBGMODINT pMod, RTLDRARCH enmArch)
6553{
6554 /*
6555 * DWARF is only supported when part of an image.
6556 */
6557 if (!pMod->pImgVt)
6558 return VERR_DBG_NO_MATCHING_INTERPRETER;
6559
6560 /*
6561 * Create the module instance data.
6562 */
6563 PRTDBGMODDWARF pThis = (PRTDBGMODDWARF)RTMemAllocZ(sizeof(*pThis));
6564 if (!pThis)
6565 return VERR_NO_MEMORY;
6566 pThis->pDbgInfoMod = pMod;
6567 pThis->pImgMod = pMod;
6568 RTListInit(&pThis->CompileUnitList);
6569
6570 /** @todo better fUseLinkAddress heuristics! */
6571 /* mach_kernel: */
6572 if ( (pMod->pszDbgFile && strstr(pMod->pszDbgFile, "mach_kernel"))
6573 || (pMod->pszImgFile && strstr(pMod->pszImgFile, "mach_kernel"))
6574 || (pMod->pszImgFileSpecified && strstr(pMod->pszImgFileSpecified, "mach_kernel")) )
6575 pThis->fUseLinkAddress = true;
6576
6577#ifdef RTDBGMODDWARF_WITH_MEM_CACHE
6578 AssertCompile(RT_ELEMENTS(pThis->aDieAllocators) == 2);
6579 pThis->aDieAllocators[0].cbMax = sizeof(RTDWARFDIE);
6580 pThis->aDieAllocators[1].cbMax = sizeof(RTDWARFDIECOMPILEUNIT);
6581 for (uint32_t i = 0; i < RT_ELEMENTS(g_aTagDescs); i++)
6582 if (g_aTagDescs[i].pDesc && g_aTagDescs[i].pDesc->cbDie > pThis->aDieAllocators[1].cbMax)
6583 pThis->aDieAllocators[1].cbMax = (uint32_t)g_aTagDescs[i].pDesc->cbDie;
6584 pThis->aDieAllocators[1].cbMax = RT_ALIGN_32(pThis->aDieAllocators[1].cbMax, sizeof(uint64_t));
6585
6586 for (uint32_t i = 0; i < RT_ELEMENTS(pThis->aDieAllocators); i++)
6587 {
6588 int rc = RTMemCacheCreate(&pThis->aDieAllocators[i].hMemCache, pThis->aDieAllocators[i].cbMax, sizeof(uint64_t),
6589 UINT32_MAX, NULL /*pfnCtor*/, NULL /*pfnDtor*/, NULL /*pvUser*/, 0 /*fFlags*/);
6590 if (RT_FAILURE(rc))
6591 {
6592 while (i-- > 0)
6593 RTMemCacheDestroy(pThis->aDieAllocators[i].hMemCache);
6594 RTMemFree(pThis);
6595 return rc;
6596 }
6597 }
6598#endif
6599
6600 /*
6601 * If the debug file name is set, let's see if it's an ELF image with DWARF
6602 * inside it. In that case we'll have to deal with two image modules, one
6603 * for segments and address translation and one for the debug information.
6604 */
6605 if (pMod->pszDbgFile != NULL)
6606 rtDbgModDwarfTryOpenDbgFile(pMod, pThis, enmArch);
6607
6608 /*
6609 * Enumerate the debug info in the module, looking for DWARF bits.
6610 */
6611 int rc = pThis->pDbgInfoMod->pImgVt->pfnEnumDbgInfo(pThis->pDbgInfoMod, rtDbgModDwarfEnumCallback, pThis);
6612 if (RT_SUCCESS(rc))
6613 {
6614 if (pThis->aSections[krtDbgModDwarfSect_info].fPresent)
6615 {
6616 /*
6617 * Extract / explode the data we want (symbols and line numbers)
6618 * storing them in a container module.
6619 */
6620 rc = RTDbgModCreate(&pThis->hCnt, pMod->pszName, 0 /*cbSeg*/, 0 /*fFlags*/);
6621 if (RT_SUCCESS(rc))
6622 {
6623 pMod->pvDbgPriv = pThis;
6624
6625 rc = rtDbgModDwarfAddSegmentsFromImage(pThis);
6626 if (RT_SUCCESS(rc))
6627 rc = rtDwarfInfo_LoadAll(pThis);
6628 if (RT_SUCCESS(rc))
6629 rc = rtDwarfSyms_LoadAll(pThis);
6630 if (RT_SUCCESS(rc))
6631 rc = rtDwarfLine_ExplodeAll(pThis);
6632 if (RT_SUCCESS(rc) && pThis->iWatcomPass == 1)
6633 {
6634 rc = rtDbgModDwarfAddSegmentsFromPass1(pThis);
6635 pThis->iWatcomPass = 2;
6636 if (RT_SUCCESS(rc))
6637 rc = rtDwarfInfo_LoadAll(pThis);
6638 if (RT_SUCCESS(rc))
6639 rc = rtDwarfSyms_LoadAll(pThis);
6640 if (RT_SUCCESS(rc))
6641 rc = rtDwarfLine_ExplodeAll(pThis);
6642 }
6643 if (RT_SUCCESS(rc))
6644 {
6645 /*
6646 * Free the cached abbreviations and unload all sections.
6647 */
6648 pThis->cCachedAbbrevsAlloced = 0;
6649 RTMemFree(pThis->paCachedAbbrevs);
6650 pThis->paCachedAbbrevs = NULL;
6651
6652 for (unsigned iSect = 0; iSect < RT_ELEMENTS(pThis->aSections); iSect++)
6653 if (pThis->aSections[iSect].pv)
6654 pThis->pDbgInfoMod->pImgVt->pfnUnmapPart(pThis->pDbgInfoMod, pThis->aSections[iSect].cb,
6655 &pThis->aSections[iSect].pv);
6656
6657 /** @todo Kill pThis->CompileUnitList and the alloc caches. */
6658 return VINF_SUCCESS;
6659 }
6660
6661 /* bail out. */
6662 RTDbgModRelease(pThis->hCnt);
6663 pMod->pvDbgPriv = NULL;
6664 }
6665 }
6666 else
6667 rc = VERR_DBG_NO_MATCHING_INTERPRETER;
6668 }
6669
6670 RTMemFree(pThis->paCachedAbbrevs);
6671
6672#ifdef RTDBGMODDWARF_WITH_MEM_CACHE
6673 uint32_t i = RT_ELEMENTS(pThis->aDieAllocators);
6674 while (i-- > 0)
6675 {
6676 RTMemCacheDestroy(pThis->aDieAllocators[i].hMemCache);
6677 pThis->aDieAllocators[i].hMemCache = NIL_RTMEMCACHE;
6678 }
6679#endif
6680
6681 RTMemFree(pThis);
6682
6683 return rc;
6684}
6685
6686
6687
6688/** Virtual function table for the DWARF debug info reader. */
6689DECL_HIDDEN_CONST(RTDBGMODVTDBG) const g_rtDbgModVtDbgDwarf =
6690{
6691 /*.u32Magic = */ RTDBGMODVTDBG_MAGIC,
6692 /*.fSupports = */ RT_DBGTYPE_DWARF,
6693 /*.pszName = */ "dwarf",
6694 /*.pfnTryOpen = */ rtDbgModDwarf_TryOpen,
6695 /*.pfnClose = */ rtDbgModDwarf_Close,
6696
6697 /*.pfnRvaToSegOff = */ rtDbgModDwarf_RvaToSegOff,
6698 /*.pfnImageSize = */ rtDbgModDwarf_ImageSize,
6699
6700 /*.pfnSegmentAdd = */ rtDbgModDwarf_SegmentAdd,
6701 /*.pfnSegmentCount = */ rtDbgModDwarf_SegmentCount,
6702 /*.pfnSegmentByIndex = */ rtDbgModDwarf_SegmentByIndex,
6703
6704 /*.pfnSymbolAdd = */ rtDbgModDwarf_SymbolAdd,
6705 /*.pfnSymbolCount = */ rtDbgModDwarf_SymbolCount,
6706 /*.pfnSymbolByOrdinal = */ rtDbgModDwarf_SymbolByOrdinal,
6707 /*.pfnSymbolByName = */ rtDbgModDwarf_SymbolByName,
6708 /*.pfnSymbolByAddr = */ rtDbgModDwarf_SymbolByAddr,
6709
6710 /*.pfnLineAdd = */ rtDbgModDwarf_LineAdd,
6711 /*.pfnLineCount = */ rtDbgModDwarf_LineCount,
6712 /*.pfnLineByOrdinal = */ rtDbgModDwarf_LineByOrdinal,
6713 /*.pfnLineByAddr = */ rtDbgModDwarf_LineByAddr,
6714
6715 /*.pfnUnwindFrame = */ rtDbgModDwarf_UnwindFrame,
6716
6717 /*.u32EndMagic = */ RTDBGMODVTDBG_MAGIC
6718};
6719
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette