blob: eef84da0e44d7dc594e3248f673e1fa72e7a3302 [file] [log] [blame]
# Copyright 2020-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 <http://www.gnu.org/licenses/> .
# This test covers some basic functionality for debugging mixed
# Fortran, C, and C++ applications. Features tested include examining
# the backtrace, and printing frame arguments in frames of different
# languages.
#
# One important aspect of this test is that we set the language in
# turn to auto, fortran, c, and c++, and carry out the full test in
# each case to ensure that trying to print objects or types from one
# language, while GDB's language is set to another, doesn't crash GDB.
require allow_fortran_tests
standard_testfile mixed-lang-stack.c mixed-lang-stack.cpp mixed-lang-stack.f90
if {[prepare_for_testing_full "failed to prepare" \
[list ${binfile} {debug f90 additional_flags=-lstdc++} \
$srcfile {debug} \
$srcfile2 {debug c++ additional_flags=-std=c++11} \
$srcfile3 {debug f90}]]} {
return -1
}
set have_index [exec_has_index_section $binfile]
# Runs the test program and examins the stack. LANG is a string, the
# value to pass to GDB's 'set language ...' command.
proc run_tests { lang } {
with_test_prefix "lang=${lang}" {
global binfile hex have_index
clean_restart ${binfile}
if ![runto_main] {
return -1
}
gdb_breakpoint "breakpt"
gdb_continue_to_breakpoint "breakpt"
if { $lang == "c" || $lang == "c++" } {
gdb_test "set language c" \
"Warning: the current language does not match this frame."
} else {
gdb_test_no_output "set language $lang"
}
# Check the backtrace.
set e_arg "\['\"\]abcdef\['\"\]"
set 1b_args "\[^\r\n\]+$e_arg\[^\r\n\]+"
set 1g_args "obj=\[^\r\n\]+"
set bt_stack \
[multi_line \
"#0\\s+breakpt \\(\\) at \[^\r\n\]+" \
"#1\\s+$hex in mixed_func_1h \\(\\) at \[^\r\n\]+" \
"#2\\s+$hex in mixed_func_1g \\($1g_args\\) at \[^\r\n\]+" \
"#3\\s+$hex in mixed_func_1f \\(\\) at \[^\r\n\]+" \
"#4\\s+$hex in mixed_func_1e \\(\\) at \[^\r\n\]+" \
"#5\\s+$hex in mixed_func_1d \\(\[^\r\n\]+\\) at \[^\r\n\]+" \
"#6\\s+$hex in mixed_func_1c \\(\[^\r\n\]+\\) at \[^\r\n\]+" \
"#7\\s+$hex in mixed_func_1b \\($1b_args\\) at \[^\r\n\]+" \
"#8\\s+$hex in mixed_func_1a \\(\\) at \[^\r\n\]+" \
"#9\\s+$hex in mixed_stack_main \\(\\) at \[^\r\n\]+" ]
set main_args "argc=1, argv=${hex}( \[^\r\n\]+)?"
set bt_stack_kfail \
[multi_line \
$bt_stack \
"#10\\s+$hex in main \\($main_args\\) at \[^\r\n\]+"]
gdb_test_multiple "bt -frame-arguments all" "" {
-re -wrap $bt_stack {
pass $gdb_test_name
}
-re -wrap $bt_stack_kfail {
if { $have_index } {
setup_kfail "gdb/24549" *-*-*
}
fail $gdb_test_name
}
}
# Check backtrace arguments in MI mode.
set mi_a "\{name=\"a\",value=\"1\"\}"
set mi_b "\{name=\"b\",value=\"2\"\}"
set mi_c "\{name=\"c\",value=\"3\"\}"
set mi_d "\{name=\"d\",value=\"\\(4,5\\)\"\}"
set mi_e "\{name=\"e\",value=\"$e_arg\"\}"
set mi_g "\{name=\"g\",value=\"\\( a = 1.5, b = 2.5 \\)\"\}"
# Hidden argument name changes depending on compiler.
set mi_hidden "\{name=\"\[^\r\n\]+\",value=\"6\"\}"
gdb_test "interpreter-exec mi '-stack-list-arguments --no-frame-filters --all-values 7 7'" \
"\\^done,stack-args=\\\[frame=\{level=\"7\",args=\\\[$mi_a,$mi_b,$mi_c,$mi_d,$mi_e,$mi_g,$mi_hidden\\\]\}\\\]"
# Check the language for frame #0.
gdb_test "info frame" "source language fortran\..*" \
"info frame in frame #0"
# Move up to the C++ frames and check the frame state, print a
# C++ object.
gdb_test "frame 2" "#2\\s+$hex in mixed_func_1g .*" \
"select frame #2"
gdb_test "info frame" "source language c\\+\\+\..*" \
"info frame in frame #2"
if { $lang == "fortran" } {
set obj_pattern " = \\( base_one = \\( num1 = 1, num2 = 2, num3 = 3 \\), base_two = \\( string = $hex 'Something in C\\+\\+\\\\000', val = 3.5 \\), xxx = 9, yyy = 10.5 \\)"
} else {
set obj_pattern " = \\{<base_one> = \\{num1 = 1, num2 = 2, num3 = 3\\}, <base_two> = \\{string = $hex \"Something in C\\+\\+\", val = 3.5\\}, xxx = 9, yyy = 10.5\\}"
}
gdb_test "print obj" "${obj_pattern}"
# Move up the stack a way, and check frame and the frame
# arguments.
gdb_test "frame 5" "#5\\s+$hex in mixed_func_1d .*" \
"select frame #5"
gdb_test "info frame" "source language fortran\..*" \
"info frame in frame #5"
gdb_test "up" "#6\\s+$hex in mixed_func_1c .*" \
"up to frame #6"
gdb_test "info frame" "source language c\..*" \
"info frame in frame #6"
if { $lang == "fortran" } {
set d_pattern "\\(4,5\\)"
set f_pattern "$hex 'abcdef\\\\000'"
} else {
set d_pattern "4 \\+ 5i"
set f_pattern "$hex \"abcdef\""
}
# When value-printing pointers in GDB, GDB will try and look for any
# associated symbol and print it after the pointer as "<SYMBOL>". For
# this test Intel and LLVM compilers move g to the .bss section, thus
# creating a symbol, while the GNU compiler stack keeps g purely on the
# stack.
set g_pattern "$hex\( <\[^\r\n\]+>\)?"
set args_pattern [multi_line \
"a = 1" \
"b = 2" \
"c = 3" \
"d = ${d_pattern}" \
"f = ${f_pattern}" \
"g = ${g_pattern}" ]
gdb_test "info args" $args_pattern \
"info args in frame #6"
if { $lang == "fortran" } {
set g_val_pattern " = \\( a = 1\\.5, b = 2\\.5 \\)"
} else {
set g_val_pattern " = \\{a = 1\\.5, b = 2\\.5\\}"
}
gdb_test "print *g" "${g_val_pattern}" \
"print object pointed to by g"
gdb_test "up" "#7\\s+$hex in mixed_func_1b .*" \
"up to frame #7"
gdb_test "info frame" "source language fortran\..*" \
"info frame in frame #7"
if { $lang == "c" || $lang == "c++" } {
set d_pattern "4 \\+ 5i"
set e_pattern "\"abcdef\""
set g_val_pattern "\{a = 1.5, b = 2.5\}"
} else {
set d_pattern "\\(4,5\\)"
set e_pattern "'abcdef'"
set g_val_pattern "\\( a = 1.5, b = 2.5 \\)"
}
# The last argument here in info args is compiler generated. It
# contains length of the passed array e (we are in mixed_func_1b here).
# For gfortran and ifx the compilers conveniently named this '_e',
# ifort however prints .tmp.E.len_V$4a. As is seems unreasonable to
# test for an artificially created name and as at the same time all
# 3 tested compilers emit this argument, we only check for its
# existence and its value (6).
set args_pattern [multi_line \
"a = 1" \
"b = 2" \
"c = 3" \
"d = ${d_pattern}" \
"e = ${e_pattern}" \
"g = ${g_val_pattern}" \
".* = 6" ]
gdb_test "info args" $args_pattern \
"info args in frame #7"
}
}
run_tests "auto"
run_tests "fortran"
run_tests "c"
run_tests "c++"