mirror of
https://sourceware.org/git/binutils-gdb.git
synced 2025-02-11 13:02:10 +08:00
I ran into: ... (gdb) PASS: gdb.python/py-record-btrace.exp: function call: \ python print(c.prev) python print(c == c.next.prev)^M Traceback (most recent call last):^M File "<string>", line 1, in <module>^M AttributeError: 'NoneType' object has no attribute 'prev'^M Error while executing Python code.^M (gdb) FAIL: gdb.python/py-record-btrace.exp: function call: \ python print(c == c.next.prev) ... due to having only 4 insn instead of 100: ... python print(len(insn))^M 4^M ... This could be caused by the same hw bug as we already have an xfail for, so expand the xfail matching. Tested on x86_64-linux. PR testsuite/30185 Bug: https://sourceware.org/bugzilla/show_bug.cgi?id=30185 Approved-By: Markus T. Metzger <markus.t.metzger@intel.com>
201 lines
6.6 KiB
Plaintext
201 lines
6.6 KiB
Plaintext
# This testcase is part of GDB, the GNU debugger.
|
|
#
|
|
# Copyright 2016-2023 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"
|
|
}
|