# Copyright 2011-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 .
# Test fork handling of an inferior that has JIT-ed objfiles.
if {[skip_shlib_tests]} {
untested "skipping shared library tests"
return -1
}
load_lib jit-elf-helpers.exp
# The main code that loads and registers JIT objects.
set main_basename "jit-elf-fork-main"
set main_srcfile ${srcdir}/${subdir}/${main_basename}.c
set main_binfile [standard_output_file ${main_basename}]
# The shared library that gets loaded as JIT objects.
set jit_solib_basename jit-elf-fork-solib
set jit_solib_srcfile ${srcdir}/${subdir}/${jit_solib_basename}.c
# Compile a shared library to use as JIT objects.
set jit_solibs_target [compile_and_download_n_jit_so \
$jit_solib_basename $jit_solib_srcfile 1]
if { $jit_solibs_target == -1 } {
return
}
# Compile the main code (which loads the JIT objects).
if { [compile_jit_main ${main_srcfile} ${main_binfile} {}] != 0 } {
return
}
# Set up for the tests.
#
# detach-on-fork and follow-fork-mode are the values to use for the GDB
# parameters of the same names.
#
# Upon return, execution is stopped at the breakpoint just after fork. Which
# inferiors exist and which inferior is the current one depends on the
# parameter.
#
# The only breakpoint left is one just before the return statement in main, so
# that the callers can continue execution until there.
proc do_setup { detach-on-fork follow-fork-mode } {
clean_restart ${::main_binfile}
gdb_test_no_output "set detach-on-fork ${detach-on-fork}"
gdb_test_no_output "set follow-fork-mode ${follow-fork-mode}"
if { ![runto_main] } {
return -1
}
# Poke desired values directly into inferior.
gdb_test_no_output "set var argc=2" "forging argc"
gdb_test_no_output "set var argv=fake_argv" "forging argv"
set jit_solib_target [lindex $::jit_solibs_target 0]
gdb_test_no_output "set var argv\[1]=\"${jit_solib_target}\"" {forging argv[1]}
# Put a breakpoint before the fork, continue there.
gdb_breakpoint [gdb_get_line_number "break before fork" $::main_srcfile]
gdb_continue_to_breakpoint "continue to before fork" ".*break before fork.*"
# We should have one JIT object loaded.
gdb_test "maint info jit" \
[multi_line \
"jit_code_entry address\\s+symfile address\\s+symfile size\\s*" \
"${::hex}\\s+${::hex}\\s+${::decimal}\\s*"] \
"jit-ed objfiles before fork"
# Put a breakpoint just after the fork, continue there.
gdb_breakpoint [gdb_get_line_number "break after fork" $::main_srcfile]
gdb_continue_to_breakpoint "continue to after fork" ".*break after fork.*"
# We should still have one JIT object loaded in whatever inferior we are
# currently stopped in, regardless of the mode.
gdb_test "maint info jit" \
[multi_line \
"jit_code_entry address\\s+symfile address\\s+symfile size\\s*" \
"${::hex}\\s+${::hex}\\s+${::decimal}\\s*"] \
"jit-ed objfiles after fork"
# Delete our breakpoints.
delete_breakpoints
# Put a breakpoint before return, for the convenience of our callers.
gdb_breakpoint [gdb_get_line_number "break before return" $::main_srcfile]
}
proc_with_prefix test_detach_on_fork_off_follow_fork_mode_parent { } {
if { [do_setup off parent] == -1 } {
return -1
}
# We are stopped in the parent.
gdb_test "inferior" "Current inferior is 1.*" "current inferior is parent"
# Switch to the child, verify there is a JIT-ed objfile.
gdb_test "inferior 2" "Switching to inferior 2.*"
gdb_test "maint info jit" \
[multi_line \
"jit_code_entry address\\s+symfile address\\s+symfile size\\s*" \
"${::hex}\\s+${::hex}\\s+${::decimal}\\s*"] \
"jit-ed objfile in child"
# Continue child past JIT unload, verify there are no more JIT-ed objfiles.
gdb_continue_to_breakpoint "continue to before return - child" ".*break before return.*"
gdb_test_no_output "maint info jit" "no more jit-ed objfiles in child"
# Go back to parent, the JIT-ed objfile should still be there.
gdb_test "inferior 1" "Switching to inferior 1.*"
gdb_test "maint info jit" \
[multi_line \
"jit_code_entry address\\s+symfile address\\s+symfile size\\s*" \
"${::hex}\\s+${::hex}\\s+${::decimal}\\s*"] \
"jit-ed objfile in parent"
# Continue parent past JIT unload, verify there are no more JIT-ed objfiles.
gdb_continue_to_breakpoint "continue to before return - parent" ".*break before return.*"
gdb_test_no_output "maint info jit" "no more jit-ed objfiles in parent"
}
proc_with_prefix test_detach_on_fork_off_follow_fork_mode_child { } {
if { [do_setup off child] == -1 } {
return -1
}
# We are stopped in the child. This is the exact same thing as
# test_detach_on_fork_off_follow_fork_mode_parent, except that we are
# stopped in the child.
gdb_test "inferior" "Current inferior is 2.*" "current inferior is child"
# Switch to the parent, verify there is a JIT-ed objfile.
gdb_test "inferior 1" "Switching to inferior 1.*"
gdb_test "maint info jit" \
[multi_line \
"jit_code_entry address\\s+symfile address\\s+symfile size\\s*" \
"${::hex}\\s+${::hex}\\s+${::decimal}\\s*"] \
"jit-ed objfile in parent"
# Continue parent past JIT unload, verify there are no more JIT-ed objfiles.
gdb_continue_to_breakpoint "continue to before return - parent" ".*break before return.*"
gdb_test_no_output "maint info jit" "no more jit-ed objfiles in parent"
# Go back to child, the JIT-ed objfile should still be there.
gdb_test "inferior 2" "Switching to inferior 2.*"
gdb_test "maint info jit" \
[multi_line \
"jit_code_entry address\\s+symfile address\\s+symfile size\\s*" \
"${::hex}\\s+${::hex}\\s+${::decimal}\\s*"] \
"jit-ed objfile in child"
# Continue child past JIT unload, verify there are no more JIT-ed objfiles.
gdb_continue_to_breakpoint "continue to before return - child" ".*break before return.*"
gdb_test_no_output "maint info jit" "no more jit-ed objfiles in child"
}
proc_with_prefix test_detach_on_fork_on_follow_fork_mode_parent { } {
if { [do_setup on parent] == -1 } {
return -1
}
# We are stopped in the parent, child is detached.
gdb_test "inferior" "Current inferior is 1.*" "current inferior is parent"
gdb_test "inferior 2" "Inferior ID 2 not known." "no inferior 2"
# Continue past JIT unload, verify there are no more JIT-ed objfiles.
gdb_continue_to_breakpoint "continue to before return" ".*break before return.*"
gdb_test_no_output "maint info jit" "no more jit-ed objfiles"
}
proc_with_prefix test_detach_on_fork_on_follow_fork_mode_child { } {
if { [do_setup on child] == -1 } {
return -1
}
# We are stopped in the child, parent is detached.
gdb_test "inferior" "Current inferior is 2.*" "current inferior is child"
gdb_test "inferior 1" "Switching to inferior 1 \\\[\\\].*"
gdb_test "inferior 2" "Switching to inferior 2.*"
# Continue past JIT unload, verify there are no more JIT-ed objfiles.
gdb_continue_to_breakpoint "continue to before return" ".*break before return.*"
gdb_test_no_output "maint info jit" "no more jit-ed objfiles"
}
test_detach_on_fork_off_follow_fork_mode_parent
test_detach_on_fork_off_follow_fork_mode_child
test_detach_on_fork_on_follow_fork_mode_parent
test_detach_on_fork_on_follow_fork_mode_child