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
29 "Provides synthetic frames via scripting",
31}
32
36
37llvm::Expected<lldb::SyntheticFrameProviderSP>
39 lldb::StackFrameListSP input_frames,
40 const ScriptedFrameProviderDescriptor &descriptor) {
41 if (!input_frames)
42 return llvm::createStringError(
43 "failed to create scripted frame provider: invalid input frames");
44
45 Thread &thread = input_frames->GetThread();
46 ProcessSP process_sp = thread.GetProcess();
47 if (!process_sp)
48 return nullptr;
49
50 if (!descriptor.IsValid())
51 return llvm::createStringError(
52 "failed to create scripted frame provider: invalid scripted metadata");
53
54 if (!descriptor.AppliesToThread(thread))
55 return nullptr;
56
57 ScriptInterpreter *script_interp =
58 process_sp->GetTarget().GetDebugger().GetScriptInterpreter();
59 if (!script_interp)
60 return llvm::createStringError("cannot create scripted frame provider: No "
61 "script interpreter installed");
62
65 if (!interface_sp)
66 return llvm::createStringError(
67 "cannot create scripted frame provider: script interpreter couldn't "
68 "create Scripted Frame Provider Interface");
69
70 const ScriptedMetadataSP scripted_metadata = descriptor.scripted_metadata_sp;
71
72 // If we shouldn't attach a frame provider to this thread, just exit early.
73 if (!interface_sp->AppliesToThread(scripted_metadata->GetClassName(),
74 thread.shared_from_this()))
75 return nullptr;
76
77 auto obj_or_err = interface_sp->CreatePluginObject(
78 scripted_metadata->GetClassName(), input_frames,
79 scripted_metadata->GetArgsSP());
80 if (!obj_or_err)
81 return obj_or_err.takeError();
82
83 StructuredData::ObjectSP object_sp = *obj_or_err;
84 if (!object_sp || !object_sp->IsValid())
85 return llvm::createStringError(
86 "cannot create scripted frame provider: failed to create valid scripted"
87 "frame provider object");
88
89 return std::make_shared<ScriptedFrameProvider>(input_frames, interface_sp,
90 descriptor);
91}
92
99
101
103 if (!m_interface_sp)
104 return {};
105
106 return m_interface_sp->GetDescription(m_descriptor.GetName());
107}
108
109llvm::Expected<StackFrameSP>
111 if (!m_interface_sp)
112 return llvm::createStringError(
113 "cannot get stack frame: scripted frame provider not initialized");
114
115 auto create_frame_from_dict =
116 [this](StructuredData::Dictionary *dict,
117 uint32_t index) -> llvm::Expected<StackFrameSP> {
119 if (!dict->GetValueForKeyAsInteger("pc", pc))
120 return llvm::createStringError(
121 "missing 'pc' key from scripted frame dictionary");
122
123 Address symbol_addr;
124 symbol_addr.SetLoadAddress(pc, &GetThread().GetProcess()->GetTarget());
125
127 const bool cfa_is_valid = false;
128 const bool artificial = false;
129 const bool behaves_like_zeroth_frame = false;
130 SymbolContext sc;
131 symbol_addr.CalculateSymbolContext(&sc);
132
133 ThreadSP thread_sp = GetThread().shared_from_this();
134 return std::make_shared<StackFrame>(thread_sp, index, index, cfa,
135 cfa_is_valid, pc,
136 StackFrame::Kind::Synthetic, artificial,
137 behaves_like_zeroth_frame, &sc);
138 };
139
140 auto create_frame_from_script_object =
141 [this](
142 StructuredData::ObjectSP object_sp) -> llvm::Expected<StackFrameSP> {
144 if (!object_sp || !object_sp->GetAsGeneric())
145 return llvm::createStringError("invalid script object");
146
147 ThreadSP thread_sp = GetThread().shared_from_this();
148 auto frame_or_error = ScriptedFrame::Create(thread_sp, nullptr, nullptr,
149 object_sp->GetAsGeneric());
150
151 if (!frame_or_error) {
153 LLVM_PRETTY_FUNCTION, toString(frame_or_error.takeError()), error);
154 return error.ToError();
155 }
156
157 return *frame_or_error;
158 };
159
160 StructuredData::ObjectSP obj_sp = m_interface_sp->GetFrameAtIndex(idx);
161
162 // None/null means no more frames or error.
163 if (!obj_sp || !obj_sp->IsValid())
164 return llvm::createStringError("invalid script object returned for frame " +
165 llvm::Twine(idx));
166
167 StackFrameSP synth_frame_sp = nullptr;
169 obj_sp->GetAsUnsignedInteger()) {
170 uint32_t real_frame_index = int_obj->GetValue();
171 if (real_frame_index < m_input_frames->GetNumFrames()) {
172 StackFrameSP real_frame_sp =
173 m_input_frames->GetFrameAtIndex(real_frame_index);
174 synth_frame_sp =
175 (real_frame_index == idx)
176 ? real_frame_sp
177 : std::make_shared<BorrowedStackFrame>(real_frame_sp, idx);
178 }
179 } else if (StructuredData::Dictionary *dict = obj_sp->GetAsDictionary()) {
180 // Check if it's a dictionary describing a frame.
181 auto frame_from_dict_or_err = create_frame_from_dict(dict, idx);
182 if (!frame_from_dict_or_err) {
183 return llvm::createStringError(llvm::Twine(
184 "couldn't create frame from dictionary at index " + llvm::Twine(idx) +
185 ": " + toString(frame_from_dict_or_err.takeError())));
186 }
187 synth_frame_sp = *frame_from_dict_or_err;
188 } else if (obj_sp->GetAsGeneric()) {
189 // It's a ScriptedFrame object.
190 auto frame_from_script_obj_or_err = create_frame_from_script_object(obj_sp);
191 if (!frame_from_script_obj_or_err) {
192 return llvm::createStringError(
193 llvm::Twine("couldn't create frame from script object at index " +
194 llvm::Twine(idx) + ": " +
195 toString(frame_from_script_obj_or_err.takeError())));
196 }
197 synth_frame_sp = *frame_from_script_obj_or_err;
198 } else {
199 return llvm::createStringError(
200 llvm::Twine("invalid return type from get_frame_at_index at index " +
201 llvm::Twine(idx)));
202 }
203
204 if (!synth_frame_sp)
205 return llvm::createStringError(
206 llvm::Twine("failed to create frame at index " + llvm::Twine(idx)));
207
208 synth_frame_sp->SetFrameIndex(idx);
209
210 return synth_frame_sp;
211}
212
213namespace lldb_private {
217
221} // namespace lldb_private
static llvm::raw_ostream & error(Stream &strm)
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::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.
void lldb_terminate_ScriptedFrameProvider()
void lldb_initialize_ScriptedFrameProvider()
const char * toString(AppleArm64ExceptionClass EC)
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.