LLDB mainline
ScriptedFrameProvider.cpp
Go to the documentation of this file.
1//===----------------------------------------------------------------------===//
2//
3// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4// See https://llvm.org/LICENSE.txt for license information.
5// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6//
7//===----------------------------------------------------------------------===//
8
11#include "lldb/Core/Debugger.h"
16#include "lldb/Target/Process.h"
18#include "lldb/Target/Thread.h"
20#include "lldb/Utility/Status.h"
21#include "llvm/Support/Error.h"
22#include <cstdint>
23
24using namespace lldb;
25using namespace lldb_private;
26
28
31 "Provides synthetic frames via scripting",
33}
34
38
39llvm::Expected<lldb::SyntheticFrameProviderSP>
41 lldb::StackFrameListSP input_frames,
42 const ScriptedFrameProviderDescriptor &descriptor) {
43 if (!input_frames)
44 return llvm::createStringError(
45 "failed to create scripted frame provider: invalid input frames");
46
47 Thread &thread = input_frames->GetThread();
48 ProcessSP process_sp = thread.GetProcess();
49 if (!process_sp)
50 return nullptr;
51
52 if (!descriptor.IsValid())
53 return llvm::createStringError(
54 "failed to create scripted frame provider: invalid scripted metadata");
55
56 if (!descriptor.AppliesToThread(thread))
57 return nullptr;
58
59 ScriptInterpreter *script_interp =
60 process_sp->GetTarget().GetDebugger().GetScriptInterpreter();
61 if (!script_interp)
62 return llvm::createStringError("cannot create scripted frame provider: No "
63 "script interpreter installed");
64
67 if (!interface_sp)
68 return llvm::createStringError(
69 "cannot create scripted frame provider: script interpreter couldn't "
70 "create Scripted Frame Provider Interface");
71
72 const ScriptedMetadataSP scripted_metadata = descriptor.scripted_metadata_sp;
73
74 // If we shouldn't attach a frame provider to this thread, just exit early.
75 if (!interface_sp->AppliesToThread(scripted_metadata->GetClassName(),
76 thread.shared_from_this()))
77 return nullptr;
78
79 auto obj_or_err = interface_sp->CreatePluginObject(
80 scripted_metadata->GetClassName(), input_frames,
81 scripted_metadata->GetArgsSP());
82 if (!obj_or_err)
83 return obj_or_err.takeError();
84
85 StructuredData::ObjectSP object_sp = *obj_or_err;
86 if (!object_sp || !object_sp->IsValid())
87 return llvm::createStringError(
88 "cannot create scripted frame provider: failed to create valid scripted"
89 "frame provider object");
90
91 return std::make_shared<ScriptedFrameProvider>(input_frames, interface_sp,
92 descriptor);
93}
94
101
103
105 if (!m_interface_sp)
106 return {};
107
108 return m_interface_sp->GetDescription(m_descriptor.GetName());
109}
110
111std::optional<uint32_t> ScriptedFrameProvider::GetPriority() const {
112 if (!m_interface_sp)
113 return std::nullopt;
114
115 return m_interface_sp->GetPriority(m_descriptor.GetName());
116}
117
118llvm::Expected<StackFrameSP>
120 if (!m_interface_sp)
121 return llvm::createStringError(
122 "cannot get stack frame: scripted frame provider not initialized");
123
124 auto create_frame_from_dict =
125 [this](StructuredData::Dictionary *dict,
126 uint32_t index) -> llvm::Expected<StackFrameSP> {
128 if (!dict->GetValueForKeyAsInteger("pc", pc))
129 return llvm::createStringError(
130 "missing 'pc' key from scripted frame dictionary");
131
132 Address symbol_addr;
133 symbol_addr.SetLoadAddress(pc, &GetThread().GetProcess()->GetTarget());
134
136 const bool cfa_is_valid = false;
137 const bool artificial = false;
138 const bool behaves_like_zeroth_frame = false;
139 SymbolContext sc;
140 symbol_addr.CalculateSymbolContext(&sc);
141
142 ThreadSP thread_sp = GetThread().shared_from_this();
143 return std::make_shared<StackFrame>(thread_sp, index, index, cfa,
144 cfa_is_valid, pc,
145 StackFrame::Kind::Synthetic, artificial,
146 behaves_like_zeroth_frame, &sc);
147 };
148
149 auto create_frame_from_script_object =
150 [this](
151 StructuredData::ObjectSP object_sp) -> llvm::Expected<StackFrameSP> {
153 if (!object_sp || !object_sp->GetAsGeneric())
154 return llvm::createStringError("invalid script object");
155
156 ThreadSP thread_sp = GetThread().shared_from_this();
157 auto frame_or_error = ScriptedFrame::Create(thread_sp, nullptr, nullptr,
158 object_sp->GetAsGeneric());
159
160 if (!frame_or_error) {
162 LLVM_PRETTY_FUNCTION, toString(frame_or_error.takeError()), error);
163 return error.ToError();
164 }
165
166 return *frame_or_error;
167 };
168
169 StructuredData::ObjectSP obj_sp = m_interface_sp->GetFrameAtIndex(idx);
170
171 // None/null means no more frames or error.
172 if (!obj_sp || !obj_sp->IsValid())
173 return llvm::createStringError("invalid script object returned for frame " +
174 llvm::Twine(idx));
175
176 StackFrameSP synth_frame_sp = nullptr;
178 obj_sp->GetAsUnsignedInteger()) {
179 uint32_t real_frame_index = int_obj->GetValue();
180 if (real_frame_index < m_input_frames->GetNumFrames()) {
181 StackFrameSP real_frame_sp =
182 m_input_frames->GetFrameAtIndex(real_frame_index);
183 synth_frame_sp =
184 (real_frame_index == idx)
185 ? real_frame_sp
186 : std::make_shared<BorrowedStackFrame>(real_frame_sp, idx);
187 }
188 } else if (StructuredData::Dictionary *dict = obj_sp->GetAsDictionary()) {
189 // Check if it's a dictionary describing a frame.
190 auto frame_from_dict_or_err = create_frame_from_dict(dict, idx);
191 if (!frame_from_dict_or_err) {
192 return llvm::createStringError(llvm::Twine(
193 "couldn't create frame from dictionary at index " + llvm::Twine(idx) +
194 ": " + toString(frame_from_dict_or_err.takeError())));
195 }
196 synth_frame_sp = *frame_from_dict_or_err;
197 } else if (obj_sp->GetAsGeneric()) {
198 // It's a ScriptedFrame object.
199 auto frame_from_script_obj_or_err = create_frame_from_script_object(obj_sp);
200 if (!frame_from_script_obj_or_err) {
201 return llvm::createStringError(
202 llvm::Twine("couldn't create frame from script object at index " +
203 llvm::Twine(idx) + ": " +
204 toString(frame_from_script_obj_or_err.takeError())));
205 }
206 synth_frame_sp = *frame_from_script_obj_or_err;
207 } else {
208 return llvm::createStringError(
209 llvm::Twine("invalid return type from get_frame_at_index at index " +
210 llvm::Twine(idx)));
211 }
212
213 if (!synth_frame_sp)
214 return llvm::createStringError(
215 llvm::Twine("failed to create frame at index " + llvm::Twine(idx)));
216
217 synth_frame_sp->SetFrameIndex(idx);
218
219 return synth_frame_sp;
220}
static llvm::raw_ostream & error(Stream &strm)
#define LLDB_PLUGIN_DEFINE(PluginName)
A section + offset based address class.
Definition Address.h:62
bool SetLoadAddress(lldb::addr_t load_addr, Target *target, bool allow_section_end=false)
Set the address to represent load_addr.
Definition Address.cpp:1035
uint32_t CalculateSymbolContext(SymbolContext *sc, lldb::SymbolContextItem resolve_scope=lldb::eSymbolContextEverything) const
Reconstruct a symbol context from an address.
Definition Address.cpp:820
static bool RegisterPlugin(llvm::StringRef name, llvm::StringRef description, ABICreateInstance create_callback)
static bool UnregisterPlugin(ABICreateInstance create_callback)
virtual lldb::ScriptedFrameProviderInterfaceSP CreateScriptedFrameProviderInterface()
lldb::ScriptedFrameProviderInterfaceSP m_interface_sp
static llvm::StringRef GetPluginNameStatic()
llvm::Expected< lldb::StackFrameSP > GetFrameAtIndex(uint32_t idx) override
Get a single stack frame at the specified index.
static llvm::Expected< lldb::SyntheticFrameProviderSP > CreateInstance(lldb::StackFrameListSP input_frames, const ScriptedFrameProviderDescriptor &descriptor)
ScriptedFrameProvider(lldb::StackFrameListSP input_frames, lldb::ScriptedFrameProviderInterfaceSP interface_sp, const ScriptedFrameProviderDescriptor &descriptor)
const ScriptedFrameProviderDescriptor & m_descriptor
std::optional< uint32_t > GetPriority() const override
Get the priority of this frame provider.
std::string GetDescription() const override
static llvm::Expected< std::shared_ptr< ScriptedFrame > > Create(lldb::ThreadSP thread_sp, lldb::ScriptedThreadInterfaceSP scripted_thread_interface_sp, StructuredData::DictionarySP args_sp, StructuredData::Generic *script_object=nullptr)
Create a ScriptedFrame from a object instanciated in the script interpreter.
static Ret ErrorWithMessage(llvm::StringRef caller_name, llvm::StringRef error_msg, Status &error, LLDBLog log_category=LLDBLog::Process)
@ Synthetic
An synthetic stack frame (e.g.
Definition StackFrame.h:72
An error handling class.
Definition Status.h:118
bool GetValueForKeyAsInteger(llvm::StringRef key, IntType &result) const
std::shared_ptr< Object > ObjectSP
Integer< uint64_t > UnsignedInteger
Defines a symbol context baton that can be handed other debug core functions.
Thread & GetThread()
Get the thread associated with this provider.
SyntheticFrameProvider(lldb::StackFrameListSP input_frames)
#define LLDB_INVALID_ADDRESS
A class that represents a running process on the host machine.
std::string toString(FormatterBytecode::OpCodes op)
std::shared_ptr< lldb_private::StackFrame > StackFrameSP
std::shared_ptr< lldb_private::Thread > ThreadSP
std::shared_ptr< lldb_private::ScriptedMetadata > ScriptedMetadataSP
std::shared_ptr< lldb_private::Process > ProcessSP
std::shared_ptr< lldb_private::ScriptedFrameProviderInterface > ScriptedFrameProviderInterfaceSP
uint64_t addr_t
Definition lldb-types.h:80
std::shared_ptr< lldb_private::StackFrameList > StackFrameListSP
This struct contains the metadata needed to instantiate a frame provider and optional filters to cont...
lldb::ScriptedMetadataSP scripted_metadata_sp
Metadata for instantiating the provider (e.g. script class name and args).
bool AppliesToThread(Thread &thread) const
Check if this descriptor applies to the given thread.
bool IsValid() const
Check if this descriptor has valid metadata for script-based providers.