180814287SRaphael Isemann //===-- ProcessKDP.cpp ----------------------------------------------------===//
2f9765acdSGreg Clayton //
32946cd70SChandler Carruth // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
42946cd70SChandler Carruth // See https://llvm.org/LICENSE.txt for license information.
52946cd70SChandler Carruth // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6f9765acdSGreg Clayton //
7f9765acdSGreg Clayton //===----------------------------------------------------------------------===//
8f9765acdSGreg Clayton
976e47d48SRaphael Isemann #include <cerrno>
1076e47d48SRaphael Isemann #include <cstdlib>
11f9765acdSGreg Clayton
12796ac80bSJonas Devlieghere #include <memory>
133f69fa6fSBenjamin Kramer #include <mutex>
143f69fa6fSBenjamin Kramer
1507e66e3eSGreg Clayton #include "lldb/Core/Debugger.h"
161f746071SGreg Clayton #include "lldb/Core/Module.h"
174bd4e7e3SJason Molenda #include "lldb/Core/ModuleSpec.h"
18b9c1b51eSKate Stone #include "lldb/Core/PluginManager.h"
1993a66fc1SZachary Turner #include "lldb/Host/ConnectionFileDescriptor.h"
20f9765acdSGreg Clayton #include "lldb/Host/Host.h"
2139de3110SZachary Turner #include "lldb/Host/ThreadLauncher.h"
22e98628ceSOleksiy Vyalov #include "lldb/Host/common/TCPSocket.h"
231d19a2f2SGreg Clayton #include "lldb/Interpreter/CommandInterpreter.h"
241d19a2f2SGreg Clayton #include "lldb/Interpreter/CommandObject.h"
251d19a2f2SGreg Clayton #include "lldb/Interpreter/CommandObjectMultiword.h"
261d19a2f2SGreg Clayton #include "lldb/Interpreter/CommandReturnObject.h"
271d19a2f2SGreg Clayton #include "lldb/Interpreter/OptionGroupString.h"
281d19a2f2SGreg Clayton #include "lldb/Interpreter/OptionGroupUInt64.h"
2941204d09SIlia K #include "lldb/Interpreter/OptionValueProperties.h"
3080552918SZachary Turner #include "lldb/Symbol/LocateSymbolFile.h"
311f746071SGreg Clayton #include "lldb/Symbol/ObjectFile.h"
327925fbbaSGreg Clayton #include "lldb/Target/RegisterContext.h"
3357508026SGreg Clayton #include "lldb/Target/Target.h"
34a63d08c9SGreg Clayton #include "lldb/Target/Thread.h"
35c34698a8SPavel Labath #include "lldb/Utility/LLDBLog.h"
36f39c2e18SJonas Devlieghere #include "lldb/Utility/Log.h"
37d821c997SPavel Labath #include "lldb/Utility/State.h"
3845788152SBruce Mitchener #include "lldb/Utility/StringExtractor.h"
39d821c997SPavel Labath #include "lldb/Utility/UUID.h"
40f9765acdSGreg Clayton
41c5f28e2aSKamil Rytarowski #include "llvm/Support/Threading.h"
42c5f28e2aSKamil Rytarowski
43510938e5SCharles Davis #define USEC_PER_SEC 1000000
44510938e5SCharles Davis
45b9c1b51eSKate Stone #include "Plugins/DynamicLoader/Darwin-Kernel/DynamicLoaderDarwinKernel.h"
46b9c1b51eSKate Stone #include "Plugins/DynamicLoader/Static/DynamicLoaderStatic.h"
47f9765acdSGreg Clayton #include "ProcessKDP.h"
48f9765acdSGreg Clayton #include "ProcessKDPLog.h"
49a63d08c9SGreg Clayton #include "ThreadKDP.h"
50f9765acdSGreg Clayton
51f9765acdSGreg Clayton using namespace lldb;
52f9765acdSGreg Clayton using namespace lldb_private;
53f9765acdSGreg Clayton
54ccad1948SJonas Devlieghere LLDB_PLUGIN_DEFINE_ADV(ProcessKDP, ProcessMacOSXKernel)
55fbb4d1e4SJonas Devlieghere
567f98240dSGreg Clayton namespace {
577f98240dSGreg Clayton
58971f9ca6SJonas Devlieghere #define LLDB_PROPERTIES_processkdp
596a253d37SJordan Rupprecht #include "ProcessKDPProperties.inc"
607f98240dSGreg Clayton
61971f9ca6SJonas Devlieghere enum {
62971f9ca6SJonas Devlieghere #define LLDB_PROPERTIES_processkdp
636a253d37SJordan Rupprecht #include "ProcessKDPPropertiesEnum.inc"
64971f9ca6SJonas Devlieghere };
657f98240dSGreg Clayton
66b9c1b51eSKate Stone class PluginProperties : public Properties {
677f98240dSGreg Clayton public:
GetSettingName()68b9c1b51eSKate Stone static ConstString GetSettingName() {
695f4980f0SPavel Labath return ConstString(ProcessKDP::GetPluginNameStatic());
707f98240dSGreg Clayton }
717f98240dSGreg Clayton
PluginProperties()72b9c1b51eSKate Stone PluginProperties() : Properties() {
73796ac80bSJonas Devlieghere m_collection_sp = std::make_shared<OptionValueProperties>(GetSettingName());
74a8ea5955SJonas Devlieghere m_collection_sp->Initialize(g_processkdp_properties);
757f98240dSGreg Clayton }
767f98240dSGreg Clayton
77*fd146460SShafik Yaghmour ~PluginProperties() override = default;
787f98240dSGreg Clayton
GetPacketTimeout()79b9c1b51eSKate Stone uint64_t GetPacketTimeout() {
80971f9ca6SJonas Devlieghere const uint32_t idx = ePropertyKDPPacketTimeout;
81b9c1b51eSKate Stone return m_collection_sp->GetPropertyAtIndexAsUInt64(
82a8ea5955SJonas Devlieghere NULL, idx, g_processkdp_properties[idx].default_uint_value);
837f98240dSGreg Clayton }
847f98240dSGreg Clayton };
857f98240dSGreg Clayton
GetGlobalPluginProperties()863d7161e3SPavel Labath static PluginProperties &GetGlobalPluginProperties() {
873d7161e3SPavel Labath static PluginProperties g_settings;
883d7161e3SPavel Labath return g_settings;
897f98240dSGreg Clayton }
907f98240dSGreg Clayton
917f98240dSGreg Clayton } // anonymous namespace end
927f98240dSGreg Clayton
93ba4e61d3SAndrew Kaylor static const lldb::tid_t g_kernel_tid = 1;
94ba4e61d3SAndrew Kaylor
GetPluginDescriptionStatic()955f4980f0SPavel Labath llvm::StringRef ProcessKDP::GetPluginDescriptionStatic() {
96b9c1b51eSKate Stone return "KDP Remote protocol based debugging plug-in for darwin kernel "
97b9c1b51eSKate Stone "debugging.";
98f9765acdSGreg Clayton }
99f9765acdSGreg Clayton
Terminate()100b9c1b51eSKate Stone void ProcessKDP::Terminate() {
101f9765acdSGreg Clayton PluginManager::UnregisterPlugin(ProcessKDP::CreateInstance);
102f9765acdSGreg Clayton }
103f9765acdSGreg Clayton
CreateInstance(TargetSP target_sp,ListenerSP listener_sp,const FileSpec * crash_file_path,bool can_connect)104b9c1b51eSKate Stone lldb::ProcessSP ProcessKDP::CreateInstance(TargetSP target_sp,
105583bbb1dSJim Ingham ListenerSP listener_sp,
10618e4272aSMichał Górny const FileSpec *crash_file_path,
10718e4272aSMichał Górny bool can_connect) {
108c3776bf2SGreg Clayton lldb::ProcessSP process_sp;
109c3776bf2SGreg Clayton if (crash_file_path == NULL)
110796ac80bSJonas Devlieghere process_sp = std::make_shared<ProcessKDP>(target_sp, listener_sp);
111c3776bf2SGreg Clayton return process_sp;
112f9765acdSGreg Clayton }
113f9765acdSGreg Clayton
CanDebug(TargetSP target_sp,bool plugin_specified_by_name)114b9c1b51eSKate Stone bool ProcessKDP::CanDebug(TargetSP target_sp, bool plugin_specified_by_name) {
115596ed24eSGreg Clayton if (plugin_specified_by_name)
116596ed24eSGreg Clayton return true;
117596ed24eSGreg Clayton
118f9765acdSGreg Clayton // For now we are just making sure the file exists for a given module
1195a3bb64fSJim Ingham Module *exe_module = target_sp->GetExecutableModulePointer();
120b9c1b51eSKate Stone if (exe_module) {
1215a3bb64fSJim Ingham const llvm::Triple &triple_ref = target_sp->GetArchitecture().GetTriple();
122b9c1b51eSKate Stone switch (triple_ref.getOS()) {
123b9c1b51eSKate Stone case llvm::Triple::Darwin: // Should use "macosx" for desktop and "ios" for
124b9c1b51eSKate Stone // iOS, but accept darwin just in case
12570512317SGreg Clayton case llvm::Triple::MacOSX: // For desktop targets
12670512317SGreg Clayton case llvm::Triple::IOS: // For arm targets
127a814f704SJason Molenda case llvm::Triple::TvOS:
128a814f704SJason Molenda case llvm::Triple::WatchOS:
129b9c1b51eSKate Stone if (triple_ref.getVendor() == llvm::Triple::Apple) {
130aa149cbdSGreg Clayton ObjectFile *exe_objfile = exe_module->GetObjectFile();
131f9765acdSGreg Clayton if (exe_objfile->GetType() == ObjectFile::eTypeExecutable &&
132f9765acdSGreg Clayton exe_objfile->GetStrata() == ObjectFile::eStrataKernel)
133f9765acdSGreg Clayton return true;
134f9765acdSGreg Clayton }
13570512317SGreg Clayton break;
13670512317SGreg Clayton
13770512317SGreg Clayton default:
13870512317SGreg Clayton break;
13970512317SGreg Clayton }
140f9765acdSGreg Clayton }
141596ed24eSGreg Clayton return false;
1423a29bdbeSGreg Clayton }
143f9765acdSGreg Clayton
144f9765acdSGreg Clayton // ProcessKDP constructor
ProcessKDP(TargetSP target_sp,ListenerSP listener_sp)145b9c1b51eSKate Stone ProcessKDP::ProcessKDP(TargetSP target_sp, ListenerSP listener_sp)
146b9c1b51eSKate Stone : Process(target_sp, listener_sp),
147f9765acdSGreg Clayton m_comm("lldb.process.kdp-remote.communication"),
1484bddaeb5SJim Ingham m_async_broadcaster(NULL, "lldb.process.kdp-remote.async-broadcaster"),
1496fa1b4ffSPavel Labath m_kernel_load_addr(LLDB_INVALID_ADDRESS), m_command_sp(),
1506fa1b4ffSPavel Labath m_kernel_thread_wp() {
151b9c1b51eSKate Stone m_async_broadcaster.SetEventName(eBroadcastBitAsyncThreadShouldExit,
152b9c1b51eSKate Stone "async thread should exit");
153b9c1b51eSKate Stone m_async_broadcaster.SetEventName(eBroadcastBitAsyncContinue,
154b9c1b51eSKate Stone "async thread continue");
155b9c1b51eSKate Stone const uint64_t timeout_seconds =
1563d7161e3SPavel Labath GetGlobalPluginProperties().GetPacketTimeout();
1577f98240dSGreg Clayton if (timeout_seconds > 0)
1585cddd608SPavel Labath m_comm.SetPacketTimeout(std::chrono::seconds(timeout_seconds));
159f9765acdSGreg Clayton }
160f9765acdSGreg Clayton
161f9765acdSGreg Clayton // Destructor
~ProcessKDP()162b9c1b51eSKate Stone ProcessKDP::~ProcessKDP() {
163f9765acdSGreg Clayton Clear();
16405097246SAdrian Prantl // We need to call finalize on the process before destroying ourselves to
16505097246SAdrian Prantl // make sure all of the broadcaster cleanup goes as planned. If we destruct
16605097246SAdrian Prantl // this class, then Process::~Process() might have problems trying to fully
16705097246SAdrian Prantl // destroy the broadcaster.
168e24c4acfSGreg Clayton Finalize();
169f9765acdSGreg Clayton }
170f9765acdSGreg Clayton
WillLaunch(Module * module)17197206d57SZachary Turner Status ProcessKDP::WillLaunch(Module *module) {
17297206d57SZachary Turner Status error;
173f9765acdSGreg Clayton error.SetErrorString("launching not supported in kdp-remote plug-in");
174f9765acdSGreg Clayton return error;
175f9765acdSGreg Clayton }
176f9765acdSGreg Clayton
WillAttachToProcessWithID(lldb::pid_t pid)17797206d57SZachary Turner Status ProcessKDP::WillAttachToProcessWithID(lldb::pid_t pid) {
17897206d57SZachary Turner Status error;
179b9c1b51eSKate Stone error.SetErrorString(
180b9c1b51eSKate Stone "attaching to a by process ID not supported in kdp-remote plug-in");
181f9765acdSGreg Clayton return error;
182f9765acdSGreg Clayton }
183f9765acdSGreg Clayton
WillAttachToProcessWithName(const char * process_name,bool wait_for_launch)18497206d57SZachary Turner Status ProcessKDP::WillAttachToProcessWithName(const char *process_name,
185b9c1b51eSKate Stone bool wait_for_launch) {
18697206d57SZachary Turner Status error;
187b9c1b51eSKate Stone error.SetErrorString(
188b9c1b51eSKate Stone "attaching to a by process name not supported in kdp-remote plug-in");
189f9765acdSGreg Clayton return error;
190f9765acdSGreg Clayton }
191f9765acdSGreg Clayton
GetHostArchitecture(ArchSpec & arch)192b9c1b51eSKate Stone bool ProcessKDP::GetHostArchitecture(ArchSpec &arch) {
193a3706888SGreg Clayton uint32_t cpu = m_comm.GetCPUType();
194b9c1b51eSKate Stone if (cpu) {
195a3706888SGreg Clayton uint32_t sub = m_comm.GetCPUSubtype();
196a3706888SGreg Clayton arch.SetArchitecture(eArchTypeMachO, cpu, sub);
197a3706888SGreg Clayton // Leave architecture vendor as unspecified unknown
198a3706888SGreg Clayton arch.GetTriple().setVendor(llvm::Triple::UnknownVendor);
199a3706888SGreg Clayton arch.GetTriple().setVendorName(llvm::StringRef());
200a3706888SGreg Clayton return true;
201a3706888SGreg Clayton }
202a3706888SGreg Clayton arch.Clear();
203a3706888SGreg Clayton return false;
204a3706888SGreg Clayton }
205a3706888SGreg Clayton
DoConnectRemote(llvm::StringRef remote_url)20632d35fb7SJonas Devlieghere Status ProcessKDP::DoConnectRemote(llvm::StringRef remote_url) {
20797206d57SZachary Turner Status error;
2083a29bdbeSGreg Clayton
20905097246SAdrian Prantl // Don't let any JIT happen when doing KDP as we can't allocate memory and we
21005097246SAdrian Prantl // don't want to be mucking with threads that might already be handling
21105097246SAdrian Prantl // exceptions
2127925fbbaSGreg Clayton SetCanJIT(false);
2137925fbbaSGreg Clayton
2143ce7e996SGreg Clayton if (remote_url.empty()) {
2153ce7e996SGreg Clayton error.SetErrorStringWithFormat("empty connection URL");
2167925fbbaSGreg Clayton return error;
2177925fbbaSGreg Clayton }
2183a29bdbeSGreg Clayton
219d5b44036SJonas Devlieghere std::unique_ptr<ConnectionFileDescriptor> conn_up(
220b9c1b51eSKate Stone new ConnectionFileDescriptor());
221d5b44036SJonas Devlieghere if (conn_up) {
2223a29bdbeSGreg Clayton // Only try once for now.
2233a29bdbeSGreg Clayton // TODO: check if we should be retrying?
2243a29bdbeSGreg Clayton const uint32_t max_retry_count = 1;
225b9c1b51eSKate Stone for (uint32_t retry_count = 0; retry_count < max_retry_count;
226b9c1b51eSKate Stone ++retry_count) {
227d5b44036SJonas Devlieghere if (conn_up->Connect(remote_url, &error) == eConnectionStatusSuccess)
2283a29bdbeSGreg Clayton break;
2293a29bdbeSGreg Clayton usleep(100000);
2303a29bdbeSGreg Clayton }
2313a29bdbeSGreg Clayton }
2323a29bdbeSGreg Clayton
233d5b44036SJonas Devlieghere if (conn_up->IsConnected()) {
234b9c1b51eSKate Stone const TCPSocket &socket =
235d5b44036SJonas Devlieghere static_cast<const TCPSocket &>(*conn_up->GetReadObject());
236014bb7daSVince Harron const uint16_t reply_port = socket.GetLocalPortNumber();
2373a29bdbeSGreg Clayton
238b9c1b51eSKate Stone if (reply_port != 0) {
239451741a9SPavel Labath m_comm.SetConnection(std::move(conn_up));
2403a29bdbeSGreg Clayton
241b9c1b51eSKate Stone if (m_comm.SendRequestReattach(reply_port)) {
242b9c1b51eSKate Stone if (m_comm.SendRequestConnect(reply_port, reply_port,
243b9c1b51eSKate Stone "Greetings from LLDB...")) {
2443a29bdbeSGreg Clayton m_comm.GetVersion();
2455a3bb64fSJim Ingham
246a3706888SGreg Clayton Target &target = GetTarget();
247a3706888SGreg Clayton ArchSpec kernel_arch;
248a3706888SGreg Clayton // The host architecture
249a3706888SGreg Clayton GetHostArchitecture(kernel_arch);
250a3706888SGreg Clayton ArchSpec target_arch = target.GetArchitecture();
251a3706888SGreg Clayton // Merge in any unspecified stuff into the target architecture in
252a3706888SGreg Clayton // case the target arch isn't set at all or incompletely.
253a3706888SGreg Clayton target_arch.MergeFrom(kernel_arch);
254a3706888SGreg Clayton target.SetArchitecture(target_arch);
2554bd4e7e3SJason Molenda
256b9c1b51eSKate Stone /* Get the kernel's UUID and load address via KDP_KERNELVERSION
257b9c1b51eSKate Stone * packet. */
258840f12cfSJason Molenda /* An EFI kdp session has neither UUID nor load address. */
2594bd4e7e3SJason Molenda
2604bd4e7e3SJason Molenda UUID kernel_uuid = m_comm.GetUUID();
2614bd4e7e3SJason Molenda addr_t kernel_load_addr = m_comm.GetLoadAddress();
2624bd4e7e3SJason Molenda
263b9c1b51eSKate Stone if (m_comm.RemoteIsEFI()) {
264b9c1b51eSKate Stone // Select an invalid plugin name for the dynamic loader so one
26505097246SAdrian Prantl // doesn't get used since EFI does its own manual loading via
26605097246SAdrian Prantl // python scripting
2676fa1b4ffSPavel Labath m_dyld_plugin_name = "none";
268a1bce2efSGreg Clayton
269a1bce2efSGreg Clayton if (kernel_uuid.IsValid()) {
27005097246SAdrian Prantl // If EFI passed in a UUID= try to lookup UUID The slide will not
27105097246SAdrian Prantl // be provided. But the UUID lookup will be used to launch EFI
27205097246SAdrian Prantl // debug scripts from the dSYM, that can load all of the symbols.
273a1bce2efSGreg Clayton ModuleSpec module_spec;
274a1bce2efSGreg Clayton module_spec.GetUUID() = kernel_uuid;
2755a3bb64fSJim Ingham module_spec.GetArchitecture() = target.GetArchitecture();
276a1bce2efSGreg Clayton
277a1bce2efSGreg Clayton // Lookup UUID locally, before attempting dsymForUUID like action
278a313ec11SZachary Turner FileSpecList search_paths =
279a313ec11SZachary Turner Target::GetDefaultDebugFileSearchPaths();
280b9c1b51eSKate Stone module_spec.GetSymbolFileSpec() =
281a313ec11SZachary Turner Symbols::LocateExecutableSymbolFile(module_spec,
282a313ec11SZachary Turner search_paths);
283b9c1b51eSKate Stone if (module_spec.GetSymbolFileSpec()) {
284b9c1b51eSKate Stone ModuleSpec executable_module_spec =
285b9c1b51eSKate Stone Symbols::LocateExecutableObjectFile(module_spec);
286dbd7fabaSJonas Devlieghere if (FileSystem::Instance().Exists(
287dbd7fabaSJonas Devlieghere executable_module_spec.GetFileSpec())) {
288b9c1b51eSKate Stone module_spec.GetFileSpec() =
289b9c1b51eSKate Stone executable_module_spec.GetFileSpec();
2908825c5c9SJason Molenda }
2918825c5c9SJason Molenda }
292b9c1b51eSKate Stone if (!module_spec.GetSymbolFileSpec() ||
293af91446aSJonas Devlieghere !module_spec.GetSymbolFileSpec()) {
294af91446aSJonas Devlieghere Status symbl_error;
295af91446aSJonas Devlieghere Symbols::DownloadObjectAndSymbolFile(module_spec, symbl_error,
296af91446aSJonas Devlieghere true);
297af91446aSJonas Devlieghere }
298a1bce2efSGreg Clayton
299dbd7fabaSJonas Devlieghere if (FileSystem::Instance().Exists(module_spec.GetFileSpec())) {
300a3706888SGreg Clayton ModuleSP module_sp(new Module(module_spec));
301b9c1b51eSKate Stone if (module_sp.get() && module_sp->GetObjectFile()) {
302a1bce2efSGreg Clayton // Get the current target executable
3035a3bb64fSJim Ingham ModuleSP exe_module_sp(target.GetExecutableModule());
304a1bce2efSGreg Clayton
305b9c1b51eSKate Stone // Make sure you don't already have the right module loaded
306b9c1b51eSKate Stone // and they will be uniqued
307a1bce2efSGreg Clayton if (exe_module_sp.get() != module_sp.get())
308f9a07e9fSJonas Devlieghere target.SetExecutableModule(module_sp, eLoadDependentsNo);
309a1bce2efSGreg Clayton }
310a1bce2efSGreg Clayton }
311a1bce2efSGreg Clayton }
312b9c1b51eSKate Stone } else if (m_comm.RemoteIsDarwinKernel()) {
313b9c1b51eSKate Stone m_dyld_plugin_name =
314b9c1b51eSKate Stone DynamicLoaderDarwinKernel::GetPluginNameStatic();
315b9c1b51eSKate Stone if (kernel_load_addr != LLDB_INVALID_ADDRESS) {
3165e8534efSJason Molenda m_kernel_load_addr = kernel_load_addr;
317a8ea4baeSJason Molenda }
3184bd4e7e3SJason Molenda }
3194bd4e7e3SJason Molenda
32097d5cf05SGreg Clayton // Set the thread ID
32197d5cf05SGreg Clayton UpdateThreadListIfNeeded();
322a63d08c9SGreg Clayton SetID(1);
32356d9a1b3SGreg Clayton GetThreadList();
324a63d08c9SGreg Clayton SetPrivateState(eStateStopped);
3255a3bb64fSJim Ingham StreamSP async_strm_sp(target.GetDebugger().GetAsyncOutputStream());
326b9c1b51eSKate Stone if (async_strm_sp) {
3275b88216dSGreg Clayton const char *cstr;
328b9c1b51eSKate Stone if ((cstr = m_comm.GetKernelVersion()) != NULL) {
3295b88216dSGreg Clayton async_strm_sp->Printf("Version: %s\n", cstr);
33007e66e3eSGreg Clayton async_strm_sp->Flush();
33107e66e3eSGreg Clayton }
3325b88216dSGreg Clayton // if ((cstr = m_comm.GetImagePath ()) != NULL)
3335b88216dSGreg Clayton // {
334b9c1b51eSKate Stone // async_strm_sp->Printf ("Image Path:
335b9c1b51eSKate Stone // %s\n", cstr);
3365b88216dSGreg Clayton // async_strm_sp->Flush();
3375b88216dSGreg Clayton // }
33807e66e3eSGreg Clayton }
339b9c1b51eSKate Stone } else {
34097d5cf05SGreg Clayton error.SetErrorString("KDP_REATTACH failed");
34197d5cf05SGreg Clayton }
342b9c1b51eSKate Stone } else {
34397d5cf05SGreg Clayton error.SetErrorString("KDP_REATTACH failed");
3443a29bdbeSGreg Clayton }
345b9c1b51eSKate Stone } else {
3463a29bdbeSGreg Clayton error.SetErrorString("invalid reply port from UDP connection");
3473a29bdbeSGreg Clayton }
348b9c1b51eSKate Stone } else {
3493a29bdbeSGreg Clayton if (error.Success())
3503ce7e996SGreg Clayton error.SetErrorStringWithFormat("failed to connect to '%s'",
3513ce7e996SGreg Clayton remote_url.str().c_str());
3523a29bdbeSGreg Clayton }
3533a29bdbeSGreg Clayton if (error.Fail())
3543a29bdbeSGreg Clayton m_comm.Disconnect();
3553a29bdbeSGreg Clayton
356f9765acdSGreg Clayton return error;
357f9765acdSGreg Clayton }
358f9765acdSGreg Clayton
359f9765acdSGreg Clayton // Process Control
DoLaunch(Module * exe_module,ProcessLaunchInfo & launch_info)36097206d57SZachary Turner Status ProcessKDP::DoLaunch(Module *exe_module,
36197206d57SZachary Turner ProcessLaunchInfo &launch_info) {
36297206d57SZachary Turner Status error;
363f9765acdSGreg Clayton error.SetErrorString("launching not supported in kdp-remote plug-in");
364f9765acdSGreg Clayton return error;
365f9765acdSGreg Clayton }
366f9765acdSGreg Clayton
36797206d57SZachary Turner Status
DoAttachToProcessWithID(lldb::pid_t attach_pid,const ProcessAttachInfo & attach_info)36897206d57SZachary Turner ProcessKDP::DoAttachToProcessWithID(lldb::pid_t attach_pid,
36997206d57SZachary Turner const ProcessAttachInfo &attach_info) {
37097206d57SZachary Turner Status error;
371b9c1b51eSKate Stone error.SetErrorString(
3724ebdee0aSBruce Mitchener "attach to process by ID is not supported in kdp remote debugging");
37384647048SHan Ming Ong return error;
37484647048SHan Ming Ong }
37584647048SHan Ming Ong
37697206d57SZachary Turner Status
DoAttachToProcessWithName(const char * process_name,const ProcessAttachInfo & attach_info)37797206d57SZachary Turner ProcessKDP::DoAttachToProcessWithName(const char *process_name,
37897206d57SZachary Turner const ProcessAttachInfo &attach_info) {
37997206d57SZachary Turner Status error;
380b9c1b51eSKate Stone error.SetErrorString(
3814ebdee0aSBruce Mitchener "attach to process by name is not supported in kdp remote debugging");
382f9765acdSGreg Clayton return error;
383f9765acdSGreg Clayton }
384f9765acdSGreg Clayton
DidAttach(ArchSpec & process_arch)385b9c1b51eSKate Stone void ProcessKDP::DidAttach(ArchSpec &process_arch) {
386bb006ce2SJim Ingham Process::DidAttach(process_arch);
387bb006ce2SJim Ingham
388c8fbf803SPavel Labath Log *log = GetLog(KDPLog::Process);
38963e5fb76SJonas Devlieghere LLDB_LOGF(log, "ProcessKDP::DidAttach()");
390b9c1b51eSKate Stone if (GetID() != LLDB_INVALID_PROCESS_ID) {
391a3706888SGreg Clayton GetHostArchitecture(process_arch);
392f9765acdSGreg Clayton }
393f9765acdSGreg Clayton }
394f9765acdSGreg Clayton
GetImageInfoAddress()395b9c1b51eSKate Stone addr_t ProcessKDP::GetImageInfoAddress() { return m_kernel_load_addr; }
3965e8534efSJason Molenda
GetDynamicLoader()397b9c1b51eSKate Stone lldb_private::DynamicLoader *ProcessKDP::GetDynamicLoader() {
398d5b44036SJonas Devlieghere if (m_dyld_up.get() == NULL)
3996fa1b4ffSPavel Labath m_dyld_up.reset(DynamicLoader::FindPlugin(this, m_dyld_plugin_name));
400d5b44036SJonas Devlieghere return m_dyld_up.get();
4015e8534efSJason Molenda }
4025e8534efSJason Molenda
WillResume()40397206d57SZachary Turner Status ProcessKDP::WillResume() { return Status(); }
404f9765acdSGreg Clayton
DoResume()40597206d57SZachary Turner Status ProcessKDP::DoResume() {
40697206d57SZachary Turner Status error;
407c8fbf803SPavel Labath Log *log = GetLog(KDPLog::Process);
4087925fbbaSGreg Clayton // Only start the async thread if we try to do any process control
409acee96aeSZachary Turner if (!m_async_thread.IsJoinable())
4107925fbbaSGreg Clayton StartAsyncThread();
4117925fbbaSGreg Clayton
41297d5cf05SGreg Clayton bool resume = false;
4137925fbbaSGreg Clayton
41497d5cf05SGreg Clayton // With KDP there is only one thread we can tell what to do
415ba4e61d3SAndrew Kaylor ThreadSP kernel_thread_sp(m_thread_list.FindThreadByProtocolID(g_kernel_tid));
416ba4e61d3SAndrew Kaylor
417b9c1b51eSKate Stone if (kernel_thread_sp) {
418b9c1b51eSKate Stone const StateType thread_resume_state =
419b9c1b51eSKate Stone kernel_thread_sp->GetTemporaryResumeState();
4206e0ff1a3SGreg Clayton
42163e5fb76SJonas Devlieghere LLDB_LOGF(log, "ProcessKDP::DoResume() thread_resume_state = %s",
422b9c1b51eSKate Stone StateAsCString(thread_resume_state));
423b9c1b51eSKate Stone switch (thread_resume_state) {
4247925fbbaSGreg Clayton case eStateSuspended:
42505097246SAdrian Prantl // Nothing to do here when a thread will stay suspended we just leave the
42605097246SAdrian Prantl // CPU mask bit set to zero for the thread
42763e5fb76SJonas Devlieghere LLDB_LOGF(log, "ProcessKDP::DoResume() = suspended???");
4287925fbbaSGreg Clayton break;
4297925fbbaSGreg Clayton
430b9c1b51eSKate Stone case eStateStepping: {
431b9c1b51eSKate Stone lldb::RegisterContextSP reg_ctx_sp(
432b9c1b51eSKate Stone kernel_thread_sp->GetRegisterContext());
4331afa68edSGreg Clayton
434b9c1b51eSKate Stone if (reg_ctx_sp) {
43563e5fb76SJonas Devlieghere LLDB_LOGF(
43663e5fb76SJonas Devlieghere log,
437b9c1b51eSKate Stone "ProcessKDP::DoResume () reg_ctx_sp->HardwareSingleStep (true);");
4381afa68edSGreg Clayton reg_ctx_sp->HardwareSingleStep(true);
43997d5cf05SGreg Clayton resume = true;
440b9c1b51eSKate Stone } else {
441b9c1b51eSKate Stone error.SetErrorStringWithFormat(
442b9c1b51eSKate Stone "KDP thread 0x%llx has no register context",
443b9c1b51eSKate Stone kernel_thread_sp->GetID());
4441afa68edSGreg Clayton }
445b9c1b51eSKate Stone } break;
4467925fbbaSGreg Clayton
447b9c1b51eSKate Stone case eStateRunning: {
448b9c1b51eSKate Stone lldb::RegisterContextSP reg_ctx_sp(
449b9c1b51eSKate Stone kernel_thread_sp->GetRegisterContext());
4501afa68edSGreg Clayton
451b9c1b51eSKate Stone if (reg_ctx_sp) {
45263e5fb76SJonas Devlieghere LLDB_LOGF(log, "ProcessKDP::DoResume () reg_ctx_sp->HardwareSingleStep "
453b9c1b51eSKate Stone "(false);");
4541afa68edSGreg Clayton reg_ctx_sp->HardwareSingleStep(false);
45597d5cf05SGreg Clayton resume = true;
456b9c1b51eSKate Stone } else {
457b9c1b51eSKate Stone error.SetErrorStringWithFormat(
458b9c1b51eSKate Stone "KDP thread 0x%llx has no register context",
459b9c1b51eSKate Stone kernel_thread_sp->GetID());
4601afa68edSGreg Clayton }
461b9c1b51eSKate Stone } break;
4627925fbbaSGreg Clayton
4637925fbbaSGreg Clayton default:
46497d5cf05SGreg Clayton // The only valid thread resume states are listed above
465a322f36cSDavid Blaikie llvm_unreachable("invalid thread resume state");
4664b1b8b3eSGreg Clayton }
4674b1b8b3eSGreg Clayton }
4687925fbbaSGreg Clayton
469b9c1b51eSKate Stone if (resume) {
47063e5fb76SJonas Devlieghere LLDB_LOGF(log, "ProcessKDP::DoResume () sending resume");
47197d5cf05SGreg Clayton
472b9c1b51eSKate Stone if (m_comm.SendRequestResume()) {
4737925fbbaSGreg Clayton m_async_broadcaster.BroadcastEvent(eBroadcastBitAsyncContinue);
4747925fbbaSGreg Clayton SetPrivateState(eStateRunning);
475b9c1b51eSKate Stone } else
47607e66e3eSGreg Clayton error.SetErrorString("KDP resume failed");
477b9c1b51eSKate Stone } else {
47897d5cf05SGreg Clayton error.SetErrorString("kernel thread is suspended");
4797925fbbaSGreg Clayton }
4807925fbbaSGreg Clayton
481f9765acdSGreg Clayton return error;
482f9765acdSGreg Clayton }
483f9765acdSGreg Clayton
GetKernelThread()484b9c1b51eSKate Stone lldb::ThreadSP ProcessKDP::GetKernelThread() {
48597d5cf05SGreg Clayton // KDP only tells us about one thread/core. Any other threads will usually
48697d5cf05SGreg Clayton // be the ones that are read from memory by the OS plug-ins.
487ba4e61d3SAndrew Kaylor
488ba4e61d3SAndrew Kaylor ThreadSP thread_sp(m_kernel_thread_wp.lock());
489b9c1b51eSKate Stone if (!thread_sp) {
490796ac80bSJonas Devlieghere thread_sp = std::make_shared<ThreadKDP>(*this, g_kernel_tid);
491ba4e61d3SAndrew Kaylor m_kernel_thread_wp = thread_sp;
492ba4e61d3SAndrew Kaylor }
49397d5cf05SGreg Clayton return thread_sp;
49497d5cf05SGreg Clayton }
49597d5cf05SGreg Clayton
DoUpdateThreadList(ThreadList & old_thread_list,ThreadList & new_thread_list)4964bb62448SWalter Erquinigo bool ProcessKDP::DoUpdateThreadList(ThreadList &old_thread_list,
497b9c1b51eSKate Stone ThreadList &new_thread_list) {
498f9765acdSGreg Clayton // locker will keep a mutex locked until it goes out of scope
499c8fbf803SPavel Labath Log *log = GetLog(KDPLog::Thread);
500250858a0SPavel Labath LLDB_LOGV(log, "pid = {0}", GetID());
501f9765acdSGreg Clayton
50239da3efdSGreg Clayton // Even though there is a CPU mask, it doesn't mean we can see each CPU
503a868c13cSBruce Mitchener // individually, there is really only one. Lets call this thread 1.
504b9c1b51eSKate Stone ThreadSP thread_sp(
505b9c1b51eSKate Stone old_thread_list.FindThreadByProtocolID(g_kernel_tid, false));
506ba4e61d3SAndrew Kaylor if (!thread_sp)
507ba4e61d3SAndrew Kaylor thread_sp = GetKernelThread();
508ba4e61d3SAndrew Kaylor new_thread_list.AddThread(thread_sp);
50997d5cf05SGreg Clayton
5109fc13556SGreg Clayton return new_thread_list.GetSize(false) > 0;
511f9765acdSGreg Clayton }
512f9765acdSGreg Clayton
RefreshStateAfterStop()513b9c1b51eSKate Stone void ProcessKDP::RefreshStateAfterStop() {
51405097246SAdrian Prantl // Let all threads recover from stopping and do any clean up based on the
51505097246SAdrian Prantl // previous thread state (if any).
516f9765acdSGreg Clayton m_thread_list.RefreshStateAfterStop();
517f9765acdSGreg Clayton }
518f9765acdSGreg Clayton
DoHalt(bool & caused_stop)51997206d57SZachary Turner Status ProcessKDP::DoHalt(bool &caused_stop) {
52097206d57SZachary Turner Status error;
521f9765acdSGreg Clayton
522b9c1b51eSKate Stone if (m_comm.IsRunning()) {
523b9c1b51eSKate Stone if (m_destroy_in_process) {
5244ebdee0aSBruce Mitchener // If we are attempting to destroy, we need to not return an error to Halt
52505097246SAdrian Prantl // or DoDestroy won't get called. We are also currently running, so send
52605097246SAdrian Prantl // a process stopped event
52797d5cf05SGreg Clayton SetPrivateState(eStateStopped);
528b9c1b51eSKate Stone } else {
52997d5cf05SGreg Clayton error.SetErrorString("KDP cannot interrupt a running kernel");
530f9765acdSGreg Clayton }
531f9765acdSGreg Clayton }
532f9765acdSGreg Clayton return error;
533f9765acdSGreg Clayton }
534f9765acdSGreg Clayton
DoDetach(bool keep_stopped)53597206d57SZachary Turner Status ProcessKDP::DoDetach(bool keep_stopped) {
53697206d57SZachary Turner Status error;
537c8fbf803SPavel Labath Log *log = GetLog(KDPLog::Process);
53863e5fb76SJonas Devlieghere LLDB_LOGF(log, "ProcessKDP::DoDetach(keep_stopped = %i)", keep_stopped);
539f9765acdSGreg Clayton
540b9c1b51eSKate Stone if (m_comm.IsRunning()) {
54105097246SAdrian Prantl // We are running and we can't interrupt a running kernel, so we need to
54205097246SAdrian Prantl // just close the connection to the kernel and hope for the best
543b9c1b51eSKate Stone } else {
544b9c1b51eSKate Stone // If we are going to keep the target stopped, then don't send the
545b9c1b51eSKate Stone // disconnect message.
546b9c1b51eSKate Stone if (!keep_stopped && m_comm.IsConnected()) {
5476e0ff1a3SGreg Clayton const bool success = m_comm.SendRequestDisconnect();
548b9c1b51eSKate Stone if (log) {
5496e0ff1a3SGreg Clayton if (success)
550b9c1b51eSKate Stone log->PutCString(
551b9c1b51eSKate Stone "ProcessKDP::DoDetach() detach packet sent successfully");
552f9765acdSGreg Clayton else
553b9c1b51eSKate Stone log->PutCString(
554b9c1b51eSKate Stone "ProcessKDP::DoDetach() connection channel shutdown failed");
555f9765acdSGreg Clayton }
5566e0ff1a3SGreg Clayton m_comm.Disconnect();
5573a29bdbeSGreg Clayton }
55897d5cf05SGreg Clayton }
559f9765acdSGreg Clayton StopAsyncThread();
56074d4193eSGreg Clayton m_comm.Clear();
561f9765acdSGreg Clayton
562f9765acdSGreg Clayton SetPrivateState(eStateDetached);
563f9765acdSGreg Clayton ResumePrivateStateThread();
564f9765acdSGreg Clayton
565f9765acdSGreg Clayton // KillDebugserverProcess ();
566f9765acdSGreg Clayton return error;
567f9765acdSGreg Clayton }
568f9765acdSGreg Clayton
DoDestroy()56997206d57SZachary Turner Status ProcessKDP::DoDestroy() {
5707925fbbaSGreg Clayton // For KDP there really is no difference between destroy and detach
571acff8950SJim Ingham bool keep_stopped = false;
572acff8950SJim Ingham return DoDetach(keep_stopped);
573f9765acdSGreg Clayton }
574f9765acdSGreg Clayton
575f9765acdSGreg Clayton // Process Queries
576f9765acdSGreg Clayton
IsAlive()577b9c1b51eSKate Stone bool ProcessKDP::IsAlive() {
578a814f704SJason Molenda return m_comm.IsConnected() && Process::IsAlive();
579f9765acdSGreg Clayton }
580f9765acdSGreg Clayton
581f9765acdSGreg Clayton // Process Memory
DoReadMemory(addr_t addr,void * buf,size_t size,Status & error)582b9c1b51eSKate Stone size_t ProcessKDP::DoReadMemory(addr_t addr, void *buf, size_t size,
58397206d57SZachary Turner Status &error) {
5848eb32817SJason Molenda uint8_t *data_buffer = (uint8_t *)buf;
585b9c1b51eSKate Stone if (m_comm.IsConnected()) {
5868eb32817SJason Molenda const size_t max_read_size = 512;
5878eb32817SJason Molenda size_t total_bytes_read = 0;
5888eb32817SJason Molenda
5898eb32817SJason Molenda // Read the requested amount of memory in 512 byte chunks
590b9c1b51eSKate Stone while (total_bytes_read < size) {
5918eb32817SJason Molenda size_t bytes_to_read_this_request = size - total_bytes_read;
592b9c1b51eSKate Stone if (bytes_to_read_this_request > max_read_size) {
5938eb32817SJason Molenda bytes_to_read_this_request = max_read_size;
5948eb32817SJason Molenda }
595b9c1b51eSKate Stone size_t bytes_read = m_comm.SendRequestReadMemory(
596b9c1b51eSKate Stone addr + total_bytes_read, data_buffer + total_bytes_read,
5978eb32817SJason Molenda bytes_to_read_this_request, error);
5988eb32817SJason Molenda total_bytes_read += bytes_read;
599b9c1b51eSKate Stone if (error.Fail() || bytes_read == 0) {
6008eb32817SJason Molenda return total_bytes_read;
6018eb32817SJason Molenda }
6028eb32817SJason Molenda }
6038eb32817SJason Molenda
6048eb32817SJason Molenda return total_bytes_read;
6058eb32817SJason Molenda }
606a63d08c9SGreg Clayton error.SetErrorString("not connected");
607f9765acdSGreg Clayton return 0;
608f9765acdSGreg Clayton }
609f9765acdSGreg Clayton
DoWriteMemory(addr_t addr,const void * buf,size_t size,Status & error)610b9c1b51eSKate Stone size_t ProcessKDP::DoWriteMemory(addr_t addr, const void *buf, size_t size,
61197206d57SZachary Turner Status &error) {
6127925fbbaSGreg Clayton if (m_comm.IsConnected())
6137925fbbaSGreg Clayton return m_comm.SendRequestWriteMemory(addr, buf, size, error);
6147925fbbaSGreg Clayton error.SetErrorString("not connected");
615f9765acdSGreg Clayton return 0;
616f9765acdSGreg Clayton }
617f9765acdSGreg Clayton
DoAllocateMemory(size_t size,uint32_t permissions,Status & error)618b9c1b51eSKate Stone lldb::addr_t ProcessKDP::DoAllocateMemory(size_t size, uint32_t permissions,
61997206d57SZachary Turner Status &error) {
620b9c1b51eSKate Stone error.SetErrorString(
6214ebdee0aSBruce Mitchener "memory allocation not supported in kdp remote debugging");
622f9765acdSGreg Clayton return LLDB_INVALID_ADDRESS;
623f9765acdSGreg Clayton }
624f9765acdSGreg Clayton
DoDeallocateMemory(lldb::addr_t addr)62597206d57SZachary Turner Status ProcessKDP::DoDeallocateMemory(lldb::addr_t addr) {
62697206d57SZachary Turner Status error;
627b9c1b51eSKate Stone error.SetErrorString(
6284ebdee0aSBruce Mitchener "memory deallocation not supported in kdp remote debugging");
629f9765acdSGreg Clayton return error;
630f9765acdSGreg Clayton }
631f9765acdSGreg Clayton
EnableBreakpointSite(BreakpointSite * bp_site)63297206d57SZachary Turner Status ProcessKDP::EnableBreakpointSite(BreakpointSite *bp_site) {
633b352e62fSTatyana Krasnukha if (bp_site->HardwareRequired())
634b352e62fSTatyana Krasnukha return Status("Hardware breakpoints are not supported.");
635b352e62fSTatyana Krasnukha
636b9c1b51eSKate Stone if (m_comm.LocalBreakpointsAreSupported()) {
63797206d57SZachary Turner Status error;
638b9c1b51eSKate Stone if (!bp_site->IsEnabled()) {
639b9c1b51eSKate Stone if (m_comm.SendRequestBreakpoint(true, bp_site->GetLoadAddress())) {
6405b88216dSGreg Clayton bp_site->SetEnabled(true);
6415b88216dSGreg Clayton bp_site->SetType(BreakpointSite::eExternal);
642b9c1b51eSKate Stone } else {
64307e66e3eSGreg Clayton error.SetErrorString("KDP set breakpoint failed");
6445b88216dSGreg Clayton }
6455b88216dSGreg Clayton }
64607e66e3eSGreg Clayton return error;
64707e66e3eSGreg Clayton }
648f9765acdSGreg Clayton return EnableSoftwareBreakpoint(bp_site);
649f9765acdSGreg Clayton }
650f9765acdSGreg Clayton
DisableBreakpointSite(BreakpointSite * bp_site)65197206d57SZachary Turner Status ProcessKDP::DisableBreakpointSite(BreakpointSite *bp_site) {
652b9c1b51eSKate Stone if (m_comm.LocalBreakpointsAreSupported()) {
65397206d57SZachary Turner Status error;
654b9c1b51eSKate Stone if (bp_site->IsEnabled()) {
6555b88216dSGreg Clayton BreakpointSite::Type bp_type = bp_site->GetType();
656b9c1b51eSKate Stone if (bp_type == BreakpointSite::eExternal) {
657b9c1b51eSKate Stone if (m_destroy_in_process && m_comm.IsRunning()) {
65897d5cf05SGreg Clayton // We are trying to destroy our connection and we are running
65997d5cf05SGreg Clayton bp_site->SetEnabled(false);
660b9c1b51eSKate Stone } else {
6615b88216dSGreg Clayton if (m_comm.SendRequestBreakpoint(false, bp_site->GetLoadAddress()))
6625b88216dSGreg Clayton bp_site->SetEnabled(false);
6635b88216dSGreg Clayton else
66407e66e3eSGreg Clayton error.SetErrorString("KDP remove breakpoint failed");
6655b88216dSGreg Clayton }
666b9c1b51eSKate Stone } else {
6675b88216dSGreg Clayton error = DisableSoftwareBreakpoint(bp_site);
6685b88216dSGreg Clayton }
6695b88216dSGreg Clayton }
67007e66e3eSGreg Clayton return error;
67107e66e3eSGreg Clayton }
672f9765acdSGreg Clayton return DisableSoftwareBreakpoint(bp_site);
673f9765acdSGreg Clayton }
674f9765acdSGreg Clayton
EnableWatchpoint(Watchpoint * wp,bool notify)67597206d57SZachary Turner Status ProcessKDP::EnableWatchpoint(Watchpoint *wp, bool notify) {
67697206d57SZachary Turner Status error;
677b9c1b51eSKate Stone error.SetErrorString(
6784ebdee0aSBruce Mitchener "watchpoints are not supported in kdp remote debugging");
679f9765acdSGreg Clayton return error;
680f9765acdSGreg Clayton }
681f9765acdSGreg Clayton
DisableWatchpoint(Watchpoint * wp,bool notify)68297206d57SZachary Turner Status ProcessKDP::DisableWatchpoint(Watchpoint *wp, bool notify) {
68397206d57SZachary Turner Status error;
684b9c1b51eSKate Stone error.SetErrorString(
6854ebdee0aSBruce Mitchener "watchpoints are not supported in kdp remote debugging");
686f9765acdSGreg Clayton return error;
687f9765acdSGreg Clayton }
688f9765acdSGreg Clayton
Clear()689b9c1b51eSKate Stone void ProcessKDP::Clear() { m_thread_list.Clear(); }
690f9765acdSGreg Clayton
DoSignal(int signo)69197206d57SZachary Turner Status ProcessKDP::DoSignal(int signo) {
69297206d57SZachary Turner Status error;
693b9c1b51eSKate Stone error.SetErrorString(
6944ebdee0aSBruce Mitchener "sending signals is not supported in kdp remote debugging");
695f9765acdSGreg Clayton return error;
696f9765acdSGreg Clayton }
697f9765acdSGreg Clayton
Initialize()698b9c1b51eSKate Stone void ProcessKDP::Initialize() {
699c5f28e2aSKamil Rytarowski static llvm::once_flag g_once_flag;
700f9765acdSGreg Clayton
701c5f28e2aSKamil Rytarowski llvm::call_once(g_once_flag, []() {
702f9765acdSGreg Clayton PluginManager::RegisterPlugin(GetPluginNameStatic(),
703b9c1b51eSKate Stone GetPluginDescriptionStatic(), CreateInstance,
704488c89edSBruce Mitchener DebuggerInitialize);
705f9765acdSGreg Clayton
7067b35b781SPavel Labath ProcessKDPLog::Initialize();
707c8d69828SDavide Italiano });
708f9765acdSGreg Clayton }
709f9765acdSGreg Clayton
DebuggerInitialize(lldb_private::Debugger & debugger)710b9c1b51eSKate Stone void ProcessKDP::DebuggerInitialize(lldb_private::Debugger &debugger) {
711b9c1b51eSKate Stone if (!PluginManager::GetSettingForProcessPlugin(
712b9c1b51eSKate Stone debugger, PluginProperties::GetSettingName())) {
7137f98240dSGreg Clayton const bool is_global_setting = true;
714b9c1b51eSKate Stone PluginManager::CreateSettingForProcessPlugin(
7153d7161e3SPavel Labath debugger, GetGlobalPluginProperties().GetValueProperties(),
7167f98240dSGreg Clayton ConstString("Properties for the kdp-remote process plug-in."),
7177f98240dSGreg Clayton is_global_setting);
7187f98240dSGreg Clayton }
7197f98240dSGreg Clayton }
7207f98240dSGreg Clayton
StartAsyncThread()721b9c1b51eSKate Stone bool ProcessKDP::StartAsyncThread() {
722c8fbf803SPavel Labath Log *log = GetLog(KDPLog::Process);
723f9765acdSGreg Clayton
72463e5fb76SJonas Devlieghere LLDB_LOGF(log, "ProcessKDP::StartAsyncThread ()");
725f9765acdSGreg Clayton
726acee96aeSZachary Turner if (m_async_thread.IsJoinable())
7277925fbbaSGreg Clayton return true;
7287925fbbaSGreg Clayton
729f39c2e18SJonas Devlieghere llvm::Expected<HostThread> async_thread = ThreadLauncher::LaunchThread(
730d0810779SPavel Labath "<lldb.process.kdp-remote.async>", [this] { return AsyncThread(); });
731f39c2e18SJonas Devlieghere if (!async_thread) {
732c8fbf803SPavel Labath LLDB_LOG_ERROR(GetLog(LLDBLog::Host), async_thread.takeError(),
733c8fbf803SPavel Labath "failed to launch host thread: {}");
734f39c2e18SJonas Devlieghere return false;
735f39c2e18SJonas Devlieghere }
736f39c2e18SJonas Devlieghere m_async_thread = *async_thread;
737acee96aeSZachary Turner return m_async_thread.IsJoinable();
738f9765acdSGreg Clayton }
739f9765acdSGreg Clayton
StopAsyncThread()740b9c1b51eSKate Stone void ProcessKDP::StopAsyncThread() {
741c8fbf803SPavel Labath Log *log = GetLog(KDPLog::Process);
742f9765acdSGreg Clayton
74363e5fb76SJonas Devlieghere LLDB_LOGF(log, "ProcessKDP::StopAsyncThread ()");
744f9765acdSGreg Clayton
745f9765acdSGreg Clayton m_async_broadcaster.BroadcastEvent(eBroadcastBitAsyncThreadShouldExit);
746f9765acdSGreg Clayton
747f9765acdSGreg Clayton // Stop the stdio thread
748acee96aeSZachary Turner if (m_async_thread.IsJoinable())
74939de3110SZachary Turner m_async_thread.Join(nullptr);
750f9765acdSGreg Clayton }
751f9765acdSGreg Clayton
AsyncThread()752d0810779SPavel Labath void *ProcessKDP::AsyncThread() {
753d0810779SPavel Labath const lldb::pid_t pid = GetID();
7547925fbbaSGreg Clayton
755c8fbf803SPavel Labath Log *log = GetLog(KDPLog::Process);
75663e5fb76SJonas Devlieghere LLDB_LOGF(log,
757d0810779SPavel Labath "ProcessKDP::AsyncThread(pid = %" PRIu64 ") thread starting...",
758d0810779SPavel Labath pid);
759f9765acdSGreg Clayton
760583bbb1dSJim Ingham ListenerSP listener_sp(Listener::MakeListener("ProcessKDP::AsyncThread"));
761f9765acdSGreg Clayton EventSP event_sp;
762b9c1b51eSKate Stone const uint32_t desired_event_mask =
763b9c1b51eSKate Stone eBroadcastBitAsyncContinue | eBroadcastBitAsyncThreadShouldExit;
764f9765acdSGreg Clayton
765d0810779SPavel Labath if (listener_sp->StartListeningForEvents(
766d0810779SPavel Labath &m_async_broadcaster, desired_event_mask) == desired_event_mask) {
767f9765acdSGreg Clayton bool done = false;
768b9c1b51eSKate Stone while (!done) {
76963e5fb76SJonas Devlieghere LLDB_LOGF(log,
77063e5fb76SJonas Devlieghere "ProcessKDP::AsyncThread (pid = %" PRIu64
771b9c1b51eSKate Stone ") listener.WaitForEvent (NULL, event_sp)...",
7727925fbbaSGreg Clayton pid);
773fafff0c5SPavel Labath if (listener_sp->GetEvent(event_sp, llvm::None)) {
7747925fbbaSGreg Clayton uint32_t event_type = event_sp->GetType();
77563e5fb76SJonas Devlieghere LLDB_LOGF(log,
77663e5fb76SJonas Devlieghere "ProcessKDP::AsyncThread (pid = %" PRIu64
777b9c1b51eSKate Stone ") Got an event of type: %d...",
778b9c1b51eSKate Stone pid, event_type);
779f9765acdSGreg Clayton
7807925fbbaSGreg Clayton // When we are running, poll for 1 second to try and get an exception
7817925fbbaSGreg Clayton // to indicate the process has stopped. If we don't get one, check to
7827925fbbaSGreg Clayton // make sure no one asked us to exit
7837925fbbaSGreg Clayton bool is_running = false;
7847925fbbaSGreg Clayton DataExtractor exc_reply_packet;
785b9c1b51eSKate Stone do {
786b9c1b51eSKate Stone switch (event_type) {
787b9c1b51eSKate Stone case eBroadcastBitAsyncContinue: {
7887925fbbaSGreg Clayton is_running = true;
789d0810779SPavel Labath if (m_comm.WaitForPacketWithTimeoutMicroSeconds(
790b9c1b51eSKate Stone exc_reply_packet, 1 * USEC_PER_SEC)) {
791d0810779SPavel Labath ThreadSP thread_sp(GetKernelThread());
792b9c1b51eSKate Stone if (thread_sp) {
793b9c1b51eSKate Stone lldb::RegisterContextSP reg_ctx_sp(
794b9c1b51eSKate Stone thread_sp->GetRegisterContext());
7951afa68edSGreg Clayton if (reg_ctx_sp)
7961afa68edSGreg Clayton reg_ctx_sp->InvalidateAllRegisters();
797b9c1b51eSKate Stone static_cast<ThreadKDP *>(thread_sp.get())
798b9c1b51eSKate Stone ->SetStopInfoFrom_KDP_EXCEPTION(exc_reply_packet);
7991afa68edSGreg Clayton }
80097d5cf05SGreg Clayton
8017925fbbaSGreg Clayton // TODO: parse the stop reply packet
8027925fbbaSGreg Clayton is_running = false;
803d0810779SPavel Labath SetPrivateState(eStateStopped);
804b9c1b51eSKate Stone } else {
8057925fbbaSGreg Clayton // Check to see if we are supposed to exit. There is no way to
8067925fbbaSGreg Clayton // interrupt a running kernel, so all we can do is wait for an
8077925fbbaSGreg Clayton // exception or detach...
808fafff0c5SPavel Labath if (listener_sp->GetEvent(event_sp,
809fafff0c5SPavel Labath std::chrono::microseconds(0))) {
8107925fbbaSGreg Clayton // We got an event, go through the loop again
8117925fbbaSGreg Clayton event_type = event_sp->GetType();
8127925fbbaSGreg Clayton }
813f9765acdSGreg Clayton }
814b9c1b51eSKate Stone } break;
815f9765acdSGreg Clayton
816f9765acdSGreg Clayton case eBroadcastBitAsyncThreadShouldExit:
81763e5fb76SJonas Devlieghere LLDB_LOGF(log,
81863e5fb76SJonas Devlieghere "ProcessKDP::AsyncThread (pid = %" PRIu64
819b9c1b51eSKate Stone ") got eBroadcastBitAsyncThreadShouldExit...",
8207925fbbaSGreg Clayton pid);
821f9765acdSGreg Clayton done = true;
8227925fbbaSGreg Clayton is_running = false;
823f9765acdSGreg Clayton break;
824f9765acdSGreg Clayton
825f9765acdSGreg Clayton default:
82663e5fb76SJonas Devlieghere LLDB_LOGF(log,
82763e5fb76SJonas Devlieghere "ProcessKDP::AsyncThread (pid = %" PRIu64
828b9c1b51eSKate Stone ") got unknown event 0x%8.8x",
829b9c1b51eSKate Stone pid, event_type);
830f9765acdSGreg Clayton done = true;
8317925fbbaSGreg Clayton is_running = false;
832f9765acdSGreg Clayton break;
833f9765acdSGreg Clayton }
8347925fbbaSGreg Clayton } while (is_running);
835b9c1b51eSKate Stone } else {
83663e5fb76SJonas Devlieghere LLDB_LOGF(log,
83763e5fb76SJonas Devlieghere "ProcessKDP::AsyncThread (pid = %" PRIu64
838b9c1b51eSKate Stone ") listener.WaitForEvent (NULL, event_sp) => false",
8397925fbbaSGreg Clayton pid);
840f9765acdSGreg Clayton done = true;
841f9765acdSGreg Clayton }
842f9765acdSGreg Clayton }
843f9765acdSGreg Clayton }
844f9765acdSGreg Clayton
845d0810779SPavel Labath LLDB_LOGF(log, "ProcessKDP::AsyncThread(pid = %" PRIu64 ") thread exiting...",
846d0810779SPavel Labath pid);
847f9765acdSGreg Clayton
848d0810779SPavel Labath m_async_thread.Reset();
849f9765acdSGreg Clayton return NULL;
850f9765acdSGreg Clayton }
851f9765acdSGreg Clayton
852b9c1b51eSKate Stone class CommandObjectProcessKDPPacketSend : public CommandObjectParsed {
8531d19a2f2SGreg Clayton private:
8541d19a2f2SGreg Clayton OptionGroupOptions m_option_group;
8551d19a2f2SGreg Clayton OptionGroupUInt64 m_command_byte;
8561d19a2f2SGreg Clayton OptionGroupString m_packet_data;
8571d19a2f2SGreg Clayton
GetOptions()85844b43a52SLogan Smith Options *GetOptions() override { return &m_option_group; }
8591d19a2f2SGreg Clayton
8601d19a2f2SGreg Clayton public:
CommandObjectProcessKDPPacketSend(CommandInterpreter & interpreter)861b9c1b51eSKate Stone CommandObjectProcessKDPPacketSend(CommandInterpreter &interpreter)
862b9c1b51eSKate Stone : CommandObjectParsed(interpreter, "process plugin packet send",
863b9c1b51eSKate Stone "Send a custom packet through the KDP protocol by "
864b9c1b51eSKate Stone "specifying the command byte and the packet "
865b9c1b51eSKate Stone "payload data. A packet will be sent with a "
866b9c1b51eSKate Stone "correct header and payload, and the raw result "
867b9c1b51eSKate Stone "bytes will be displayed as a string value. ",
8681d19a2f2SGreg Clayton NULL),
869e1cfbc79STodd Fiala m_option_group(),
870b9c1b51eSKate Stone m_command_byte(LLDB_OPT_SET_1, true, "command", 'c', 0, eArgTypeNone,
871b9c1b51eSKate Stone "Specify the command byte to use when sending the KDP "
872b9c1b51eSKate Stone "request packet.",
873b9c1b51eSKate Stone 0),
874b9c1b51eSKate Stone m_packet_data(LLDB_OPT_SET_1, false, "payload", 'p', 0, eArgTypeNone,
875b9c1b51eSKate Stone "Specify packet payload bytes as a hex ASCII string with "
876b9c1b51eSKate Stone "no spaces or hex prefixes.",
877b9c1b51eSKate Stone NULL) {
8781d19a2f2SGreg Clayton m_option_group.Append(&m_command_byte, LLDB_OPT_SET_ALL, LLDB_OPT_SET_1);
8791d19a2f2SGreg Clayton m_option_group.Append(&m_packet_data, LLDB_OPT_SET_ALL, LLDB_OPT_SET_1);
8801d19a2f2SGreg Clayton m_option_group.Finalize();
8811d19a2f2SGreg Clayton }
8821d19a2f2SGreg Clayton
883*fd146460SShafik Yaghmour ~CommandObjectProcessKDPPacketSend() override = default;
8841d19a2f2SGreg Clayton
DoExecute(Args & command,CommandReturnObject & result)88544b43a52SLogan Smith bool DoExecute(Args &command, CommandReturnObject &result) override {
886b9c1b51eSKate Stone if (!m_command_byte.GetOptionValue().OptionWasSet()) {
887b9c1b51eSKate Stone result.AppendError(
888b9c1b51eSKate Stone "the --command option must be set to a valid command byte");
889b9c1b51eSKate Stone } else {
890b9c1b51eSKate Stone const uint64_t command_byte =
891b9c1b51eSKate Stone m_command_byte.GetOptionValue().GetUInt64Value(0);
892b9c1b51eSKate Stone if (command_byte > 0 && command_byte <= UINT8_MAX) {
893b9c1b51eSKate Stone ProcessKDP *process =
894b9c1b51eSKate Stone (ProcessKDP *)m_interpreter.GetExecutionContext().GetProcessPtr();
895b9c1b51eSKate Stone if (process) {
8961d19a2f2SGreg Clayton const StateType state = process->GetState();
8971d19a2f2SGreg Clayton
898b9c1b51eSKate Stone if (StateIsStoppedState(state, true)) {
8991d19a2f2SGreg Clayton std::vector<uint8_t> payload_bytes;
900b9c1b51eSKate Stone const char *ascii_hex_bytes_cstr =
901b9c1b51eSKate Stone m_packet_data.GetOptionValue().GetCurrentValue();
902b9c1b51eSKate Stone if (ascii_hex_bytes_cstr && ascii_hex_bytes_cstr[0]) {
9031d19a2f2SGreg Clayton StringExtractor extractor(ascii_hex_bytes_cstr);
904b9c1b51eSKate Stone const size_t ascii_hex_bytes_cstr_len =
905b9c1b51eSKate Stone extractor.GetStringRef().size();
906b9c1b51eSKate Stone if (ascii_hex_bytes_cstr_len & 1) {
907b9c1b51eSKate Stone result.AppendErrorWithFormat("payload data must contain an "
908b9c1b51eSKate Stone "even number of ASCII hex "
909b9c1b51eSKate Stone "characters: '%s'",
910b9c1b51eSKate Stone ascii_hex_bytes_cstr);
9111d19a2f2SGreg Clayton return false;
9121d19a2f2SGreg Clayton }
9131d19a2f2SGreg Clayton payload_bytes.resize(ascii_hex_bytes_cstr_len / 2);
914b9c1b51eSKate Stone if (extractor.GetHexBytes(payload_bytes, '\xdd') !=
915b9c1b51eSKate Stone payload_bytes.size()) {
916b9c1b51eSKate Stone result.AppendErrorWithFormat("payload data must only contain "
917b9c1b51eSKate Stone "ASCII hex characters (no "
918b9c1b51eSKate Stone "spaces or hex prefixes): '%s'",
919b9c1b51eSKate Stone ascii_hex_bytes_cstr);
9201d19a2f2SGreg Clayton return false;
9211d19a2f2SGreg Clayton }
9221d19a2f2SGreg Clayton }
92397206d57SZachary Turner Status error;
9241d19a2f2SGreg Clayton DataExtractor reply;
925b9c1b51eSKate Stone process->GetCommunication().SendRawRequest(
926b9c1b51eSKate Stone command_byte,
9271d19a2f2SGreg Clayton payload_bytes.empty() ? NULL : payload_bytes.data(),
928b9c1b51eSKate Stone payload_bytes.size(), reply, error);
9291d19a2f2SGreg Clayton
930b9c1b51eSKate Stone if (error.Success()) {
9311d19a2f2SGreg Clayton // Copy the binary bytes into a hex ASCII string for the result
9321d19a2f2SGreg Clayton StreamString packet;
933b9c1b51eSKate Stone packet.PutBytesAsRawHex8(
934b9c1b51eSKate Stone reply.GetDataStart(), reply.GetByteSize(),
935b9c1b51eSKate Stone endian::InlHostByteOrder(), endian::InlHostByteOrder());
936c156427dSZachary Turner result.AppendMessage(packet.GetString());
9371d19a2f2SGreg Clayton result.SetStatus(eReturnStatusSuccessFinishResult);
9381d19a2f2SGreg Clayton return true;
939b9c1b51eSKate Stone } else {
9401d19a2f2SGreg Clayton const char *error_cstr = error.AsCString();
9411d19a2f2SGreg Clayton if (error_cstr && error_cstr[0])
9421d19a2f2SGreg Clayton result.AppendError(error_cstr);
9431d19a2f2SGreg Clayton else
944b9c1b51eSKate Stone result.AppendErrorWithFormat("unknown error 0x%8.8x",
945b9c1b51eSKate Stone error.GetError());
9461d19a2f2SGreg Clayton return false;
9471d19a2f2SGreg Clayton }
948b9c1b51eSKate Stone } else {
949b9c1b51eSKate Stone result.AppendErrorWithFormat("process must be stopped in order "
950b9c1b51eSKate Stone "to send KDP packets, state is %s",
951b9c1b51eSKate Stone StateAsCString(state));
9521d19a2f2SGreg Clayton }
953b9c1b51eSKate Stone } else {
9541d19a2f2SGreg Clayton result.AppendError("invalid process");
9551d19a2f2SGreg Clayton }
956b9c1b51eSKate Stone } else {
957b9c1b51eSKate Stone result.AppendErrorWithFormat("invalid command byte 0x%" PRIx64
958b9c1b51eSKate Stone ", valid values are 1 - 255",
959b9c1b51eSKate Stone command_byte);
9601d19a2f2SGreg Clayton }
9611d19a2f2SGreg Clayton }
9621d19a2f2SGreg Clayton return false;
9631d19a2f2SGreg Clayton }
9641d19a2f2SGreg Clayton };
9651d19a2f2SGreg Clayton
966b9c1b51eSKate Stone class CommandObjectProcessKDPPacket : public CommandObjectMultiword {
9671d19a2f2SGreg Clayton private:
9681d19a2f2SGreg Clayton public:
CommandObjectProcessKDPPacket(CommandInterpreter & interpreter)969b9c1b51eSKate Stone CommandObjectProcessKDPPacket(CommandInterpreter &interpreter)
970b9c1b51eSKate Stone : CommandObjectMultiword(interpreter, "process plugin packet",
9711d19a2f2SGreg Clayton "Commands that deal with KDP remote packets.",
972b9c1b51eSKate Stone NULL) {
973b9c1b51eSKate Stone LoadSubCommand(
974b9c1b51eSKate Stone "send",
975b9c1b51eSKate Stone CommandObjectSP(new CommandObjectProcessKDPPacketSend(interpreter)));
9761d19a2f2SGreg Clayton }
9771d19a2f2SGreg Clayton
978*fd146460SShafik Yaghmour ~CommandObjectProcessKDPPacket() override = default;
9791d19a2f2SGreg Clayton };
9801d19a2f2SGreg Clayton
981b9c1b51eSKate Stone class CommandObjectMultiwordProcessKDP : public CommandObjectMultiword {
9821d19a2f2SGreg Clayton public:
CommandObjectMultiwordProcessKDP(CommandInterpreter & interpreter)9837428a18cSKate Stone CommandObjectMultiwordProcessKDP(CommandInterpreter &interpreter)
984b9c1b51eSKate Stone : CommandObjectMultiword(
985b9c1b51eSKate Stone interpreter, "process plugin",
986b9c1b51eSKate Stone "Commands for operating on a ProcessKDP process.",
987b9c1b51eSKate Stone "process plugin <subcommand> [<subcommand-options>]") {
988b9c1b51eSKate Stone LoadSubCommand("packet", CommandObjectSP(new CommandObjectProcessKDPPacket(
989b9c1b51eSKate Stone interpreter)));
9901d19a2f2SGreg Clayton }
9911d19a2f2SGreg Clayton
992*fd146460SShafik Yaghmour ~CommandObjectMultiwordProcessKDP() override = default;
9931d19a2f2SGreg Clayton };
9941d19a2f2SGreg Clayton
GetPluginCommandObject()995b9c1b51eSKate Stone CommandObject *ProcessKDP::GetPluginCommandObject() {
9961d19a2f2SGreg Clayton if (!m_command_sp)
997796ac80bSJonas Devlieghere m_command_sp = std::make_shared<CommandObjectMultiwordProcessKDP>(
998796ac80bSJonas Devlieghere GetTarget().GetDebugger().GetCommandInterpreter());
9991d19a2f2SGreg Clayton return m_command_sp.get();
10001d19a2f2SGreg Clayton }
1001