LLDB mainline
DynamicLoader.cpp
Go to the documentation of this file.
1//===-- DynamicLoader.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#include "lldb/Core/Debugger.h"
12#include "lldb/Core/Module.h"
16#include "lldb/Core/Progress.h"
17#include "lldb/Core/Section.h"
21#include "lldb/Target/Process.h"
22#include "lldb/Target/Target.h"
25#include "lldb/Utility/Log.h"
27
28#include "llvm/ADT/StringRef.h"
29
30#include <memory>
31
32#include <cassert>
33
34using namespace lldb;
35using namespace lldb_private;
36
38 llvm::StringRef plugin_name) {
39 DynamicLoaderCreateInstance create_callback = nullptr;
40 if (!plugin_name.empty()) {
41 create_callback =
43 if (create_callback) {
44 std::unique_ptr<DynamicLoader> instance_up(
45 create_callback(process, true));
46 if (instance_up)
47 return instance_up.release();
48 }
49 } else {
50 for (uint32_t idx = 0;
51 (create_callback =
53 nullptr;
54 ++idx) {
55 std::unique_ptr<DynamicLoader> instance_up(
56 create_callback(process, false));
57 if (instance_up)
58 return instance_up.release();
59 }
60 }
61 return nullptr;
62}
63
65
66// Accessors to the global setting as to whether to stop at image (shared
67// library) loading/unloading.
68
70 return m_process->GetStopOnSharedLibraryEvents();
71}
72
74 m_process->SetStopOnSharedLibraryEvents(stop);
75}
76
78 Target &target = m_process->GetTarget();
79 ModuleSP executable = target.GetExecutableModule();
80
81 if (executable) {
82 if (FileSystem::Instance().Exists(executable->GetFileSpec())) {
83 ModuleSpec module_spec(executable->GetFileSpec(),
84 executable->GetArchitecture());
85 auto module_sp = std::make_shared<Module>(module_spec);
86 // If we're a coredump and we already have a main executable, we don't
87 // need to reload the module list that target already has
88 if (!m_process->IsLiveDebugSession()) {
89 return executable;
90 }
91 // Check if the executable has changed and set it to the target
92 // executable if they differ.
93 if (module_sp && module_sp->GetUUID().IsValid() &&
94 executable->GetUUID().IsValid()) {
95 if (module_sp->GetUUID() != executable->GetUUID())
96 executable.reset();
97 } else if (executable->FileHasChanged()) {
98 executable.reset();
99 }
100
101 if (!executable) {
102 executable = target.GetOrCreateModule(module_spec, true /* notify */);
103 if (executable.get() != target.GetExecutableModulePointer()) {
104 // Don't load dependent images since we are in dyld where we will
105 // know and find out about all images that are loaded
106 target.SetExecutableModule(executable, eLoadDependentsNo);
107 }
108 }
109 }
110 }
111 return executable;
112}
113
115 addr_t base_addr,
116 bool base_addr_is_offset) {
117 UpdateLoadedSectionsCommon(module, base_addr, base_addr_is_offset);
118}
119
121 addr_t base_addr,
122 bool base_addr_is_offset) {
123 bool changed;
124 module->SetLoadAddress(m_process->GetTarget(), base_addr, base_addr_is_offset,
125 changed);
126}
127
129 UnloadSectionsCommon(module);
130}
131
133 Target &target = m_process->GetTarget();
134 const SectionList *sections = GetSectionListFromModule(module);
135
136 assert(sections && "SectionList missing from unloaded module.");
137
138 const size_t num_sections = sections->GetSize();
139 for (size_t i = 0; i < num_sections; ++i) {
140 SectionSP section_sp(sections->GetSectionAtIndex(i));
141 target.SetSectionUnloaded(section_sp);
142 }
143}
144
145const SectionList *
147 SectionList *sections = nullptr;
148 if (module) {
149 ObjectFile *obj_file = module->GetObjectFile();
150 if (obj_file != nullptr) {
151 sections = obj_file->GetSectionList();
152 }
153 }
154 return sections;
155}
156
158 Target &target = m_process->GetTarget();
159 ModuleSpec module_spec(file, target.GetArchitecture());
160 if (UUID uuid = m_process->FindModuleUUID(file.GetPath())) {
161 // Process may be able to augment the module_spec with UUID, e.g. ELF core.
162 module_spec.GetUUID() = uuid;
163 }
164
165 if (ModuleSP module_sp = target.GetImages().FindFirstModule(module_spec))
166 return module_sp;
167
168 if (ModuleSP module_sp =
169 target.GetOrCreateModule(module_spec, /*notify=*/false))
170 return module_sp;
171
172 return nullptr;
173}
174
176 addr_t link_map_addr,
177 addr_t base_addr,
178 bool base_addr_is_offset) {
179 if (ModuleSP module_sp = FindModuleViaTarget(file)) {
180 UpdateLoadedSections(module_sp, link_map_addr, base_addr,
181 base_addr_is_offset);
182 return module_sp;
183 }
184
185 return nullptr;
186}
187
189 llvm::StringRef name) {
190 char namebuf[80];
191 if (name.empty()) {
192 snprintf(namebuf, sizeof(namebuf), "memory-image-0x%" PRIx64, addr);
193 name = namebuf;
194 }
195 return process->ReadModuleFromMemory(FileSpec(name), addr);
196}
197
199 Process *process, llvm::StringRef name, UUID uuid, addr_t value,
200 bool value_is_offset, bool force_symbol_search, bool notify,
201 bool set_address_in_target, bool allow_memory_image_last_resort) {
202 ModuleSP memory_module_sp;
203 ModuleSP module_sp;
204 PlatformSP platform_sp = process->GetTarget().GetPlatform();
205 Target &target = process->GetTarget();
207
208 StreamString prog_str;
209 if (!name.empty()) {
210 prog_str << name.str() << " ";
211 }
212 if (uuid.IsValid())
213 prog_str << uuid.GetAsString();
214 if (value_is_offset == 0 && value != LLDB_INVALID_ADDRESS) {
215 prog_str << " at 0x";
216 prog_str.PutHex64(value);
217 }
218
219 if (!uuid.IsValid() && !value_is_offset) {
220 memory_module_sp = ReadUnnamedMemoryModule(process, value, name);
221
222 if (memory_module_sp) {
223 uuid = memory_module_sp->GetUUID();
224 if (uuid.IsValid()) {
225 prog_str << " ";
226 prog_str << uuid.GetAsString();
227 }
228 }
229 }
230 ModuleSpec module_spec;
231 module_spec.SetTarget(target.shared_from_this());
232 module_spec.GetUUID() = uuid;
233 FileSpec name_filespec(name);
234 if (FileSystem::Instance().Exists(name_filespec))
235 module_spec.GetFileSpec() = name_filespec;
236
237 if (uuid.IsValid()) {
238 Progress progress("Locating binary", prog_str.GetString().str());
239
240 // Has lldb already seen a module with this UUID?
241 // Or have external lookup enabled in DebugSymbols on macOS.
242 if (!module_sp)
243 error =
244 ModuleList::GetSharedModule(module_spec, module_sp, nullptr, nullptr);
245
246 // Can lldb's symbol/executable location schemes
247 // find an executable and symbol file.
248 if (!module_sp) {
250 StatisticsMap symbol_locator_map;
251 module_spec.GetSymbolFileSpec() =
252 PluginManager::LocateExecutableSymbolFile(module_spec, search_paths,
253 symbol_locator_map);
254 ModuleSpec objfile_module_spec =
256 symbol_locator_map);
257 module_spec.GetFileSpec() = objfile_module_spec.GetFileSpec();
258 if (FileSystem::Instance().Exists(module_spec.GetFileSpec()) &&
259 FileSystem::Instance().Exists(module_spec.GetSymbolFileSpec())) {
260 module_sp = std::make_shared<Module>(module_spec);
261 }
262
263 if (module_sp) {
264 module_sp->GetSymbolLocatorStatistics().merge(symbol_locator_map);
265 }
266 }
267
268 // If we haven't found a binary, or we don't have a SymbolFile, see
269 // if there is an external search tool that can find it.
270 if (!module_sp || !module_sp->GetSymbolFileFileSpec()) {
272 force_symbol_search);
273 if (FileSystem::Instance().Exists(module_spec.GetFileSpec())) {
274 module_sp = std::make_shared<Module>(module_spec);
275 } else if (force_symbol_search && error.AsCString("") &&
276 error.AsCString("")[0] != '\0') {
277 *target.GetDebugger().GetAsyncErrorStream() << error.AsCString();
278 }
279 }
280
281 // If we only found the executable, create a Module based on that.
282 if (!module_sp && FileSystem::Instance().Exists(module_spec.GetFileSpec()))
283 module_sp = std::make_shared<Module>(module_spec);
284 }
285
286 // If we couldn't find the binary anywhere else, as a last resort,
287 // read it out of memory.
288 if (allow_memory_image_last_resort && !module_sp.get() &&
289 value != LLDB_INVALID_ADDRESS && !value_is_offset) {
290 if (!memory_module_sp)
291 memory_module_sp = ReadUnnamedMemoryModule(process, value, name);
292 if (memory_module_sp)
293 module_sp = memory_module_sp;
294 }
295
297 if (module_sp.get()) {
298 // Ensure the Target has an architecture set in case
299 // we need it while processing this binary/eh_frame/debug info.
300 if (!target.GetArchitecture().IsValid())
301 target.SetArchitecture(module_sp->GetArchitecture());
302 target.GetImages().AppendIfNeeded(module_sp, false);
303
304 bool changed = false;
305 if (set_address_in_target) {
306 if (module_sp->GetObjectFile()) {
307 if (value != LLDB_INVALID_ADDRESS) {
308 LLDB_LOGF(log,
309 "DynamicLoader::LoadBinaryWithUUIDAndAddress Loading "
310 "binary %s UUID %s at %s 0x%" PRIx64,
311 name.str().c_str(), uuid.GetAsString().c_str(),
312 value_is_offset ? "offset" : "address", value);
313 module_sp->SetLoadAddress(target, value, value_is_offset, changed);
314 } else {
315 // No address/offset/slide, load the binary at file address,
316 // offset 0.
317 LLDB_LOGF(log,
318 "DynamicLoader::LoadBinaryWithUUIDAndAddress Loading "
319 "binary %s UUID %s at file address",
320 name.str().c_str(), uuid.GetAsString().c_str());
321 module_sp->SetLoadAddress(target, 0, true /* value_is_slide */,
322 changed);
323 }
324 } else {
325 // In-memory image, load at its true address, offset 0.
326 LLDB_LOGF(log,
327 "DynamicLoader::LoadBinaryWithUUIDAndAddress Loading binary "
328 "%s UUID %s from memory at address 0x%" PRIx64,
329 name.str().c_str(), uuid.GetAsString().c_str(), value);
330 module_sp->SetLoadAddress(target, 0, true /* value_is_slide */,
331 changed);
332 }
333 }
334
335 if (notify) {
336 ModuleList added_module;
337 added_module.Append(module_sp, false);
338 target.ModulesDidLoad(added_module);
339 }
340 } else {
341 if (force_symbol_search) {
343 s->Printf("Unable to find file");
344 if (!name.empty())
345 s->Printf(" %s", name.str().c_str());
346 if (uuid.IsValid())
347 s->Printf(" with UUID %s", uuid.GetAsString().c_str());
348 if (value != LLDB_INVALID_ADDRESS) {
349 if (value_is_offset)
350 s->Printf(" with slide 0x%" PRIx64, value);
351 else
352 s->Printf(" at address 0x%" PRIx64, value);
353 }
354 s->Printf("\n");
355 }
356 LLDB_LOGF(log,
357 "Unable to find binary %s with UUID %s and load it at "
358 "%s 0x%" PRIx64,
359 name.str().c_str(), uuid.GetAsString().c_str(),
360 value_is_offset ? "offset" : "address", value);
361 }
362
363 return module_sp;
364}
365
367 int size_in_bytes) {
369 uint64_t value =
370 m_process->ReadUnsignedIntegerFromMemory(addr, size_in_bytes, 0, error);
371 if (error.Fail())
372 return -1;
373 else
374 return (int64_t)value;
375}
376
379 addr_t value = m_process->ReadPointerFromMemory(addr, error);
380 if (error.Fail())
382 else
383 return value;
384}
385
387{
388 if (m_process)
389 m_process->LoadOperatingSystemPlugin(flush);
390}
static llvm::raw_ostream & error(Stream &strm)
static ModuleSP ReadUnnamedMemoryModule(Process *process, addr_t addr, llvm::StringRef name)
#define LLDB_LOGF(log,...)
Definition Log.h:376
bool IsValid() const
Tests if this ArchSpec is valid.
Definition ArchSpec.h:366
lldb::StreamUP GetAsyncErrorStream()
void LoadOperatingSystemPlugin(bool flush)
void SetStopWhenImagesChange(bool stop)
Set whether the process should stop when images change.
int64_t ReadUnsignedIntWithSizeInBytes(lldb::addr_t addr, int size_in_bytes)
lldb::addr_t ReadPointer(lldb::addr_t addr)
Process * m_process
The process that this dynamic loader plug-in is tracking.
void UpdateLoadedSectionsCommon(lldb::ModuleSP module, lldb::addr_t base_addr, bool base_addr_is_offset)
lldb::ModuleSP GetTargetExecutable()
Checks to see if the target module has changed, updates the target accordingly and returns the target...
bool GetStopWhenImagesChange() const
Get whether the process should stop when images change.
static lldb::ModuleSP LoadBinaryWithUUIDAndAddress(Process *process, llvm::StringRef name, UUID uuid, lldb::addr_t value, bool value_is_offset, bool force_symbol_search, bool notify, bool set_address_in_target, bool allow_memory_image_last_resort)
Find/load a binary into lldb given a UUID and the address where it is loaded in memory,...
virtual lldb::ModuleSP LoadModuleAtAddress(const lldb_private::FileSpec &file, lldb::addr_t link_map_addr, lldb::addr_t base_addr, bool base_addr_is_offset)
Locates or creates a module given by file and updates/loads the resulting module at the virtual base ...
virtual void UpdateLoadedSections(lldb::ModuleSP module, lldb::addr_t link_map_addr, lldb::addr_t base_addr, bool base_addr_is_offset)
Updates the load address of every allocatable section in module.
DynamicLoader(Process *process)
Construct with a process.
const lldb_private::SectionList * GetSectionListFromModule(const lldb::ModuleSP module) const
static DynamicLoader * FindPlugin(Process *process, llvm::StringRef plugin_name)
Find a dynamic loader plugin for a given process.
void UnloadSectionsCommon(const lldb::ModuleSP module)
lldb::ModuleSP FindModuleViaTarget(const FileSpec &file)
Find a module in the target that matches the given file.
virtual void UnloadSections(const lldb::ModuleSP module)
Removes the loaded sections from the target in module.
A file collection class.
A file utility class.
Definition FileSpec.h:57
size_t GetPath(char *path, size_t max_path_length, bool denormalize=true) const
Extract the full path to the file.
Definition FileSpec.cpp:374
static FileSystem & Instance()
A collection class for Module objects.
Definition ModuleList.h:104
lldb::ModuleSP FindFirstModule(const ModuleSpec &module_spec) const
Finds the first module whose file specification matches module_spec.
bool AppendIfNeeded(const lldb::ModuleSP &new_module, bool notify=true)
Append a module to the module list, if it is not already there.
static Status GetSharedModule(const ModuleSpec &module_spec, lldb::ModuleSP &module_sp, llvm::SmallVectorImpl< lldb::ModuleSP > *old_modules, bool *did_create_ptr, bool always_create=false, bool invoke_locate_callback=true)
void Append(const lldb::ModuleSP &module_sp, bool notify=true)
Append a module to the module list.
FileSpec & GetFileSpec()
Definition ModuleSpec.h:55
FileSpec & GetSymbolFileSpec()
Definition ModuleSpec.h:79
void SetTarget(std::shared_ptr< Target > target)
Set the target to be used when resolving a module.
Definition ModuleSpec.h:139
A plug-in interface definition class for object file parsers.
Definition ObjectFile.h:45
virtual SectionList * GetSectionList(bool update_module_section_list=true)
Gets the section list for the currently selected architecture (and object for archives).
static bool DownloadObjectAndSymbolFile(ModuleSpec &module_spec, Status &error, bool force_lookup=true, bool copy_executable=true)
static FileSpec LocateExecutableSymbolFile(const ModuleSpec &module_spec, const FileSpecList &default_search_paths, StatisticsMap &map)
static DynamicLoaderCreateInstance GetDynamicLoaderCreateCallbackAtIndex(uint32_t idx)
static DynamicLoaderCreateInstance GetDynamicLoaderCreateCallbackForPluginName(llvm::StringRef name)
static ModuleSpec LocateExecutableObjectFile(const ModuleSpec &module_spec, StatisticsMap &map)
A plug-in interface definition class for debugging a process.
Definition Process.h:354
lldb::ModuleSP ReadModuleFromMemory(const FileSpec &file_spec, lldb::addr_t header_addr, size_t size_to_read=512)
Definition Process.cpp:2563
Target & GetTarget()
Get the target object pointer for this module.
Definition Process.h:1267
A Progress indicator helper class.
Definition Progress.h:60
size_t GetSize() const
Definition Section.h:77
lldb::SectionSP GetSectionAtIndex(size_t idx) const
Definition Section.cpp:557
A class to count time for plugins.
Definition Statistics.h:94
An error handling class.
Definition Status.h:118
llvm::StringRef GetString() const
size_t PutHex64(uint64_t uvalue, lldb::ByteOrder byte_order=lldb::eByteOrderInvalid)
Definition Stream.cpp:307
void ModulesDidLoad(ModuleList &module_list)
This call may preload module symbols, and may do so in parallel depending on the following target set...
Definition Target.cpp:1857
Module * GetExecutableModulePointer()
Definition Target.cpp:1539
Debugger & GetDebugger() const
Definition Target.h:1122
bool SetSectionUnloaded(const lldb::SectionSP &section_sp)
Definition Target.cpp:3380
lldb::ModuleSP GetOrCreateModule(const ModuleSpec &module_spec, bool notify, Status *error_ptr=nullptr)
Find a binary on the system and return its Module, or return an existing Module that is already in th...
Definition Target.cpp:2348
bool SetArchitecture(const ArchSpec &arch_spec, bool set_platform=false, bool merge=true)
Set the architecture for this target.
Definition Target.cpp:1703
lldb::ModuleSP GetExecutableModule()
Gets the module for the main executable.
Definition Target.cpp:1523
static FileSpecList GetDefaultDebugFileSearchPaths()
Definition Target.cpp:2798
lldb::PlatformSP GetPlatform()
Definition Target.h:1576
const ModuleList & GetImages() const
Get accessor for the images for this process.
Definition Target.h:1039
const ArchSpec & GetArchitecture() const
Definition Target.h:1081
void SetExecutableModule(lldb::ModuleSP &module_sp, LoadDependentFiles load_dependent_files=eLoadDependentsDefault)
Set the main executable module.
Definition Target.cpp:1574
Represents UUID's of various sizes.
Definition UUID.h:27
std::string GetAsString(llvm::StringRef separator="-") const
Definition UUID.cpp:54
bool IsValid() const
Definition UUID.h:69
#define LLDB_INVALID_ADDRESS
A class that represents a running process on the host machine.
Log * GetLog(Cat mask)
Retrieve the Log object for the channel associated with the given log enum.
Definition Log.h:332
DynamicLoader *(* DynamicLoaderCreateInstance)(Process *process, bool force)
std::shared_ptr< lldb_private::Platform > PlatformSP
std::shared_ptr< lldb_private::Section > SectionSP
uint64_t addr_t
Definition lldb-types.h:80
std::unique_ptr< lldb_private::Stream > StreamUP
std::shared_ptr< lldb_private::Module > ModuleSP