binutils-gdb/gdb/testsuite/gdb.python/py-record-btrace.exp
Tom Tromey a207f6b3a3 Rewrite "python" command exception handling
The "python" command (and the Python implementation of the gdb
"source" command) does not handle Python exceptions in the same way as
other gdb-facing Python code.  In particular, exceptions are turned
into a generic error rather than being routed through
gdbpy_handle_exception, which takes care of converting to 'quit' as
appropriate.

I think this was done this way because PyRun_SimpleFile and friends do
not propagate the Python exception -- they simply indicate that one
occurred.

This patch reimplements these functions to respect the general gdb
convention here.  As a bonus, some Windows-specific code can be
removed, as can the _execute_file function.

The bulk of this change is tweaking the test suite to match the new
way that exceptions are displayed.  These changes are largely
uninteresting.  However, it's worth pointing out the py-error.exp
change.  Here, the failure changes because the test changes the host
charset to something that isn't supported by Python.  This then
results in a weird error in the new setup.

Bug: https://sourceware.org/bugzilla/show_bug.cgi?id=31354
Acked-By: Tom de Vries <tdevries@suse.de>
Reviewed-By: Eli Zaretskii <eliz@gnu.org>
2024-02-27 09:46:31 -07:00

200 lines
6.6 KiB
Text

