# Copyright 2013-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 .
load_lib "range-stepping-support.exp"
standard_testfile
set executable $testfile
if { [prepare_for_testing "failed to prepare" $testfile $srcfile {debug}] } {
return -1
}
if ![runto_main] {
return -1
}
if ![gdb_range_stepping_enabled] {
unsupported "range stepping not supported by the target"
return -1
}
# Check that range stepping can step a range of multiple instructions.
with_test_prefix "multi insns" {
gdb_breakpoint [gdb_get_line_number "location 1"]
gdb_continue_to_breakpoint "location 1"
set pc_before_stepping ""
set test "pc before stepping"
gdb_test_multiple "print/x \$pc" $test {
-re "\\\$$decimal = (\[^\r\n\]*)\r\n$gdb_prompt $" {
set pc_before_stepping $expect_out(1,string)
pass $test
}
}
# When "next" is executed, GDB should send one vCont;s and vCont;r
# and receive two stop replies:
#
# --> vCont;s (step over breakpoint)
# <-- T05
# --> vCont;rSTART,END (range step)
# <-- T05
set result [exec_cmd_expect_vCont_count "next" 1]
if { $result } {
# This is the first range-stepping test, and the simplest
# one. If it fails, probably the rest of the tests would
# fail too, and the huge number of rsp packets in the test
# with the time-consuming loop would blow up the gdb.log file.
# Skip the rest of the tests.
return
}
set pc_after_stepping ""
set msg "pc after stepping"
gdb_test_multiple "print/x \$pc" $msg {
-re "\\\$$decimal = (\[^\r\n\]*)\r\n$gdb_prompt $" {
set pc_after_stepping $expect_out(1,string)
pass $msg
}
}
# There should be at least two instructions between
# PC_BEFORE_STEPPING and PC_AFTER_STEPPING.
gdb_test "disassemble ${pc_before_stepping},${pc_after_stepping}" \
"${hex} :.*${hex} :.*" \
"stepped multiple insns"
}
# Check that range stepping can step over a function.
with_test_prefix "step over func" {
set line_num [gdb_get_line_number "location 2"]
gdb_test "where" "main \\(\\) at .*${srcfile}:${line_num}.*"
# It's expected to get three stops and two 'vCont;r's. In the C
# code, the line of C source produces roughly the following
# instructions:
#
# addr1:
# insn1
# insn2
# ...
# call func1
# addr2:
# ...
# insn3
# addr3:
# insn4
#
# Something like this will happen:
# --> vCont;rADDR1,ADDR3 (range step from ADDR1 to ADDR3)
# <-- T05 (target single-stepped to func, which is out of the step range)
# --> $Z0,ADDR2 (place step-resume breakpoint at ADDR2)
# --> vCont;c (resume)
# <-- T05 (target stops at ADDR2)
# --> vCont;rADDR1,ADDR3 (continues range stepping)
# <-- T05
exec_cmd_expect_vCont_count "next" 2
}
# Check that breakpoints interrupt range stepping correctly.
with_test_prefix "breakpoint" {
gdb_breakpoint "func1"
# Something like this will happen:
# --> vCont;rADDR1,ADDR3
# <-- T05 (target single-steps to func1, which is out of the step range)
# --> $Z0,ADDR2 (step-resume breakpoint at ADDR2)
# --> vCont;c (resume)
# <-- T05 (target hits the breakpoint at func1)
exec_cmd_expect_vCont_count "next" 1
gdb_test "backtrace" "#0 .* func1 .*#1 .* main .*" \
"backtrace from func1"
# A cancelled range step should not confuse the following
# execution commands.
exec_cmd_expect_vCont_count "stepi" 0
gdb_test "finish" ".*"
gdb_test "next" ".*"
delete_breakpoints
}
# Check that range stepping works well even when there's a loop in the
# step range.
with_test_prefix "loop" {
# GDB should send one vCont;r and receive one stop reply:
# --> vCont;rSTART,END (range step)
# <-- T05
exec_cmd_expect_vCont_count "next" 1
# Confirm the loop completed.
gdb_test "print a" " = 15"
gdb_test "print e" " = 105"
}
# Check that range stepping works well even when the target's PC was
# already within the loop's body.
with_test_prefix "loop 2" {
# Stepi into the loop body. 15 should be large enough to make
# sure the program stops within the loop's body.
gdb_test "stepi 15" ".*"
# GDB should send one vCont;r and receive one stop reply:
# --> vCont;rSTART,END (range step)
# <-- T05
exec_cmd_expect_vCont_count "next" 1
# Confirm the loop completed.
gdb_test "print a" " = 15"
gdb_test "print e" " = 105"
}
# Check that range stepping works well even when it is interrupted by
# ctrl-c.
if ![target_info exists gdb,nointerrupts] {
with_test_prefix "interrupt" {
gdb_test_no_output "set debug remote 1"
send_gdb "next\n"
sleep 1
send_gdb "\003"
# GDB should send one vCont;r and receive one stop reply for
# SIGINT:
# --> vCont;rSTART,END (range step)
# <-- T02 (SIGINT)
set vcont_r_counter 0
set test "send ctrl-c to GDB"
gdb_test_multiple "" $test {
-re "vCont;r\[^\r\n\]*\.\.\." {
incr vcont_r_counter
exp_continue
}
-re "Program received signal SIGINT.*$gdb_prompt $" {
pass $test
}
}
gdb_test_no_output "set debug remote 0"
# Check the number of 'vCont;r' packets.
if { $vcont_r_counter == 1 } {
pass "${test}: 1 vCont;r"
} else {
fail "${test}: 1 vCont;r"
}
# Break the loop earlier and continue range stepping.
gdb_test "set variable c = 0"
exec_cmd_expect_vCont_count "next" 1
}
}
# Check that range stepping doesn't break software watchpoints. With
# those, GDB needs to be notified of all single-steps, to evaluate
# whether the watched value changes at each step.
with_test_prefix "software watchpoint" {
gdb_test "step" "soft-watch.*" "step into multiple instruction line"
# A software watchpoint at PC makes the thread stop before the
# whole line range is over (after one single-step, actually).
gdb_test "watch \$pc" ".*" "set watchpoint"
gdb_test "step" "soft-watch.*" "step still in same line"
}
return 0