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
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;
{
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)
# 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"