blob: 813999b1adc7be7ff33ba1a56211d2ee1dedca22 [file] [log] [blame] [edit]
# Copyright 2023-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 <http://www.gnu.org/licenses/>.
# Test "pause" in DAP.
require allow_dap_tests
load_lib dap-support.exp
standard_testfile
if {[build_executable ${testfile}.exp $testfile $srcfile] == -1} {
return
}
if {[dap_initialize] == ""} {
return
}
set launch_id [dap_launch $testfile]
# Set a conditional breakpoint that will never fire. This is done to
# test the state-tracking in events -- an inferior call from a
# breakpoint condition should not cause any sort of stop or continue
# events.
set line [gdb_get_line_number "STOP"]
dap_check_request_and_response "set conditional breakpoint" \
setBreakpoints \
[format {o source [o path [%s]] \
breakpoints [a [o line [i %d] \
condition [s "return_false()"]]]} \
[list s $srcfile] $line]
dap_check_request_and_response "configurationDone" configurationDone
dap_check_response "launch response" launch $launch_id
dap_wait_for_event_and_check "process event generated" process \
"body startMethod" process
dap_wait_for_event_and_check "inferior started" thread "body reason" started
set resp [lindex [dap_request_and_response evaluate {o expression [s 23]}] \
0]
gdb_assert {[dict get $resp success] == "false"} \
"evaluate failed while inferior executing"
gdb_assert {[dict get $resp message] == "notStopped"} \
"evaluate issued notStopped"
dap_check_request_and_response pause pause \
{o threadId [i 1]}
dap_wait_for_event_and_check "stopped by pause" stopped \
"body reason" pause
set result [dap_request_and_response evaluate {o expression [s do_nothing()]}]
gdb_assert {[dict get [lindex $result 0] body result] == 91} \
"check result of evaluation"
set seen fail
foreach event [lindex $result 1] {
if {[dict get $event type] != "event"} {
continue
}
if {[dict get $event event] == "continued"} {
set seen pass
break
}
}
gdb_assert {$seen == "pass"} "continue event from inferior call"
#
# Test that a repl evaluation that causes a continue can be canceled.
#
set cont_id [dap_send_request evaluate \
{o expression [s continue] context [s repl]}]
dap_wait_for_event_and_check "continued" continued
set cancel_id [dap_send_request cancel \
[format {o requestId [i %d]} $cont_id]]
# The stop event will come before any responses to the requests.
dap_wait_for_event_and_check "stopped by cancel" stopped
# Now we can wait for the 'continue' request to complete, and then the
# 'cancel' request.
dap_read_response evaluate $cont_id
dap_read_response cancel $cancel_id
#
# Test that a repl evaluation of a long-running gdb command (that does
# not continue the inferior) can be canceled.
#
proc write_file {suffix contents} {
global testfile
set gdbfile [standard_output_file ${testfile}.$suffix]
set ofd [open $gdbfile w]
puts $ofd $contents
close $ofd
return $gdbfile
}
set gdbfile [write_file gdb "set \$x = 0\nwhile 1\nset \$x = \$x + 1\nend"]
set cont_id [dap_send_request evaluate \
[format {o expression [s "source %s"] context [s repl]} \
$gdbfile]]
# Wait a little to try to ensure the command is running.
sleep 0.2
set cancel_id [dap_send_request cancel \
[format {o requestId [i %d]} $cont_id]]
set info [lindex [dap_read_response evaluate $cont_id] 0]
gdb_assert {[dict get $info success] == "false"} "gdb command failed"
gdb_assert {[dict get $info message] == "cancelled"} "gdb command canceled"
dap_read_response cancel $cancel_id
#
# Test that a repl evaluation of a long-running Python command (that
# does not continue the inferior) can be canceled.
#
set gdbfile [write_file py "while True:\n pass"]
set cont_id [dap_send_request evaluate \
[format {o expression [s "source %s"] context [s repl]} \
$gdbfile]]
# Wait a little to try to ensure the command is running.
sleep 0.2
set cancel_id [dap_send_request cancel \
[format {o requestId [i %d]} $cont_id]]
set info [lindex [dap_read_response evaluate $cont_id] 0]
gdb_assert {[dict get $info success] == "false"} "python command failed"
gdb_assert {[dict get $info message] == "cancelled"} "python command canceled"
dap_read_response cancel $cancel_id
dap_shutdown