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.
210 lines
3.6 KiB
C
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;
|
|
}
|