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 (auto create_callback :
52 std::unique_ptr<DynamicLoader> instance_up(
53 create_callback(process, false));
54 if (instance_up)
55 return instance_up.release();
56 }
57 }
58 return nullptr;
59}
60
62
63// Accessors to the global setting as to whether to stop at image (shared
64// library) loading/unloading.
65
67 return m_process->GetStopOnSharedLibraryEvents();
68}
69
71 m_process->SetStopOnSharedLibraryEvents(stop);
72}
73
75 Target &target = m_process->GetTarget();
76 ModuleSP executable = target.GetExecutableModule();
77
78 if (executable) {
79 if (FileSystem::Instance().Exists(executable->GetFileSpec())) {
80 ModuleSpec module_spec(executable->GetFileSpec(),
81 executable->GetArchitecture());
82 auto module_sp = std::make_shared<Module>(module_spec);
83 // If we're a coredump and we already have a main executable, we don't
84 // need to reload the module list that target already has
85 if (!m_process->IsLiveDebugSession()) {
86 return executable;
87 }
88 // Check if the executable has changed and set it to the target
89 // executable if they differ.
90 if (module_sp && module_sp->GetUUID().IsValid() &&
91 executable->GetUUID().IsValid()) {
92 if (module_sp->GetUUID() != executable->GetUUID())
93 executable.reset();
94 } else if (executable->FileHasChanged()) {
95 executable.reset();
96 }
97
98 if (!executable) {
99 executable = target.GetOrCreateModule(module_spec, true /* notify */);
100 if (executable.get() != target.GetExecutableModulePointer()) {
101 // Don't load dependent images since we are in dyld where we will
102 // know and find out about all images that are loaded
103 target.SetExecutableModule(executable, eLoadDependentsNo);
104 }
105 }
106 }
107 }
108 return executable;
109}
110
112 addr_t base_addr,
113 bool base_addr_is_offset) {
114 UpdateLoadedSectionsCommon(module, base_addr, base_addr_is_offset);
115}
116
118 addr_t base_addr,
119 bool base_addr_is_offset) {
120 bool changed;
121 module->SetLoadAddress(m_process->GetTarget(), base_addr, base_addr_is_offset,
122 changed);
123}
124
126 UnloadSectionsCommon(module);
127}
128
130 Target &target = m_process->GetTarget();
131 const SectionList *sections = GetSectionListFromModule(module);
132
133 assert(sections && "SectionList missing from unloaded module.");
134
135 const size_t num_sections = sections->GetSize();
136 for (size_t i = 0; i < num_sections; ++i) {
137 SectionSP section_sp(sections->GetSectionAtIndex(i));
138 target.SetSectionUnloaded(section_sp);
139 }
140}
141
142const SectionList *
144 SectionList *sections = nullptr;
145 if (module) {
146 ObjectFile *obj_file = module->GetObjectFile();
147 if (obj_file != nullptr) {
148 sections = obj_file->GetSectionList();
149 }
150 }
151 return sections;
152}
153
155 Target &target = m_process->GetTarget();
156 ModuleSpec module_spec(file, target.GetArchitecture());
157 if (UUID uuid = m_process->FindModuleUUID(file.GetPath())) {
158 // Process may be able to augment the module_spec with UUID, e.g. ELF core.
159 module_spec.GetUUID() = uuid;
160 }
161
162 if (ModuleSP module_sp = target.GetImages().FindFirstModule(module_spec))
163 return module_sp;
164
165 if (ModuleSP module_sp =
166 target.GetOrCreateModule(module_spec, /*notify=*/false))
167 return module_sp;
168
169 return nullptr;
170}
171
173 addr_t link_map_addr,
174 addr_t base_addr,
175 bool base_addr_is_offset) {
176 ModuleSP module_sp = FindModuleViaTarget(file);
177 // We have a core file, try to load the image from memory if we didn't find
178 // the module.
179 if (!module_sp && !m_process->IsLiveDebugSession()) {
180 llvm::Expected<ModuleSP> memory_module_sp_or_err =
181 m_process->ReadModuleFromMemory(file, base_addr);
182 if (auto err = memory_module_sp_or_err.takeError())
184 "Failed to read module from memory: {0}");
185 else {
186 module_sp = *memory_module_sp_or_err;
187 m_process->GetTarget().GetImages().AppendIfNeeded(module_sp, false);
188 }
189 }
190 if (module_sp)
191 UpdateLoadedSections(module_sp, link_map_addr, base_addr,
192 base_addr_is_offset);
193 return module_sp;
194}
195
197 llvm::StringRef name) {
198 char namebuf[80];
199 if (name.empty()) {
200 snprintf(namebuf, sizeof(namebuf), "memory-image-0x%" PRIx64, addr);
201 name = namebuf;
202 }
203 llvm::Expected<ModuleSP> module_sp_or_err =
204 process->ReadModuleFromMemory(FileSpec(name), addr);
205 if (auto err = module_sp_or_err.takeError()) {
207 "Failed to read module from memory: {0}");
208 return {};
209 }
210 return *module_sp_or_err;
211}
212
214 Process *process, llvm::StringRef name, UUID uuid, addr_t value,
215 bool value_is_offset, bool force_symbol_search, bool notify,
216 bool set_address_in_target, bool allow_memory_image_last_resort) {
217 ModuleSP memory_module_sp;
218 ModuleSP module_sp;
219 PlatformSP platform_sp = process->GetTarget().GetPlatform();
220 Target &target = process->GetTarget();
222
223 StreamString prog_str;
224 if (!name.empty()) {
225 prog_str << name.str() << " ";
226 }
227 if (uuid.IsValid())
228 prog_str << uuid.GetAsString();
229 if (value_is_offset == 0 && value != LLDB_INVALID_ADDRESS) {
230 prog_str << " at 0x";
231 prog_str.PutHex64(value);
232 }
233
234 if (!uuid.IsValid() && !value_is_offset) {
235 memory_module_sp = ReadUnnamedMemoryModule(process, value, name);
236
237 if (memory_module_sp) {
238 uuid = memory_module_sp->GetUUID();
239 if (uuid.IsValid()) {
240 prog_str << " ";
241 prog_str << uuid.GetAsString();
242 }
243 }
244 }
245 ModuleSpec module_spec;
246 module_spec.SetTarget(target.shared_from_this());
247 module_spec.GetUUID() = uuid;
248 FileSpec name_filespec(name);
249 if (FileSystem::Instance().Exists(name_filespec))
250 module_spec.GetFileSpec() = name_filespec;
251
252 if (uuid.IsValid()) {
253 Progress progress("Locating binary", prog_str.GetString().str());
254
255 // Has lldb already seen a module with this UUID?
256 // Or have external lookup enabled in DebugSymbols on macOS.
257 if (!module_sp)
258 error =
259 ModuleList::GetSharedModule(module_spec, module_sp, nullptr, nullptr);
260
261 // Can lldb's symbol/executable location schemes
262 // find an executable and symbol file.
263 if (!module_sp) {
265 StatisticsMap symbol_locator_map;
266 module_spec.GetSymbolFileSpec() =
267 PluginManager::LocateExecutableSymbolFile(module_spec, search_paths,
268 symbol_locator_map);
269 ModuleSpec objfile_module_spec =
271 symbol_locator_map);
272 module_spec.GetFileSpec() = objfile_module_spec.GetFileSpec();
273 if (FileSystem::Instance().Exists(module_spec.GetFileSpec()) &&
274 FileSystem::Instance().Exists(module_spec.GetSymbolFileSpec())) {
275 module_sp = std::make_shared<Module>(module_spec);
276 }
277
278 if (module_sp) {
279 module_sp->GetSymbolLocatorStatistics().merge(symbol_locator_map);
280 }
281 }
282
283 // If we haven't found a binary, or we don't have a SymbolFile, see
284 // if there is an external search tool that can find it.
285 if (!module_sp || !module_sp->GetSymbolFileFileSpec()) {
287 force_symbol_search);
288 if (FileSystem::Instance().Exists(module_spec.GetFileSpec())) {
289 module_sp = std::make_shared<Module>(module_spec);
290 } else if (force_symbol_search && error.AsCString("") &&
291 error.AsCString("")[0] != '\0') {
292 *target.GetDebugger().GetAsyncErrorStream() << error.AsCString();
293 }
294 }
295
296 // If we only found the executable, create a Module based on that.
297 if (!module_sp && FileSystem::Instance().Exists(module_spec.GetFileSpec()))
298 module_sp = std::make_shared<Module>(module_spec);
299 }
300
301 // If we couldn't find the binary anywhere else, as a last resort,
302 // read it out of memory.
303 if (allow_memory_image_last_resort && !module_sp.get() &&
304 value != LLDB_INVALID_ADDRESS && !value_is_offset) {
305 if (!memory_module_sp)
306 memory_module_sp = ReadUnnamedMemoryModule(process, value, name);
307 if (memory_module_sp)
308 module_sp = memory_module_sp;
309 }
310
312 if (module_sp.get()) {
313 // Ensure the Target has an architecture set in case
314 // we need it while processing this binary/eh_frame/debug info.
315 if (!target.GetArchitecture().IsValid())
316 target.SetArchitecture(module_sp->GetArchitecture());
317 target.GetImages().AppendIfNeeded(module_sp, false);
318
319 bool changed = false;
320 if (set_address_in_target) {
321 if (module_sp->GetObjectFile()) {
322 if (value != LLDB_INVALID_ADDRESS) {
323 LLDB_LOGF(log,
324 "DynamicLoader::LoadBinaryWithUUIDAndAddress Loading "
325 "binary %s UUID %s at %s 0x%" PRIx64,
326 name.str().c_str(), uuid.GetAsString().c_str(),
327 value_is_offset ? "offset" : "address", value);
328 module_sp->SetLoadAddress(target, value, value_is_offset, changed);
329 } else {
330 // No address/offset/slide, load the binary at file address,
331 // offset 0.
332 LLDB_LOGF(log,
333 "DynamicLoader::LoadBinaryWithUUIDAndAddress Loading "
334 "binary %s UUID %s at file address",
335 name.str().c_str(), uuid.GetAsString().c_str());
336 module_sp->SetLoadAddress(target, 0, true /* value_is_slide */,
337 changed);
338 }
339 } else {
340 // In-memory image, load at its true address, offset 0.
341 LLDB_LOGF(log,
342 "DynamicLoader::LoadBinaryWithUUIDAndAddress Loading binary "
343 "%s UUID %s from memory at address 0x%" PRIx64,
344 name.str().c_str(), uuid.GetAsString().c_str(), value);
345 module_sp->SetLoadAddress(target, 0, true /* value_is_slide */,
346 changed);
347 }
348 }
349
350 if (notify) {
351 ModuleList added_module;
352 added_module.Append(module_sp, false);
353 target.ModulesDidLoad(added_module);
354 }
355 } else {
356 if (force_symbol_search) {
358 s->Printf("Unable to find file");
359 if (!name.empty())
360 s->Printf(" %s", name.str().c_str());
361 if (uuid.IsValid())
362 s->Printf(" with UUID %s", uuid.GetAsString().c_str());
363 if (value != LLDB_INVALID_ADDRESS) {
364 if (value_is_offset)
365 s->Printf(" with slide 0x%" PRIx64, value);
366 else
367 s->Printf(" at address 0x%" PRIx64, value);
368 }
369 s->Printf("\n");
370 }
371 LLDB_LOGF(log,
372 "Unable to find binary %s with UUID %s and load it at "
373 "%s 0x%" PRIx64,
374 name.str().c_str(), uuid.GetAsString().c_str(),
375 value_is_offset ? "offset" : "address", value);
376 }
377
378 return module_sp;
379}
380
382 int size_in_bytes) {
384 uint64_t value =
385 m_process->ReadUnsignedIntegerFromMemory(addr, size_in_bytes, 0, error);
386 if (error.Fail())
387 return -1;
388 else
389 return (int64_t)value;
390}
391
394 addr_t value = m_process->ReadPointerFromMemory(addr, error);
395 if (error.Fail())
397 else
398 return value;
399}
400
402{
403 if (m_process)
404 m_process->LoadOperatingSystemPlugin(flush);
405}
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
#define LLDB_LOG_ERROR(log, error,...)
Definition Log.h:392
bool IsValid() const
Tests if this ArchSpec is valid.
Definition ArchSpec.h:367
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:125
static Status GetSharedModule(const ModuleSpec &module_spec, lldb::ModuleSP &module_sp, llvm::SmallVectorImpl< lldb::ModuleSP > *old_modules, bool *did_create_ptr, bool invoke_locate_callback=true)
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.
void Append(const lldb::ModuleSP &module_sp, bool notify=true)
Append a module to the module list.
FileSpec & GetFileSpec()
Definition ModuleSpec.h:57
FileSpec & GetSymbolFileSpec()
Definition ModuleSpec.h:81
void SetTarget(lldb::TargetSP target)
Set the target to be used when resolving a module.
Definition ModuleSpec.h:141
A plug-in interface definition class for object file parsers.
Definition ObjectFile.h:46
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 GetDynamicLoaderCreateCallbackForPluginName(llvm::StringRef name)
static ModuleSpec LocateExecutableObjectFile(const ModuleSpec &module_spec, StatisticsMap &map)
static llvm::SmallVector< DynamicLoaderCreateInstance > GetDynamicLoaderCreateCallbacks()
A plug-in interface definition class for debugging a process.
Definition Process.h:354
llvm::Expected< lldb::ModuleSP > ReadModuleFromMemory(const FileSpec &file_spec, lldb::addr_t header_addr, size_t size_to_read=512)
Creates and populates a module using an in-memory object file.
Definition Process.cpp:2599
Target & GetTarget()
Get the target object pointer for this module.
Definition Process.h:1250
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:553
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:1859
Module * GetExecutableModulePointer()
Definition Target.cpp:1541
Debugger & GetDebugger() const
Definition Target.h:1224
bool SetSectionUnloaded(const lldb::SectionSP &section_sp)
Definition Target.cpp:3382
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:2352
bool SetArchitecture(const ArchSpec &arch_spec, bool set_platform=false, bool merge=true)
Set the architecture for this target.
Definition Target.cpp:1705
lldb::ModuleSP GetExecutableModule()
Gets the module for the main executable.
Definition Target.cpp:1525
static FileSpecList GetDefaultDebugFileSearchPaths()
Definition Target.cpp:2803
lldb::PlatformSP GetPlatform()
Definition Target.h:1678
const ModuleList & GetImages() const
Get accessor for the images for this process.
Definition Target.h:1141
const ArchSpec & GetArchitecture() const
Definition Target.h:1183
void SetExecutableModule(lldb::ModuleSP &module_sp, LoadDependentFiles load_dependent_files=eLoadDependentsDefault)
Set the main executable module.
Definition Target.cpp:1576
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