gdb: fix failures in gdb.mi/mi-exec-run.exp with native-extended-gdbserver
authorAndrew Burgess <aburgess@redhat.com>
Wed, 27 Apr 2022 14:40:39 +0000 (15:40 +0100)
committerAndrew Burgess <aburgess@redhat.com>
Tue, 3 May 2022 09:30:33 +0000 (10:30 +0100)
When running the gdb.mi/mi-exec-run.exp test using the
native-extended-gdbserver I see failures like this:

  FAIL: gdb.mi/mi-exec-run.exp: inferior-tty=main: mi=main: force-fail=1: run failure detected
  FAIL: gdb.mi/mi-exec-run.exp: inferior-tty=main: mi=separate: force-fail=1: run failure detected
  FAIL: gdb.mi/mi-exec-run.exp: inferior-tty=separate: mi=separate: force-fail=1: run failure detected

There's a race condition here, so you might see a slightly different
set of failures, but I always see some from the 'run failure detected'
test.

NOTE: I also see an additional test failure:

 FAIL: gdb.mi/mi-exec-run.exp: inferior-tty=separate: mi=separate: force-fail=0: breakpoint hit reported on console (timeout)

but that is a completely different issue, and is not being addressed
in this commit.

The problem for the 'run failure detected' test is that we end up
in gdb_expect looking for output from two spawn-ids, one from
gdbserver, and one from gdb.  We're looking for one output pattern
from each spawn-id, and for the test to pass we need to see both
patterns.

Now, if gdb exits then this is a test failure (this would indicate gdb
crashing, which is bad), so we have an eof pattern associated with
the gdb spawn-id.

However, in this particular test we expect gdbserver to fail to
execute the binary (the test binary is set non-executable), and so we
get an error message from gdbserver (which matches the pattern), and
then gdbserver exits, this is expected.

The problem is that after spotting the pattern from gdbserver, we
often see the eof from gdbserver before we see the pattern from gdb.
If this happens then we drop out of the gdb_expect without ever seeing
the pattern from gdb, and fail the test.

In this commit, I place the spawn-id of gdbserver into a global
variable, and then use this global variable as the -i option within
the gdb_expect.

Now, once we have seen the expected pattern on the gdbserver spawn-id,
the global variable is cleared.  After this the gdb_expect no longer
checks the gdbserver spawn-id for additional output, and so never sees
the eof event.  This leaves the gdb_expect running, which allows the
pattern from gdb to be seen, and for the test to pass.

I now see no failures relating to 'run failure detected'.

gdb/testsuite/gdb.mi/mi-exec-run.exp

index f397159078f70c7123c24385b7e87f625633c6f4..ffbe6bcd36eb5c1d6611c859aef3f52dc936be5e 100644 (file)
@@ -93,10 +93,27 @@ proc test {inftty_mode mi_mode force_fail} {
        set test "run failure detected"
        send_gdb "-exec-run --start\n"
 
+       # Redirect through SPAWN_LIST global.  If the
+       # inferior_spawn_id is not the same as gdb_spawn_id, e.g. when
+       # testing with gdbserver, the gdbserver can exit after
+       # emitting it's error message.
+       #
+       # If inferior_spawn_id exits then we may see the eof from that
+       # spawn-id before we see the pattern from the gdb_spawn_id,
+       # which will kick us out of the gdb_expect, and cause us to
+       # fail the test.
+       #
+       # Instead we clean SPAWN_LIST once we've seen the expected
+       # pattern from that spawn-id, and after that we no longer care
+       # when gdbserver exits.
+       global spawn_list
+       set spawn_list "$inferior_spawn_id"
+
        gdb_expect {
-           -i "$inferior_spawn_id"
+           -i spawn_list
            -re ".*Cannot exec.*Permission denied" {
                set saw_perm_error 1
+               set spawn_list ""
                verbose -log "saw perm error"
                if {!$saw_mi_error} {
                    exp_continue