# Copyright 2020-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 . # 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. if {[skip_fortran_tests]} { return -1 } 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++} \ $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 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 " = \\{ = \\{num1 = 1, num2 = 2, num3 = 3\\}, = \\{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 "". 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 \\)" } set args_pattern [multi_line \ "a = 1" \ "b = 2" \ "c = 3" \ "d = ${d_pattern}" \ "e = ${e_pattern}" \ "g = ${g_val_pattern}" \ "_e = 6" ] gdb_test "info args" $args_pattern \ "info args in frame #7" } } run_tests "auto" run_tests "fortran" run_tests "c" run_tests "c++"