34#include "llvm/ADT/ScopeExit.h"
35#include "llvm/Support/ConvertUTF.h"
48 const bool is_host =
false;
51 if (!create && arch && arch->IsValid()) {
52 const llvm::Triple &triple = arch->GetTriple();
53 switch (triple.getVendor()) {
54 case llvm::Triple::PC:
58 case llvm::Triple::UnknownVendor:
59 create = !arch->TripleVendorWasSpecified();
67 switch (triple.getOS()) {
68 case llvm::Triple::Win32:
71 case llvm::Triple::UnknownOS:
72 create = arch->TripleOSWasSpecified();
87 return is_host ?
"Local Windows user platform plug-in."
88 :
"Remote Windows user platform plug-in.";
98 default_platform_sp->SetSystemArchitecture(HostInfo::GetArchitecture());
120 const auto &AddArch = [&](
const ArchSpec &spec) {
122 return spec.IsExactMatch(rhs);
128 AddArch(HostInfo::GetArchitecture(HostInfo::eArchKindDefault));
129 AddArch(HostInfo::GetArchitecture(HostInfo::eArchKind32));
130 AddArch(HostInfo::GetArchitecture(HostInfo::eArchKind64));
136 "can't connect to the host platform '{0}', always connected",
146 "failed to create a 'remote-gdb-server' platform");
157 const std::vector<std::string> *paths,
162 loaded_image->
Clear();
167 "LoadLibrary error: no thread available to invoke LoadLibrary");
172 thread->CalculateExecutionContext(context);
179 if (loader ==
nullptr)
185 "LoadLibrary error: could not get function caller");
190 llvm::SmallVector<llvm::UTF16, 261> name;
191 if (!llvm::convertUTF8ToUTF16String(remote_file.
GetPath(), name)) {
193 "LoadLibrary error: could not convert path to UCS2");
196 name.emplace_back(L
'\0');
201 ePermissionsReadable | ePermissionsWritable,
205 "LoadLibrary error: unable to allocate memory for name: %s",
210 llvm::scope_exit name_cleanup(
214 name.size() *
sizeof(llvm::UTF16), status);
217 "LoadLibrary error: unable to write name: %s", status.
AsCString());
223 std::optional<llvm::scope_exit<std::function<void()>>> paths_cleanup;
225 llvm::SmallVector<llvm::UTF16, 261> search_paths;
227 for (
const auto &path : *paths) {
231 llvm::SmallVector<llvm::UTF16, 261> buffer;
232 if (!llvm::convertUTF8ToUTF16String(path, buffer))
235 search_paths.append(std::begin(buffer), std::end(buffer));
236 search_paths.emplace_back(L
'\0');
238 search_paths.emplace_back(L
'\0');
241 process->
AllocateMemory(search_paths.size() *
sizeof(llvm::UTF16),
242 ePermissionsReadable | ePermissionsWritable,
246 "LoadLibrary error: unable to allocate memory for paths: %s",
251 paths_cleanup.emplace([process, injected_paths]() {
255 process->
WriteMemory(injected_paths, search_paths.data(),
256 search_paths.size() *
sizeof(llvm::UTF16), status);
259 "LoadLibrary error: unable to write paths: %s", status.
AsCString());
268 unsigned injected_length = 261;
272 ePermissionsReadable | ePermissionsWritable,
276 "LoadLibrary error: unable to allocate memory for module location: %s",
281 llvm::scope_exit injected_module_path_cleanup(
282 [process, injected_module_path]() {
290 ePermissionsReadable | ePermissionsWritable,
294 "LoadLibrary error: could not allocate memory for result: %s",
299 llvm::scope_exit result_cleanup([process, injected_result]() {
304 injected_module_path, status);
307 "LoadLibrary error: could not initialize result: %s",
314 Scalar{injected_length},
sizeof(unsigned),
318 "LoadLibrary error: could not initialize result: %s",
332 parameters, diagnostics)) {
335 "LoadLibrary error: unable to write function parameters:"));
339 llvm::scope_exit parameter_cleanup(
340 [invocation, &context, injected_parameters]() {
346 if (!scratch_ts_sp) {
348 "LoadLibrary error: unable to get (clang) type system");
378 "LoadLibrary error: failed to execute LoadLibrary helper:"));
386 "LoadLibrary error: could not read the result: %s", status.
AsCString());
393 injected_result + 2 *
word_size +
sizeof(
unsigned),
sizeof(
unsigned), 0,
397 "LoadLibrary error: could not read error status: %s",
407 std::string module_path;
411 "LoadLibrary error: could not read module path: %s",
417 loaded_image->
SetFile(module_path, llvm::sys::path::Style::native);
427 expression.
Printf(
"FreeLibrary((HMODULE)0x%" PRIx64
")", address);
435 if (value->GetError().Fail())
436 return value->GetError().Clone();
439 if (value->ResolveValue(scalar)) {
454 "can't disconnect from the host platform '{0}', always connected",
500 return Attach(attach_info, debugger, &target,
error);
513 error = process_sp->Launch(launch_info);
515 if (
error.Success()) {
516 process_sp->SetPseudoConsoleHandle();
519 LLDB_LOGF(log,
"Platform::%s LaunchProcess() failed: %s", __FUNCTION__,
542 if (target ==
nullptr) {
546 target = new_target_sp.get();
549 if (!target ||
error.Fail())
558 error = process_sp->Attach(attach_info);
567 llvm::VersionTuple version = HostInfo::GetOSVersion();
568 strm <<
" Host: Windows " << version.getAsString() <<
'\n';
588 const uint8_t *trap_opcode =
nullptr;
589 size_t trap_opcode_size = 0;
592 case llvm::Triple::aarch64: {
593 static const uint8_t g_aarch64_opcode[] = {0x00, 0x00, 0x3e, 0xd4};
594 trap_opcode = g_aarch64_opcode;
595 trap_opcode_size =
sizeof(g_aarch64_opcode);
598 return trap_opcode_size;
602 case llvm::Triple::arm:
603 case llvm::Triple::thumb: {
604 static const uint8_t g_thumb_opcode[] = {0xfe, 0xde};
605 trap_opcode = g_thumb_opcode;
606 trap_opcode_size =
sizeof(g_thumb_opcode);
609 return trap_opcode_size;
618std::unique_ptr<UtilityFunction>
622 static constexpr const char kLoaderDecls[] = R
"(
626// `LOAD_LIBRARY_SEARCH_APPLICATION_DIR | LOAD_LIBRARY_SEARCH_SYSTEM32 | LOAD_LIBRARY_SEARCH_USER_DIRS`
628// Directories in the standard search path are not searched. This value cannot
629// be combined with `LOAD_WITH_ALTERED_SEARCH_PATH`.
631// This value represents the recommended maximum number of directories an
632// application should include in its DLL search path.
633#define LOAD_LIBRARY_SEARCH_DEFAULT_DIRS 0x00001000
635// If this value is used, and lpFileName specifies an absolute path, the system
636// uses the alternate file search strategy to find associated executable
638#define LOAD_WITH_ALTERED_SEARCH_PATH 0x00000008
640// WINBASEAPI DWORD WINAPI GetLastError(VOID);
641/* __declspec(dllimport) */ uint32_t __stdcall GetLastError();
645// WINBASEAPI DLL_DIRECTORY_COOKIE WINAPI AddDllDirectory(LPCWSTR);
646/* __declspec(dllimport) */ void * __stdcall AddDllDirectory(const wchar_t *);
648// WINBASEAPI BOOL WINAPI FreeModule(HMODULE);
649/* __declspec(dllimport) */ int __stdcall FreeModule(void *hLibModule);
651// WINBASEAPI DWORD WINAPI GetModuleFileNameA(HMODULE hModule, LPSTR lpFilename, DWORD nSize);
652/* __declspec(dllimport) */ uint32_t GetModuleFileNameA(void *, char *, uint32_t);
654// WINBASEAPI HMODULE WINAPI LoadLibraryExW(LPCWSTR, HANDLE, DWORD);
655/* __declspec(dllimport) */ void * __stdcall LoadLibraryExW(const wchar_t *, void *, uint32_t);
659// _ACRTIMP size_t __cdecl wcslen(wchar_t const *_String);
660/* __declspec(dllimport) */ size_t __cdecl wcslen(const wchar_t *);
664struct __lldb_LoadLibraryResult {
671_Static_assert(sizeof(struct __lldb_LoadLibraryResult) <= 3 * sizeof(void *),
672 "__lldb_LoadLibraryResult size mismatch");
674void * __lldb_LoadLibraryHelper(const wchar_t *name, const wchar_t *paths,
675 __lldb_LoadLibraryResult *result) {
676 for (const wchar_t *path = paths; path && *path; ) {
677 (void)AddDllDirectory(path);
678 path += wcslen(path) + 1;
681 result->ImageBase = LoadLibraryExW(name, nullptr,
682 LOAD_LIBRARY_SEARCH_DEFAULT_DIRS);
684 // Fallback: if the AddDllDirectory + LOAD_LIBRARY_SEARCH_DEFAULT_DIRS path
685 // failed to find the library, iterate the search paths ourselves and
686 // load by absolute path using LOAD_WITH_ALTERED_SEARCH_PATH, which makes
687 // Windows use the loaded DLL's own directory to resolve its sibling imports.
688 if (result->ImageBase == nullptr) {
690 for (const wchar_t *path = paths; path && *path; path += wcslen(path) + 1) {
691 size_t plen = wcslen(path);
692 size_t nlen = wcslen(name);
693 // Need room for: path + '\\' + name + '\0'
694 if (plen + 1 + nlen + 1 > 4096)
697 for (size_t i = 0; i < plen; ++i)
700 for (size_t i = 0; i <= nlen; ++i) // Copy name including trailing '\0'.
702 result->ImageBase = LoadLibraryExW(full, nullptr,
703 LOAD_WITH_ALTERED_SEARCH_PATH);
704 if (result->ImageBase != nullptr)
709 if (result->ImageBase == nullptr)
710 result->ErrorCode = GetLastError();
712 result->Length = GetModuleFileNameA(result->ImageBase, result->ModulePath,
715 return result->ImageBase;
720 static constexpr const char kName[] =
"__lldb_LoadLibraryHelper";
723 Target &target = process->GetTarget();
729 std::string
error = llvm::toString(function.takeError());
731 "LoadLibrary error: could not create utility function: %s",
759 std::unique_ptr<UtilityFunction> utility{std::move(*function)};
760 utility->MakeFunctionCaller(VoidPtrTy, parameters, context.
GetThreadSP(),
764 "LoadLibrary error: could not create function caller: %s",
769 if (!utility->GetFunctionCaller()) {
771 "LoadLibrary error: could not get function caller");
779 const char *expression,
782 static constexpr const char kLoaderDecls[] = R
"(
786// WINBASEAPI DLL_DIRECTORY_COOKIE WINAPI AddDllDirectory(LPCWSTR);
787/* __declspec(dllimport) */ void * __stdcall AddDllDirectory(const wchar_t *);
789// WINBASEAPI BOOL WINAPI FreeModule(HMODULE);
790/* __declspec(dllimport) */ int __stdcall FreeModule(void *);
792// WINBASEAPI DWORD WINAPI GetModuleFileNameA(HMODULE, LPSTR, DWORD);
793/* __declspec(dllimport) */ uint32_t GetModuleFileNameA(void *, char *, uint32_t);
795// WINBASEAPI HMODULE WINAPI LoadLibraryExW(LPCWSTR, HANDLE, DWORD);
796/* __declspec(dllimport) */ void * __stdcall LoadLibraryExW(const wchar_t *, void *, uint32_t);
801 Status result = loader->CanLoadImage();
814 ExecutionContext context;
815 frame->CalculateExecutionContext(context);
817 EvaluateExpressionOptions options;
829 context, options, expression, kLoaderDecls, value);
831 return value ? value->GetError().Clone() :
Status(
"unknown error");
833 if (value && value->GetError().Fail())
834 return value->GetError().Clone();
static const size_t word_size
static llvm::raw_ostream & error(Stream &strm)
#define LLDB_LOGF(log,...)
#define LLDB_PLUGIN_DEFINE(PluginName)
static constexpr llvm::StringLiteral kName
An architecture specification class.
bool IsValid() const
Tests if this ArchSpec is valid.
llvm::Triple::ArchType GetMachine() const
Returns a machine family for the current architecture.
A command line argument class.
Class that manages the actual breakpoint that will be inserted into the running program.
bool SetTrapOpcode(const uint8_t *trap_opcode, uint32_t trap_opcode_size)
Sets the trap opcode.
Generic representation of a type in a programming language.
CompilerType GetPointerType() const
Return a new CompilerType that is a pointer to this type.
A uniqued constant string class.
bool IsEmpty() const
Test for empty string.
const char * GetCString() const
Get the string value as a C string.
A class to manage flag bits.
TargetList & GetTargetList()
Get accessor for the target list.
llvm::Error GetAsError(lldb::ExpressionResults result, llvm::Twine message={}) const
Returns an ExpressionError with arg as error code.
void SetUnwindOnError(bool unwind=false)
void SetExecutionPolicy(ExecutionPolicy policy=eExecutionPolicyAlways)
void SetLanguage(lldb::LanguageType language_type)
void SetTrapExceptions(bool b)
void SetTimeout(const Timeout< std::micro > &timeout)
void SetIsForUtilityExpr(bool b)
void SetIgnoreBreakpoints(bool ignore=false)
"lldb/Target/ExecutionContext.h" A class that contains an execution context.
const lldb::ProcessSP & GetProcessSP() const
Get accessor to get the process shared pointer.
const lldb::ThreadSP & GetThreadSP() const
Get accessor to get the thread shared pointer.
void SetFile(llvm::StringRef path, Style style)
Change the file specified with a new path.
size_t GetPath(char *path, size_t max_path_length, bool denormalize=true) const
Extract the full path to the file.
void Clear()
Clears the object state.
ValueType Set(ValueType mask)
Set one or more flags by logical OR'ing mask with the current flags.
Encapsulates a function that can be called.
ValueList GetArgumentValues() const
void DeallocateFunctionResults(ExecutionContext &exe_ctx, lldb::addr_t args_addr)
Deallocate the arguments structure.
lldb::ExpressionResults ExecuteFunction(ExecutionContext &exe_ctx, lldb::addr_t *args_addr_ptr, const EvaluateExpressionOptions &options, DiagnosticManager &diagnostic_manager, Value &results)
Run the function this FunctionCaller was created with.
bool WriteFunctionArguments(ExecutionContext &exe_ctx, lldb::addr_t &args_addr_ref, DiagnosticManager &diagnostic_manager)
Insert the default function argument struct.
static bool RegisterPlugin(llvm::StringRef name, llvm::StringRef description, ABICreateInstance create_callback)
static bool UnregisterPlugin(ABICreateInstance create_callback)
lldb::ListenerSP GetListenerForProcess(Debugger &debugger)
llvm::StringRef GetProcessPluginName() const
lldb::ListenerSP GetHijackListener() const
lldb::pid_t GetProcessID() const
lldb::ListenerSP GetListener() const
llvm::StringRef GetProcessPluginName() const
std::chrono::seconds GetUtilityExpressionTimeout() const
A plug-in interface definition class for debugging a process.
UtilityFunction * GetLoadImageUtilityFunction(Platform *platform, llvm::function_ref< std::unique_ptr< UtilityFunction >()> factory)
Get the cached UtilityFunction that assists in loading binary images into the process.
void ResetImageToken(size_t token)
ThreadList & GetThreadList()
lldb::addr_t AllocateMemory(size_t size, uint32_t permissions, Status &error)
The public interface to allocating memory in the process.
size_t ReadCStringFromMemory(lldb::addr_t vm_addr, char *cstr, size_t cstr_max_len, Status &error)
Read a NULL terminated C string from memory.
bool WritePointerToMemory(lldb::addr_t vm_addr, lldb::addr_t ptr_value, Status &error)
size_t AddImageToken(lldb::addr_t image_ptr)
uint64_t ReadUnsignedIntegerFromMemory(lldb::addr_t load_addr, size_t byte_size, uint64_t fail_value, Status &error)
Reads an unsigned integer of the specified byte size from process memory.
lldb::addr_t ReadPointerFromMemory(lldb::addr_t vm_addr, Status &error)
Status DeallocateMemory(lldb::addr_t ptr)
The public interface to deallocating memory in the process.
lldb::addr_t GetImagePtrFromToken(size_t token) const
uint32_t GetAddressByteSize() const
size_t WriteMemory(lldb::addr_t vm_addr, const void *buf, size_t size, Status &error)
Write memory to a process.
virtual DynamicLoader * GetDynamicLoader()
Get the dynamic loader plug-in for this process.
size_t WriteScalarToMemory(lldb::addr_t vm_addr, const Scalar &scalar, size_t size, Status &error)
Write all or part of a scalar value to memory.
Target & GetTarget()
Get the target object pointer for this module.
unsigned int UInt(unsigned int fail_value=0) const
static lldb::TypeSystemClangSP GetForTarget(Target &target, std::optional< IsolatedASTKind > ast_kind=DefaultAST, bool create_on_demand=true)
Returns the scratch TypeSystemClang for the given target.
static Status FromErrorStringWithFormat(const char *format,...) __attribute__((format(printf
static Status FromErrorString(const char *str)
bool Fail() const
Test for error condition.
const char * AsCString(const char *default_error_str="unknown error") const
Get the error string associated with the current error.
static Status static Status FromErrorStringWithFormatv(const char *format, Args &&...args)
static Status FromError(llvm::Error error)
Avoid using this in new code. Migrate APIs to llvm::Expected instead.
const char * GetData() const
llvm::StringRef GetString() const
A stream class that can stream formatted output to a file.
size_t Printf(const char *format,...) __attribute__((format(printf
Output printf formatted output to the stream.
Status CreateTarget(Debugger &debugger, llvm::StringRef user_exe_path, llvm::StringRef triple_str, LoadDependentFiles get_dependent_modules, const OptionGroupPlatform *platform_options, lldb::TargetSP &target_sp)
Create a new Target.
const lldb::ProcessSP & CreateProcess(lldb::ListenerSP listener_sp, llvm::StringRef plugin_name, const FileSpec *crash_file, bool can_connect)
llvm::Expected< std::unique_ptr< UtilityFunction > > CreateUtilityFunction(std::string expression, std::string name, lldb::LanguageType language, ExecutionContext &exe_ctx)
Creates and installs a UtilityFunction for the given language.
const ArchSpec & GetArchitecture() const
lldb::ThreadSP GetExpressionExecutionThread()
static lldb::ExpressionResults Evaluate(ExecutionContext &exe_ctx, const EvaluateExpressionOptions &options, llvm::StringRef expr_cstr, llvm::StringRef expr_prefix, lldb::ValueObjectSP &result_valobj_sp, std::string *fixed_expression=nullptr, ValueObject *ctx_obj=nullptr)
Evaluate one expression in the scratch context of the target passed in the exe_ctx and return its res...
"lldb/Expression/UtilityFunction.h" Encapsulates a bit of source code that provides a function that i...
FunctionCaller * GetFunctionCaller()
void PushValue(const Value &value)
Value * GetValueAtIndex(size_t idx)
const Scalar & GetScalar() const
See comment on m_scalar to understand what GetScalar returns.
@ Scalar
A raw scalar value.
void SetCompilerType(const CompilerType &compiler_type)
void SetValueType(ValueType value_type)
#define LLDB_INVALID_IMAGE_TOKEN
#define LLDB_INVALID_ADDRESS
#define LLDB_INVALID_PROCESS_ID
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.
std::shared_ptr< lldb_private::StackFrame > StackFrameSP
std::shared_ptr< lldb_private::Thread > ThreadSP
std::shared_ptr< lldb_private::ValueObject > ValueObjectSP
std::shared_ptr< lldb_private::Platform > PlatformSP
@ eLanguageTypeC_plus_plus
ISO C++:1998.
ExpressionResults
The results of expression evaluation.
std::shared_ptr< lldb_private::Process > ProcessSP
std::shared_ptr< lldb_private::TypeSystemClang > TypeSystemClangSP
std::shared_ptr< lldb_private::Target > TargetSP