8sa1-binutils-gdb/gdb/testsuite/gdb.opt/inline-break.c
Keith Seitz ddfe970e6b Don't elide all inlined frames
This patch essentially causes GDB to treat inlined frames like "normal"
frames from the user's perspective.  This means, for example, that when a
user sets a breakpoint in an inlined function, GDB will now actually stop
"in" that function.

Using the test case from breakpoints/17534,

3	static inline void NVIC_EnableIRQ(int IRQn)
4	{
5	  volatile int y;
6	  y = IRQn;
7	}
8
9	__attribute__( ( always_inline ) ) static inline void __WFI(void)
10	{
11	    __asm volatile ("nop");
12	}
13
14	int main(void) {
15
16	    x= 42;
17
18	    if (x)
19	      NVIC_EnableIRQ(16);
20	    else
21	      NVIC_EnableIRQ(18);
(gdb) b NVIC_EnableIRQ
Breakpoint 1 at 0x4003e4: NVIC_EnableIRQ. (2 locations)
(gdb) r
Starting program: 17534

Breakpoint 1, main () at 17534.c:19
19	      NVIC_EnableIRQ(16);

Because skip_inline_frames currently skips every inlined frame, GDB "stops"
in the caller.  This patch adds a new parameter to skip_inline_frames
that allows us to pass in a bpstat stop chain.  The breakpoint locations
on the stop chain can be used to determine if we've stopped inside an inline
function (due to a user breakpoint).  If we have, we do not elide the frame.

With this patch, GDB now reports that the inferior has stopped inside the
inlined function:

(gdb) r
Starting program: 17534

Breakpoint 1, NVIC_EnableIRQ (IRQn=16) at 17534.c:6
6	  y = IRQn;

Many thanks to Jan and Pedro for guidance on this.

gdb/ChangeLog:

	* breakpoint.c (build_bpstat_chain): New function, moved from
	bpstat_stop_status.
	(bpstat_stop_status): Add optional parameter, `stop_chain'.
	If no stop chain is passed, call build_bpstat_chain to build it.
	* breakpoint.h (build_bpstat_chain): Declare.
	(bpstat_stop_status): Move documentation here from breakpoint.c.
	* infrun.c (handle_signal_stop): Before eliding inlined frames,
	build the stop chain and pass it to skip_inline_frames.
	Pass this stop chain to bpstat_stop_status.
	* inline-frame.c: Include breakpoint.h.
	(stopped_by_user_bp_inline_frame): New function.
	(skip_inline_frames): Add parameter `stop_chain'.
	Move documention to inline-frame.h.
	If non-NULL, use stopped_by_user_bp_inline_frame to determine
	whether the frame should be elided.
	* inline-frame.h (skip_inline_frames): Add parameter `stop_chain'.
	Add moved documentation and update for new parameter.

gdb/testsuite/ChangeLog:

	* gdb.ada/bp_inlined_func.exp: Update inlined frame locations
	in expected breakpoint stop locations.
	* gdb.dwarf2/implptr.exp (implptr_test_baz): Use up/down to
	move to proper scope to test variable values.
	* gdb.opt/inline-break.c (inline_func1, not_inline_func1)
	(inline_func2, not_inline_func2, inline_func3, not_inline_func3):
	New functions.
	(main): Call not_inline_func3.
	* gdb.opt/inline-break.exp: Start inferior and set breakpoints at
	inline_func1, inline_func2, and inline_func3.  Test that when each
	breakpoint is hit, GDB properly reports both the stop location
	and the backtrace. Repeat tests for temporary breakpoints.
2018-05-17 12:15:11 -07:00

210 lines
3.6 KiB
C

/* This testcase is part of GDB, the GNU debugger.
Copyright (C) 2012-2018 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/>. */
/* The file ../gdb.dwarf2/inline-break.S was generated manually from
this file, and should be regenerated if this file is modified. */
#ifdef __GNUC__
# define ATTR __attribute__((gnu_inline)) __attribute__((always_inline))
#else
# define ATTR
#endif
/* A static inlined function that is called once. */
static inline ATTR int
func1 (int x)
{
return x * 23;
}
/* A non-static inlined function that is called once. */
inline ATTR int
func2 (int x)
{
return x * 17;
}
/* A static inlined function that calls another static inlined
function. */
static inline ATTR int
func3b (int x)
{
return x < 14 ? 1 : 2;
}
static inline ATTR int
func3a (int x)
{
return func3b (x * 23);
}
/* A non-static inlined function that calls a static inlined
function. */
static inline ATTR int
func4b (int x)
{
return x < 13 ? 1 : 2;
}
inline ATTR int
func4a (int x)
{
return func4b (x * 17);
}
/* A static inlined function that calls a non-static inlined
function. */
inline ATTR int
func5b (int x)
{
return x < 12 ? 1 : 2;
}
static inline ATTR int
func5a (int x)
{
return func5b (x * 23);
}
/* A non-static inlined function that calls another non-static inlined
function. */
inline ATTR int
func6b (int x)
{
return x < 14 ? 3 : 2;
}
inline ATTR int
func6a (int x)
{
return func6b (x * 17);
}
/* A static inlined function that is called more than once. */
static inline ATTR int
func7b (int x)
{
return x < 23 ? 1 : 4;
}
static inline ATTR int
func7a (int x)
{
return func7b (x * 29);
}
/* A non-static inlined function that is called more than once. */
inline ATTR int
func8b (int x)
{
return x < 7 ? 11 : 9;
}
static inline ATTR int
func8a (int x)
{
return func8b (x * 31);
}
static inline ATTR int
inline_func1 (int x)
{
int y = 1; /* inline_func1 */
return y + x;
}
static int
not_inline_func1 (int x)
{
int y = 2; /* not_inline_func1 */
return y + inline_func1 (x);
}
inline ATTR int
inline_func2 (int x)
{
int y = 3; /* inline_func2 */
return y + not_inline_func1 (x);
}
int
not_inline_func2 (int x)
{
int y = 4; /* not_inline_func2 */
return y + inline_func2 (x);
}
static inline ATTR int
inline_func3 (int x)
{
int y = 5; /* inline_func3 */
return y + not_inline_func2 (x);
}
static int
not_inline_func3 (int x)
{
int y = 6; /* not_inline_func3 */
return y + inline_func3 (x);
}
/* Entry point. */
int
main (int argc, char *argv[])
{
/* Declaring x as volatile here prevents GCC from combining calls.
If GCC is allowed to combine calls then some of them end up with
no instructions at all, so there is no specific address for GDB
to set a breakpoint at. */
volatile int x = argc;
x = func1 (x);
x = func2 (x);
x = func3a (x);
x = func4a (x);
x = func5a (x);
x = func6a (x);
x = func7a (x) + func7b (x);
x = func8a (x) + func8b (x);
x = not_inline_func3 (-21);
return x;
}