# Copyright 2022-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 setting a breakpoint at "f(std::string)". # # GDB should be able to expand the std::string typedef, and then set # the breakpoint using the resulting name. In the Itanium ABI's # mangling scheme, "std::string", "std::istream", "std::iostream", # "std::ostream" are special, though, they have corresponding standard # abbreviations. The libiberty demangler only expands these standard # abbreviations to their full non-typedef underlying type if the # DMGL_VERBOSE option is requested. By default it expands them to the # user-friendly "std::string", etc. typedefs. GDB didn't use to use # that option, and would instead prevent expansion of the # "std::string" (etc.) standard-abbreviation typedefs at # breakpoint-set type, such that the function name used for function # lookup would match the "std::string" present in the function's # non-DMGL_VERBOSE demangled name. # # For example (DMGL_VERBOSE): # # $ echo "_Z1fSs" | c++filt # f(std::basic_string, std::allocator >) # # vs (no DMGL_VERBOSE): # # $ echo "_Z1fSs" | c++filt --no-verbose # f(std::string) # # This design broke setting a breakpoint at "f(std::string)" when the # libstdc++ C++11 ABI was introduced, as the "f(std::string)" # function's mangled name no longer uses a standard substitution for # std::string... # # I.e., with the libstdc++ C++11 ABI, we now have (and DMGL_VERBOSE # makes no difference): # # $ echo _Z1fNSt7__cxx1112basic_stringIcSt11char_traitsIcESaIcEEE | c++filt # f(std::__cxx11::basic_string, std::allocator >) # # So nowadays, GDB always uses DMGL_VERBOSE and no longer prevents # std::string (etc.) typedef expansion. This test exercises both # pre-C++11 and C++11 ABIs for this reason. On non-libstdc++ systems # where _GLIBCXX_USE_CXX11_ABI has no effect, we just end up running # the test twice with whatever ABI is used. standard_testfile .cc if { [skip_cplus_tests] } { continue } # CXX11_ABI specifies the value to define _GLIBCXX_USE_CXX11_ABI as. proc test {cxx11_abi} { global srcdir subdir srcfile binfile testfile set options \ [list c++ debug additional_flags=-D_GLIBCXX_USE_CXX11_ABI=$cxx11_abi] if { [gdb_compile \ "${srcdir}/${subdir}/${srcfile}" "${binfile}-${cxx11_abi}.o" \ object $options] != "" } { untested "failed to compile" return -1 } clean_restart ${testfile}-${cxx11_abi}.o # Since we're debugging an .o file, GDB doesn't figure out we're # debugging C++ code and the current language when auto, is # guessed as C. gdb_test_no_output "set language c++" # Get the type std::string is a typedef for. We'll try to set a # breakpoint using the expanded type too. set realtype "" set type "std::string" gdb_test_multiple "whatis /r $type" "" { -re -wrap "type = (\[^\r\n\]+)" { set realtype $expect_out(1,string) gdb_assert {![string eq "$realtype" "$type"]} \ $gdb_test_name } } # GDB should be able to expand the std::string typedef in the # function prototype using C++ logic even if the current language # is C. foreach_with_prefix lang {"c" "c++"} { gdb_test_no_output "set language $lang" gdb_test "break f($type)" "$srcfile, line $::decimal\\." if { $realtype != "" } { gdb_test "break f($realtype)" "$srcfile, line $::decimal\\." } } } foreach_with_prefix _GLIBCXX_USE_CXX11_ABI {0 1} { test $_GLIBCXX_USE_CXX11_ABI }