binutils-gdb/gdb/testsuite/gdb.python/py-finish-breakpoint.c
Hannes Domani 80ffe72264 Fix gdb.FinishBreakpoint when returning to an inlined function
Currently, when creating a gdb.FinishBreakpoint in a function
called from an inline frame, it will never be hit:
```
(gdb) py fb=gdb.FinishBreakpoint()
Temporary breakpoint 1 at 0x13f1917b4: file C:/src/repos/binutils-gdb.git/gdb/testsuite/gdb.python/py-finish-breakpoint.c, line 47.
(gdb) c
Continuing.
Thread-specific breakpoint 1 deleted - thread 1 no longer in the thread list.
[Inferior 1 (process 1208) exited normally]
```

The reason is that the frame_id of a breakpoint has to be the
ID of a real frame, ignoring any inline frames.

With this fixed, it's working correctly:
```
(gdb) py fb=gdb.FinishBreakpoint()
Temporary breakpoint 1 at 0x13f5617b4: file C:/src/repos/binutils-gdb.git/gdb/testsuite/gdb.python/py-finish-breakpoint.c, line 47.
(gdb) c
Continuing.

Breakpoint 1, increase_inlined (a=0x40fa5c) at C:/src/repos/binutils-gdb.git/gdb/testsuite/gdb.python/py-finish-breakpoint.c:47
(gdb) py print(fb.return_value)
-8
```

Approved-By: Tom Tromey <tom@tromey.com>
2023-12-12 15:57:14 +01:00

119 lines
2.1 KiB
C

/* This testcase is part of GDB, the GNU debugger.
Copyright 2011-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 <http://www.gnu.org/licenses/>. */
#include <setjmp.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
/* Defined in py-events-shlib.h. */
extern void do_nothing (void);
int increase_1 (int *a)
{
*a += 1;
return -5;
}
void increase (int *a)
{
increase_1 (a);
}
int increase_2 (int *a)
{
*a += 10;
return -8;
}
inline void __attribute__((always_inline))
increase_inlined (int *a)
{
increase_2 (a);
*a += 5;
}
int
test_1 (int i, int j)
{
return i == j;
}
int
test (int i, int j)
{
return test_1 (i, j);
}
int
call_longjmp_1 (jmp_buf *buf)
{
longjmp (*buf, 1);
}
int
call_longjmp (jmp_buf *buf)
{
call_longjmp_1 (buf);
return 0;
}
void
test_exec_exit (const char *self_exec)
{
if (self_exec == NULL)
exit (0);
else
execl (self_exec, self_exec, "exit", (char *)0);
}
int main (int argc, char *argv[])
{
jmp_buf env;
int foo = 5;
int bar = 42;
int i, j;
if (argc == 2 && strcmp (argv[1], "exit") == 0)
return 0;
do_nothing ();
i = 0;
/* Break at increase. */
increase (&i);
increase (&i);
increase (&i);
increase_inlined (&i);
for (i = 0; i < 10; i++)
{
j += 1; /* Condition Break. */
}
if (setjmp (env) == 0) /* longjmp caught */
{
call_longjmp (&env);
}
else
j += 1; /* after longjmp. */
test_exec_exit (argv[0]);
return j; /* Break at end. */
}