blob: 77618dea82436066ee3323713488a3b93ff881d4 (
plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
|
# Copyright 2023-2024 Free Software Foundation, Inc.
# This program is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation; either version 3 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program. If not, see <http://www.gnu.org/licenses/>.
# Create a compilation unit containing two functions (the CU's
# low/high pc range will include both functions), but define a line
# table that only covers one of the functions.
#
# Use GDB to try and place a breakpoint in both functions. GDB should
# correctly find some address within both functions at which to place
# a breakpoint.
load_lib dwarf.exp
# This test can only be run on targets which support the DWARF
# assembler.
require dwarf2_support
standard_testfile .c -dw.S
get_func_info main
get_func_info foo
set asm_file [standard_output_file $srcfile2]
Dwarf::assemble $asm_file {
declare_labels Llines
cu {} {
compile_unit {
{language @DW_LANG_C}
{name missing-line-table.c}
{stmt_list $Llines DW_FORM_sec_offset}
{DW_AT_low_pc $::foo_start DW_FORM_addr}
{DW_AT_high_pc $::main_end DW_FORM_addr}
} {
subprogram {
{external 1 flag}
{MACRO_AT_func {main}}
}
subprogram {
{external 1 flag}
{MACRO_AT_func {foo}}
}
}
}
lines {version 2} Llines {
include_dir "${::srcdir}/${::subdir}"
file_name "$::srcfile" 1
program {
DW_LNE_set_address main
line [gdb_get_line_number "L1"]
DW_LNS_copy
DW_LNE_set_address main_label
line [gdb_get_line_number "L2"]
DW_LNS_copy
DW_LNE_set_address main_label_2
line [gdb_get_line_number "L3"]
DW_LNS_copy
DW_LNE_set_address "$::main_start + $::main_len"
DW_LNE_end_sequence
}
}
}
if { [prepare_for_testing "failed to prepare" ${testfile} \
[list $srcfile $asm_file] {nodebug}] } {
return -1
}
# Place a breakpoint on FUNC. Check that the address at which the
# breakpoint is placed lies within FUNC which is starts at address
# START, and address END is the first address after FUNC.
proc check_breakpoint { func start end } {
with_test_prefix "func=$func" {
set addr "*unknown*"
gdb_test_multiple "break $func" "place breakpoint" {
-re -wrap "Breakpoint $::decimal at ($::hex)(?:: \[^\r\n\]+)?" {
set addr $expect_out(1,string)
pass $gdb_test_name
}
}
verbose -log "breakpoint at: $addr"
verbose -log "$func start: $start, $func end: $end"
gdb_assert {$addr >= $start && $addr < $end} \
"check breakpoint address is within function"
}
}
# The get_func_info calls declare the function start/end in terms of a
# program label and offsets, but we now need these as actual
# addresses. As the program is running, this can be done easily by
# asking GDB to evaluate each in turn.
foreach func {foo main} {
foreach part {start end} {
set ${func}_${part} \
[get_hexadecimal_valueof [set ${func}_${part}] "??" \
"get value of ${func}_${part}"]
}
}
# Place breakpoints within 'foo' and 'main', and check that the
# breakpoint is actually placed within the function.
check_breakpoint "foo" $foo_start $foo_end
check_breakpoint "main" $main_start $main_end
|