LLDB mainline
ProcessWasm.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
9#include "ProcessWasm.h"
10#include "ThreadWasm.h"
11#include "lldb/Core/Module.h"
13#include "lldb/Core/Value.h"
15
17
18using namespace lldb;
19using namespace lldb_private;
21using namespace lldb_private::wasm;
22
24
26 : ProcessGDBRemote(target_sp, listener_sp) {
27 assert(target_sp);
28 // Wasm doesn't have any Unix-like signals as a platform concept, but pretend
29 // like it does to appease LLDB.
30 m_unix_signals_sp = UnixSignals::Create(target_sp->GetArchitecture());
31}
32
34 static llvm::once_flag g_once_flag;
35
36 llvm::call_once(g_once_flag, []() {
40 });
41}
42
46
47llvm::StringRef ProcessWasm::GetPluginName() { return GetPluginNameStatic(); }
48
49llvm::StringRef ProcessWasm::GetPluginNameStatic() { return "wasm"; }
50
52 return "GDB Remote protocol based WebAssembly debugging plug-in.";
53}
54
58
60 ListenerSP listener_sp,
61 const FileSpec *crash_file_path,
62 bool can_connect) {
63 if (crash_file_path == nullptr)
64 return std::make_shared<ProcessWasm>(target_sp, listener_sp);
65 return {};
66}
67
69 bool plugin_specified_by_name) {
70 if (plugin_specified_by_name)
71 return true;
72
73 if (Module *exe_module = target_sp->GetExecutableModulePointer()) {
74 if (ObjectFile *exe_objfile = exe_module->GetObjectFile())
75 return exe_objfile->GetArchitecture().GetMachine() ==
76 llvm::Triple::wasm32;
77 }
78
79 // However, if there is no wasm module, we return false, otherwise,
80 // we might use ProcessWasm to attach gdb remote.
81 return false;
82}
83
84std::shared_ptr<ThreadGDBRemote> ProcessWasm::CreateThread(lldb::tid_t tid) {
85 return std::make_shared<ThreadWasm>(*this, tid);
86}
87
88size_t ProcessWasm::ReadMemory(lldb::addr_t vm_addr, void *buf, size_t size,
89 Status &error) {
90 wasm_addr_t wasm_addr(vm_addr);
91
92 switch (wasm_addr.GetType()) {
95 return ProcessGDBRemote::ReadMemory(vm_addr, buf, size, error);
97 error.FromErrorStringWithFormat(
98 "Wasm read failed for invalid address 0x%" PRIx64, vm_addr);
99 return 0;
100 }
101 llvm_unreachable("Fully covered switch above");
102}
103
104llvm::Expected<std::vector<lldb::addr_t>>
106 StreamString packet;
107 packet.Printf("qWasmCallStack:");
108 packet.Printf("%" PRIx64, tid);
109
111 if (m_gdb_comm.SendPacketAndWaitForResponse(packet.GetString(), response) !=
113 return llvm::createStringError("failed to send qWasmCallStack");
114
115 if (!response.IsNormalResponse())
116 return llvm::createStringError("failed to get response for qWasmCallStack");
117
118 WritableDataBufferSP data_buffer_sp =
119 std::make_shared<DataBufferHeap>(response.GetStringRef().size() / 2, 0);
120 const size_t bytes = response.GetHexBytes(data_buffer_sp->GetData(), '\xcc');
121 if (bytes == 0 || bytes % sizeof(uint64_t) != 0)
122 return llvm::createStringError("invalid response for qWasmCallStack");
123
124 // To match the Wasm specification, the addresses are encoded in little endian
125 // byte order.
126 DataExtractor data(data_buffer_sp, lldb::eByteOrderLittle,
128 lldb::offset_t offset = 0;
129 std::vector<lldb::addr_t> call_stack_pcs;
130 while (offset < bytes)
131 call_stack_pcs.push_back(data.GetU64(&offset));
132
133 return call_stack_pcs;
134}
135
136llvm::Expected<lldb::DataBufferSP>
138 int index) {
139 StreamString packet;
140 switch (kind) {
141 case eWasmTagLocal:
142 packet.Printf("qWasmLocal:");
143 break;
144 case eWasmTagGlobal:
145 packet.Printf("qWasmGlobal:");
146 break;
148 packet.PutCString("qWasmStackValue:");
149 break;
151 return llvm::createStringError("not a Wasm location");
152 }
153 packet.Printf("%d;%d", frame_index, index);
154
156 if (m_gdb_comm.SendPacketAndWaitForResponse(packet.GetString(), response) !=
158 return llvm::createStringError("failed to send Wasm variable");
159
160 if (!response.IsNormalResponse())
161 return llvm::createStringError("failed to get response for Wasm variable");
162
163 WritableDataBufferSP buffer_sp(
164 new DataBufferHeap(response.GetStringRef().size() / 2, 0));
165 response.GetHexBytes(buffer_sp->GetData(), '\xcc');
166 return buffer_sp;
167}
static llvm::raw_ostream & error(Stream &strm)
#define LLDB_PLUGIN_DEFINE(PluginName)
size_t GetHexBytes(llvm::MutableArrayRef< uint8_t > dest, uint8_t fail_fill_value)
llvm::StringRef GetStringRef() const
A subclass of DataBuffer that stores a data buffer on the heap.
An data extractor class.
uint64_t GetU64(lldb::offset_t *offset_ptr) const
Extract a uint64_t value from *offset_ptr.
A file utility class.
Definition FileSpec.h:57
A class that describes an executable image and its associated object and symbol files.
Definition Module.h:90
A plug-in interface definition class for object file parsers.
Definition ObjectFile.h:45
static bool RegisterPlugin(llvm::StringRef name, llvm::StringRef description, ABICreateInstance create_callback)
static bool UnregisterPlugin(ABICreateInstance create_callback)
virtual size_t ReadMemory(lldb::addr_t vm_addr, void *buf, size_t size, Status &error)
Read of memory from a process.
Definition Process.cpp:1930
lldb::UnixSignalsSP m_unix_signals_sp
Definition Process.h:3189
uint32_t GetAddressByteSize() const
Definition Process.cpp:3620
friend class Debugger
Definition Process.h:359
An error handling class.
Definition Status.h:118
llvm::StringRef GetString() const
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:65
static lldb::UnixSignalsSP Create(const ArchSpec &arch)
ProcessGDBRemote(lldb::TargetSP target_sp, lldb::ListenerSP listener_sp)
ProcessWasm provides the access to the Wasm program state retrieved from the Wasm engine.
Definition ProcessWasm.h:47
llvm::StringRef GetPluginName() override
std::shared_ptr< process_gdb_remote::ThreadGDBRemote > CreateThread(lldb::tid_t tid) override
static llvm::StringRef GetPluginNameStatic()
bool CanDebug(lldb::TargetSP target_sp, bool plugin_specified_by_name) override
Check if a plug-in instance can debug the file in module.
static void DebuggerInitialize(Debugger &debugger)
static lldb::ProcessSP CreateInstance(lldb::TargetSP target_sp, lldb::ListenerSP listener_sp, const FileSpec *crash_file_path, bool can_connect)
llvm::Expected< std::vector< lldb::addr_t > > GetWasmCallStack(lldb::tid_t tid)
Retrieve the current call stack from the WebAssembly remote process.
ProcessWasm(lldb::TargetSP target_sp, lldb::ListenerSP listener_sp)
llvm::Expected< lldb::DataBufferSP > GetWasmVariable(WasmVirtualRegisterKinds kind, int frame_index, int index)
Query the value of a WebAssembly variable from the WebAssembly remote process.
static llvm::StringRef GetPluginDescriptionStatic()
size_t ReadMemory(lldb::addr_t vm_addr, void *buf, size_t size, Status &error) override
Read of memory from a process.
A class that represents a running process on the host machine.
uint64_t offset_t
Definition lldb-types.h:85
std::shared_ptr< lldb_private::Process > ProcessSP
std::shared_ptr< lldb_private::Listener > ListenerSP
std::shared_ptr< lldb_private::WritableDataBuffer > WritableDataBufferSP
uint64_t addr_t
Definition lldb-types.h:80
std::shared_ptr< lldb_private::Target > TargetSP
uint64_t tid_t
Definition lldb-types.h:84
For the purpose of debugging, we can represent all these separated 32-bit address spaces with a singl...
Definition ProcessWasm.h:26