diff --git a/gdb/python/py-finishbreakpoint.c b/gdb/python/py-finishbreakpoint.c index 0b5dbeb5413..ea6662f0a86 100644 --- a/gdb/python/py-finishbreakpoint.c +++ b/gdb/python/py-finishbreakpoint.c @@ -55,6 +55,10 @@ struct finish_breakpoint_object the function; Py_None if the value is not computable; NULL if GDB is not stopped at a FinishBreakpoint. */ PyObject *return_value; + + /* The initiating frame for this operation, used to decide when we have + left this frame. */ + struct frame_id initiating_frame; }; extern PyTypeObject finish_breakpoint_object_type @@ -310,6 +314,7 @@ bpfinishpy_init (PyObject *self, PyObject *args, PyObject *kwargs) self_bpfinish->py_bp.bp->frame_id = frame_id; self_bpfinish->py_bp.is_finish_bp = 1; + self_bpfinish->initiating_frame = get_frame_id (frame); /* Bind the breakpoint with the current program space. */ self_bpfinish->py_bp.bp->pspace = current_program_space; @@ -360,9 +365,11 @@ bpfinishpy_detect_out_scope_cb (struct breakpoint *b, { try { + struct frame_id initiating_frame = finish_bp->initiating_frame; + if (b->pspace == current_inferior ()->pspace && (!target_has_registers () - || frame_find_by_id (b->frame_id) == NULL)) + || frame_find_by_id (initiating_frame) == NULL)) bpfinishpy_out_of_scope (finish_bp); } catch (const gdb_exception &except) diff --git a/gdb/testsuite/gdb.python/py-finish-breakpoint2.cc b/gdb/testsuite/gdb.python/py-finish-breakpoint2.cc index 84234238f74..7e53d5393ae 100644 --- a/gdb/testsuite/gdb.python/py-finish-breakpoint2.cc +++ b/gdb/testsuite/gdb.python/py-finish-breakpoint2.cc @@ -17,9 +17,13 @@ #include +int i; + void throw_exception_1 (int e) { + i += 1; /* Finish breakpoint is set here. */ + i += 1; /* Break before exception. */ throw new int (e); } @@ -32,7 +36,6 @@ throw_exception (int e) int main (void) { - int i; try { throw_exception_1 (10); diff --git a/gdb/testsuite/gdb.python/py-finish-breakpoint2.exp b/gdb/testsuite/gdb.python/py-finish-breakpoint2.exp index 30758473d40..bd1e96bb4fe 100644 --- a/gdb/testsuite/gdb.python/py-finish-breakpoint2.exp +++ b/gdb/testsuite/gdb.python/py-finish-breakpoint2.exp @@ -38,21 +38,60 @@ set pyfile [gdb_remote_download host \ # Check FinishBreakpoints against C++ exceptions # +gdb_breakpoint [gdb_get_line_number "Break before exception"] gdb_breakpoint [gdb_get_line_number "Break after exception 2"] gdb_test "source $pyfile" ".*Python script imported.*" \ "import python scripts" gdb_breakpoint "throw_exception_1" + +# +# Part 1. +# + gdb_test "continue" "Breakpoint .*throw_exception_1.*" "run to exception 1" -gdb_test "python print (len(gdb.breakpoints()))" "3" "check BP count" +# Count breakpoints before setting finishbreakpoint. +gdb_test "python print (len(gdb.breakpoints()))" "4" "check BP count" + gdb_test "python ExceptionFinishBreakpoint(gdb.newest_frame())" \ "init ExceptionFinishBreakpoint" "set FinishBP after the exception" -gdb_test "continue" ".*stopped at ExceptionFinishBreakpoint.*" "check FinishBreakpoint in catch()" -gdb_test "python print (len(gdb.breakpoints()))" "3" "check finish BP removal" -gdb_test "continue" ".*Breakpoint.* throw_exception_1.*" "continue to second exception" +gdb_test "continue" "Break before exception.*" \ + "break before exception" + +set need_continue 0 +gdb_test_multiple "continue" "finishBP out-of-scope" { + -re -wrap "exception did not finish.*" { + # Out-of-scope. For instance on x86_64 with unix/-m32. + pass $gdb_test_name + } + -re -wrap "stopped at ExceptionFinishBreakpoint.*" { + # Triggered despite the fact that the function call never finished. + # It just so happens to be that the frame return address at which the + # breakpoint is set, is also the first instruction after the exception + # has been handled. For instance on x86_64 with unix/-m64. + kfail python/29909 $gdb_test_name + set need_continue 1 + } +} + +# Count breakpoints, check that the finishbreakpoint has been removed. +gdb_test "python print (len(gdb.breakpoints()))" "4" "check finish BP removal" + +# +# Part 2. +# + +if { $need_continue } { + gdb_test "continue" ".*Breakpoint.* throw_exception_1.*" \ + "continue to second exception" +} gdb_test "python ExceptionFinishBreakpoint(gdb.newest_frame())" \ "init ExceptionFinishBreakpoint" "set FinishBP after the exception again" + +gdb_test "continue" "Break before exception.*" \ + "break before exception again" + gdb_test "continue" ".*exception did not finish.*" "FinishBreakpoint with exception thrown not caught"