# This testcase is part of GDB, the GNU debugger.
#
# Copyright 2016-2024 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/>.
# Skip this test if btrace is disabled.
require allow_btrace_tests allow_python_tests
load_lib gdb-python.exp
standard_testfile
if [prepare_for_testing "failed to prepare" $testfile $srcfile] { return -1 }
if {![runto_main]} {
return -1
}
with_test_prefix "no or double record" {
gdb_test "python print(gdb.current_recording())" "None"
gdb_test_no_output "python gdb.start_recording(\"btrace\")"
gdb_test "python gdb.start_recording(\"btrace\")" \
".*gdb\.error.*: The process is already being recorded\..*" \
"already recording"
gdb_test_no_output "python gdb.stop_recording()" "first"
gdb_test "python gdb.stop_recording()" \
".*gdb\.error.*: No recording is currently active\..*" "second"
}
with_test_prefix "preopened record btrace" {
gdb_test_no_output "record btrace"
gdb_test "python print(gdb.current_recording().method)" "btrace"
gdb_test "python print(gdb.current_recording().format)" "pt|bts"
gdb_test_no_output "python gdb.stop_recording()"
}
with_test_prefix "prepare record" {
gdb_test_no_output "python r = gdb.start_recording(\"btrace\")"
gdb_test "python print(r.method)" "btrace"
gdb_test "python print(r.format)" "pt|bts"
gdb_test "stepi 100" ".*"
# There's a HW bug affecting Processor Trace on some Intel processors.
# The bug was exposed by linux kernel commit 670638477aed
# ("perf/x86/intel/pt: Opportunistically use single range output mode"),
# added in version v5.5.0, and was worked around by commit ce0d998be927
# ("perf/x86/intel/pt: Fix sampling using single range output") in version
# 6.1.0. Detect the kernel version range for which the problem may
# manifest.
set have_xfail 0
set v [linux_kernel_version]
if { $v != {} } {
set have_xfail \
[expr [version_compare [list 5 5 0] <= $v] \
&& [version_compare $v < [list 6 1 0]]]
}
set nonl_re \[^\r\n\]
set xfail_re \
[join \
[list \
"warning: Decode error \\($nonl_re*\\) at instruction $decimal" \
"\\(offset = $hex, pc = $hex\\):" \
"$nonl_re*\\."]]
set xfail_re_2 \
[join \
[list \
"warning: Non-contiguous trace at instruction $decimal" \
"\\(offset = $hex\\)\\."]]
set got_xfail 0
set cmd "python insn = r.instruction_history"
gdb_test_multiple $cmd "" {
-re "^[string_to_regexp $cmd]\r\n$::gdb_prompt $" {
pass $gdb_test_name
}
-re -wrap "($xfail_re|$xfail_re_2)" {
if { $have_xfail } {
xfail $gdb_test_name
set got_xfail 1
} else {
fail $gdb_test_name
}
}
}
if { $got_xfail } {
return
}
# Also handle the case that we're running into the hw bug without
# triggering a decode error.
gdb_test_multiple "python print(len(insn))" "" {
-re -wrap "100" {
pass $gdb_test_name
}
-re -wrap "" {
if { $have_xfail } {
xfail $gdb_test_name
set got_xfail 1
} else {
fail $gdb_test_name
}
}
}
if { $got_xfail } {
return
}
gdb_test_no_output "python call = r.function_call_history"
gdb_test_no_output "python i = insn\[0\]"
gdb_test_no_output "python c = call\[0\]"
}
with_test_prefix "replay begin" {
gdb_test "python print(r.replay_position)" "None"
gdb_test "python r.goto(r.begin)"
gdb_test "python print(r.replay_position.number)" "1"
}
with_test_prefix "replay end" {
gdb_test "python r.goto(r.end)"
gdb_test "python print(r.replay_position)" "None"
}
with_test_prefix "instruction " {
gdb_test "python print(i.number)" "1"
gdb_test "python print(i.sal)" "symbol and line for .*"
gdb_test "python print(i.pc)" "$decimal"
gdb_test "python print(repr(i.data))" "<memory at $hex>"
gdb_test "python print(i.decoded)" ".*"
gdb_test "python print(i.size)" "$decimal"
gdb_test "python print(i.is_speculative)" "False"
}
with_test_prefix "function call" {
gdb_test "python print(c.number)" "1"
gdb_test "python print(c.symbol)" "main"
gdb_test "python print(c.level)" "$decimal"
gdb_test "python print(len(c.instructions))" "$decimal"
gdb_test "python print(c.up)" "None"
gdb_test "python print(c.prev)" "None"
gdb_test "python print(c == c.next.prev)" "True"
}
with_test_prefix "list" {
gdb_test "python print(len(insn\[23:65\]))" "42"
gdb_test "python print(insn\[17:\]\[2\].number)" "20"
gdb_test "python print(i in insn)" "True"
gdb_test "python print(i in call)" "False"
gdb_test "python print(c in insn)" "False"
gdb_test "python print(c in call)" "True"
gdb_test "python print(insn.index(i))" "0"
gdb_test "python print(insn.count(i))" "1"
}
with_test_prefix "sublist" {
gdb_test_no_output "python s1 = insn\[3:72:5\]"
gdb_test_no_output "python s2 = s1\[2:13:3\]"
gdb_test_no_output "python s3 = s1\[13:2:-3\]"
gdb_test_no_output "python s4 = insn\[::-1\]"
gdb_test "python print(\[i.number for i in s1\])" "\\\[4, 9, 14, 19, 24, 29, 34, 39, 44, 49, 54, 59, 64, 69\\\]"
gdb_test "python print(\[i.number for i in s2\])" "\\\[14, 29, 44, 59\\\]"
gdb_test "python print(\[i.number for i in s3\])" "\\\[69, 54, 39, 24\\\]"
gdb_test "python print(len(s1))" "14"
gdb_test "python print(len(s2))" "4"
gdb_test "python print(len(s3))" "4"
gdb_test "python print(len(s4))" "100"
gdb_test "python print(s4\[5\].number)" "95"
gdb_test "python print(s4\[-5\].number)" "5"
gdb_test "python print(s4\[100\].number)" ".*IndexError.*"
gdb_test "python print(s4\[-101\].number)" ".*IndexError.*"
}
with_test_prefix "level" {
gdb_test_no_output "python gdb.stop_recording()"
gdb_test "break inner" "Breakpoint.*"
gdb_test "continue" "Continuing\..*"
gdb_test_no_output "record btrace"
gdb_test "step" "outer ().*" "step one"
gdb_test "step" "main ().*" "step two"
gdb_test "python print(gdb.current_recording().function_call_history\[0\].level)" "1"
gdb_test "python print(gdb.current_recording().function_call_history\[1\].level)" "0"
}