mirror of
https://sourceware.org/git/binutils-gdb.git
synced 2025-01-12 12:16:04 +08:00
4295e285ef
I noticed that if input is already pending on the new-ui TTY, gdb internal-errors. E.g., create /dev/pts/2, and type anything there (even just <return> is sufficient). Now start GDB creating a new UI on that TTY, while at the same time, running a synchronous execution command. Something like: $ gdb program -ex "new-ui console /dev/pts/2" -ex "start" Back on /dev/pts/2, we get: (gdb) .../src/gdb/event-top.c:360: internal-error: double prompt A problem internal to GDB has been detected, further debugging may prove unreliable. While the main UI was waiting for "start" to finish, gdb kepts pumping events, including the input fd of the extra console. The problem is that stdin_event_handler doesn't restore the current UI back to what it was, assuming that it's only ever called from the top level event loop. However, in this case, it's being called from the nested event loop from within maybe_wait_sync_command_done. When finally the "start" command is done, we reach the code that prints the prompt in the main UI, just before starting the main event loop. Since now the current UI is pointing at the extra console (by mistake), we find ourselves printing a double prompt on the extra console. This is caught by the assertion that fails, as shown above. Since other event handlers also don't restore the UI (e.g., signal event handlers), I think it's better if whatever is pumping events to take care to restore the UI, if it cares. That's what this patch does. New test included. gdb/ChangeLog: 2016-09-06 Pedro Alves <palves@redhat.com> * top.c (wait_sync_command_done): Don't assume current_ui doesn't change across events. Restore the current UI before returning. (gdb_readline_wrapper): Restore the current UI before returning. gdb/testsuite/ChangeLog: 2016-09-06 Pedro Alves <palves@redhat.com> * gdb.base/new-ui-pending-input.c: New file. * gdb.base/new-ui-pending-input.exp: New file. * gdb.exp (clear_gdb_spawn_id): New procedure. (with_spawn_id): Check whether gdb_spawn_id exists before referencing it. If gdb_spawn_id didn't exist on entry, clear it on exit.
124 lines
3.3 KiB
Plaintext
124 lines
3.3 KiB
Plaintext
# Copyright 2016 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 'gdb "-ex new-ui TTY" -ex "start"' (or any other synchronous
|
|
# execution command), when TTY already has input pending. GDB used to
|
|
# internal error in this situation.
|
|
|
|
standard_testfile
|
|
|
|
set compile_options "debug"
|
|
if {[build_executable $testfile.exp $testfile ${srcfile} ${compile_options}] == -1} {
|
|
untested "failed to compile $testfile"
|
|
return -1
|
|
}
|
|
|
|
# See intro.
|
|
|
|
proc test_command_line_new_ui_pending_input {} {
|
|
global gdb_prompt
|
|
global binfile
|
|
|
|
# This test requires running a synchronous execution command from
|
|
# the command line.
|
|
if {[use_gdb_stub] || [target_info gdb_protocol] == "extended-remote" } {
|
|
unsupported "can't run from the command line"
|
|
return 0
|
|
}
|
|
|
|
spawn -pty
|
|
set extra_spawn_id $spawn_id
|
|
set extra_tty_name $spawn_out(slave,name)
|
|
|
|
# An arbitrary number of prints.
|
|
set nprints 3
|
|
|
|
# Queue a few commands before GDB is started.
|
|
with_spawn_id $extra_spawn_id {
|
|
for {set i 1} {$i <= $nprints} {incr i} {
|
|
send_gdb "print $i\n"
|
|
}
|
|
}
|
|
pass "commands pending"
|
|
|
|
# Now spawn GDB, creating a new-ui and at the same time running a
|
|
# synchronous command.
|
|
set test "spawn gdb"
|
|
|
|
set bpline [gdb_get_line_number "set breakpoint here"]
|
|
|
|
set options ""
|
|
append options " -iex \"set height 0\""
|
|
append options " -iex \"set width 0\""
|
|
append options " -iex \"new-ui console $extra_tty_name\""
|
|
append options " -ex \"b $bpline\""
|
|
append options " -ex \"run\""
|
|
|
|
if {[gdb_spawn_with_cmdline_opts "$options $binfile"] != 0} {
|
|
fail $test
|
|
return
|
|
} else {
|
|
pass $test
|
|
}
|
|
|
|
# Consume the initial prompt on the extra console.
|
|
with_spawn_id $extra_spawn_id {
|
|
set test "initial prompt on extra console"
|
|
gdb_test_multiple "" $test {
|
|
-re "$gdb_prompt $" {
|
|
pass $test
|
|
}
|
|
}
|
|
}
|
|
|
|
# Check that we see the result of the print commands in the extra
|
|
# UI.
|
|
with_spawn_id $extra_spawn_id {
|
|
for {set i 1} {$i <= $nprints} {incr i} {
|
|
set test "print $i on extra console"
|
|
gdb_test_multiple "" $test {
|
|
-re " = $i\r\n$gdb_prompt " {
|
|
pass "$test"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
# Now check that we reach the breakpoint successfully.
|
|
set test "run to breakpoint on main console"
|
|
gdb_test_multiple "" $test {
|
|
-re "Breakpoint .* main .*set breakpoint here.*$gdb_prompt $" {
|
|
pass $test
|
|
}
|
|
}
|
|
|
|
# And likewise on the extra console. No prompt is expected.
|
|
with_spawn_id $extra_spawn_id {
|
|
set test "run to breakpoint on extra console"
|
|
gdb_test_multiple "" $test {
|
|
-re "Breakpoint .* main .*set breakpoint here" {
|
|
pass $test
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
# The driver. Calls all tests.
|
|
proc testcase_driver {} {
|
|
test_command_line_new_ui_pending_input
|
|
}
|
|
|
|
testcase_driver
|