# Copyright 2017-2020 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 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]} {
untested "could not run to 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\]"
}
}
}