| # Copyright (C) 2024 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/>. |
| |
| import gdb |
| from gdb.unwinder import FrameId, Unwinder |
| |
| # Cached FrameId. See set_break_bt_here_frame_id for details. |
| break_bt_here_frame_id = None |
| |
| |
| def set_break_bt_here_frame_id(pc, cfa): |
| """Call this to pre-calculate the FrameId for the frame our unwinder |
| is going to claim, this avoids us having to actually figure out a |
| frame-id within the unwinder, something which is going to be hard |
| to do in a cross-target way. |
| |
| Instead we first run the test without the Python unwinder in |
| place, use 'maint print frame-id' to record the frame-id, then, |
| after loading this Python script, we all this function to record |
| the frame-id that the unwinder should use.""" |
| global break_bt_here_frame_id |
| break_bt_here_frame_id = FrameId(cfa, pc) |
| |
| |
| class break_unwinding(Unwinder): |
| """An unwinder for the function 'break_bt_here'. This unwinder will |
| claim any frame for the function in question, but doesn't provide |
| any unwound register values. Importantly, we don't provide a |
| previous $pc value, this means that if we are stopped in |
| 'break_bt_here' then we should fail to unwind beyond frame #0.""" |
| |
| def __init__(self): |
| Unwinder.__init__(self, "break unwinding") |
| |
| def __call__(self, pending_frame): |
| pc_desc = pending_frame.architecture().registers().find("pc") |
| pc = pending_frame.read_register(pc_desc) |
| |
| if pc.is_optimized_out: |
| return None |
| |
| block = gdb.block_for_pc(pc) |
| if block == None: |
| return None |
| func = block.function |
| if func == None: |
| return None |
| if str(func) != "break_bt_here": |
| return None |
| |
| global break_bt_here_frame_id |
| if break_bt_here_frame_id is None: |
| return None |
| |
| return pending_frame.create_unwind_info(break_bt_here_frame_id) |
| |
| |
| gdb.unwinder.register_unwinder(None, break_unwinding(), True) |