# This testcase is part of GDB, the GNU debugger. # # Copyright 2016-2025 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 . # 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))" "" 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" } # Note: GDB can incrementally add to the recording from the raw trace data. # After a clear(), GDB might not have all the raw trace data available in its # buffer to recreate the full recording it had before the clear(). # So, do this testing last to avoid disturbing subsequent tests. with_test_prefix "clear" { gdb_test_no_output "python r.clear()" gdb_test "python insn = r.instruction_history" gdb_test_no_output "python i = insn\[0\]" gdb_test "python print(i.size)" "$decimal" }