[gdb/symtab] Fix gdb.base/fission-macro.exp with unix/-m32
[binutils-gdb.git] / gdb / testsuite / gdb.threads / process-dies-while-detaching.exp
blob5c54a36a582a7e8f740486fb14d342c891cd7102
1 # Copyright 2016-2024 Free Software Foundation, Inc.
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.
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.
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/>.
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.
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.
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.
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.
45 standard_testfile
47 # Test that GDBserver exits.
49 proc test_server_exit {} {
50     global server_spawn_id
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     }
66 # If RESULT is not zero, make the caller return.
68 proc return_if_fail { result } {
69     if {$result != 0} {
70         return -code return
71     }
74 # Detach from a process, and ensure that it exits after detaching.
75 # This relies on inferior I/O.  INF_OUTPUT_RE is the pattern that
76 # matches the expected inferior output.
78 proc detach_and_expect_exit {inf_output_re test} {
79     global decimal
80     global gdb_spawn_id
81     global inferior_spawn_id
82     global gdb_prompt
84     return_if_fail [gdb_test_multiple "detach" $test {
85         -re "Detaching from .*, process $decimal" {
86         }
87     }]
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"
97     set saw_prompt 0
98     set saw_inf_exit 0
99     while { !$saw_prompt || ! $saw_inf_exit } {
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 {
107             -i daee_spawn_id_list
108             -re "($inf_output_re)|($gdb_prompt )" {
109                 if {[info exists expect_out(1,string)]} {
110                     verbose -log "saw inferior exit"
111                     set saw_inf_exit 1
112                     set daee_spawn_id_list "$gdb_spawn_id"
113                 } elseif {[info exists expect_out(2,string)]} {
114                     verbose -log "saw prompt"
115                     set saw_prompt 1
116                     set daee_spawn_id_list "$inferior_spawn_id"
117                 }
118                 array unset expect_out
119             }
120         }]
121     }
123     pass $test
126 # Run to _exit in the child.
128 proc continue_to_exit_bp {} {
129     gdb_breakpoint "_exit" temporary
130     return [gdb_continue_to_breakpoint "_exit" ".*_exit.*"]
133 # If testing single-process, simply detach from the process.
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.
140 # If connected with "target remote", make sure gdbserver exits.
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
144 # exit.
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.
150 proc do_detach {multi_process cmd child_exit} {
151     global decimal
152     global server_spawn_id
154     if {$child_exit == "normal"} {
155         set continue_re "exited normally.*"
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"
166     }
168     set is_remote [expr {[target_info exists gdb_protocol]
169                          && [target_info gdb_protocol] == "remote"}]
171     if {$multi_process} {
172         gdb_test "detach" "Detaching from .*, process $decimal\r\n\\\[Inferior $decimal \\(.*\\) detached\\\]" \
173             "detach child"
175         gdb_test "inferior 1" "\[Switching to inferior $decimal\].*" \
176             "switch to parent"
178         if {$cmd == "detach"} {
179             # Make sure that detach works and that the parent process
180             # exits cleanly.
181             detach_and_expect_exit $inf_output_re "detach parent"
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"} {
196             gdb_test "detach" "Detaching from .*, process ${decimal}\r\n\\\[Inferior $decimal \\(.*\\) detached\\\]$extra"
197         } elseif {$cmd == "continue"} {
198             gdb_test "continue" $continue_re
199         } else {
200             perror "unhandled command: $cmd"
201         }
202     }
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     }
211 # Test detaching from a process that dies just while GDB is detaching.
213 proc test_detach {multi_process cmd} {
214     with_test_prefix "detach" {
215         global binfile
217         clean_restart ${binfile}
219         if ![runto_main] {
220             return -1
221         }
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         }
228         # Run to _exit in the child.
229         return_if_fail [continue_to_exit_bp]
231         do_detach $multi_process $cmd "normal"
232     }
235 # Same as test_detach, except set a watchpoint before detaching.
237 proc test_detach_watch {wp multi_process cmd} {
238     if { $wp == "hw" && ![allow_hw_watchpoint_tests] } {
239         unsupported "hw watchpoint"
240         return
241     }
242     with_test_prefix "watchpoint:$wp" {
243         global binfile decimal
245         clean_restart ${binfile}
247         if ![runto_main] {
248             return -1
249         }
251         if {$multi_process} {
252             gdb_test_no_output "set detach-on-fork off"
253             gdb_test_no_output "set follow-fork-mode child"
255             gdb_breakpoint "child_function" temporary
256             gdb_continue_to_breakpoint "child_function" ".*"
257         }
259         if { $wp == "hw" } {
260             # Set a watchpoint in the child.
261             gdb_test "watch globalvar" ".* watchpoint $decimal: globalvar"
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).
268             return_if_fail [continue_to_exit_bp]
269         } else {
270             # Force software watchpoints.
271             gdb_test_no_output "set can-use-hw-watchpoints 0"
273             # As above, but flip order, other wise things take too long.
274             return_if_fail [continue_to_exit_bp]
275             gdb_test "watch globalvar" "Watchpoint $decimal: globalvar"
277             if { $multi_process == 0 && $cmd == "continue" } {
278                 setup_kfail "gdb/28375" "*-*-*"
279             }
280         }
282         do_detach $multi_process $cmd "normal"
283     }
286 # Test detaching from a process that dies _before_ GDB starts
287 # detaching.
289 proc test_detach_killed_outside {multi_process cmd} {
290     with_test_prefix "killed outside" {
291         global binfile
293         clean_restart ${binfile}
295         if ![runto_main] {
296             return -1
297         }
299         gdb_test_no_output "set breakpoint always-inserted on"
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         }
306         # Run to _exit in the child.
307         return_if_fail [continue_to_exit_bp]
309         set childpid [get_integer_valueof "mypid" -1]
310         if { $childpid == -1 } {
311             untested "failed to extract child pid"
312             return -1
313         }
315         remote_exec target "kill -9 ${childpid}"
317         # Give it some time to die.
318         sleep 2
320         do_detach $multi_process $cmd "signal"
321     }
324 # The test proper.  MULTI_PROCESS is true if testing the multi-process
325 # variant.
327 proc do_test {multi_process cmd} {
328     global testfile srcfile binfile
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     }
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     }
343     if {[build_executable "failed to build" \
344              $testfile-$multi_process-$cmd $srcfile $options] == -1} {
345         return -1
346     }
348     test_detach $multi_process $cmd
349     foreach wp {"sw" "hw"} {
350         test_detach_watch $wp $multi_process $cmd
351     }
352     test_detach_killed_outside $multi_process $cmd
355 foreach multi_process {0 1} {
356     set mode [expr {$multi_process ? "multi-process" : "single-process"}]
357     foreach cmd {"detach" "continue"} {
358         with_test_prefix "$mode: $cmd" {
359             do_test $multi_process $cmd
360         }
361     }