|  | # Copyright (C) 2016-2021 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/>. | 
|  |  | 
|  | # This unwinder never does any unwinding.  It'll (pretend to) "sniff" | 
|  | # the frame and ultimately return None, indicating that actual unwinding | 
|  | # should be performed by some other unwinder. | 
|  | # | 
|  | # But, prior to returning None, it will attempt to obtain the value | 
|  | # associated with a symbol via a call to gdb.parse_and_eval().  In | 
|  | # the course of doing this evaluation, GDB will potentially access | 
|  | # some frames, leading to the possibility of a recursive invocation of | 
|  | # this unwinder.  If that should happen, code contained herein detects | 
|  | # that and prints a message which will cause some of the associated | 
|  | # tests to FAIL. | 
|  |  | 
|  | import gdb | 
|  | from gdb.unwinder import Unwinder | 
|  |  | 
|  |  | 
|  | class TestUnwinder(Unwinder): | 
|  |  | 
|  | count = 0 | 
|  |  | 
|  | @classmethod | 
|  | def reset_count(cls): | 
|  | cls.count = 0 | 
|  |  | 
|  | @classmethod | 
|  | def inc_count(cls): | 
|  | cls.count += 1 | 
|  |  | 
|  | test = "check_undefined_symbol" | 
|  |  | 
|  | @classmethod | 
|  | def set_test(cls, test): | 
|  | cls.test = test | 
|  |  | 
|  | def __init__(self): | 
|  | Unwinder.__init__(self, "test unwinder") | 
|  | self.recurse_level = 0 | 
|  |  | 
|  | def __call__(self, pending_frame): | 
|  |  | 
|  | if self.recurse_level > 0: | 
|  | gdb.write("TestUnwinder: Recursion detected - returning early.\n") | 
|  | return None | 
|  |  | 
|  | self.recurse_level += 1 | 
|  | TestUnwinder.inc_count() | 
|  |  | 
|  | if TestUnwinder.test == "check_user_reg_pc": | 
|  |  | 
|  | pc = pending_frame.read_register("pc") | 
|  | pc_as_int = int(pc.cast(gdb.lookup_type("int"))) | 
|  | # gdb.write("In unwinder: pc=%x\n" % pc_as_int) | 
|  |  | 
|  | elif TestUnwinder.test == "check_pae_pc": | 
|  |  | 
|  | pc = gdb.parse_and_eval("$pc") | 
|  | pc_as_int = int(pc.cast(gdb.lookup_type("int"))) | 
|  | # gdb.write("In unwinder: pc=%x\n" % pc_as_int) | 
|  |  | 
|  | elif TestUnwinder.test == "check_undefined_symbol": | 
|  |  | 
|  | try: | 
|  | val = gdb.parse_and_eval("undefined_symbol") | 
|  |  | 
|  | except Exception as arg: | 
|  | pass | 
|  |  | 
|  | self.recurse_level -= 1 | 
|  |  | 
|  | return None | 
|  |  | 
|  |  | 
|  | gdb.unwinder.register_unwinder(None, TestUnwinder(), True) | 
|  | gdb.write("Python script imported\n") |