]> git.ipfire.org Git - thirdparty/binutils-gdb.git/blob - gdb/testsuite/gdb.base/attach-pie-misread.exp
run copyright.sh for 2011.
[thirdparty/binutils-gdb.git] / gdb / testsuite / gdb.base / attach-pie-misread.exp
1 # Copyright 2010, 2011 Free Software Foundation, Inc.
2
3 # This program is free software; you can redistribute it and/or modify
4 # it under the terms of the GNU General Public License as published by
5 # the Free Software Foundation; either version 3 of the License, or
6 # (at your option) any later version.
7 #
8 # This program is distributed in the hope that it will be useful,
9 # but WITHOUT ANY WARRANTY; without even the implied warranty of
10 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11 # GNU General Public License for more details.
12 #
13 # You should have received a copy of the GNU General Public License
14 # along with this program. If not, see <http://www.gnu.org/licenses/>.
15
16 # This test only works on GNU/Linux.
17 if { ![isnative] || [is_remote host] || ![istarget *-linux*] || [skip_shlib_tests]} {
18 continue
19 }
20
21 load_lib prelink-support.exp
22
23 set test "attach-pie-misread"
24 set srcfile ${test}.c
25 set genfile ${objdir}/${subdir}/${test}-gen.h
26 set executable ${test}
27 set binfile ${objdir}/${subdir}/${executable}
28
29 if {[build_executable_own_libs ${test}.exp $executable $srcfile [list "additional_flags=-fPIE -pie"]] == ""} {
30 return -1
31 }
32
33 # Program Headers:
34 # Type Offset VirtAddr PhysAddr FileSiz MemSiz Flg Align
35 # LOAD 0x000000 0x0000000000400000 0x0000000000400000 0x134f5ec 0x134f5ec R E 0x200000
36 # LOAD 0x134f5f0 0x000000000194f5f0 0x000000000194f5f0 0x1dbc60 0x214088 RW 0x200000
37 # DYNAMIC 0x134f618 0x000000000194f618 0x000000000194f618 0x000200 0x000200 RW 0x8
38 #
39 proc read_phdr {binfile test} {
40 set readelf_program [transform readelf]
41 set command "exec $readelf_program -Wl $binfile"
42 verbose -log "command is $command"
43 set result [catch $command output]
44 verbose -log "result is $result"
45 verbose -log "output is $output"
46 if {$result != 0} {
47 fail $test
48 return
49 }
50 if ![regexp {\nProgram Headers:\n *Type [^\n]* Align\n(.*?)\n\n} $output trash phdr] {
51 fail "$test (no Program Headers)"
52 return
53 }
54 if ![regexp -line {^ *DYNAMIC +0x[0-9a-f]+ +(0x[0-9a-f]+) } $phdr trash dynamic_vaddr] {
55 fail "$test (no DYNAMIC found)"
56 return
57 }
58 verbose -log "dynamic_vaddr is $dynamic_vaddr"
59 set align_max -1
60 foreach {trash align} [regexp -line -all -inline {^ *LOAD .* (0x[0-9]+)$} $phdr] {
61 if {$align_max < $align} {
62 set align_max $align
63 }
64 }
65 verbose -log "align_max is $align_max"
66 if {$align_max == -1} {
67 fail "$test (no LOAD found)"
68 return
69 }
70 pass $test
71 return [list $dynamic_vaddr $align_max]
72 }
73
74 set phdr [read_phdr $binfile "readelf initial scan"]
75 set dynamic_vaddr [lindex $phdr 0]
76 set align_max [lindex $phdr 1]
77
78 set stub_size [format 0x%x [expr "2 * $align_max - ($dynamic_vaddr & ($align_max - 1))"]]
79 verbose -log "stub_size is $stub_size"
80
81 # On x86_64 it is commonly about 4MB.
82 if {$stub_size > 25000000} {
83 xfail "stub size $stub_size is too large"
84 return
85 }
86
87 set test "generate stub"
88 set command "exec $binfile $stub_size >$genfile"
89 verbose -log "command is $command"
90 set result [catch $command output]
91 verbose -log "result is $result"
92 verbose -log "output is $output"
93 if {$result == 0} {
94 pass $test
95 } else {
96 fail $test
97 }
98
99 set prelink_args [build_executable_own_libs ${test}.exp $executable $srcfile [list "additional_flags=-fPIE -pie -DGEN=\"$genfile\""]]
100 if {$prelink_args == ""} {
101 return -1
102 }
103
104 # x86_64 file has 25MB, no need to keep it.
105 file delete -- $genfile
106
107 set phdr [read_phdr $binfile "readelf rebuilt with stub_size"]
108 set dynamic_vaddr_prelinkno [lindex $phdr 0]
109
110 if ![prelink_yes $prelink_args] {
111 return -1
112 }
113
114 set phdr [read_phdr $binfile "readelf with prelink -R"]
115 set dynamic_vaddr_prelinkyes [lindex $phdr 0]
116
117 set first_offset [format 0x%x [expr $dynamic_vaddr_prelinkyes - $dynamic_vaddr_prelinkno]]
118 verbose -log "first_offset is $first_offset"
119
120 set test "first offset is non-zero"
121 if {$first_offset == 0} {
122 fail "$test (-fPIE -pie in effect?)"
123 } else {
124 pass $test
125 }
126
127 set test "start inferior"
128 gdb_exit
129
130 set res [remote_spawn host $binfile];
131 if { $res < 0 || $res == "" } {
132 perror "Spawning $binfile failed."
133 fail $test
134 return
135 }
136 set pid [exp_pid -i $res]
137 gdb_expect {
138 -re "sleeping\r\n" {
139 pass $test
140 }
141 eof {
142 fail "$test (eof)"
143 remote_exec host "kill -9 $pid"
144 return
145 }
146 timeout {
147 fail "$test (timeout)"
148 remote_exec host "kill -9 $pid"
149 return
150 }
151 }
152
153 # Due to alignments it was reproducible with 1 on x86_64 but 2 on i686.
154 foreach align_mult {1 2} {
155 set old_ldprefix $pf_prefix
156 lappend pf_prefix "shift-by-$align_mult:"
157
158 # FIXME: We believe there is enough room under FIRST_OFFSET.
159 set shifted_offset [format 0x%x [expr "$first_offset - $align_mult * $align_max"]]
160 verbose -log "shifted_offset is $shifted_offset"
161
162 # For normal prelink (prelink_yes call), we need to supply $prelink_args.
163 # For the prelink `-r' option below, $prelink_args is not required.
164 # Moreover, if it was used, the problem would not longer be reproducible
165 # as the libraries would also get relocated.
166 set command "exec /usr/sbin/prelink -q -N --no-exec-shield -r $shifted_offset $binfile"
167 verbose -log "command is $command"
168 set result [catch $command output]
169 verbose -log "result is $result"
170 verbose -log "output is $output"
171
172 set test "prelink -r"
173 if {$result == 0 && $output == ""} {
174 pass $test
175 } else {
176 fail $test
177 }
178
179 clean_restart $executable
180
181 set test "attach"
182 gdb_test_multiple "attach $pid" $test {
183 -re "Attaching to program: .*, process $pid\r\n" {
184 # Missing "$gdb_prompt $" is intentional.
185 pass $test
186 }
187 }
188
189 set test "error on Cannot access memory at address"
190 gdb_test_multiple "" $test {
191 -re "\r\nCannot access memory at address .*$gdb_prompt $" {
192 fail $test
193 }
194 -re "$gdb_prompt $" {
195 pass $test
196 }
197 }
198
199 gdb_test "detach" "Detaching from program: .*"
200
201 set pf_prefix $old_ldprefix
202 }
203
204 remote_exec host "kill -9 $pid"