]> git.ipfire.org Git - thirdparty/binutils-gdb.git/blame - gdb/testsuite/gdb.threads/process-dies-while-detaching.exp
Update copyright year range in header of all files managed by GDB
[thirdparty/binutils-gdb.git] / gdb / testsuite / gdb.threads / process-dies-while-detaching.exp
CommitLineData
1d506c26 1# Copyright 2016-2024 Free Software Foundation, Inc.
ced2dffb
PA
2# This program is free software; you can redistribute it and/or modify
3# it under the terms of the GNU General Public License as published by
4# the Free Software Foundation; either version 3 of the License, or
5# (at your option) any later version.
6#
7# This program is distributed in the hope that it will be useful,
8# but WITHOUT ANY WARRANTY; without even the implied warranty of
9# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
10# GNU General Public License for more details.
11#
12# You should have received a copy of the GNU General Public License
13# along with this program. If not, see <http://www.gnu.org/licenses/>.
14
15# This test spawns a few threads that immediately exit the whole
16# process. On targets where the debugger needs to detach from each
17# thread individually (such as on the Linux kernel), the debugger must
18# handle the case of the process exiting while the detach is ongoing.
19#
20# Similarly, the process can also be killed from outside the debugger
21# (e.g., with SIGKILL), _before_ the user requests a detach. The
22# debugger must likewise detach gracefully.
23#
24# The testcase actually builds two variants of the test program:
25# single-process, and multi-process. In the multi-process variant,
26# the test program forks, and it's the fork child that spawns threads
27# that exit just while the process is being detached from. The fork
28# parent waits for its child to exit, so if GDB fails to detach from
29# the child correctly, the parent hangs. Because continuing the
30# parent can mask failure to detach from the child correctly (e.g.,
31# due to waitpid(-1,...) calls deep in the target layers managing to
32# reap the child), we try immediately detaching from the parent too,
33# and observing whether the parent exits via standard output.
34#
35# Normally, if testing with "target remote" against gdbserver, then
36# after detaching from all attached processes, gdbserver exits.
37# However, when gdbserver detaches from a process that is its own
38# direct child, gdbserver does not exit immediately. Instead it
39# "joins" (waits for) the child, only exiting when the child itself
40# exits too. Thus, on Linux, if gdbserver fails to detach from the
41# zombie child's threads correctly (or rather, reap them), it'll hang,
42# because the leader thread will only return an exit status after all
43# threads are reaped. We test that as well.
44
45standard_testfile
46
47# Test that GDBserver exits.
48
49proc test_server_exit {} {
50 global server_spawn_id
51
52 set test "server exits"
53 gdb_expect {
54 -i $server_spawn_id
55 eof {
56 pass $test
57 wait -i $server_spawn_id
58 unset server_spawn_id
59 }
60 timeout {
61 fail "$test (timeout)"
62 }
63 }
64}
65
66# If RESULT is not zero, make the caller return.
67
68proc return_if_fail { result } {
69 if {$result != 0} {
70 return -code return
71 }
72}
73
74# Detach from a process, and ensure that it exits after detaching.
f0fb2488
PA
75# This relies on inferior I/O. INF_OUTPUT_RE is the pattern that
76# matches the expected inferior output.
ced2dffb 77
f0fb2488 78proc detach_and_expect_exit {inf_output_re test} {
ced2dffb
PA
79 global decimal
80 global gdb_spawn_id
81 global inferior_spawn_id
82 global gdb_prompt
83
84 return_if_fail [gdb_test_multiple "detach" $test {
85 -re "Detaching from .*, process $decimal" {
86 }
87 }]
88
f0fb2488
PA
89 # Use an indirect spawn id list, and remove inferior spawn id from
90 # the expected output as soon as it matches, so that if
91 # $inf_inferior_spawn_id is $server_spawn_id and we're testing in
92 # "target remote" mode, the eof caused by gdbserver exiting is
93 # left for the caller to handle.
94 global daee_spawn_id_list
95 set daee_spawn_id_list "$inferior_spawn_id $gdb_spawn_id"
96
ced2dffb
PA
97 set saw_prompt 0
98 set saw_inf_exit 0
f0fb2488 99 while { !$saw_prompt || ! $saw_inf_exit } {
ced2dffb
PA
100 # We don't know what order the interesting things will arrive in.
101 # Using a pattern of the form 'x|y|z' instead of -re x ... -re y
102 # ... -re z ensures that expect always chooses the match that
103 # occurs leftmost in the input, and not the pattern appearing
104 # first in the script that occurs anywhere in the input, so that
105 # we don't skip anything.
106 return_if_fail [gdb_test_multiple "" $test {
f0fb2488
PA
107 -i daee_spawn_id_list
108 -re "($inf_output_re)|($gdb_prompt )" {
ced2dffb
PA
109 if {[info exists expect_out(1,string)]} {
110 verbose -log "saw inferior exit"
111 set saw_inf_exit 1
f0fb2488 112 set daee_spawn_id_list "$gdb_spawn_id"
ced2dffb
PA
113 } elseif {[info exists expect_out(2,string)]} {
114 verbose -log "saw prompt"
115 set saw_prompt 1
f0fb2488 116 set daee_spawn_id_list "$inferior_spawn_id"
ced2dffb
PA
117 }
118 array unset expect_out
119 }
120 }]
121 }
122
123 pass $test
124}
125
126# Run to _exit in the child.
127
128proc continue_to_exit_bp {} {
129 gdb_breakpoint "_exit" temporary
0f3efefb 130 return [gdb_continue_to_breakpoint "_exit" ".*_exit.*"]
ced2dffb
PA
131}
132
133# If testing single-process, simply detach from the process.
134#
135# If testing multi-process, first detach from the child, then detach
136# from the parent and confirm that the parent exits, thus ensuring
137# we've detached from the child successfully, as the parent hangs in
138# its waitpid call otherwise.
139#
140# If connected with "target remote", make sure gdbserver exits.
141#
142# CMD indicates what to do with the parent after detaching the child.
143# Can be either "detach" to detach, or "continue", to continue to
f0fb2488
PA
144# exit.
145#
146# CHILD_EXIT indicates how is the child expected to exit. Can be
147# either "normal" for normal exit, or "signal" for killed with signal
148# SIGKILL.
ced2dffb 149#
f0fb2488 150proc do_detach {multi_process cmd child_exit} {
ced2dffb
PA
151 global decimal
152 global server_spawn_id
153
f0fb2488 154 if {$child_exit == "normal"} {
ced2dffb 155 set continue_re "exited normally.*"
f0fb2488
PA
156 set inf_output_re "exited, status=0"
157 } elseif {$child_exit == "signal"} {
158 if {$multi_process} {
159 set continue_re "exited with code 02.*"
160 } else {
161 set continue_re "terminated with signal SIGKILL.*"
162 }
163 set inf_output_re "signaled, sig=9"
164 } else {
165 error "unhandled \$child_exit: $child_exit"
ced2dffb
PA
166 }
167
168 set is_remote [expr {[target_info exists gdb_protocol]
169 && [target_info gdb_protocol] == "remote"}]
170
171 if {$multi_process} {
f67c0c91 172 gdb_test "detach" "Detaching from .*, process $decimal\r\n\\\[Inferior $decimal \\(.*\\) detached\\\]" \
ced2dffb
PA
173 "detach child"
174
175 gdb_test "inferior 1" "\[Switching to inferior $decimal\].*" \
176 "switch to parent"
177
178 if {$cmd == "detach"} {
179 # Make sure that detach works and that the parent process
180 # exits cleanly.
f0fb2488 181 detach_and_expect_exit $inf_output_re "detach parent"
ced2dffb
PA
182 } elseif {$cmd == "continue"} {
183 # Make sure that continuing works and that the parent process
184 # exits cleanly.
185 gdb_test "continue" $continue_re
186 } else {
187 perror "unhandled command: $cmd"
188 }
189 } else {
190 if $is_remote {
191 set extra "\r\nEnding remote debugging\."
192 } else {
193 set extra ""
194 }
195 if {$cmd == "detach"} {
f67c0c91 196 gdb_test "detach" "Detaching from .*, process ${decimal}\r\n\\\[Inferior $decimal \\(.*\\) detached\\\]$extra"
ced2dffb
PA
197 } elseif {$cmd == "continue"} {
198 gdb_test "continue" $continue_re
199 } else {
200 perror "unhandled command: $cmd"
201 }
202 }
203
204 # When connected in "target remote" mode, the server should exit
205 # when there are no processes left to debug.
206 if { $is_remote && [info exists server_spawn_id]} {
207 test_server_exit
208 }
209}
210
211# Test detaching from a process that dies just while GDB is detaching.
212
213proc test_detach {multi_process cmd} {
214 with_test_prefix "detach" {
215 global binfile
216
217 clean_restart ${binfile}
218
219 if ![runto_main] {
ced2dffb
PA
220 return -1
221 }
222
223 if {$multi_process} {
224 gdb_test_no_output "set detach-on-fork off"
225 gdb_test_no_output "set follow-fork-mode child"
226 }
227
228 # Run to _exit in the child.
0f3efefb 229 return_if_fail [continue_to_exit_bp]
ced2dffb 230
f0fb2488 231 do_detach $multi_process $cmd "normal"
ced2dffb
PA
232 }
233}
234
235# Same as test_detach, except set a watchpoint before detaching.
236
203a9824 237proc test_detach_watch {wp multi_process cmd} {
e379cbb1 238 if { $wp == "hw" && ![allow_hw_watchpoint_tests] } {
203a9824
TV
239 unsupported "hw watchpoint"
240 return
241 }
242 with_test_prefix "watchpoint:$wp" {
ced2dffb
PA
243 global binfile decimal
244
245 clean_restart ${binfile}
246
247 if ![runto_main] {
ced2dffb
PA
248 return -1
249 }
250
251 if {$multi_process} {
252 gdb_test_no_output "set detach-on-fork off"
253 gdb_test_no_output "set follow-fork-mode child"
254
255 gdb_breakpoint "child_function" temporary
256 gdb_continue_to_breakpoint "child_function" ".*"
257 }
258
203a9824
TV
259 if { $wp == "hw" } {
260 # Set a watchpoint in the child.
261 gdb_test "watch globalvar" ".* watchpoint $decimal: globalvar"
ced2dffb 262
203a9824
TV
263 # Continue to the _exit breakpoint. This arms the watchpoint
264 # registers in all threads. Detaching will thus need to clear
265 # them out, and handle the case of the thread disappearing
266 # while doing that (on targets that need to detach from each
267 # thread individually).
0f3efefb 268 return_if_fail [continue_to_exit_bp]
203a9824
TV
269 } else {
270 # Force software watchpoints.
271 gdb_test_no_output "set can-use-hw-watchpoints 0"
272
273 # As above, but flip order, other wise things take too long.
0f3efefb 274 return_if_fail [continue_to_exit_bp]
203a9824
TV
275 gdb_test "watch globalvar" "Watchpoint $decimal: globalvar"
276
277 if { $multi_process == 0 && $cmd == "continue" } {
278 setup_kfail "gdb/28375" "*-*-*"
279 }
280 }
ced2dffb 281
f0fb2488 282 do_detach $multi_process $cmd "normal"
ced2dffb
PA
283 }
284}
285
286# Test detaching from a process that dies _before_ GDB starts
287# detaching.
288
289proc test_detach_killed_outside {multi_process cmd} {
290 with_test_prefix "killed outside" {
291 global binfile
292
293 clean_restart ${binfile}
294
295 if ![runto_main] {
ced2dffb
PA
296 return -1
297 }
298
299 gdb_test_no_output "set breakpoint always-inserted on"
300
301 if {$multi_process} {
302 gdb_test_no_output "set detach-on-fork off"
303 gdb_test_no_output "set follow-fork-mode child"
304 }
305
306 # Run to _exit in the child.
0f3efefb 307 return_if_fail [continue_to_exit_bp]
ced2dffb
PA
308
309 set childpid [get_integer_valueof "mypid" -1]
310 if { $childpid == -1 } {
311 untested "failed to extract child pid"
312 return -1
313 }
314
315 remote_exec target "kill -9 ${childpid}"
316
317 # Give it some time to die.
318 sleep 2
319
f0fb2488 320 do_detach $multi_process $cmd "signal"
ced2dffb
PA
321 }
322}
323
324# The test proper. MULTI_PROCESS is true if testing the multi-process
325# variant.
326
327proc do_test {multi_process cmd} {
328 global testfile srcfile binfile
329
330 if {$multi_process && $cmd == "detach"
331 && [target_info exists gdb,noinferiorio]} {
332 # This requires inferior I/O to tell whether both the parent
333 # and child exit successfully.
334 return
335 }
336
337 set binfile [standard_output_file ${testfile}-$multi_process-$cmd]
338 set options {debug pthreads}
339 if {$multi_process} {
340 lappend options "additional_flags=-DMULTIPROCESS"
341 }
342
343 if {[build_executable "failed to build" \
344 $testfile-$multi_process-$cmd $srcfile $options] == -1} {
345 return -1
346 }
347
348 test_detach $multi_process $cmd
203a9824
TV
349 foreach wp {"sw" "hw"} {
350 test_detach_watch $wp $multi_process $cmd
351 }
ced2dffb
PA
352 test_detach_killed_outside $multi_process $cmd
353}
354
355foreach multi_process {0 1} {
41bfcd63 356 set mode [expr {$multi_process ? "multi-process" : "single-process"}]
ced2dffb
PA
357 foreach cmd {"detach" "continue"} {
358 with_test_prefix "$mode: $cmd" {
359 do_test $multi_process $cmd
360 }
361 }
362}