mirror of
https://sourceware.org/git/binutils-gdb.git
synced 2025-01-24 12:35:55 +08:00
f67c0c9171
This patch aims to turn 'set print inferior-events' always on, and do some cleanup on the messages printed by GDB when various inferior events happen (attach, detach, fork, kill, exit). To make sure that the patch is correct, I've tested it with a handful of combinations of 'set follow-fork-mode', 'set detach-on-fork' and 'set print inferior-events'. In the end, I decided to make my hand-made test into an official testcase. More on that below. Using the following program as an example: #include <unistd.h> int main () { fork (); return 0; } We see the following outputs from the patched GDB: - With 'set print inferior-events on': (gdb) r Starting program: a.out [Detaching after fork from child process 27749] [Inferior 1 (process 27745) exited normally] (gdb) - With 'set print inferior-events off': (gdb) r Starting program: a.out [Inferior 1 (process 27823) exited normally] (gdb) Comparing this against an unpatched GDB: - With 'set print inferior-events off' and 'set follow-fork-mode child': (gdb) r Starting program: a.out [Inferior 2 (process 5993) exited normally] (gdb) Compare this against an unpatched GDB: (unpatched-gdb) r Starting program: a.out [New process 5702] [Inferior 2 (process 5702) exited normally] (unpatched-gdb) It is possible to notice that, in this scenario, the patched GDB will lose the '[New process %d]' message. - With 'set print inferior-events on', 'set follow-fork-mode child' and 'set detach-on-fork on': (gdb) r Starting program: a.out [Attaching after process 27905 fork to child process 27909] [New inferior 2 (process 27909)] [Detaching after fork from parent process 27905] [Inferior 1 (process 27905) detached] [Inferior 2 (process 27909) exited normally] (gdb) Compare this output with an unpatched GDB, using the same settings: (unpatched-gdb) r Starting program: a.out [New inferior 28033] [Inferior 28029 detached] [New process 28033] [Inferior 2 (process 28033) exited normally] [Inferior 28033 exited] (unpatched-gdb) As can be seen above, I've also made a few modifications to messages that are printed when 'set print inferior-events' is on. For example, a few of the messages did not contain the '[' and ']' as prefix/suffix, which led to a few inconsistencies like: Attaching after process 22995 fork to child process 22999. [New inferior 22999] Detaching after fork from child process 22999. [Inferior 22995 detached] [Inferior 2 (process 22999) exited normally] So I took the opportunity and included the square brackets where applicable. I have also made the existing messages more uniform, by always printing "Inferior %d (process %d)..." where applicable. This makes it easier to identify the inferior number and the PID number from the messages. As suggested by Pedro, the "[Inferior %d exited]" message from 'exit_inferior' has been removed, because it got duplicated when 'inferior-events' is on. I'm also using the 'add_{thread,inferior}_silent' versions (instead of their verbose counterparts) on some locations, also to avoid duplicated messages. For example, a patched GDB with 'set print inferior-events on', 'set detach-on-fork on' and 'set follow-fork-mode child', but using 'add_thread', would print: (gdb) run Starting program: a.out [Attaching after process 25088 fork to child process 25092.] [New inferior 25092] <--- duplicated [Detaching after fork from child process 25092.] [Inferior 25088 detached] [New process 25092] <--- duplicated [Inferior 2 (process 25092) exited normally] But if we use 'add_thread_silent' (with the same configuration as before): (gdb) run Starting program: a.out [Attaching after process 31606 fork to child process 31610] [New inferior 2 (process 31610)] [Detaching after fork from parent process 31606] [Inferior 1 (process 31606) detached] [Inferior 2 (process 31610) exited normally] As for the tests, the configuration options being exercised are: - follow-fork-mode: child/parent - detach-on-fork: on/off - print inferior-events: on/off It was also necessary to perform adjustments on several testcases, because the expected messages changed considerably. Built and regtested on BuildBot, without regressions. gdb/ChangeLog: 2018-04-24 Jan Kratochvil <jan.kratochvil@redhat.com> Sergio Durigan Junior <sergiodj@redhat.com> Pedro Alves <palves@redhat.com> * infcmd.c (kill_command): Print message when inferior has been killed. * inferior.c (print_inferior_events): Remove 'static'. Set as '1'. (add_inferior): Improve message printed when 'print_inferior_events' is on. (exit_inferior): Remove message printed when 'print_inferior_events' is on. (detach_inferior): Improve message printed when 'print_inferior_events' is on. (initialize_inferiors): Use 'add_inferior_silent' to set 'current_inferior_'. * inferior.h (print_inferior_events): Declare here as 'extern'. * infrun.c (follow_fork_inferior): Print '[Attaching...]' or '[Detaching...]' messages when 'print_inferior_events' is on. Use 'add_thread_silent' instead of 'add_thread'. Add '[' and ']' as prefix/suffix for messages. Remove periods. Fix erroneous 'Detaching after fork from child...', replace it by '... from parent...'. (handle_vfork_child_exec_or_exit): Add '[' and ']' as prefix/suffix when printing 'Detaching...' messages. Print them when 'print_inferior_events' is on. * remote.c (remote_detach_1): Print message when detaching from inferior and '!is_fork_parent'. gdb/testsuite/ChangeLog: 2018-04-24 Jan Kratochvil <jan.kratochvil@redhat.com> Sergio Durigan Junior <sergiodj@redhat.com> Pedro Alves <palves@redhat.com> * gdb.base/attach-non-pgrp-leader.exp: Adjust 'Detaching...' regexps to expect for '[Inferior ... detached]' as well. * gdb.base/attach.exp: Likewise. * gdb.base/catch-syscall.exp (check_for_program_end): Adjust "gdb_continue_to_end". (test_catch_syscall_with_wrong_args): Likewise. * gdb.base/foll-fork.exp: Adjust regexps to match '[' and ']'. Don't set 'verbose' on. * gdb.base/foll-vfork.exp: Likewise. * gdb.base/fork-print-inferior-events.c: New file. * gdb.base/fork-print-inferior-events.exp: New file. * gdb.base/hook-stop.exp: Adjust regexps to expect for new '[Inferior ... has been killed]' message. * gdb.base/kill-after-signal.exp: Likewise. * gdb.base/solib-overlap.exp: Adjust regexps to expect for new detach message. * gdb.threads/kill.exp: Adjust regexps to expect for new kill message. * gdb.threads/clone-attach-detach.exp: Adjust 'Detaching...' regexps to expect for '[Inferior ... detached]' as well. * gdb.threads/process-dies-while-detaching.exp: Likewise.
347 lines
10 KiB
Plaintext
347 lines
10 KiB
Plaintext
# Copyright 2016-2018 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/>.
|
|
|
|
# This test spawns a few threads that immediately exit the whole
|
|
# process. On targets where the debugger needs to detach from each
|
|
# thread individually (such as on the Linux kernel), the debugger must
|
|
# handle the case of the process exiting while the detach is ongoing.
|
|
#
|
|
# Similarly, the process can also be killed from outside the debugger
|
|
# (e.g., with SIGKILL), _before_ the user requests a detach. The
|
|
# debugger must likewise detach gracefully.
|
|
#
|
|
# The testcase actually builds two variants of the test program:
|
|
# single-process, and multi-process. In the multi-process variant,
|
|
# the test program forks, and it's the fork child that spawns threads
|
|
# that exit just while the process is being detached from. The fork
|
|
# parent waits for its child to exit, so if GDB fails to detach from
|
|
# the child correctly, the parent hangs. Because continuing the
|
|
# parent can mask failure to detach from the child correctly (e.g.,
|
|
# due to waitpid(-1,...) calls deep in the target layers managing to
|
|
# reap the child), we try immediately detaching from the parent too,
|
|
# and observing whether the parent exits via standard output.
|
|
#
|
|
# Normally, if testing with "target remote" against gdbserver, then
|
|
# after detaching from all attached processes, gdbserver exits.
|
|
# However, when gdbserver detaches from a process that is its own
|
|
# direct child, gdbserver does not exit immediately. Instead it
|
|
# "joins" (waits for) the child, only exiting when the child itself
|
|
# exits too. Thus, on Linux, if gdbserver fails to detach from the
|
|
# zombie child's threads correctly (or rather, reap them), it'll hang,
|
|
# because the leader thread will only return an exit status after all
|
|
# threads are reaped. We test that as well.
|
|
|
|
standard_testfile
|
|
|
|
# Test that GDBserver exits.
|
|
|
|
proc test_server_exit {} {
|
|
global server_spawn_id
|
|
|
|
set test "server exits"
|
|
gdb_expect {
|
|
-i $server_spawn_id
|
|
eof {
|
|
pass $test
|
|
wait -i $server_spawn_id
|
|
unset server_spawn_id
|
|
}
|
|
timeout {
|
|
fail "$test (timeout)"
|
|
}
|
|
}
|
|
}
|
|
|
|
# If RESULT is not zero, make the caller return.
|
|
|
|
proc return_if_fail { result } {
|
|
if {$result != 0} {
|
|
return -code return
|
|
}
|
|
}
|
|
|
|
# Detach from a process, and ensure that it exits after detaching.
|
|
# This relies on inferior I/O. INF_OUTPUT_RE is the pattern that
|
|
# matches the expected inferior output.
|
|
|
|
proc detach_and_expect_exit {inf_output_re test} {
|
|
global decimal
|
|
global gdb_spawn_id
|
|
global inferior_spawn_id
|
|
global gdb_prompt
|
|
|
|
return_if_fail [gdb_test_multiple "detach" $test {
|
|
-re "Detaching from .*, process $decimal" {
|
|
}
|
|
}]
|
|
|
|
# Use an indirect spawn id list, and remove inferior spawn id from
|
|
# the expected output as soon as it matches, so that if
|
|
# $inf_inferior_spawn_id is $server_spawn_id and we're testing in
|
|
# "target remote" mode, the eof caused by gdbserver exiting is
|
|
# left for the caller to handle.
|
|
global daee_spawn_id_list
|
|
set daee_spawn_id_list "$inferior_spawn_id $gdb_spawn_id"
|
|
|
|
set saw_prompt 0
|
|
set saw_inf_exit 0
|
|
while { !$saw_prompt || ! $saw_inf_exit } {
|
|
# We don't know what order the interesting things will arrive in.
|
|
# Using a pattern of the form 'x|y|z' instead of -re x ... -re y
|
|
# ... -re z ensures that expect always chooses the match that
|
|
# occurs leftmost in the input, and not the pattern appearing
|
|
# first in the script that occurs anywhere in the input, so that
|
|
# we don't skip anything.
|
|
return_if_fail [gdb_test_multiple "" $test {
|
|
-i daee_spawn_id_list
|
|
-re "($inf_output_re)|($gdb_prompt )" {
|
|
if {[info exists expect_out(1,string)]} {
|
|
verbose -log "saw inferior exit"
|
|
set saw_inf_exit 1
|
|
set daee_spawn_id_list "$gdb_spawn_id"
|
|
} elseif {[info exists expect_out(2,string)]} {
|
|
verbose -log "saw prompt"
|
|
set saw_prompt 1
|
|
set daee_spawn_id_list "$inferior_spawn_id"
|
|
}
|
|
array unset expect_out
|
|
}
|
|
}]
|
|
}
|
|
|
|
pass $test
|
|
}
|
|
|
|
# Run to _exit in the child.
|
|
|
|
proc continue_to_exit_bp {} {
|
|
gdb_breakpoint "_exit" temporary
|
|
gdb_continue_to_breakpoint "_exit" ".*_exit.*"
|
|
}
|
|
|
|
# If testing single-process, simply detach from the process.
|
|
#
|
|
# If testing multi-process, first detach from the child, then detach
|
|
# from the parent and confirm that the parent exits, thus ensuring
|
|
# we've detached from the child successfully, as the parent hangs in
|
|
# its waitpid call otherwise.
|
|
#
|
|
# If connected with "target remote", make sure gdbserver exits.
|
|
#
|
|
# CMD indicates what to do with the parent after detaching the child.
|
|
# Can be either "detach" to detach, or "continue", to continue to
|
|
# exit.
|
|
#
|
|
# CHILD_EXIT indicates how is the child expected to exit. Can be
|
|
# either "normal" for normal exit, or "signal" for killed with signal
|
|
# SIGKILL.
|
|
#
|
|
proc do_detach {multi_process cmd child_exit} {
|
|
global decimal
|
|
global server_spawn_id
|
|
|
|
if {$child_exit == "normal"} {
|
|
set continue_re "exited normally.*"
|
|
set inf_output_re "exited, status=0"
|
|
} elseif {$child_exit == "signal"} {
|
|
if {$multi_process} {
|
|
set continue_re "exited with code 02.*"
|
|
} else {
|
|
set continue_re "terminated with signal SIGKILL.*"
|
|
}
|
|
set inf_output_re "signaled, sig=9"
|
|
} else {
|
|
error "unhandled \$child_exit: $child_exit"
|
|
}
|
|
|
|
set is_remote [expr {[target_info exists gdb_protocol]
|
|
&& [target_info gdb_protocol] == "remote"}]
|
|
|
|
if {$multi_process} {
|
|
gdb_test "detach" "Detaching from .*, process $decimal\r\n\\\[Inferior $decimal \\(.*\\) detached\\\]" \
|
|
"detach child"
|
|
|
|
gdb_test "inferior 1" "\[Switching to inferior $decimal\].*" \
|
|
"switch to parent"
|
|
|
|
if {$cmd == "detach"} {
|
|
# Make sure that detach works and that the parent process
|
|
# exits cleanly.
|
|
detach_and_expect_exit $inf_output_re "detach parent"
|
|
} elseif {$cmd == "continue"} {
|
|
# Make sure that continuing works and that the parent process
|
|
# exits cleanly.
|
|
gdb_test "continue" $continue_re
|
|
} else {
|
|
perror "unhandled command: $cmd"
|
|
}
|
|
} else {
|
|
if $is_remote {
|
|
set extra "\r\nEnding remote debugging\."
|
|
} else {
|
|
set extra ""
|
|
}
|
|
if {$cmd == "detach"} {
|
|
gdb_test "detach" "Detaching from .*, process ${decimal}\r\n\\\[Inferior $decimal \\(.*\\) detached\\\]$extra"
|
|
} elseif {$cmd == "continue"} {
|
|
gdb_test "continue" $continue_re
|
|
} else {
|
|
perror "unhandled command: $cmd"
|
|
}
|
|
}
|
|
|
|
# When connected in "target remote" mode, the server should exit
|
|
# when there are no processes left to debug.
|
|
if { $is_remote && [info exists server_spawn_id]} {
|
|
test_server_exit
|
|
}
|
|
}
|
|
|
|
# Test detaching from a process that dies just while GDB is detaching.
|
|
|
|
proc test_detach {multi_process cmd} {
|
|
with_test_prefix "detach" {
|
|
global binfile
|
|
|
|
clean_restart ${binfile}
|
|
|
|
if ![runto_main] {
|
|
fail "can't run to main"
|
|
return -1
|
|
}
|
|
|
|
if {$multi_process} {
|
|
gdb_test_no_output "set detach-on-fork off"
|
|
gdb_test_no_output "set follow-fork-mode child"
|
|
}
|
|
|
|
# Run to _exit in the child.
|
|
continue_to_exit_bp
|
|
|
|
do_detach $multi_process $cmd "normal"
|
|
}
|
|
}
|
|
|
|
# Same as test_detach, except set a watchpoint before detaching.
|
|
|
|
proc test_detach_watch {multi_process cmd} {
|
|
with_test_prefix "watchpoint" {
|
|
global binfile decimal
|
|
|
|
clean_restart ${binfile}
|
|
|
|
if ![runto_main] {
|
|
fail "can't run to main"
|
|
return -1
|
|
}
|
|
|
|
if {$multi_process} {
|
|
gdb_test_no_output "set detach-on-fork off"
|
|
gdb_test_no_output "set follow-fork-mode child"
|
|
|
|
gdb_breakpoint "child_function" temporary
|
|
gdb_continue_to_breakpoint "child_function" ".*"
|
|
}
|
|
|
|
# Set a watchpoint in the child.
|
|
gdb_test "watch globalvar" ".* watchpoint $decimal: globalvar"
|
|
|
|
# Continue to the _exit breakpoint. This arms the watchpoint
|
|
# registers in all threads. Detaching will thus need to clear
|
|
# them out, and handle the case of the thread disappearing
|
|
# while doing that (on targets that need to detach from each
|
|
# thread individually).
|
|
continue_to_exit_bp
|
|
|
|
do_detach $multi_process $cmd "normal"
|
|
}
|
|
}
|
|
|
|
# Test detaching from a process that dies _before_ GDB starts
|
|
# detaching.
|
|
|
|
proc test_detach_killed_outside {multi_process cmd} {
|
|
with_test_prefix "killed outside" {
|
|
global binfile
|
|
|
|
clean_restart ${binfile}
|
|
|
|
if ![runto_main] {
|
|
fail "can't run to main"
|
|
return -1
|
|
}
|
|
|
|
gdb_test_no_output "set breakpoint always-inserted on"
|
|
|
|
if {$multi_process} {
|
|
gdb_test_no_output "set detach-on-fork off"
|
|
gdb_test_no_output "set follow-fork-mode child"
|
|
}
|
|
|
|
# Run to _exit in the child.
|
|
continue_to_exit_bp
|
|
|
|
set childpid [get_integer_valueof "mypid" -1]
|
|
if { $childpid == -1 } {
|
|
untested "failed to extract child pid"
|
|
return -1
|
|
}
|
|
|
|
remote_exec target "kill -9 ${childpid}"
|
|
|
|
# Give it some time to die.
|
|
sleep 2
|
|
|
|
do_detach $multi_process $cmd "signal"
|
|
}
|
|
}
|
|
|
|
# The test proper. MULTI_PROCESS is true if testing the multi-process
|
|
# variant.
|
|
|
|
proc do_test {multi_process cmd} {
|
|
global testfile srcfile binfile
|
|
|
|
if {$multi_process && $cmd == "detach"
|
|
&& [target_info exists gdb,noinferiorio]} {
|
|
# This requires inferior I/O to tell whether both the parent
|
|
# and child exit successfully.
|
|
return
|
|
}
|
|
|
|
set binfile [standard_output_file ${testfile}-$multi_process-$cmd]
|
|
set options {debug pthreads}
|
|
if {$multi_process} {
|
|
lappend options "additional_flags=-DMULTIPROCESS"
|
|
}
|
|
|
|
if {[build_executable "failed to build" \
|
|
$testfile-$multi_process-$cmd $srcfile $options] == -1} {
|
|
return -1
|
|
}
|
|
|
|
test_detach $multi_process $cmd
|
|
test_detach_watch $multi_process $cmd
|
|
test_detach_killed_outside $multi_process $cmd
|
|
}
|
|
|
|
foreach multi_process {0 1} {
|
|
set mode [expr {$multi_process ? "multi-process" : "single-process"}]
|
|
foreach cmd {"detach" "continue"} {
|
|
with_test_prefix "$mode: $cmd" {
|
|
do_test $multi_process $cmd
|
|
}
|
|
}
|
|
}
|