b3ccfe11d3
A patch in the target cleanup series caused a regression when using record with target-async. Version 4 of the patch is here: https://sourceware.org/ml/gdb-patches/2014-03/msg00159.html The immediate problem is that record supplies to_can_async_p and to_is_async_p methods, but does not supply a to_async method. So, when target-async is set, record claims to support async -- but if the underlying target does not support async, then the to_async method call will end up in that method's default implementation, namely tcomplain. This worked previously because the record target used to provide a to_async method; one that (erroneously, only at push time) checked the other members of the target stack, and then simply dropped to_async calls in the "does not implement async" case. My first thought was to simply drop tcomplain as the default for to_async. This works, but Pedro pointed out that the only reason record has to supply to_can_async_p and to_is_async_p is that these default to using the find_default_run_target machinery -- and these defaults are only needed by "run" and "attach". So, a nicer solution presents itself: change run and attach to explicitly call into the default run target when needed; and change to_is_async_p and to_can_async_p to default to "return 0". This makes the target stack simpler to use and lets us remove the method implementations from record. This is also in harmony with other plans for the target stack; namely trying to reduce the impact of find_default_run_target. This approach makes it clear that find_default_is_async_p is not needed -- it is asking whether a target that may not even be pushed is actually async, which seems like a nonsensical question. While an improvement, this approach proved to introduce the same bug when using the core target. Looking a bit deeper, the issue is that code in "attach" and "run" may need to use either the current target stack or the default run target -- but different calls into the target API in those functions could wind up querying different targets. This new patch makes the target to use more explicit in "run" and "attach". Then these commands explicitly make the needed calls against that target. This ensures that a single target is used for all relevant operations. This lets us remove a couple find_default_* functions from various targets, including the dummy target. I think this is a decent understandability improvement. One issue I see with this patch is that the new calls in "run" and "attach" are not very much like the rest of the target API. I think fundamentally this is due to bad factoring in the target API, which may need to be fixed for multi-target. Tackling that seemed ambitious for a regression fix. While working on this I noticed that there don't seem to be any test cases that involve both target-async and record, so this patch changes break-precsave.exp to add some. It also changes corefile.exp to add some target-async tests; these pass with current trunk and with this patch applied, but fail with the v1 patch. This patch differs from v4 in that it moves initialization of to_can_async_p and to_supports_non_stop into inf-child, adds some assertions to complete_target_initialization, and adds some comments to target.h. Built and regtested on x86-64 Fedora 20. 2014-03-12 Tom Tromey <tromey@redhat.com> * inf-child.c (return_zero): New function. (inf_child_target): Set to_can_async_p, to_supports_non_stop. * aix-thread.c (aix_thread_inferior_created): New function. (aix_thread_attach): Remove. (init_aix_thread_ops): Don't set to_attach. (_initialize_aix_thread): Register inferior_created observer. * corelow.c (init_core_ops): Don't set to_attach or to_create_inferior. * exec.c (init_exec_ops): Don't set to_attach or to_create_inferior. * infcmd.c (run_command_1): Use find_run_target. Make direct target calls. (attach_command): Use find_attach_target. Make direct target calls. * record-btrace.c (init_record_btrace_ops): Don't set to_create_inferior. * record-full.c (record_full_can_async_p, record_full_is_async_p): Remove. (init_record_full_ops, init_record_full_core_ops): Update. Don't set to_create_inferior. * target.c (complete_target_initialization): Add assertion. (target_create_inferior): Remove. (find_default_attach, find_default_create_inferior): Remove. (find_attach_target, find_run_target): New functions. (find_default_is_async_p, find_default_can_async_p) (target_supports_non_stop, target_attach): Remove. (init_dummy_target): Don't set to_create_inferior or to_supports_non_stop. * target.h (struct target_ops) <to_attach>: Add comment. Remove TARGET_DEFAULT_FUNC. <to_create_inferior>: Add comment. <to_can_async_p, to_is_async_p, to_supports_non_stop>: Use TARGET_DEFAULT_RETURN. <to_can_async_p, to_supports_non_stop, to_can_run>: Add comments. (find_attach_target, find_run_target): Declare. (target_create_inferior): Remove. (target_has_execution_1): Update comment. (target_supports_non_stop): Remove. * target-delegates.c: Rebuild. 2014-03-12 Tom Tromey <tromey@redhat.com> * gdb.base/corefile.exp (corefile_test_run, corefile_test_attach): New procs. Add target-async tests. * gdb.reverse/break-precsave.exp (precsave_tests): New proc. Add target-async tests.
310 lines
9 KiB
Text
310 lines
9 KiB
Text
# Copyright 1992-2014 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 file was written by Fred Fish. (fnf@cygnus.com)
|
|
|
|
|
|
# are we on a target board
|
|
if ![isnative] then {
|
|
return
|
|
}
|
|
|
|
standard_testfile coremaker.c
|
|
|
|
# Create and source the file that provides information about the compiler
|
|
# used to compile the test case.
|
|
if [get_compiler_info] {
|
|
return -1
|
|
}
|
|
|
|
if {[build_executable $testfile.exp $testfile $srcfile debug] == -1} {
|
|
untested $testfile.exp
|
|
return -1
|
|
}
|
|
|
|
set corefile [core_find $binfile {coremmap.data}]
|
|
if {$corefile == ""} {
|
|
return 0
|
|
}
|
|
|
|
# Test that we can simply startup with a "-core=$corefile" command line arg
|
|
# and recognize that the core file is a valid, usable core file.
|
|
# To do this, we must shutdown the currently running gdb and restart
|
|
# with the -core args. We can't use gdb_start because it looks for
|
|
# the first gdb prompt, and the message we are looking for occurs
|
|
# before the first prompt.
|
|
#
|
|
# Another problem is that on some systems (solaris for example), there
|
|
# is apparently a limit on the length of a fully specified path to
|
|
# the corefile executable, at about 80 chars. For this case, consider
|
|
# it a pass, but note that the program name is bad.
|
|
|
|
gdb_exit
|
|
if $verbose>1 then {
|
|
send_user "Spawning $GDB $INTERNAL_GDBFLAGS $GDBFLAGS -core=$corefile\n"
|
|
}
|
|
|
|
set oldtimeout $timeout
|
|
set timeout [expr "$timeout + 60"]
|
|
verbose "Timeout is now $timeout seconds" 2
|
|
eval "spawn $GDB $INTERNAL_GDBFLAGS $GDBFLAGS -core=$corefile"
|
|
expect {
|
|
-re "Couldn't find .* registers in core file.*$gdb_prompt $" {
|
|
fail "args: -core=[file tail $corefile] (couldn't find regs)"
|
|
}
|
|
-re "Core was generated by .*corefile.*\r\n\#0 .*\(\).*\r\n$gdb_prompt $" {
|
|
pass "args: -core=[file tail $corefile]"
|
|
}
|
|
-re "Core was generated by .*\r\n\#0 .*\(\).*\r\n$gdb_prompt $" {
|
|
pass "args: -core=[file tail $corefile] (with bad program name)"
|
|
}
|
|
-re ".*registers from core file: File in wrong format.* $" {
|
|
fail "args: -core=[file tail $corefile] (could not read registers from core file)"
|
|
}
|
|
-re ".*$gdb_prompt $" { fail "args: -core=[file tail $corefile]" }
|
|
timeout { fail "(timeout) starting with -core" }
|
|
}
|
|
|
|
|
|
#
|
|
# Test that startup with both an executable file and -core argument.
|
|
# See previous comments above, they are still applicable.
|
|
#
|
|
|
|
close
|
|
|
|
if $verbose>1 then {
|
|
send_user "Spawning $GDB $INTERNAL_GDBFLAGS $GDBFLAGS $binfile -core=$corefile\n"
|
|
}
|
|
|
|
|
|
eval "spawn $GDB $INTERNAL_GDBFLAGS $GDBFLAGS $binfile -core=$corefile"
|
|
expect {
|
|
-re "Core was generated by .*corefile.*\r\n\#0 .*\(\).*\r\n$gdb_prompt $" {
|
|
pass "args: execfile -core=[file tail $corefile]"
|
|
}
|
|
-re "Core was generated by .*\r\n\#0 .*\(\).*\r\n$gdb_prompt $" {
|
|
pass "args: execfile -core=[file tail $corefile] (with bad program name)"
|
|
}
|
|
-re ".*registers from core file: File in wrong format.* $" {
|
|
fail "args: execfile -core=[file tail $corefile] (could not read registers from core file)"
|
|
}
|
|
-re ".*$gdb_prompt $" { fail "args: execfile -core=[file tail $corefile]" }
|
|
timeout { fail "(timeout) starting with -core" }
|
|
}
|
|
set timeout $oldtimeout
|
|
verbose "Timeout is now $timeout seconds" 2
|
|
|
|
close
|
|
|
|
# Now restart normally.
|
|
|
|
gdb_start
|
|
gdb_reinitialize_dir $srcdir/$subdir
|
|
gdb_load ${binfile}
|
|
|
|
# Test basic corefile recognition via core-file command.
|
|
|
|
gdb_test_multiple "core-file $corefile" "core-file command" {
|
|
-re ".* program is being debugged already.*y or n. $" {
|
|
# gdb_load may connect us to a gdbserver.
|
|
send_gdb "y\n"
|
|
exp_continue
|
|
}
|
|
-re "Core was generated by .*corefile.*\r\n\#0 .*\(\).*\r\n$gdb_prompt $" {
|
|
pass "core-file command"
|
|
}
|
|
-re "Core was generated by .*\r\n\#0 .*\(\).*\r\n$gdb_prompt $" {
|
|
pass "core-file command (with bad program name)"
|
|
}
|
|
-re ".*registers from core file: File in wrong format.* $" {
|
|
fail "core-file command (could not read registers from core file)"
|
|
}
|
|
}
|
|
|
|
# Test correct mapping of corefile sections by printing some variables.
|
|
|
|
gdb_test "print coremaker_data" "\\\$$decimal = 202"
|
|
gdb_test "print coremaker_bss" "\\\$$decimal = 10"
|
|
gdb_test "print coremaker_ro" "\\\$$decimal = 201"
|
|
|
|
gdb_test "print func2::coremaker_local" "\\\$$decimal = \\{0, 1, 2, 3, 4\\}"
|
|
|
|
# Test the presence and the correct values of $_exitsignal and
|
|
# $_exitcode variables. The corefile is generated with a SIGABRT,
|
|
# which is "6" in the Linux kernel.
|
|
|
|
gdb_test "print \$_exitsignal" " = 6" \
|
|
"\$_exitsignal prints SIGABRT (6)"
|
|
|
|
gdb_test "print \$_exitcode" " = void" \
|
|
"\$_exitcode is void"
|
|
|
|
# Somehow we better test the ability to read the registers out of the core
|
|
# file correctly. I don't think the other tests do this.
|
|
|
|
gdb_test "bt" "abort.*func2.*func1.*main.*" "backtrace in corefile.exp"
|
|
gdb_test "up" "#\[0-9\]* *\[0-9xa-fH'\]* in .* \\(.*\\).*" "up in corefile.exp"
|
|
|
|
# Test ability to read mmap'd data
|
|
|
|
gdb_test "x/8bd buf1" ".*:.*0.*1.*2.*3.*4.*5.*6.*7" "accessing original mmap data in core file"
|
|
setup_xfail "*-*-sunos*" "*-*-ultrix*" "*-*-aix*"
|
|
set test "accessing mmapped data in core file"
|
|
gdb_test_multiple "x/8bd buf2" "$test" {
|
|
-re ".*:.*0.*1.*2.*3.*4.*5.*6.*7.*$gdb_prompt $" {
|
|
pass "$test"
|
|
}
|
|
-re "0x\[f\]*:.*Cannot access memory at address 0x\[f\]*.*$gdb_prompt $" {
|
|
fail "$test (mapping failed at runtime)"
|
|
}
|
|
-re "0x.*:.*Cannot access memory at address 0x.*$gdb_prompt $" {
|
|
fail "$test (mapping address not found in core file)"
|
|
}
|
|
}
|
|
|
|
# test reinit_frame_cache
|
|
|
|
gdb_load ${binfile}
|
|
gdb_test "up" "#\[0-9\]* *\[0-9xa-fH'\]* in .* \\(.*\\).*" "up in corefile.exp (reinit)"
|
|
|
|
gdb_test "core" "No core file now."
|
|
|
|
|
|
# Test a run (start) command will clear any loaded core file.
|
|
|
|
proc corefile_test_run {} {
|
|
global corefile gdb_prompt
|
|
|
|
gdb_test "core-file $corefile" "Core was generated by .*" "run: load core again"
|
|
gdb_test "info files" "\r\nLocal core dump file:\r\n.*" "run: sanity check we see the core file"
|
|
|
|
set test "run: with core"
|
|
if [runto_main] {
|
|
pass $test
|
|
} else {
|
|
fail $test
|
|
}
|
|
|
|
set test "run: core file is cleared"
|
|
gdb_test_multiple "info files" $test {
|
|
-re "\r\nLocal core dump file:\r\n.*\r\n$gdb_prompt $" {
|
|
fail $test
|
|
}
|
|
-re "\r\n$gdb_prompt $" {
|
|
pass $test
|
|
}
|
|
}
|
|
|
|
set test "quit with a process"
|
|
gdb_test_multiple "quit" $test {
|
|
-re "A debugging session is active.\r\n.*\r\nQuit anyway\\? \\(y or n\\) $" {
|
|
pass $test
|
|
gdb_test "n" {Not confirmed\.} "quit with processes: n"
|
|
}
|
|
}
|
|
|
|
gdb_exit
|
|
}
|
|
|
|
corefile_test_run
|
|
|
|
# Verify there is no question if only a core file is loaded.
|
|
|
|
gdb_start
|
|
gdb_test "core-file $corefile" "Core was generated by .*" "no question: load core"
|
|
|
|
set test "quit with a core file"
|
|
gdb_test_multiple "quit" $test {
|
|
-re "A debugging session is active.\r\n.*\r\nQuit anyway\\? \\(y or n\\) $" {
|
|
fail $test
|
|
gdb_test "n" {Not confirmed\.} "quit with processes: n"
|
|
}
|
|
eof {
|
|
pass $test
|
|
}
|
|
}
|
|
|
|
gdb_exit
|
|
|
|
# Test an attach command will clear any loaded core file.
|
|
|
|
proc corefile_test_attach {{async 0}} {
|
|
global binfile corefile gdb_prompt
|
|
|
|
if ![is_remote target] {
|
|
set test "attach: spawn sleep"
|
|
set res [remote_spawn host "$binfile sleep"]
|
|
if { $res < 0 || $res == "" } {
|
|
fail $test
|
|
return
|
|
}
|
|
set pid [exp_pid -i $res]
|
|
# We don't care whether the program is still in the startup phase when we
|
|
# attach.
|
|
|
|
gdb_start
|
|
|
|
if {$async} {
|
|
gdb_test_no_output "set target-async on" \
|
|
"enable target-async for attach tests"
|
|
}
|
|
|
|
gdb_test "core-file $corefile" "Core was generated by .*" "attach: load core again"
|
|
gdb_test "info files" "\r\nLocal core dump file:\r\n.*" "attach: sanity check we see the core file"
|
|
|
|
gdb_test "attach $pid" "Attaching to process $pid\r\n.*" "attach: with core"
|
|
|
|
set test "attach: core file is cleared"
|
|
gdb_test_multiple "info files" $test {
|
|
-re "\r\nLocal core dump file:\r\n.*\r\n$gdb_prompt $" {
|
|
fail $test
|
|
}
|
|
-re "\r\n$gdb_prompt $" {
|
|
pass $test
|
|
}
|
|
}
|
|
|
|
gdb_exit
|
|
}
|
|
}
|
|
|
|
corefile_test_attach
|
|
|
|
# Test warning-free core file load. E.g., a Linux vDSO used to
|
|
# trigger this warning:
|
|
# warning: Can't read pathname for load map: Input/output error.
|
|
|
|
clean_restart ${testfile}
|
|
|
|
set test "core-file warning-free"
|
|
gdb_test_multiple "core-file $corefile" $test {
|
|
-re "warning: .*\r\n.*\r\n$gdb_prompt $" {
|
|
fail $test
|
|
}
|
|
-re "\r\n$gdb_prompt $" {
|
|
pass $test
|
|
}
|
|
}
|
|
|
|
|
|
# Try a couple tests again with target-async.
|
|
with_test_prefix "target-async" {
|
|
clean_restart ${testfile}
|
|
|
|
gdb_test_no_output "set target-async on"
|
|
corefile_test_run
|
|
corefile_test_attach 1
|
|
}
|