| # Copyright 2023-2024 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/>. |
| |
| # Test amd64 displaced stepping over a call instruction that calls to |
| # itself. This is pretty unlikely to be seen in the wild, but does |
| # test a corner case of our displaced step handling. |
| |
| require is_x86_64_m64_target |
| |
| set newline "\[\r\n\]*" |
| |
| set opts {debug nopie} |
| standard_testfile .S -alarm.c |
| |
| if { [prepare_for_testing "failed to prepare" $testfile "$srcfile $srcfile2" $opts] } { |
| return -1 |
| } |
| |
| gdb_test "set displaced-stepping on" "" |
| gdb_test "show displaced-stepping" ".* displaced stepping .* is on.*" |
| |
| if {![runto_main]} { |
| return 0 |
| } |
| |
| # Proceed to the test function. |
| gdb_breakpoint "test_call" |
| gdb_continue_to_breakpoint "test_call" |
| |
| # Get the current stack pointer value. |
| set sp [get_hexadecimal_valueof "\$sp" "*UNKNOWN*"] |
| |
| # Get the address of the next instruction. |
| set next_insn_addr "" |
| gdb_test_multiple "x/2i \$pc" "get address of next insn" { |
| -re "\r\n=> $hex \[^\r\n\]+\r\n" { |
| exp_continue |
| } |
| -re "^ ($hex) \[^\r\n\]+\r\n" { |
| set next_insn_addr $expect_out(1,string) |
| exp_continue |
| } |
| -re "^$::gdb_prompt $" { |
| gdb_assert {![string equal $next_insn_addr ""]} \ |
| $gdb_test_name |
| } |
| } |
| |
| # Clear the slot on the stack and confirm it was set to zero. |
| set sp [expr $sp - 0x8] |
| gdb_test_no_output "set {unsigned long long} $sp = 0" \ |
| "clear stack slot" |
| set zero_val 0x[format %016x 0] |
| gdb_test "x/1gx 0x[format %x $sp]" "$hex:\\s+${zero_val}" \ |
| "check return address slot was set to zero" |
| |
| # Single step. |
| gdb_test "stepi" \ |
| "Breakpoint $decimal, test_call \\(\\) at .*" |
| |
| # Check stack pointer was updated to the expected value. |
| set new_sp [get_hexadecimal_valueof "\$sp" "*UNKNOWN*" \ |
| "get stack pointer after step"] |
| gdb_assert {[expr $sp == $new_sp]} \ |
| "check stack pointer was updated as expected" |
| |
| # Check the contents of the stack were updated to the expected value. |
| set next_insn_addr 0x[format %016X $next_insn_addr] |
| gdb_test "x/1gx 0x[format %x $sp]" "$hex:\\s+$next_insn_addr" \ |
| "check return address was updated correctly" |