# 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 . # Tests of core file memory accesses when mmap() has been used to # create a "hole" of zeroes over pre-existing memory regions. See # coremaker2.c for details. # are we on a target board if {![isnative]} { return } # Some of these tests will only work on GNU/Linux due to the # fact that Linux core files includes a section describing # memory address to file mappings. We'll use set_up_xfail for the # affected tests. As other targets become supported, the condition # can be changed accordingly. set xfail 0 if { ![istarget *-linux*] } { set xfail 1 } standard_testfile coremaker2.c if {[build_executable $testfile.exp $testfile $srcfile debug] == -1} { untested "failed to compile" return -1 } set corefile [core_find $binfile {}] if {$corefile == ""} { return 0 } gdb_start gdb_reinitialize_dir $srcdir/$subdir gdb_load ${binfile} # Attempt to load the core file. gdb_test_multiple "core-file $corefile" "core-file command" { -re ".* program is being debugged already.*y or n. $" { # gdb_load may connect us to a gdbserver. send_gdb "y\n" exp_continue } -re "Core was generated by .*corefile.*\r\n\#0 .*\(\).*\r\n$gdb_prompt $" { pass "core-file command" } -re "Core was generated by .*\r\n\#0 .*\(\).*\r\n$gdb_prompt $" { pass "core-file command (with bad program name)" } -re ".*registers from core file: File in wrong format.* $" { fail "core-file command (could not read registers from core file)" } } # Perform the "interesting" tests which check the contents of certain # memory regions. proc do_tests { } { global xfail # Check contents of beginning of buf_rw and buf_ro. gdb_test {print/x buf_rw[0]@4} {\{0x6b, 0x6b, 0x6b, 0x6b\}} gdb_test {print/x buf_ro[0]@4} {\{0xc5, 0xc5, 0xc5, 0xc5\}} # Check for correct contents at beginning of mbuf_rw and mbuf_ro. gdb_test {print/x mbuf_rw[0]@4} {\{0x0, 0x0, 0x0, 0x0\}} if { $xfail } { setup_xfail "*-*-*" } gdb_test {print/x mbuf_ro[0]@4} {\{0x0, 0x0, 0x0, 0x0\}} # Check contents of mbuf_rw and mbuf_ro at the end of these regions. gdb_test {print/x mbuf_rw[pagesize-4]@4} {\{0x0, 0x0, 0x0, 0x0\}} if { $xfail } { setup_xfail "*-*-*" } gdb_test {print/x mbuf_ro[pagesize-4]@4} {\{0x0, 0x0, 0x0, 0x0\}} # Check contents of mbuf_rw and mbuf_ro, right before the hole, # overlapping into the beginning of these mmap'd regions. gdb_test {print/x mbuf_rw[-3]@6} {\{0x6b, 0x6b, 0x6b, 0x0, 0x0, 0x0\}} if { $xfail } { setup_xfail "*-*-*" } gdb_test {print/x mbuf_ro[-3]@6} {\{0xc5, 0xc5, 0xc5, 0x0, 0x0, 0x0\}} # Likewise, at the end of the mbuf_rw and mbuf_ro, with overlap. # If this test FAILs, it's probably a genuine bug unrelated to whether # the core file includes a section describing memory address to file # mappings or not. (So don't xfail it!) gdb_test {print/x mbuf_rw[pagesize-3]@6} {\{0x0, 0x0, 0x0, 0x6b, 0x6b, 0x6b\}} if { $xfail } { setup_xfail "*-*-*" } gdb_test {print/x mbuf_ro[pagesize-3]@6} {\{0x0, 0x0, 0x0, 0xc5, 0xc5, 0xc5\}} # Check contents of (what should be) buf_rw and buf_ro immediately after # mbuf_rw and mbuf_ro holes. gdb_test {print/x mbuf_rw[pagesize]@4} {\{0x6b, 0x6b, 0x6b, 0x6b\}} gdb_test {print/x mbuf_ro[pagesize]@4} {\{0xc5, 0xc5, 0xc5, 0xc5\}} # Check contents at ends of buf_rw and buf_rw. gdb_test {print/x buf_rw[sizeof(buf_rw)-4]@4} {\{0x6b, 0x6b, 0x6b, 0x6b\}} gdb_test {print/x buf_ro[sizeof(buf_ro)-4]@4} {\{0xc5, 0xc5, 0xc5, 0xc5\}} } # Run tests with kernel-produced core file. with_test_prefix "kernel core" { do_tests } # Verify that "maint print core-file-backed-mappings" exists and does # not crash GDB. If it produces any output at all, make sure that # that output at least mentions binfile. set test "maint print core-file-backed-mappings" gdb_test_multiple $test "" { -re ".*$binfile.*$gdb_prompt $" { pass $test } -re "^$test\[\r\n\]*$gdb_prompt $" { pass "$test (no output)" } } # Test again with executable renamed during loading of core file. with_test_prefix "renamed binfile" { # Don't load $binfile in this call to clean_restart. (BFD will # complain that $binfile has disappeared after the rename if it's # loaded first.) clean_restart # Rename $binfile so that it won't be found during loading of # the core file. set hide_binfile [standard_output_file "${testfile}.hide"] remote_exec host "mv -f $binfile $hide_binfile" # Load core file - check that a warning is printed. global xfail if { $xfail } { setup_xfail "*-*-*" } gdb_test "core-file $corefile" \ "warning: Can't open file.*during.* note processing.*Core was generated by .*\#0 .*\(\)" \ "load core file without having first loaded binfile" # Restore $binfile and then load it. remote_exec host "mv -f $hide_binfile $binfile" gdb_load ${binfile} do_tests } # Restart and run to the abort call. clean_restart $binfile if {![runto_main]} { return } gdb_breakpoint [gdb_get_line_number "abort"] gdb_continue_to_breakpoint "at abort" # Do not execute abort call; instead, invoke gcore command to make a # gdb-produced core file. set corefile [standard_output_file gcore.test] set core_supported [gdb_gcore_cmd "$corefile" "save a corefile"] if {!$core_supported} { return } # maint print-core-file-backed-mappings shouldn't produce any output # when not debugging a core file. gdb_test_no_output "maint print core-file-backed-mappings" \ "maint print core-file-backed-mapping with no core file" clean_restart $binfile set core_loaded [gdb_core_cmd "$corefile" "re-load generated corefile"] if { $core_loaded == -1 } { # No use proceeding from here. return } # Run tests using gcore-produced core file. with_test_prefix "gcore core" { do_tests }