LLDB mainline
BreakpointResolverScripted.cpp
Go to the documentation of this file.
1//===-- BreakpointResolverScripted.cpp ------------------------------------===//
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
10
11
13#include "lldb/Core/Debugger.h"
14#include "lldb/Core/Module.h"
15#include "lldb/Core/Section.h"
19#include "lldb/Target/Process.h"
20#include "lldb/Target/Target.h"
21#include "lldb/Utility/Log.h"
23
24using namespace lldb;
25using namespace lldb_private;
26
27// BreakpointResolverScripted:
29 const BreakpointSP &bkpt, const llvm::StringRef class_name,
30 lldb::SearchDepth depth, const StructuredDataImpl &args_data)
32 m_class_name(std::string(class_name)), m_depth(depth), m_args(args_data) {
33 if (bkpt)
35}
36
38 BreakpointSP breakpoint_sp) {
39 // This version has to be called with a valid breakpoint_sp
40 // But the interface might have been made before we sent the breakpoint to
41 // the interface. If so, do that here:
42 assert(breakpoint_sp);
43 if (m_interface_sp) {
44 if (!m_breakpoint_sent) {
45 m_interface_sp->SetBreakpoint(breakpoint_sp);
46 m_breakpoint_sent = true;
47 }
48 return;
49 }
50
51 if (m_class_name.empty())
52 return;
53
54 if (!breakpoint_sp)
55 return;
56
57 TargetSP target_sp = breakpoint_sp->GetTargetSP();
58 if (target_sp)
59 CreateImplementationIfNeeded(*target_sp.get(), breakpoint_sp);
60}
61
63 Target &target, BreakpointSP breakpoint_sp) {
64 if (m_interface_sp) {
65 if (!m_breakpoint_sent && breakpoint_sp) {
66 m_interface_sp->SetBreakpoint(breakpoint_sp);
67 m_breakpoint_sent = true;
68 }
69 return;
70 }
71
72 ScriptInterpreter *script_interp =
74 if (!script_interp)
75 return;
76
77 if (!m_interface_sp)
79
80 if (!m_interface_sp) {
82 "BreakpointResolverScripted::%s () - ERROR: %s", __FUNCTION__,
83 "Script interpreter couldn't create Scripted Breakpoint Interface");
84 return;
85 }
86
87 StructuredData::ObjectSP args_obj = m_args.GetObjectSP();
89 if (args_obj && args_obj->GetType() == lldb::eStructuredDataTypeDictionary)
90 args_dict_sp =
91 std::static_pointer_cast<StructuredData::Dictionary>(args_obj);
92 ScriptedMetadata scripted_metadata(m_class_name, args_dict_sp);
93 auto obj_or_err =
94 m_interface_sp->CreatePluginObject(scripted_metadata, breakpoint_sp);
95 if (!obj_or_err) {
96 m_interface_sp.reset();
97 m_error = Status::FromError(obj_or_err.takeError());
98 return;
99 }
100 StructuredData::ObjectSP object_sp = *obj_or_err;
101 if (!object_sp || !object_sp->IsValid()) {
103 "ScriptedBreakpoint::%s () - ERROR: %s", __FUNCTION__,
104 "Failed to create valid script object");
105 }
106 if (breakpoint_sp)
107 m_breakpoint_sent = true;
108}
109
111 Target &target, BreakpointResolverSP original_sp) {
112 // At this point neither resolver has been assigned a breakpoint, so pass
113 // in an empty one.
115 if (!m_interface_sp)
116 return false;
117
118 StructuredData::ObjectSP serialized_sp =
119 original_sp->SerializeToStructuredData();
120 StructuredDataImpl impl(serialized_sp);
121 return m_interface_sp->OverridesResolver(target, impl);
122}
123
127
129 const StructuredData::Dictionary &options_dict, Status &error) {
130 llvm::StringRef class_name;
131 bool success;
132
133 success = options_dict.GetValueForKeyAsString(
135 if (!success) {
136 error =
137 Status::FromErrorString("BRFL::CFSD: Couldn't find class name entry.");
138 return nullptr;
139 }
140 // The Python function will actually provide the search depth, this is a
141 // placeholder.
143
144 StructuredDataImpl args_data_impl;
145 StructuredData::Dictionary *args_dict = nullptr;
147 args_dict))
148 args_data_impl.SetObjectSP(args_dict->shared_from_this());
149 return std::make_shared<BreakpointResolverScripted>(nullptr, class_name,
150 depth, args_data_impl);
151}
152
155 StructuredData::DictionarySP options_dict_sp(
157
158 options_dict_sp->AddStringItem(GetKey(OptionNames::PythonClassName),
160 if (m_args.IsValid())
161 options_dict_sp->AddItem(GetKey(OptionNames::ScriptArgs),
162 m_args.GetObjectSP());
163
164 return WrapOptionsDict(options_dict_sp);
165}
166
168 return GetBreakpoint()->GetTarget().GetDebugger().GetScriptInterpreter();
169}
170
172 SearchFilter &filter, SymbolContext &context, Address *addr) {
173 bool should_continue = true;
174 if (!m_interface_sp)
176
177 should_continue = m_interface_sp->ResolverCallback(context);
178 if (should_continue)
180
182}
183
187 if (m_interface_sp)
188 depth = m_interface_sp->GetDepth();
189
190 return depth;
191}
192
194 StructuredData::GenericSP generic_sp;
195 std::optional<std::string> short_help;
196
198
199 if (m_interface_sp) {
200 short_help = m_interface_sp->GetShortHelp();
201 }
202 if (short_help && !short_help->empty())
203 s->PutCString(short_help->c_str());
204 else
205 s->Printf("python class = %s", m_class_name.c_str());
206}
207
211 if (m_interface_sp)
212 return m_interface_sp->GetLocationDescription(bp_loc_sp, level);
213 return {};
214}
215
218 lldb::BreakpointLocationSP bp_loc_sp) {
219 if (m_interface_sp)
220 return m_interface_sp->WasHit(frame_sp, bp_loc_sp);
221 return {};
222}
223
225
228 return std::make_shared<BreakpointResolverScripted>(breakpoint, m_class_name,
229 m_depth, m_args);
230}
static llvm::raw_ostream & error(Stream &strm)
A section + offset based address class.
Definition Address.h:62
BreakpointResolverScripted(const lldb::BreakpointSP &bkpt, const llvm::StringRef class_name, lldb::SearchDepth depth, const StructuredDataImpl &args_data)
lldb::BreakpointResolverSP CopyForBreakpoint(lldb::BreakpointSP &breakpoint) override
std::optional< std::string > GetLocationDescription(lldb::BreakpointLocationSP bp_loc_sp, lldb::DescriptionLevel level)
void GetDescription(Stream *s) override
Prints a canonical description for the breakpoint to the stream s.
StructuredData::ObjectSP SerializeToStructuredData() override
void Dump(Stream *s) const override
Standard "Dump" method. At present it does nothing.
Searcher::CallbackReturn SearchCallback(SearchFilter &filter, SymbolContext &context, Address *addr) override
void CreateImplementationIfNeeded(lldb::BreakpointSP bkpt)
lldb::BreakpointLocationSP WasHit(lldb::StackFrameSP frame_sp, lldb::BreakpointLocationSP bp_loc_sp)
lldb::ScriptedBreakpointInterfaceSP m_interface_sp
bool OverridesResolver(Target &target, lldb::BreakpointResolverSP original_sp) override
The resolver_sp won't have had its breakpoint set by the time we are checking the Override,...
static lldb::BreakpointResolverSP CreateFromStructuredData(const StructuredData::Dictionary &options_dict, Status &error)
static const char * GetKey(OptionNames enum_value)
StructuredData::DictionarySP WrapOptionsDict(StructuredData::DictionarySP options_dict_sp)
lldb::BreakpointSP GetBreakpoint() const
This gets the breakpoint for this resolver.
BreakpointResolver(const lldb::BreakpointSP &bkpt, unsigned char resolverType, lldb::addr_t offset=0, bool offset_is_insn_count=false)
The breakpoint resolver need to have a breakpoint for "ResolveBreakpoint to make sense.
ScriptInterpreter * GetScriptInterpreter(bool can_create=true, std::optional< lldb::ScriptLanguage > language={})
virtual lldb::ScriptedBreakpointInterfaceSP CreateScriptedBreakpointInterface()
General Outline: Provides the callback and search depth for the SearchFilter search.
An error handling class.
Definition Status.h:118
static Status FromErrorStringWithFormat(const char *format,...) __attribute__((format(printf
Definition Status.cpp:106
static Status FromErrorString(const char *str)
Definition Status.h:141
static Status FromError(llvm::Error error)
Avoid using this in new code. Migrate APIs to llvm::Expected instead.
Definition Status.cpp:136
A stream class that can stream formatted output to a file.
Definition Stream.h:28
size_t Printf(const char *format,...) __attribute__((format(printf
Output printf formatted output to the stream.
Definition Stream.cpp:134
size_t PutCString(llvm::StringRef cstr)
Output a C string to the stream.
Definition Stream.cpp:63
void SetObjectSP(const StructuredData::ObjectSP &obj)
bool GetValueForKeyAsString(llvm::StringRef key, llvm::StringRef &result) const
bool GetValueForKeyAsDictionary(llvm::StringRef key, Dictionary *&result) const
std::shared_ptr< Generic > GenericSP
std::shared_ptr< Dictionary > DictionarySP
std::shared_ptr< Object > ObjectSP
Defines a symbol context baton that can be handed other debug core functions.
Debugger & GetDebugger() const
Definition Target.h:1324
A class that represents a running process on the host machine.
std::shared_ptr< lldb_private::StackFrame > StackFrameSP
std::shared_ptr< lldb_private::BreakpointResolver > BreakpointResolverSP
std::shared_ptr< lldb_private::BreakpointLocation > BreakpointLocationSP
DescriptionLevel
Description levels for "void GetDescription(Stream *, DescriptionLevel)" calls.
std::shared_ptr< lldb_private::Breakpoint > BreakpointSP
std::shared_ptr< lldb_private::Target > TargetSP
@ eStructuredDataTypeDictionary