From 2755241d02d7f129848adb6d76316b8116f346f8 Mon Sep 17 00:00:00 2001 From: Tom Tromey Date: Mon, 12 Feb 2024 10:12:26 -0700 Subject: Add return value to DAP scope MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit A bug report in the DAP specification repository pointed out that it is typical for DAP implementations to put a function's return value into the outermost scope. This patch changes gdb to follow this convention. Bug: https://sourceware.org/bugzilla/show_bug.cgi?id=31341 Reviewed-By: Kévin Le Gouguec --- gdb/python/lib/gdb/dap/events.py | 3 +++ gdb/python/lib/gdb/dap/scopes.py | 39 +++++++++++++++++++++++++++++++++++++-- 2 files changed, 40 insertions(+), 2 deletions(-) (limited to 'gdb/python') diff --git a/gdb/python/lib/gdb/dap/events.py b/gdb/python/lib/gdb/dap/events.py index 4130222..928f23f 100644 --- a/gdb/python/lib/gdb/dap/events.py +++ b/gdb/python/lib/gdb/dap/events.py @@ -15,6 +15,7 @@ import gdb +from .scopes import set_finish_value from .server import send_event from .startup import exec_and_log, in_gdb_thread, log from .modules import is_module, make_module @@ -218,6 +219,8 @@ def _on_stop(event): } if isinstance(event, gdb.BreakpointEvent): obj["hitBreakpointIds"] = [x.number for x in event.breakpoints] + if hasattr(event, "details") and "finish-value" in event.details: + set_finish_value(event.details["finish-value"]) global _expected_pause global _expected_stop_reason diff --git a/gdb/python/lib/gdb/dap/scopes.py b/gdb/python/lib/gdb/dap/scopes.py index ff55325..be2c382 100644 --- a/gdb/python/lib/gdb/dap/scopes.py +++ b/gdb/python/lib/gdb/dap/scopes.py @@ -25,17 +25,32 @@ from .varref import BaseReference frame_to_scope = {} +# If the most recent stop was due to a 'finish', and the return value +# could be computed, then this holds that value. Otherwise it holds +# None. +_last_return_value = None + + # When the inferior is re-started, we erase all scope references. See # the section "Lifetime of Objects References" in the spec. @in_gdb_thread def clear_scopes(event): global frame_to_scope frame_to_scope = {} + global _last_return_value + _last_return_value = None gdb.events.cont.connect(clear_scopes) +@in_gdb_thread +def set_finish_value(val): + """Set the current 'finish' value on a stop.""" + global _last_return_value + _last_return_value = val + + # A helper function to compute the value of a symbol. SYM is either a # gdb.Symbol, or an object implementing the SymValueWrapper interface. # FRAME is a frame wrapper, as produced by a frame filter. Returns a @@ -76,7 +91,7 @@ class _ScopeReference(BaseReference): result["presentationHint"] = self.hint # How would we know? result["expensive"] = False - result["namedVariables"] = len(self.var_list) + result["namedVariables"] = self.child_count() if self.line is not None: result["line"] = self.line # FIXME construct a Source object @@ -93,6 +108,22 @@ class _ScopeReference(BaseReference): return symbol_value(self.var_list[idx], self.frame) +# A _ScopeReference that prepends the most recent return value. Note +# that this object is only created if such a value actually exists. +class _FinishScopeReference(_ScopeReference): + def __init__(self, *args): + super().__init__(*args) + + def child_count(self): + return super().child_count() + 1 + + def fetch_one_child(self, idx): + if idx == 0: + global _last_return_value + return ("(return)", _last_return_value) + return super().fetch_one_child(idx - 1) + + class _RegisterReference(_ScopeReference): def __init__(self, name, frame): super().__init__( @@ -109,6 +140,7 @@ class _RegisterReference(_ScopeReference): @request("scopes") def scopes(*, frameId: int, **extra): + global _last_return_value global frame_to_scope if frameId in frame_to_scope: scopes = frame_to_scope[frameId] @@ -120,10 +152,13 @@ def scopes(*, frameId: int, **extra): args = tuple(frame.frame_args() or ()) if args: scopes.append(_ScopeReference("Arguments", "arguments", frame, args)) + has_return_value = frameId == 0 and _last_return_value is not None # Make sure to handle the None case as well as the empty # iterator case. locs = tuple(frame.frame_locals() or ()) - if locs: + if has_return_value: + scopes.append(_FinishScopeReference("Locals", "locals", frame, locs)) + elif locs: scopes.append(_ScopeReference("Locals", "locals", frame, locs)) scopes.append(_RegisterReference("Registers", frame)) frame_to_scope[frameId] = scopes -- cgit v1.1