jarin updated this revision to Diff 375865.
jarin added a comment.
Rewrote the recursive parser to use a plain method.
Pruned and annotated the test.
Added other test cases:
- all parameters unused,
- inlining from two different functions,
- stack trace.
This still uses `frame variable` rather than `image lookup`, mostly because
`frame variable` tests better the user experience and the cognitive overhead
for making the code runnable does not seem to be too high.
CHANGES SINCE LAST ACTION
https://reviews.llvm.org/D110571/new/
https://reviews.llvm.org/D110571
Files:
lldb/source/Plugins/SymbolFile/DWARF/SymbolFileDWARF.cpp
lldb/source/Plugins/SymbolFile/DWARF/SymbolFileDWARF.h
lldb/test/Shell/SymbolFile/DWARF/x86/Inputs/unused-inlined-params.s
lldb/test/Shell/SymbolFile/DWARF/x86/unused-inlined-params.test
Index: lldb/test/Shell/SymbolFile/DWARF/x86/unused-inlined-params.test
===================================================================
--- /dev/null
+++ lldb/test/Shell/SymbolFile/DWARF/x86/unused-inlined-params.test
@@ -0,0 +1,69 @@
+# UNSUPPORTED: system-darwin, system-windows
+
+# REQUIRES: lld
+
+# RUN: llvm-mc -filetype=obj %S/Inputs/unused-inlined-params.s \
+# RUN: -triple x86_64-pc-linux -o %t.o
+# RUN: ld.lld %t.o -o %t
+# RUN: %lldb %t -s %s -o exit | FileCheck %s
+
+
+# In this test we verify that inlined functions still mention
+# all their parameters in `frame variable`, even when those
+# parameters were completely optimized away from the concrete
+# instance of the inlined function in the debug info.
+# The debugger should look up the parameters in the abstract
+# origin of the concrete instance.
+
+breakpoint set -n main
+process launch
+# CHECK: Process {{.*}} stopped
+
+# Let us check that unused parameters of an inlined function are listed
+# at the inlined function entry.
+breakpoint set -n break_at_inlined_f_in_main
+continue
+frame variable
+# CHECK-LABEL: frame variable
+# CHECK: (void *) unused1 = <
+# CHECK: (int) used = 42
+# CHECK: (int) unused2 = <
+# CHECK: (int) partial = 1
+# CHECK: (int) unused3 = <
+
+# Step out of the live range of the 'partial' parameter and verify that
+# all variables are still displayed.
+breakpoint set -n break_at_inlined_f_in_main_between_printfs
+continue
+frame variable
+# CHECK-LABEL: frame variable
+# CHECK: (void *) unused1 = <
+# CHECK: (int) used = 43
+# CHECK: (int) unused2 = <
+# CHECK: (int) partial = <
+# CHECK: (int) unused3 = <
+
+# Check that we show parameters even if all of them are compiled away.
+breakpoint set -n break_at_inlined_g_in_main
+continue
+frame variable
+# CHECK-LABEL: frame variable
+# CHECK: (int) unused = <
+
+# Check that even the other inlined instance of f displays correct
+# parameters.
+breakpoint set -n break_at_inlined_f_in_other
+continue
+frame variable
+# CHECK-LABEL: frame variable
+# CHECK: (void *) unused1 = <
+# CHECK: (int) used = 1
+# CHECK: (int) unused2 = <
+# CHECK: (int) partial = 2
+# CHECK: (int) unused3 = <
+
+# Check that the stack trace contains the inlined function with
+# all the arguments
+thread backtrace
+CHECK: other [inlined] f(unused1=<unavailable>, used=1, unused2=<unavailable>, \
+partial=2, unused3=<unavailable>)
\ No newline at end of file
Index: lldb/test/Shell/SymbolFile/DWARF/x86/Inputs/unused-inlined-params.s
===================================================================
--- /dev/null
+++ lldb/test/Shell/SymbolFile/DWARF/x86/Inputs/unused-inlined-params.s
@@ -0,0 +1,458 @@
+# The below program is roughly derived from the following C program.
+# To see the annotated debug info, look for the section
+# '.section .debug_info' below.
+#
+# __attribute__((always_inline))
+# void f(void* unused1, int used, int unused2, int partial, int unused3) {
+# used += partial;
+# printf("f %i", partial);
+# printf("f %i", used); // |partial| is not live at this line.
+# }
+#
+# void g(int unused) {
+# printf("Hello");
+# }
+#
+# __attribute__((noinline))
+# void other() {
+# f(nullptr, 1, 0, 2, 0);
+# }
+#
+# int main(int argc, char** argv) {
+# f(argv, 42, 1, argc, 2);
+# g(1);
+# other();
+# return 0;
+# }
+
+ .text
+ .file "unused-inlined-params.c"
+
+.Lcu_begin:
+
+ .globl other
+other:
+ nop
+.Linlined_f_in_other:
+break_at_inlined_f_in_other:
+ callq printf # Omitted the setup of arguments.
+.Linlined_f_in_other_between_printfs:
+ callq printf # Omitted the setup of arguments.
+.Linlined_f_in_other_end:
+ retq
+.Lother_end:
+ .size other, .Lother_end-other
+
+ .globl main
+main:
+ .file 1 "/example" "unused-inlined-params.c"
+ movl $1, %esi
+.Linlined_f:
+break_at_inlined_f_in_main:
+ leal 42(%rsi), %ebx
+.Linlined_f_before_printf:
+ callq printf # Omitted the setup of arguments.
+.Linlined_f_between_printfs:
+break_at_inlined_f_in_main_between_printfs:
+ callq printf # Omitted the setup of arguments.
+.Linlined_f_end:
+.Linlined_g:
+break_at_inlined_g_in_main:
+ callq printf # Omitted the setup of arguments.
+.Linlined_g_end:
+ callq other
+ retq
+.Lmain_end:
+ .size main, .Lmain_end-main
+
+# Dummy printf implementation.
+printf:
+ retq
+
+# Simple entry point to make the linker happy.
+ .globl _start
+_start:
+ jmp main
+
+.Lcu_end:
+
+
+ .section .debug_loc,"",@progbits
+.Ldebug_loc_partial:
+ .quad .Linlined_f-.Lcu_begin
+ .quad .Linlined_f_between_printfs-.Lcu_begin
+ .short 1 # Loc expr size
+ .byte 84 # super-register DW_OP_reg4
+ .quad 0
+ .quad 0
+.Ldebug_loc_used:
+ .quad .Linlined_f-.Lcu_begin
+ .quad .Linlined_f_before_printf-.Lcu_begin
+ .short 3 # Loc expr size
+ .byte 17 # DW_OP_consts
+ .byte 42 # value
+ .byte 159 # DW_OP_stack_value
+ .quad .Linlined_f_before_printf-.Lcu_begin
+ .quad .Linlined_f_end-.Lcu_begin
+ .short 1 # Loc expr size
+ .byte 83 # super-register DW_OP_reg3
+ .quad 0
+ .quad 0
+.Ldebug_loc_partial_in_other:
+ .quad .Linlined_f_in_other-.Lcu_begin
+ .quad .Linlined_f_in_other_between_printfs-.Lcu_begin
+ .short 3 # Loc expr size
+ .byte 17 # DW_OP_consts
+ .byte 2 # value
+ .byte 159 # DW_OP_stack_value
+ .quad 0
+ .quad 0
+.Ldebug_loc_used_in_other:
+ .quad .Linlined_f_in_other-.Lcu_begin
+ .quad .Linlined_f_in_other_end-.Lcu_begin
+ .short 3 # Loc expr size
+ .byte 17 # DW_OP_consts
+ .byte 1 # value
+ .byte 159 # DW_OP_stack_value
+ .quad 0
+ .quad 0
+
+ .section .debug_abbrev,"",@progbits
+ .byte 1 # Abbreviation Code
+ .byte 17 # DW_TAG_compile_unit
+ .byte 1 # DW_CHILDREN_yes
+ .byte 3 # DW_AT_name
+ .byte 14 # DW_FORM_strp
+ .byte 16 # DW_AT_stmt_list
+ .byte 23 # DW_FORM_sec_offset
+ .byte 17 # DW_AT_low_pc
+ .byte 1 # DW_FORM_addr
+ .byte 18 # DW_AT_high_pc
+ .byte 6 # DW_FORM_data4
+ .byte 0 # EOM(1)
+ .byte 0 # EOM(2)
+
+ .byte 4 # Abbreviation Code
+ .byte 5 # DW_TAG_formal_parameter
+ .byte 0 # DW_CHILDREN_no
+ .byte 2 # DW_AT_location
+ .byte 23 # DW_FORM_sec_offset
+ .byte 49 # DW_AT_abstract_origin
+ .byte 19 # DW_FORM_ref4
+ .byte 0 # EOM(1)
+ .byte 0 # EOM(2)
+
+ .byte 5 # Abbreviation Code
+ .byte 46 # DW_TAG_subprogram
+ .byte 1 # DW_CHILDREN_yes
+ .byte 3 # DW_AT_name
+ .byte 14 # DW_FORM_strp
+ .byte 58 # DW_AT_decl_file
+ .byte 11 # DW_FORM_data1
+ .byte 59 # DW_AT_decl_line
+ .byte 11 # DW_FORM_data1
+ .byte 39 # DW_AT_prototyped
+ .byte 25 # DW_FORM_flag_present
+ .byte 63 # DW_AT_external
+ .byte 25 # DW_FORM_flag_present
+ .byte 32 # DW_AT_inline
+ .byte 11 # DW_FORM_data1
+ .byte 0 # EOM(1)
+ .byte 0 # EOM(2)
+
+ .byte 6 # Abbreviation Code
+ .byte 5 # DW_TAG_formal_parameter
+ .byte 0 # DW_CHILDREN_no
+ .byte 3 # DW_AT_name
+ .byte 14 # DW_FORM_strp
+ .byte 58 # DW_AT_decl_file
+ .byte 11 # DW_FORM_data1
+ .byte 59 # DW_AT_decl_line
+ .byte 11 # DW_FORM_data1
+ .byte 73 # DW_AT_type
+ .byte 19 # DW_FORM_ref4
+ .byte 0 # EOM(1)
+ .byte 0 # EOM(2)
+
+ .byte 7 # Abbreviation Code
+ .byte 15 # DW_TAG_pointer_type
+ .byte 0 # DW_CHILDREN_no
+ .byte 0 # EOM(1)
+ .byte 0 # EOM(2)
+
+ .byte 8 # Abbreviation Code
+ .byte 36 # DW_TAG_base_type
+ .byte 0 # DW_CHILDREN_no
+ .byte 3 # DW_AT_name
+ .byte 14 # DW_FORM_strp
+ .byte 62 # DW_AT_encoding
+ .byte 11 # DW_FORM_data1
+ .byte 11 # DW_AT_byte_size
+ .byte 11 # DW_FORM_data1
+ .byte 0 # EOM(1)
+ .byte 0 # EOM(2)
+
+ .byte 9 # Abbreviation Code
+ .byte 46 # DW_TAG_subprogram
+ .byte 1 # DW_CHILDREN_yes
+ .byte 17 # DW_AT_low_pc
+ .byte 1 # DW_FORM_addr
+ .byte 18 # DW_AT_high_pc
+ .byte 6 # DW_FORM_data4
+ .byte 64 # DW_AT_frame_base
+ .byte 24 # DW_FORM_exprloc
+ .byte 3 # DW_AT_name
+ .byte 14 # DW_FORM_strp
+ .byte 58 # DW_AT_decl_file
+ .byte 11 # DW_FORM_data1
+ .byte 59 # DW_AT_decl_line
+ .byte 11 # DW_FORM_data1
+ .byte 39 # DW_AT_prototyped
+ .byte 25 # DW_FORM_flag_present
+ .byte 73 # DW_AT_type
+ .byte 19 # DW_FORM_ref4
+ .byte 63 # DW_AT_external
+ .byte 25 # DW_FORM_flag_present
+ .byte 0 # EOM(1)
+ .byte 0 # EOM(2)
+
+ .byte 10 # Abbreviation Code
+ .byte 5 # DW_TAG_formal_parameter
+ .byte 0 # DW_CHILDREN_no
+ .byte 2 # DW_AT_location
+ .byte 23 # DW_FORM_sec_offset
+ .byte 3 # DW_AT_name
+ .byte 14 # DW_FORM_strp
+ .byte 58 # DW_AT_decl_file
+ .byte 11 # DW_FORM_data1
+ .byte 59 # DW_AT_decl_line
+ .byte 11 # DW_FORM_data1
+ .byte 73 # DW_AT_type
+ .byte 19 # DW_FORM_ref4
+ .byte 0 # EOM(1)
+ .byte 0 # EOM(2)
+
+ .byte 11 # Abbreviation Code
+ .byte 29 # DW_TAG_inlined_subroutine
+ .byte 1 # DW_CHILDREN_yes
+ .byte 49 # DW_AT_abstract_origin
+ .byte 19 # DW_FORM_ref4
+ .byte 17 # DW_AT_low_pc
+ .byte 1 # DW_FORM_addr
+ .byte 18 # DW_AT_high_pc
+ .byte 6 # DW_FORM_data4
+ .byte 88 # DW_AT_call_file
+ .byte 11 # DW_FORM_data1
+ .byte 89 # DW_AT_call_line
+ .byte 11 # DW_FORM_data1
+ .byte 87 # DW_AT_call_column
+ .byte 11 # DW_FORM_data1
+ .byte 0 # EOM(1)
+ .byte 0 # EOM(2)
+
+ .byte 12 # Abbreviation Code
+ .byte 15 # DW_TAG_pointer_type
+ .byte 0 # DW_CHILDREN_no
+ .byte 73 # DW_AT_type
+ .byte 19 # DW_FORM_ref4
+ .byte 0 # EOM(1)
+ .byte 0 # EOM(2)
+ .byte 0 # EOM(3)
+
+ .section .debug_info,"",@progbits
+.Ldi_cu_begin:
+ .long .Ldebug_info_end0-.Ldebug_info_start0 # Length of Unit
+.Ldebug_info_start0:
+ .short 4 # DWARF version number
+ .long .debug_abbrev # Offset Into Abbrev. Section
+ .byte 8 # Address Size (in bytes)
+ .byte 1 # Abbrev [1] 0xb:0x11c DW_TAG_compile_unit
+ .long .Linfo_string_fname # DW_AT_name
+ .long .Lline_table_start0 # DW_AT_stmt_list
+ .quad .Lcu_begin # DW_AT_low_pc
+ .long .Lcu_end-.Lcu_begin # DW_AT_high_pc
+
+# Debug info for |f| (abstract version with all parameters).
+
+.Ldebug_info_f:
+ .byte 5 # Abbrev [5] 0x5f:0x40 DW_TAG_subprogram
+ .long .Linfo_string_f # DW_AT_name
+ .byte 1 # DW_AT_decl_file
+ .byte 4 # DW_AT_decl_line
+ # DW_AT_prototyped
+ # DW_AT_external
+ .byte 1 # DW_AT_inline
+.Ldebug_info_param1:
+ .byte 6 # Abbrev [6] 0x67:0xb DW_TAG_formal_parameter
+ .long .Linfo_string_unused1 # DW_AT_name
+ .byte 1 # DW_AT_decl_file
+ .byte 4 # DW_AT_decl_line
+ .long .Ldebug_info_void_ptr-.Ldi_cu_begin
+ # DW_AT_type
+.Ldebug_info_param2:
+ .byte 6 # Abbrev [6] 0x72:0xb DW_TAG_formal_parameter
+ .long .Linfo_string_used # DW_AT_name
+ .byte 1 # DW_AT_decl_file
+ .byte 4 # DW_AT_decl_line
+ .long .Ldebug_info_int-.Ldi_cu_begin # DW_AT_type
+.Ldebug_info_param3:
+ .byte 6 # Abbrev [6] 0x7d:0xb DW_TAG_formal_parameter
+ .long .Linfo_string_unused2 # DW_AT_name
+ .byte 1 # DW_AT_decl_file
+ .byte 4 # DW_AT_decl_line
+ .long .Ldebug_info_int-.Ldi_cu_begin # DW_AT_type
+.Ldebug_info_param4:
+ .byte 6 # Abbrev [6] 0x88:0xb DW_TAG_formal_parameter
+ .long .Linfo_string_partial # DW_AT_name
+ .byte 1 # DW_AT_decl_file
+ .byte 4 # DW_AT_decl_line
+ .long .Ldebug_info_int-.Ldi_cu_begin # DW_AT_type
+.Ldebug_info_param5:
+ .byte 6 # Abbrev [6] 0x93:0xb DW_TAG_formal_parameter
+ .long .Linfo_string_unused3 # DW_AT_name
+ .byte 1 # DW_AT_decl_file
+ .byte 4 # DW_AT_decl_line
+ .long .Ldebug_info_int-.Ldi_cu_begin # DW_AT_type
+ .byte 0 # End Of Children Mark (DW_TAG_subprogram)
+
+# Debug info for |g| (abstract version with all parameters).
+
+.Ldebug_info_g:
+ .byte 5 # Abbrev [5] 0x5f:0x40 DW_TAG_subprogram
+ .long .Linfo_string_g # DW_AT_name
+ .byte 1 # DW_AT_decl_file
+ .byte 4 # DW_AT_decl_line
+ # DW_AT_prototyped
+ # DW_AT_external
+ .byte 1 # DW_AT_inline
+.Ldebug_info_g_param1:
+ .byte 6 # Abbrev [6] 0x67:0xb DW_TAG_formal_parameter
+ .long .Linfo_string_unused # DW_AT_name
+ .byte 1 # DW_AT_decl_file
+ .byte 10 # DW_AT_decl_line
+ .long .Ldebug_info_int-.Ldi_cu_begin
+ .byte 0 # End Of Children Mark (DW_TAG_subprogram)
+
+# Debug info for |main|.
+
+ .byte 9 # Abbrev [9] 0xa7:0x6e DW_TAG_subprogram
+ .quad main # DW_AT_low_pc
+ .long .Lmain_end-main # DW_AT_high_pc
+ .byte 1 # DW_AT_frame_base
+ .byte 87
+ .long .Linfo_string_main # DW_AT_name
+ .byte 1 # DW_AT_decl_file
+ .byte 18 # DW_AT_decl_line
+ # DW_AT_prototyped
+ .long .Ldebug_info_int-.Ldi_cu_begin # DW_AT_type
+ # DW_AT_external
+
+# Debug info for concrete |f| inlined into |main|.
+
+ .byte 11 # Abbrev [11] DW_TAG_inlined_subroutine
+ .long .Ldebug_info_f-.Ldi_cu_begin
+ # DW_AT_abstract_origin
+ .quad .Linlined_f # DW_AT_low_pc
+ .long .Linlined_f_end-.Linlined_f # DW_AT_high_pc
+ .byte 1 # DW_AT_call_file
+ .byte 20 # DW_AT_call_line
+ .byte 3 # DW_AT_call_column
+ .byte 4 # Abbrev [4] DW_TAG_formal_parameter
+ .long .Ldebug_loc_used # DW_AT_location
+ .long .Ldebug_info_param2-.Ldi_cu_begin
+ # DW_AT_abstract_origin
+ .byte 4 # Abbrev [4] DW_TAG_formal_parameter
+ .long .Ldebug_loc_partial # DW_AT_location
+ .long .Ldebug_info_param4-.Ldi_cu_begin
+ # DW_AT_abstract_origin
+ .byte 0 # End Of Children Mark (DW_TAG_inlined_subroutine)
+
+# Debug info for concrete |g| inlined into |main|.
+
+ .byte 11 # Abbrev [11] DW_TAG_inlined_subroutine
+ .long .Ldebug_info_g-.Ldi_cu_begin
+ # DW_AT_abstract_origin
+ .quad .Linlined_g # DW_AT_low_pc
+ .long .Linlined_g_end-.Linlined_g # DW_AT_high_pc
+ .byte 1 # DW_AT_call_file
+ .byte 21 # DW_AT_call_line
+ .byte 3 # DW_AT_call_column
+ .byte 0 # End Of Children Mark (DW_TAG_inlined_subroutine)
+
+ .byte 0 # End Of Children Mark (DW_TAG_subprogram)
+
+# Debug info for |other|.
+
+ .byte 9 # Abbrev [9] 0xa7:0x6e DW_TAG_subprogram
+ .quad other # DW_AT_low_pc
+ .long .Lother_end-other # DW_AT_high_pc
+ .byte 1 # DW_AT_frame_base
+ .byte 87
+ .long .Linfo_string_other # DW_AT_name
+ .byte 1 # DW_AT_decl_file
+ .byte 15 # DW_AT_decl_line
+ # DW_AT_prototyped
+ .long .Ldebug_info_int-.Ldi_cu_begin # DW_AT_type
+ # DW_AT_external
+
+# Debug info for concrete |f| inlined into |other|.
+
+ .byte 11 # Abbrev [11] DW_TAG_inlined_subroutine
+ .long .Ldebug_info_f-.Ldi_cu_begin
+ # DW_AT_abstract_origin
+ .quad .Linlined_f_in_other # DW_AT_low_pc
+ .long .Linlined_f_in_other_end-.Linlined_f_in_other
+ # DW_AT_high_pc
+ .byte 1 # DW_AT_call_file
+ .byte 16 # DW_AT_call_line
+ .byte 3 # DW_AT_call_column
+ .byte 4 # Abbrev [4] DW_TAG_formal_parameter
+ .long .Ldebug_loc_used_in_other # DW_AT_location
+ .long .Ldebug_info_param2-.Ldi_cu_begin
+ # DW_AT_abstract_origin
+ .byte 4 # Abbrev [4] DW_TAG_formal_parameter
+ .long .Ldebug_loc_partial_in_other # DW_AT_location
+ .long .Ldebug_info_param4-.Ldi_cu_begin
+ # DW_AT_abstract_origin
+ .byte 0 # End Of Children Mark (DW_TAG_inlined_subroutine)
+ .byte 0 # End Of Children Mark (DW_TAG_subprogram)
+
+.Ldebug_info_void_ptr:
+ .byte 7 # Abbrev [7] DW_TAG_pointer_type
+.Ldebug_info_int:
+ .byte 8 # Abbrev [8] DW_TAG_base_type
+ .long .Linfo_string_int # DW_AT_name
+ .byte 5 # DW_AT_encoding
+ .byte 4 # DW_AT_byte_size
+
+ .byte 0 # End Of Children Mark (DW_TAG_compile_unit)
+.Ldebug_info_end0:
+ .section .debug_str,"MS",@progbits,1
+.Linfo_string_fname:
+ .asciz "unused-inlined-params.c"
+.Linfo_string_f:
+ .asciz "f"
+.Linfo_string_unused1:
+ .asciz "unused1"
+.Linfo_string_used:
+ .asciz "used"
+.Linfo_string_int:
+ .asciz "int"
+.Linfo_string_unused2:
+ .asciz "unused2"
+.Linfo_string_partial:
+ .asciz "partial"
+.Linfo_string_unused3:
+ .asciz "unused3"
+.Linfo_string_main:
+ .asciz "main"
+.Linfo_string_g:
+ .asciz "g"
+.Linfo_string_unused:
+ .asciz "unused"
+.Linfo_string_other:
+ .asciz "other"
+ .section ".note.GNU-stack","",@progbits
+ .addrsig
+ .section .debug_line,"",@progbits
+.Lline_table_start0:
Index: lldb/source/Plugins/SymbolFile/DWARF/SymbolFileDWARF.h
===================================================================
--- lldb/source/Plugins/SymbolFile/DWARF/SymbolFileDWARF.h
+++ lldb/source/Plugins/SymbolFile/DWARF/SymbolFileDWARF.h
@@ -389,10 +389,11 @@
const DWARFDIE &die,
const lldb::addr_t func_low_pc);
+ using DIEVector = llvm::SmallVector<DWARFDIE, 2>;
size_t ParseVariablesInFunctionContextRecursive(
const lldb_private::SymbolContext &sc, const DWARFDIE &die,
- const lldb::addr_t func_low_pc,
- lldb_private::VariableList &variable_list);
+ const lldb::addr_t func_low_pc, lldb_private::VariableList &variable_list,
+ DIEVector &abstract_formal_parameters, size_t &abstract_parameter_index);
bool ClassOrStructIsVirtual(const DWARFDIE &die);
Index: lldb/source/Plugins/SymbolFile/DWARF/SymbolFileDWARF.cpp
===================================================================
--- lldb/source/Plugins/SymbolFile/DWARF/SymbolFileDWARF.cpp
+++ lldb/source/Plugins/SymbolFile/DWARF/SymbolFileDWARF.cpp
@@ -3498,7 +3498,9 @@
return;
}
- // We haven't already parsed it, lets do that now.
+ // We haven't parsed the variable yet, lets do that now. Also, let us include
+ // the variable in the relevant compilation unit's variable list, if it
+ // exists.
VariableListSP variable_list_sp;
DWARFDIE sc_parent_die = GetParentSymbolContextDIE(die);
dw_tag_t parent_tag = sc_parent_die.Tag();
@@ -3544,16 +3546,46 @@
sc.function->GetBlock(/*can_create=*/true).FindBlockByID(die.GetID());
const bool can_create = false;
VariableListSP variable_list_sp = block->GetBlockVariableList(can_create);
- return ParseVariablesInFunctionContextRecursive(sc, die, func_low_pc,
- *variable_list_sp);
+ DIEVector abstract_formal_parameters;
+ size_t abstract_parameter_index = 0;
+ return ParseVariablesInFunctionContextRecursive(
+ sc, die, func_low_pc, *variable_list_sp, abstract_formal_parameters,
+ abstract_parameter_index);
}
size_t SymbolFileDWARF::ParseVariablesInFunctionContextRecursive(
const lldb_private::SymbolContext &sc, const DWARFDIE &die,
- const lldb::addr_t func_low_pc, VariableList &variable_list) {
+ const lldb::addr_t func_low_pc, VariableList &variable_list,
+ DIEVector &abstract_formal_parameters, size_t &abstract_parameter_index) {
size_t vars_added = 0;
dw_tag_t tag = die.Tag();
+ // If we are parsing a formal parameter, let us make sure that we insert
+ // preceding abstract parameters if they were omitted from the concrete
+ // function instance.
+ if (tag == DW_TAG_formal_parameter) {
+ while (abstract_parameter_index < abstract_formal_parameters.size()) {
+ const DWARFDIE &abstract_parameter =
+ abstract_formal_parameters[abstract_parameter_index++];
+ DWARFDIE parameter = die;
+ bool found = false;
+ while (parameter) {
+ parameter = parameter.GetReferencedDIE(DW_AT_abstract_origin);
+ if (parameter == abstract_parameter) {
+ found = true;
+ break;
+ }
+ }
+ if (found)
+ break;
+ VariableSP var_sp(ParseVariableDIE(sc, abstract_parameter, func_low_pc));
+ if (var_sp) {
+ variable_list.AddVariableIfUnique(var_sp);
+ ++vars_added;
+ }
+ }
+ }
+
if ((tag == DW_TAG_variable) || (tag == DW_TAG_constant) ||
(tag == DW_TAG_formal_parameter)) {
VariableSP var_sp(ParseVariableDIE(sc, die, func_low_pc));
@@ -3592,10 +3624,43 @@
block_variable_list_sp = std::make_shared<VariableList>();
block->SetVariableList(block_variable_list_sp);
}
+
+ DIEVector block_abstract_formal_parameters;
+ if (tag == DW_TAG_inlined_subroutine) {
+ // Walk abstract origins until we find DW_TAG_subprogram and extract
+ // its formal parameters.
+ DWARFDIE abs_die = die;
+ while (abs_die && abs_die.Tag() != DW_TAG_subprogram) {
+ abs_die = abs_die.GetReferencedDIE(DW_AT_abstract_origin);
+ }
+ if (abs_die && abs_die.HasChildren()) {
+ for (DWARFDIE child = abs_die.GetFirstChild(); child;
+ child = child.GetSibling()) {
+ if (child.Tag() == DW_TAG_formal_parameter) {
+ block_abstract_formal_parameters.push_back(child);
+ }
+ }
+ }
+ }
+
+ size_t block_parameter_index = 0;
for (DWARFDIE child = die.GetFirstChild(); child;
child = child.GetSibling()) {
vars_added += ParseVariablesInFunctionContextRecursive(
- sc, child, func_low_pc, *block_variable_list_sp);
+ sc, child, func_low_pc, *block_variable_list_sp,
+ block_abstract_formal_parameters, block_parameter_index);
+ }
+
+ // If there any remaining abstract formal parameters, parse and insert
+ // them as well.
+ while (block_parameter_index < block_abstract_formal_parameters.size()) {
+ const DWARFDIE &abstract_parameter =
+ block_abstract_formal_parameters[block_parameter_index++];
+ VariableSP var_sp(ParseVariableDIE(sc, abstract_parameter, func_low_pc));
+ if (var_sp) {
+ block_variable_list_sp->AddVariableIfUnique(var_sp);
+ ++vars_added;
+ }
}
break;
@@ -3606,7 +3671,8 @@
for (DWARFDIE child = die.GetFirstChild(); child;
child = child.GetSibling()) {
vars_added += ParseVariablesInFunctionContextRecursive(
- sc, child, func_low_pc, variable_list);
+ sc, child, func_low_pc, variable_list, abstract_formal_parameters,
+ abstract_parameter_index);
}
break;
_______________________________________________
lldb-commits mailing list
[email protected]
https://lists.llvm.org/cgi-bin/mailman/listinfo/lldb-commits