# Copyright 2017-2024 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 . # Common routines for testing multi-target features. load_lib gdbserver-support.exp standard_testfile multi-target.c # Keep a list of (inferior ID, spawn ID). set server_spawn_ids [list] proc connect_target_extended_remote {binfile num} { set res [gdbserver_start "--multi" ""] global server_spawn_ids server_spawn_id lappend server_spawn_ids $num $server_spawn_id set gdbserver_gdbport [lindex $res 1] return [gdb_target_cmd "extended-remote" $gdbserver_gdbport] } # Add and start inferior number NUM. Returns true on success, false # otherwise. proc add_inferior {num target binfile {gcorefile ""}} { # Start another inferior. gdb_test "add-inferior -no-connection" "Added inferior $num" \ "add empty inferior $num" gdb_test "inferior $num" "Switching to inferior $num.*" \ "switch to inferior $num" gdb_test "file ${binfile}" ".*" "load file in inferior $num" gdb_test_no_output "set remote exec-file ${binfile}" \ "set remote-exec file in inferior $num" if {$target == "core"} { gdb_test "core $gcorefile" "Core was generated by.*" \ "core [file tail $gcorefile], inf $num" return 1 } if {$target == "extended-remote"} { if {[connect_target_extended_remote $binfile $num]} { return 0 } } if ![runto "all_started"] then { return 0 } delete_breakpoints return 1 } proc prepare_core {} { global gcorefile gcore_created global binfile clean_restart ${binfile} if ![runto all_started] then { return -1 } global testfile set gcorefile [standard_output_file $testfile.gcore] set gcore_created [gdb_gcore_cmd $gcorefile "save a core file"] } proc next_live_inferior {inf} { incr inf if {$inf == 3} { # 3 is a core. return 4 } if {$inf > 5} { # 6 is a core. return 1 } return $inf } # Clean up the server_spawn_ids. proc cleanup_gdbservers { } { global server_spawn_id global server_spawn_ids foreach { inferior_id spawn_id } $server_spawn_ids { set server_spawn_id $spawn_id gdb_test "inferior $inferior_id" gdbserver_exit 0 } set server_spawn_ids [list] } # Return true on success, false otherwise. proc setup {non-stop {multi_process ""}} { global gcorefile gcore_created global binfile cleanup_gdbservers save_vars { ::GDBFLAGS } { # Make GDB read files from the local file system, not through the # remote targets, to speed things up. set ::GDBFLAGS "${::GDBFLAGS} -ex \"set sysroot\"" clean_restart ${binfile} } # multi-target depends on target running in non-stop mode. Force # it on for remote targets, until this is the default. gdb_test_no_output "maint set target-non-stop on" gdb_test_no_output "set non-stop ${non-stop}" if {${multi_process} ne ""} then { gdb_test \ "set remote multiprocess-feature-packet $multi_process" \ "Support for the 'multiprocess-feature' packet on future remote targets is set to \"${multi_process}\"." } if ![runto all_started] then { return 0 } delete_breakpoints # inferior 1 -> native # inferior 2 -> extended-remote # inferior 3 -> core # inferior 4 -> native # inferior 5 -> extended-remote # inferior 6 -> core if {![add_inferior 2 "extended-remote" $binfile]} { return 0 } if {![add_inferior 3 "core" $binfile $gcorefile]} { return 0 } if {![add_inferior 4 "native" $binfile]} { return 0 } if {![add_inferior 5 "extended-remote" $binfile]} { return 0 } if {![add_inferior 6 "core" $binfile $gcorefile]} { return 0 } # For debugging. gdb_test "info threads" ".*" # Make "continue" resume all inferiors. if {${non-stop} == "off"} { gdb_test_no_output "set schedule-multiple on" } return 1 } proc multi_target_prepare {} { global binfile srcfile if { ![allow_gdbserver_tests] } { return 0 } # The plain remote target can't do multiple inferiors. if {[target_info gdb_protocol] != ""} { return 0 } if { [prepare_for_testing "failed to prepare" ${binfile} "${srcfile}" \ {debug pthreads}] } { return 0 } # Make a core file with two threads upfront. Several tests load # the same core file. prepare_core return 1 } proc multi_target_cleanup {} { cleanup_gdbservers }