# Copyright 2017-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 file is part of the gdb testsuite. # Tests which verify (or not) that GDB can access in-scope variables # when stopped within an OpenMP parallel region. standard_testfile set have_nested_function_support 0 set opts {openmp debug} if [support_nested_function_tests] { lappend opts "additional_flags=-DHAVE_NESTED_FUNCTION_SUPPORT" set have_nested_function_support 1 } if {[prepare_for_testing "failed to prepare" $testfile $srcfile $opts]} { return -1 } # gdb_openmp_setup may be defined to set auto-load safe-path and possibly # sysroot. These settings are required for gdb to be able to find # the libgomp python plugin. (sysroot only needs to be defined for # remote debugging.) # # This approach has both pros and cons. On the plus side, it's easy # to automatically set a precise auto-load safe-path. (It's easy because # the output of ldd on the binary may be examined to learn the location # of libgomp.so.) # # However, making these settings is also a drawback due to potentially # overriding settings made by a board file. Therefore, this proc # is optional and will only be called if it's defined. if {[info procs gdb_openmp_setup] != ""} { if {[gdb_openmp_setup $binfile] != ""} { untested "could not set up OpenMP environment" return -1 } } if {![runto_main]} { return -1 } # We want to invoke setup_kfail (and in some cases setup_xfail) when # GDB does not yet have support for finding the values of variables in # (non-master) threads. We'll check this by looking at the output of # "maint print thread-parent". If this command is undefined, then GDB # does not yet have thread parent support, and it makes sense to kfail # tests which won't work. It's possible for GDB to have this support, # but not work. E.g. it may be the case that the plugin doesn't # exist or is not found. We may eventually need to add additional # constraints related to setting allow_kfail to 0. But, for the moment, # this simple test should be sufficient. set allow_kfail 1 gdb_test_multiple "maint print thread-parent" "maint print thread-parent" { -re "Undefined maintenance print command.*$gdb_prompt" { pass "maint print thread-parent (does not exist)" } -re "No parent found.*" { pass "maint print thread-parent" set allow_kfail 0 } } # Determine whether to xfail some of the tests based on GCC version. # # This may need to be tweaked somewhat. Testing shows that GCC 7.3.1 # needs the xfails. GCC 8.3.1 and 9.1.1 do not. The assumption made # below is that all versions of gcc 8 and above won't require the # XFAIL setup and that all versions of gcc 7 and below will, but it's # possible that there are versions in between 7.3.1 and 8.3.1 for # which this assumption is invalid. set have_older_gcc 0 if {[test_compiler_info {gcc-[0-7]-*}]} { set have_older_gcc 1 } # maybe_setup_kfail will set up a kfail for gdb/22214 when COND holds in # addition to considering the values of $have_older_gcc and $allow_kfail. # # When $have_older_gcc evaluates to true, setup_xfail will invoked # instead. proc maybe_setup_kfail {cond} { global have_older_gcc allow_kfail if {$have_older_gcc} { setup_xfail *-*-* } elseif {[uplevel 1 [list expr $cond]] && $allow_kfail} { setup_kfail "gdb/22214" *-*-* } } with_test_prefix "single_scope" { gdb_breakpoint [gdb_get_line_number "single_scope: thread_num="] gdb_breakpoint [gdb_get_line_number "single_scope: s1="] foreach pref {"first thread" "second thread"} { with_test_prefix $pref { gdb_continue_to_breakpoint "at printf" if {$have_older_gcc} { setup_xfail "*-*-*" } set thread_num [get_valueof "" thread_num "unknown"] if {$have_older_gcc} { setup_xfail "*-*-*" } gdb_test "print s1" "= -41" gdb_test "print s2" "= \[12\]02" if {$have_older_gcc} { setup_xfail "*-*-*" } gdb_test "print s3" "= -43" gdb_test "print i1" "= 11" gdb_test "print i2" "= \[12]12" maybe_setup_kfail {$thread_num != 0} gdb_test "print i3" "= 13" } } with_test_prefix "after parallel region" { gdb_continue_to_breakpoint "at printf" gdb_test "print s1" "= -41" gdb_test "print s2" "= -42" gdb_test "print s3" "= -43" gdb_test "print i1" "= 11" gdb_test "print i2" "= 12" gdb_test "print i3" "= 13" } } with_test_prefix "multi_scope" { gdb_breakpoint [gdb_get_line_number "multi_scope: thread_num="] gdb_breakpoint [gdb_get_line_number "multi_scope: i01="] foreach pref {"first thread" "second thread"} { with_test_prefix $pref { gdb_continue_to_breakpoint "at printf" if {$have_older_gcc} { setup_xfail "*-*-*" } set thread_num [get_valueof "" thread_num "unknown"] gdb_test "print i01" "= 1" maybe_setup_kfail {$thread_num != 0} gdb_test "print i02" "= 2" gdb_test "print i11" "= 11" maybe_setup_kfail {$thread_num != 0} gdb_test "print i12" "= 12" gdb_test "print i21" "= \[12\]21" maybe_setup_kfail {$thread_num != 0} gdb_test "print i22" "= 22" gdb_test "print file_scope_var" "= 9876" } } with_test_prefix "after parallel" { gdb_continue_to_breakpoint "at printf" gdb_test "print i01" "= 1" gdb_test "print i02" "= 2" gdb_test "print i11" "= 11" gdb_test "print i12" "= 12" gdb_test "print i21" "= -21" gdb_test "print i22" "= 22" gdb_test "print file_scope_var" "= 9876" } } # Nested functions in C are a GNU extension, so only do the nested function # tests if compiling with -DHAVE_NESTED_FUNCTION_SUPPORT was successful. if $have_nested_function_support { with_test_prefix "nested_func" { gdb_breakpoint [gdb_get_line_number "nested_func: tn="] foreach call_prefix {"1st call" "2nd call"} { with_test_prefix $call_prefix { foreach thread_prefix {"1st thread" "2nd thread"} { with_test_prefix $thread_prefix { gdb_continue_to_breakpoint "at printf" if {$have_older_gcc} { setup_xfail "*-*-*" } set thread_num [get_valueof "" "tn" "unknown"] gdb_test "print file_scope_var" "= 9876" if {$have_older_gcc} { setup_xfail *-*-* } gdb_test "print s1" "= -42" if {$call_prefix eq "1st call"} { gdb_test "print i" "= 1" } else { gdb_test "print i" "= 101" } gdb_test "print j" "= \[12\]000" maybe_setup_kfail {$thread_num != 0} if {$call_prefix eq "1st call"} { gdb_test "print k" "= 3" } else { gdb_test "print k" "= 103" } if {$call_prefix eq "1st call"} { gdb_test "print p" "= 10" } else { gdb_test "print p" "= 20" } gdb_test "print q" "= \[12\]001" maybe_setup_kfail {$thread_num != 0} if {$call_prefix eq "1st call"} { gdb_test "print r" "= 12" } else { gdb_test "print r" "= 22" } gdb_test "print x" "= 4" gdb_test "print y" "= \[12\]002" maybe_setup_kfail {$thread_num != 0} gdb_test "print z" "= 6" if {$have_older_gcc} { setup_xfail "*-*-*" } gdb_test "print tn" "= \[01\]" } } } } } } with_test_prefix "nested_parallel" { gdb_breakpoint [gdb_get_line_number "nested_parallel (inner threads)"] with_test_prefix "inner_threads" { foreach pref {"1st stop" "2nd stop" "3rd stop" "4th stop"} { with_test_prefix $pref { gdb_continue_to_breakpoint "at printf" # Don't need setup_xfail here due to fact that num is made # made known to the inner parallel region. set thread_num [get_valueof "" "num" "unknown"] if {$have_older_gcc} { setup_xfail "*-*-*" } set inner_thread_num [get_valueof "" "inner_num" "unknown"] gdb_test "print file_scope_var" "= 9876" gdb_test "print num" "= \[01\]" maybe_setup_kfail {$thread_num != 0 || $inner_thread_num != 0} gdb_test "print i" "= 1" maybe_setup_kfail {$thread_num != 0 || $inner_thread_num != 0} gdb_test "print j" "= 2" if {$have_older_gcc || ($inner_thread_num != 0 && $allow_kfail)} { setup_xfail *-*-* } gdb_test "print l" "= 10\[24\]" if {$have_older_gcc ||( $inner_thread_num != 0 && $allow_kfail)} { setup_xfail *-*-* } gdb_test "print k" "= 10\[13\]" } } } with_test_prefix "outer_threads" { gdb_breakpoint [gdb_get_line_number "nested_parallel (outer threads)"] with_test_prefix "outer stop" { gdb_continue_to_breakpoint "at printf" if {$have_older_gcc} { setup_xfail "*-*-*" } # Use get_local_valueof instead of get_valueof to avoid picking up # some random 'num' in a shared library. set thread_num [get_local_valueof "num" "unknown"] gdb_test "print file_scope_var" "= 9876" if {$have_older_gcc} { setup_xfail "*-*-*" } gdb_test "print num" "= \[01\]" maybe_setup_kfail {$thread_num != 0} gdb_test "print i" "= 1" maybe_setup_kfail {$thread_num != 0} gdb_test "print j" "= 2" gdb_test "print l" "= 10\[24\]" if {$have_older_gcc} { setup_xfail "*-*-*" } gdb_test "print k" "= 10\[13\]" } } }