159ec512cSGreg Clayton //===-- ProcessKDP.cpp ------------------------------------------*- C++ -*-===// 2f9765acdSGreg Clayton // 3f9765acdSGreg Clayton // The LLVM Compiler Infrastructure 4f9765acdSGreg Clayton // 5f9765acdSGreg Clayton // This file is distributed under the University of Illinois Open Source 6f9765acdSGreg Clayton // License. See LICENSE.TXT for details. 7f9765acdSGreg Clayton // 8f9765acdSGreg Clayton //===----------------------------------------------------------------------===// 9f9765acdSGreg Clayton 10f9765acdSGreg Clayton // C Includes 11f9765acdSGreg Clayton #include <errno.h> 12f9765acdSGreg Clayton #include <stdlib.h> 13f9765acdSGreg Clayton 14f9765acdSGreg Clayton // C++ Includes 153f69fa6fSBenjamin Kramer #include <mutex> 163f69fa6fSBenjamin Kramer 17f9765acdSGreg Clayton // Other libraries and framework includes 1807e66e3eSGreg Clayton #include "lldb/Core/Debugger.h" 19f9765acdSGreg Clayton #include "lldb/Core/PluginManager.h" 201f746071SGreg Clayton #include "lldb/Core/Module.h" 214bd4e7e3SJason Molenda #include "lldb/Core/ModuleSpec.h" 22f9765acdSGreg Clayton #include "lldb/Core/State.h" 234bd4e7e3SJason Molenda #include "lldb/Core/UUID.h" 2493a66fc1SZachary Turner #include "lldb/Host/ConnectionFileDescriptor.h" 25f9765acdSGreg Clayton #include "lldb/Host/Host.h" 264bd4e7e3SJason Molenda #include "lldb/Host/Symbols.h" 2739de3110SZachary Turner #include "lldb/Host/ThreadLauncher.h" 28e98628ceSOleksiy Vyalov #include "lldb/Host/common/TCPSocket.h" 291d19a2f2SGreg Clayton #include "lldb/Interpreter/CommandInterpreter.h" 301d19a2f2SGreg Clayton #include "lldb/Interpreter/CommandObject.h" 311d19a2f2SGreg Clayton #include "lldb/Interpreter/CommandObjectMultiword.h" 321d19a2f2SGreg Clayton #include "lldb/Interpreter/CommandReturnObject.h" 331d19a2f2SGreg Clayton #include "lldb/Interpreter/OptionGroupString.h" 341d19a2f2SGreg Clayton #include "lldb/Interpreter/OptionGroupUInt64.h" 3541204d09SIlia K #include "lldb/Interpreter/OptionValueProperties.h" 361f746071SGreg Clayton #include "lldb/Symbol/ObjectFile.h" 377925fbbaSGreg Clayton #include "lldb/Target/RegisterContext.h" 3857508026SGreg Clayton #include "lldb/Target/Target.h" 39a63d08c9SGreg Clayton #include "lldb/Target/Thread.h" 4045788152SBruce Mitchener #include "lldb/Utility/StringExtractor.h" 41f9765acdSGreg Clayton 42510938e5SCharles Davis #define USEC_PER_SEC 1000000 43510938e5SCharles Davis 44f9765acdSGreg Clayton // Project includes 45f9765acdSGreg Clayton #include "ProcessKDP.h" 46f9765acdSGreg Clayton #include "ProcessKDPLog.h" 47a63d08c9SGreg Clayton #include "ThreadKDP.h" 485e8534efSJason Molenda #include "Plugins/DynamicLoader/Darwin-Kernel/DynamicLoaderDarwinKernel.h" 49840f12cfSJason Molenda #include "Plugins/DynamicLoader/Static/DynamicLoaderStatic.h" 50f9765acdSGreg Clayton 51f9765acdSGreg Clayton using namespace lldb; 52f9765acdSGreg Clayton using namespace lldb_private; 53f9765acdSGreg Clayton 547f98240dSGreg Clayton namespace { 557f98240dSGreg Clayton 567f98240dSGreg Clayton static PropertyDefinition 577f98240dSGreg Clayton g_properties[] = 587f98240dSGreg Clayton { 597f98240dSGreg Clayton { "packet-timeout" , OptionValue::eTypeUInt64 , true , 5, NULL, NULL, "Specify the default packet timeout in seconds." }, 607f98240dSGreg Clayton { NULL , OptionValue::eTypeInvalid, false, 0, NULL, NULL, NULL } 617f98240dSGreg Clayton }; 627f98240dSGreg Clayton 637f98240dSGreg Clayton enum 647f98240dSGreg Clayton { 657f98240dSGreg Clayton ePropertyPacketTimeout 667f98240dSGreg Clayton }; 677f98240dSGreg Clayton 687f98240dSGreg Clayton class PluginProperties : public Properties 697f98240dSGreg Clayton { 707f98240dSGreg Clayton public: 717f98240dSGreg Clayton 727f98240dSGreg Clayton static ConstString 737f98240dSGreg Clayton GetSettingName () 747f98240dSGreg Clayton { 757f98240dSGreg Clayton return ProcessKDP::GetPluginNameStatic(); 767f98240dSGreg Clayton } 777f98240dSGreg Clayton 787f98240dSGreg Clayton PluginProperties() : 797f98240dSGreg Clayton Properties () 807f98240dSGreg Clayton { 817f98240dSGreg Clayton m_collection_sp.reset (new OptionValueProperties(GetSettingName())); 827f98240dSGreg Clayton m_collection_sp->Initialize(g_properties); 837f98240dSGreg Clayton } 847f98240dSGreg Clayton 857f98240dSGreg Clayton virtual 867f98240dSGreg Clayton ~PluginProperties() 877f98240dSGreg Clayton { 887f98240dSGreg Clayton } 897f98240dSGreg Clayton 907f98240dSGreg Clayton uint64_t 917f98240dSGreg Clayton GetPacketTimeout() 927f98240dSGreg Clayton { 937f98240dSGreg Clayton const uint32_t idx = ePropertyPacketTimeout; 947f98240dSGreg Clayton return m_collection_sp->GetPropertyAtIndexAsUInt64(NULL, idx, g_properties[idx].default_uint_value); 957f98240dSGreg Clayton } 967f98240dSGreg Clayton }; 977f98240dSGreg Clayton 987f98240dSGreg Clayton typedef std::shared_ptr<PluginProperties> ProcessKDPPropertiesSP; 997f98240dSGreg Clayton 1007f98240dSGreg Clayton static const ProcessKDPPropertiesSP & 1017f98240dSGreg Clayton GetGlobalPluginProperties() 1027f98240dSGreg Clayton { 1037f98240dSGreg Clayton static ProcessKDPPropertiesSP g_settings_sp; 1047f98240dSGreg Clayton if (!g_settings_sp) 1057f98240dSGreg Clayton g_settings_sp.reset (new PluginProperties ()); 1067f98240dSGreg Clayton return g_settings_sp; 1077f98240dSGreg Clayton } 1087f98240dSGreg Clayton 1097f98240dSGreg Clayton } // anonymous namespace end 1107f98240dSGreg Clayton 111ba4e61d3SAndrew Kaylor static const lldb::tid_t g_kernel_tid = 1; 112ba4e61d3SAndrew Kaylor 11357abc5d6SGreg Clayton ConstString 114f9765acdSGreg Clayton ProcessKDP::GetPluginNameStatic() 115f9765acdSGreg Clayton { 11657abc5d6SGreg Clayton static ConstString g_name("kdp-remote"); 11757abc5d6SGreg Clayton return g_name; 118f9765acdSGreg Clayton } 119f9765acdSGreg Clayton 120f9765acdSGreg Clayton const char * 121f9765acdSGreg Clayton ProcessKDP::GetPluginDescriptionStatic() 122f9765acdSGreg Clayton { 123f9765acdSGreg Clayton return "KDP Remote protocol based debugging plug-in for darwin kernel debugging."; 124f9765acdSGreg Clayton } 125f9765acdSGreg Clayton 126f9765acdSGreg Clayton void 127f9765acdSGreg Clayton ProcessKDP::Terminate() 128f9765acdSGreg Clayton { 129f9765acdSGreg Clayton PluginManager::UnregisterPlugin (ProcessKDP::CreateInstance); 130f9765acdSGreg Clayton } 131f9765acdSGreg Clayton 132f9765acdSGreg Clayton 133c3776bf2SGreg Clayton lldb::ProcessSP 1345a3bb64fSJim Ingham ProcessKDP::CreateInstance (TargetSP target_sp, 135583bbb1dSJim Ingham ListenerSP listener_sp, 136c3776bf2SGreg Clayton const FileSpec *crash_file_path) 137f9765acdSGreg Clayton { 138c3776bf2SGreg Clayton lldb::ProcessSP process_sp; 139c3776bf2SGreg Clayton if (crash_file_path == NULL) 140583bbb1dSJim Ingham process_sp.reset(new ProcessKDP (target_sp, listener_sp)); 141c3776bf2SGreg Clayton return process_sp; 142f9765acdSGreg Clayton } 143f9765acdSGreg Clayton 144f9765acdSGreg Clayton bool 1455a3bb64fSJim Ingham ProcessKDP::CanDebug(TargetSP target_sp, bool plugin_specified_by_name) 146f9765acdSGreg Clayton { 147596ed24eSGreg Clayton if (plugin_specified_by_name) 148596ed24eSGreg Clayton return true; 149596ed24eSGreg Clayton 150f9765acdSGreg Clayton // For now we are just making sure the file exists for a given module 1515a3bb64fSJim Ingham Module *exe_module = target_sp->GetExecutableModulePointer(); 152aa149cbdSGreg Clayton if (exe_module) 153f9765acdSGreg Clayton { 1545a3bb64fSJim Ingham const llvm::Triple &triple_ref = target_sp->GetArchitecture().GetTriple(); 15570512317SGreg Clayton switch (triple_ref.getOS()) 15670512317SGreg Clayton { 15770512317SGreg Clayton case llvm::Triple::Darwin: // Should use "macosx" for desktop and "ios" for iOS, but accept darwin just in case 15870512317SGreg Clayton case llvm::Triple::MacOSX: // For desktop targets 15970512317SGreg Clayton case llvm::Triple::IOS: // For arm targets 160a814f704SJason Molenda case llvm::Triple::TvOS: 161a814f704SJason Molenda case llvm::Triple::WatchOS: 16270512317SGreg Clayton if (triple_ref.getVendor() == llvm::Triple::Apple) 163f9765acdSGreg Clayton { 164aa149cbdSGreg Clayton ObjectFile *exe_objfile = exe_module->GetObjectFile(); 165f9765acdSGreg Clayton if (exe_objfile->GetType() == ObjectFile::eTypeExecutable && 166f9765acdSGreg Clayton exe_objfile->GetStrata() == ObjectFile::eStrataKernel) 167f9765acdSGreg Clayton return true; 168f9765acdSGreg Clayton } 16970512317SGreg Clayton break; 17070512317SGreg Clayton 17170512317SGreg Clayton default: 17270512317SGreg Clayton break; 17370512317SGreg Clayton } 174f9765acdSGreg Clayton } 175596ed24eSGreg Clayton return false; 1763a29bdbeSGreg Clayton } 177f9765acdSGreg Clayton 178f9765acdSGreg Clayton //---------------------------------------------------------------------- 179f9765acdSGreg Clayton // ProcessKDP constructor 180f9765acdSGreg Clayton //---------------------------------------------------------------------- 181583bbb1dSJim Ingham ProcessKDP::ProcessKDP(TargetSP target_sp, ListenerSP listener_sp) : 182583bbb1dSJim Ingham Process (target_sp, listener_sp), 183f9765acdSGreg Clayton m_comm("lldb.process.kdp-remote.communication"), 1844bddaeb5SJim Ingham m_async_broadcaster (NULL, "lldb.process.kdp-remote.async-broadcaster"), 1855e8534efSJason Molenda m_dyld_plugin_name (), 1861d19a2f2SGreg Clayton m_kernel_load_addr (LLDB_INVALID_ADDRESS), 187ba4e61d3SAndrew Kaylor m_command_sp(), 188ba4e61d3SAndrew Kaylor m_kernel_thread_wp() 189f9765acdSGreg Clayton { 1907925fbbaSGreg Clayton m_async_broadcaster.SetEventName (eBroadcastBitAsyncThreadShouldExit, "async thread should exit"); 1917925fbbaSGreg Clayton m_async_broadcaster.SetEventName (eBroadcastBitAsyncContinue, "async thread continue"); 1927f98240dSGreg Clayton const uint64_t timeout_seconds = GetGlobalPluginProperties()->GetPacketTimeout(); 1937f98240dSGreg Clayton if (timeout_seconds > 0) 1947f98240dSGreg Clayton m_comm.SetPacketTimeout(timeout_seconds); 195f9765acdSGreg Clayton } 196f9765acdSGreg Clayton 197f9765acdSGreg Clayton //---------------------------------------------------------------------- 198f9765acdSGreg Clayton // Destructor 199f9765acdSGreg Clayton //---------------------------------------------------------------------- 200f9765acdSGreg Clayton ProcessKDP::~ProcessKDP() 201f9765acdSGreg Clayton { 202f9765acdSGreg Clayton Clear(); 203e24c4acfSGreg Clayton // We need to call finalize on the process before destroying ourselves 204e24c4acfSGreg Clayton // to make sure all of the broadcaster cleanup goes as planned. If we 205e24c4acfSGreg Clayton // destruct this class, then Process::~Process() might have problems 206e24c4acfSGreg Clayton // trying to fully destroy the broadcaster. 207e24c4acfSGreg Clayton Finalize(); 208f9765acdSGreg Clayton } 209f9765acdSGreg Clayton 210f9765acdSGreg Clayton //---------------------------------------------------------------------- 211f9765acdSGreg Clayton // PluginInterface 212f9765acdSGreg Clayton //---------------------------------------------------------------------- 21357abc5d6SGreg Clayton lldb_private::ConstString 214f9765acdSGreg Clayton ProcessKDP::GetPluginName() 215f9765acdSGreg Clayton { 216f9765acdSGreg Clayton return GetPluginNameStatic(); 217f9765acdSGreg Clayton } 218f9765acdSGreg Clayton 219f9765acdSGreg Clayton uint32_t 220f9765acdSGreg Clayton ProcessKDP::GetPluginVersion() 221f9765acdSGreg Clayton { 222f9765acdSGreg Clayton return 1; 223f9765acdSGreg Clayton } 224f9765acdSGreg Clayton 225f9765acdSGreg Clayton Error 226f9765acdSGreg Clayton ProcessKDP::WillLaunch (Module* module) 227f9765acdSGreg Clayton { 228f9765acdSGreg Clayton Error error; 229f9765acdSGreg Clayton error.SetErrorString ("launching not supported in kdp-remote plug-in"); 230f9765acdSGreg Clayton return error; 231f9765acdSGreg Clayton } 232f9765acdSGreg Clayton 233f9765acdSGreg Clayton Error 234f9765acdSGreg Clayton ProcessKDP::WillAttachToProcessWithID (lldb::pid_t pid) 235f9765acdSGreg Clayton { 236f9765acdSGreg Clayton Error error; 237f9765acdSGreg Clayton error.SetErrorString ("attaching to a by process ID not supported in kdp-remote plug-in"); 238f9765acdSGreg Clayton return error; 239f9765acdSGreg Clayton } 240f9765acdSGreg Clayton 241f9765acdSGreg Clayton Error 242f9765acdSGreg Clayton ProcessKDP::WillAttachToProcessWithName (const char *process_name, bool wait_for_launch) 243f9765acdSGreg Clayton { 244f9765acdSGreg Clayton Error error; 245f9765acdSGreg Clayton error.SetErrorString ("attaching to a by process name not supported in kdp-remote plug-in"); 246f9765acdSGreg Clayton return error; 247f9765acdSGreg Clayton } 248f9765acdSGreg Clayton 249a3706888SGreg Clayton bool 250a3706888SGreg Clayton ProcessKDP::GetHostArchitecture(ArchSpec &arch) 251a3706888SGreg Clayton { 252a3706888SGreg Clayton uint32_t cpu = m_comm.GetCPUType(); 253a3706888SGreg Clayton if (cpu) 254a3706888SGreg Clayton { 255a3706888SGreg Clayton uint32_t sub = m_comm.GetCPUSubtype(); 256a3706888SGreg Clayton arch.SetArchitecture(eArchTypeMachO, cpu, sub); 257a3706888SGreg Clayton // Leave architecture vendor as unspecified unknown 258a3706888SGreg Clayton arch.GetTriple().setVendor(llvm::Triple::UnknownVendor); 259a3706888SGreg Clayton arch.GetTriple().setVendorName(llvm::StringRef()); 260a3706888SGreg Clayton return true; 261a3706888SGreg Clayton } 262a3706888SGreg Clayton arch.Clear(); 263a3706888SGreg Clayton return false; 264a3706888SGreg Clayton } 265a3706888SGreg Clayton 266f9765acdSGreg Clayton Error 2674bd4e7e3SJason Molenda ProcessKDP::DoConnectRemote (Stream *strm, const char *remote_url) 268f9765acdSGreg Clayton { 269f9765acdSGreg Clayton Error error; 2703a29bdbeSGreg Clayton 2717925fbbaSGreg Clayton // Don't let any JIT happen when doing KDP as we can't allocate 2727925fbbaSGreg Clayton // memory and we don't want to be mucking with threads that might 2737925fbbaSGreg Clayton // already be handling exceptions 2747925fbbaSGreg Clayton SetCanJIT(false); 2757925fbbaSGreg Clayton 2763a29bdbeSGreg Clayton if (remote_url == NULL || remote_url[0] == '\0') 2777925fbbaSGreg Clayton { 2787925fbbaSGreg Clayton error.SetErrorStringWithFormat ("invalid connection URL '%s'", remote_url); 2797925fbbaSGreg Clayton return error; 2807925fbbaSGreg Clayton } 2813a29bdbeSGreg Clayton 2827b0992d9SGreg Clayton std::unique_ptr<ConnectionFileDescriptor> conn_ap(new ConnectionFileDescriptor()); 2833a29bdbeSGreg Clayton if (conn_ap.get()) 2843a29bdbeSGreg Clayton { 2853a29bdbeSGreg Clayton // Only try once for now. 2863a29bdbeSGreg Clayton // TODO: check if we should be retrying? 2873a29bdbeSGreg Clayton const uint32_t max_retry_count = 1; 2883a29bdbeSGreg Clayton for (uint32_t retry_count = 0; retry_count < max_retry_count; ++retry_count) 2893a29bdbeSGreg Clayton { 2903a29bdbeSGreg Clayton if (conn_ap->Connect(remote_url, &error) == eConnectionStatusSuccess) 2913a29bdbeSGreg Clayton break; 2923a29bdbeSGreg Clayton usleep (100000); 2933a29bdbeSGreg Clayton } 2943a29bdbeSGreg Clayton } 2953a29bdbeSGreg Clayton 2963a29bdbeSGreg Clayton if (conn_ap->IsConnected()) 2973a29bdbeSGreg Clayton { 298e98628ceSOleksiy Vyalov const TCPSocket& socket = static_cast<const TCPSocket&>(*conn_ap->GetReadObject()); 299014bb7daSVince Harron const uint16_t reply_port = socket.GetLocalPortNumber(); 3003a29bdbeSGreg Clayton 3013a29bdbeSGreg Clayton if (reply_port != 0) 3023a29bdbeSGreg Clayton { 3033a29bdbeSGreg Clayton m_comm.SetConnection(conn_ap.release()); 3043a29bdbeSGreg Clayton 3053a29bdbeSGreg Clayton if (m_comm.SendRequestReattach(reply_port)) 3063a29bdbeSGreg Clayton { 3073a29bdbeSGreg Clayton if (m_comm.SendRequestConnect(reply_port, reply_port, "Greetings from LLDB...")) 3083a29bdbeSGreg Clayton { 3093a29bdbeSGreg Clayton m_comm.GetVersion(); 3105a3bb64fSJim Ingham 311a3706888SGreg Clayton Target &target = GetTarget(); 312a3706888SGreg Clayton ArchSpec kernel_arch; 313a3706888SGreg Clayton // The host architecture 314a3706888SGreg Clayton GetHostArchitecture(kernel_arch); 315a3706888SGreg Clayton ArchSpec target_arch = target.GetArchitecture(); 316a3706888SGreg Clayton // Merge in any unspecified stuff into the target architecture in 317a3706888SGreg Clayton // case the target arch isn't set at all or incompletely. 318a3706888SGreg Clayton target_arch.MergeFrom(kernel_arch); 319a3706888SGreg Clayton target.SetArchitecture(target_arch); 3204bd4e7e3SJason Molenda 321840f12cfSJason Molenda /* Get the kernel's UUID and load address via KDP_KERNELVERSION packet. */ 322840f12cfSJason Molenda /* An EFI kdp session has neither UUID nor load address. */ 3234bd4e7e3SJason Molenda 3244bd4e7e3SJason Molenda UUID kernel_uuid = m_comm.GetUUID (); 3254bd4e7e3SJason Molenda addr_t kernel_load_addr = m_comm.GetLoadAddress (); 3264bd4e7e3SJason Molenda 327840f12cfSJason Molenda if (m_comm.RemoteIsEFI ()) 328840f12cfSJason Molenda { 329a1bce2efSGreg Clayton // Select an invalid plugin name for the dynamic loader so one doesn't get used 330a1bce2efSGreg Clayton // since EFI does its own manual loading via python scripting 331a1bce2efSGreg Clayton static ConstString g_none_dynamic_loader("none"); 332a1bce2efSGreg Clayton m_dyld_plugin_name = g_none_dynamic_loader; 333a1bce2efSGreg Clayton 334a1bce2efSGreg Clayton if (kernel_uuid.IsValid()) { 335a1bce2efSGreg Clayton // If EFI passed in a UUID= try to lookup UUID 336a1bce2efSGreg Clayton // The slide will not be provided. But the UUID 337a1bce2efSGreg Clayton // lookup will be used to launch EFI debug scripts 338a1bce2efSGreg Clayton // from the dSYM, that can load all of the symbols. 339a1bce2efSGreg Clayton ModuleSpec module_spec; 340a1bce2efSGreg Clayton module_spec.GetUUID() = kernel_uuid; 3415a3bb64fSJim Ingham module_spec.GetArchitecture() = target.GetArchitecture(); 342a1bce2efSGreg Clayton 343a1bce2efSGreg Clayton // Lookup UUID locally, before attempting dsymForUUID like action 344a1bce2efSGreg Clayton module_spec.GetSymbolFileSpec() = Symbols::LocateExecutableSymbolFile(module_spec); 345a1bce2efSGreg Clayton if (module_spec.GetSymbolFileSpec()) 3468825c5c9SJason Molenda { 3478825c5c9SJason Molenda ModuleSpec executable_module_spec = Symbols::LocateExecutableObjectFile (module_spec); 3488825c5c9SJason Molenda if (executable_module_spec.GetFileSpec().Exists()) 3498825c5c9SJason Molenda { 3508825c5c9SJason Molenda module_spec.GetFileSpec() = executable_module_spec.GetFileSpec(); 3518825c5c9SJason Molenda } 3528825c5c9SJason Molenda } 353a1bce2efSGreg Clayton if (!module_spec.GetSymbolFileSpec() || !module_spec.GetSymbolFileSpec()) 354a1bce2efSGreg Clayton Symbols::DownloadObjectAndSymbolFile (module_spec, true); 355a1bce2efSGreg Clayton 356a1bce2efSGreg Clayton if (module_spec.GetFileSpec().Exists()) 357a1bce2efSGreg Clayton { 358a3706888SGreg Clayton ModuleSP module_sp(new Module (module_spec)); 359a3706888SGreg Clayton if (module_sp.get() && module_sp->GetObjectFile()) 360a1bce2efSGreg Clayton { 361a1bce2efSGreg Clayton // Get the current target executable 3625a3bb64fSJim Ingham ModuleSP exe_module_sp (target.GetExecutableModule ()); 363a1bce2efSGreg Clayton 364a1bce2efSGreg Clayton // Make sure you don't already have the right module loaded and they will be uniqued 365a1bce2efSGreg Clayton if (exe_module_sp.get() != module_sp.get()) 3665a3bb64fSJim Ingham target.SetExecutableModule (module_sp, false); 367a1bce2efSGreg Clayton } 368a1bce2efSGreg Clayton } 369a1bce2efSGreg Clayton } 370840f12cfSJason Molenda } 371ca2ffa7eSJason Molenda else if (m_comm.RemoteIsDarwinKernel ()) 372a8ea4baeSJason Molenda { 373ca2ffa7eSJason Molenda m_dyld_plugin_name = DynamicLoaderDarwinKernel::GetPluginNameStatic(); 374a8ea4baeSJason Molenda if (kernel_load_addr != LLDB_INVALID_ADDRESS) 3754bd4e7e3SJason Molenda { 3765e8534efSJason Molenda m_kernel_load_addr = kernel_load_addr; 377a8ea4baeSJason Molenda } 3784bd4e7e3SJason Molenda } 3794bd4e7e3SJason Molenda 38097d5cf05SGreg Clayton // Set the thread ID 38197d5cf05SGreg Clayton UpdateThreadListIfNeeded (); 382a63d08c9SGreg Clayton SetID (1); 38356d9a1b3SGreg Clayton GetThreadList (); 384a63d08c9SGreg Clayton SetPrivateState (eStateStopped); 3855a3bb64fSJim Ingham StreamSP async_strm_sp(target.GetDebugger().GetAsyncOutputStream()); 38607e66e3eSGreg Clayton if (async_strm_sp) 38707e66e3eSGreg Clayton { 3885b88216dSGreg Clayton const char *cstr; 3895b88216dSGreg Clayton if ((cstr = m_comm.GetKernelVersion ()) != NULL) 39007e66e3eSGreg Clayton { 3915b88216dSGreg Clayton async_strm_sp->Printf ("Version: %s\n", cstr); 39207e66e3eSGreg Clayton async_strm_sp->Flush(); 39307e66e3eSGreg Clayton } 3945b88216dSGreg Clayton // if ((cstr = m_comm.GetImagePath ()) != NULL) 3955b88216dSGreg Clayton // { 3965b88216dSGreg Clayton // async_strm_sp->Printf ("Image Path: %s\n", cstr); 3975b88216dSGreg Clayton // async_strm_sp->Flush(); 3985b88216dSGreg Clayton // } 39907e66e3eSGreg Clayton } 4003a29bdbeSGreg Clayton } 40197d5cf05SGreg Clayton else 40297d5cf05SGreg Clayton { 40397d5cf05SGreg Clayton error.SetErrorString("KDP_REATTACH failed"); 40497d5cf05SGreg Clayton } 4053a29bdbeSGreg Clayton } 4063a29bdbeSGreg Clayton else 4073a29bdbeSGreg Clayton { 40897d5cf05SGreg Clayton error.SetErrorString("KDP_REATTACH failed"); 4093a29bdbeSGreg Clayton } 4103a29bdbeSGreg Clayton } 4113a29bdbeSGreg Clayton else 4123a29bdbeSGreg Clayton { 4133a29bdbeSGreg Clayton error.SetErrorString("invalid reply port from UDP connection"); 4143a29bdbeSGreg Clayton } 4153a29bdbeSGreg Clayton } 4163a29bdbeSGreg Clayton else 4173a29bdbeSGreg Clayton { 4183a29bdbeSGreg Clayton if (error.Success()) 4193a29bdbeSGreg Clayton error.SetErrorStringWithFormat ("failed to connect to '%s'", remote_url); 4203a29bdbeSGreg Clayton } 4213a29bdbeSGreg Clayton if (error.Fail()) 4223a29bdbeSGreg Clayton m_comm.Disconnect(); 4233a29bdbeSGreg Clayton 424f9765acdSGreg Clayton return error; 425f9765acdSGreg Clayton } 426f9765acdSGreg Clayton 427f9765acdSGreg Clayton //---------------------------------------------------------------------- 428f9765acdSGreg Clayton // Process Control 429f9765acdSGreg Clayton //---------------------------------------------------------------------- 430f9765acdSGreg Clayton Error 431982c9762SGreg Clayton ProcessKDP::DoLaunch (Module *exe_module, 4327782de92SJean-Daniel Dupas ProcessLaunchInfo &launch_info) 433f9765acdSGreg Clayton { 434f9765acdSGreg Clayton Error error; 435f9765acdSGreg Clayton error.SetErrorString ("launching not supported in kdp-remote plug-in"); 436f9765acdSGreg Clayton return error; 437f9765acdSGreg Clayton } 438f9765acdSGreg Clayton 439f9765acdSGreg Clayton Error 44084647048SHan Ming Ong ProcessKDP::DoAttachToProcessWithID (lldb::pid_t attach_pid, const ProcessAttachInfo &attach_info) 44184647048SHan Ming Ong { 44284647048SHan Ming Ong Error error; 44384647048SHan Ming Ong error.SetErrorString ("attach to process by ID is not suppported in kdp remote debugging"); 44484647048SHan Ming Ong return error; 44584647048SHan Ming Ong } 44684647048SHan Ming Ong 44784647048SHan Ming Ong Error 4489c517c0dSJean-Daniel Dupas ProcessKDP::DoAttachToProcessWithName (const char *process_name, const ProcessAttachInfo &attach_info) 449f9765acdSGreg Clayton { 450f9765acdSGreg Clayton Error error; 451f9765acdSGreg Clayton error.SetErrorString ("attach to process by name is not suppported in kdp remote debugging"); 452f9765acdSGreg Clayton return error; 453f9765acdSGreg Clayton } 454f9765acdSGreg Clayton 455f9765acdSGreg Clayton 456f9765acdSGreg Clayton void 457bb006ce2SJim Ingham ProcessKDP::DidAttach (ArchSpec &process_arch) 458f9765acdSGreg Clayton { 459bb006ce2SJim Ingham Process::DidAttach(process_arch); 460bb006ce2SJim Ingham 4615160ce5cSGreg Clayton Log *log (ProcessKDPLog::GetLogIfAllCategoriesSet (KDP_LOG_PROCESS)); 462f9765acdSGreg Clayton if (log) 46354cb8f83SJohnny Chen log->Printf ("ProcessKDP::DidAttach()"); 464f9765acdSGreg Clayton if (GetID() != LLDB_INVALID_PROCESS_ID) 465f9765acdSGreg Clayton { 466a3706888SGreg Clayton GetHostArchitecture(process_arch); 467f9765acdSGreg Clayton } 468f9765acdSGreg Clayton } 469f9765acdSGreg Clayton 4705e8534efSJason Molenda addr_t 4715e8534efSJason Molenda ProcessKDP::GetImageInfoAddress() 4725e8534efSJason Molenda { 4735e8534efSJason Molenda return m_kernel_load_addr; 4745e8534efSJason Molenda } 4755e8534efSJason Molenda 4765e8534efSJason Molenda lldb_private::DynamicLoader * 4775e8534efSJason Molenda ProcessKDP::GetDynamicLoader () 4785e8534efSJason Molenda { 4795e8534efSJason Molenda if (m_dyld_ap.get() == NULL) 4802e56a254SJason Molenda m_dyld_ap.reset (DynamicLoader::FindPlugin(this, m_dyld_plugin_name.IsEmpty() ? NULL : m_dyld_plugin_name.GetCString())); 4815e8534efSJason Molenda return m_dyld_ap.get(); 4825e8534efSJason Molenda } 4835e8534efSJason Molenda 484f9765acdSGreg Clayton Error 485f9765acdSGreg Clayton ProcessKDP::WillResume () 486f9765acdSGreg Clayton { 487f9765acdSGreg Clayton return Error(); 488f9765acdSGreg Clayton } 489f9765acdSGreg Clayton 490f9765acdSGreg Clayton Error 491f9765acdSGreg Clayton ProcessKDP::DoResume () 492f9765acdSGreg Clayton { 493f9765acdSGreg Clayton Error error; 4945160ce5cSGreg Clayton Log *log (ProcessKDPLog::GetLogIfAllCategoriesSet (KDP_LOG_PROCESS)); 4957925fbbaSGreg Clayton // Only start the async thread if we try to do any process control 496acee96aeSZachary Turner if (!m_async_thread.IsJoinable()) 4977925fbbaSGreg Clayton StartAsyncThread(); 4987925fbbaSGreg Clayton 49997d5cf05SGreg Clayton bool resume = false; 5007925fbbaSGreg Clayton 50197d5cf05SGreg Clayton // With KDP there is only one thread we can tell what to do 502ba4e61d3SAndrew Kaylor ThreadSP kernel_thread_sp (m_thread_list.FindThreadByProtocolID(g_kernel_tid)); 503ba4e61d3SAndrew Kaylor 50497d5cf05SGreg Clayton if (kernel_thread_sp) 5054b1b8b3eSGreg Clayton { 50697d5cf05SGreg Clayton const StateType thread_resume_state = kernel_thread_sp->GetTemporaryResumeState(); 5076e0ff1a3SGreg Clayton 5086e0ff1a3SGreg Clayton if (log) 5096e0ff1a3SGreg Clayton log->Printf ("ProcessKDP::DoResume() thread_resume_state = %s", StateAsCString(thread_resume_state)); 5107925fbbaSGreg Clayton switch (thread_resume_state) 5114b1b8b3eSGreg Clayton { 5127925fbbaSGreg Clayton case eStateSuspended: 5137925fbbaSGreg Clayton // Nothing to do here when a thread will stay suspended 5147925fbbaSGreg Clayton // we just leave the CPU mask bit set to zero for the thread 5156e0ff1a3SGreg Clayton if (log) 5166e0ff1a3SGreg Clayton log->Printf ("ProcessKDP::DoResume() = suspended???"); 5177925fbbaSGreg Clayton break; 5187925fbbaSGreg Clayton 5197925fbbaSGreg Clayton case eStateStepping: 5201afa68edSGreg Clayton { 5211afa68edSGreg Clayton lldb::RegisterContextSP reg_ctx_sp (kernel_thread_sp->GetRegisterContext()); 5221afa68edSGreg Clayton 5231afa68edSGreg Clayton if (reg_ctx_sp) 5241afa68edSGreg Clayton { 5256e0ff1a3SGreg Clayton if (log) 5266e0ff1a3SGreg Clayton log->Printf ("ProcessKDP::DoResume () reg_ctx_sp->HardwareSingleStep (true);"); 5271afa68edSGreg Clayton reg_ctx_sp->HardwareSingleStep (true); 52897d5cf05SGreg Clayton resume = true; 5291afa68edSGreg Clayton } 5301afa68edSGreg Clayton else 5311afa68edSGreg Clayton { 5321afa68edSGreg Clayton error.SetErrorStringWithFormat("KDP thread 0x%llx has no register context", kernel_thread_sp->GetID()); 5331afa68edSGreg Clayton } 5341afa68edSGreg Clayton } 5357925fbbaSGreg Clayton break; 5367925fbbaSGreg Clayton 53797d5cf05SGreg Clayton case eStateRunning: 5381afa68edSGreg Clayton { 5391afa68edSGreg Clayton lldb::RegisterContextSP reg_ctx_sp (kernel_thread_sp->GetRegisterContext()); 5401afa68edSGreg Clayton 5411afa68edSGreg Clayton if (reg_ctx_sp) 5421afa68edSGreg Clayton { 5436e0ff1a3SGreg Clayton if (log) 5446e0ff1a3SGreg Clayton log->Printf ("ProcessKDP::DoResume () reg_ctx_sp->HardwareSingleStep (false);"); 5451afa68edSGreg Clayton reg_ctx_sp->HardwareSingleStep (false); 54697d5cf05SGreg Clayton resume = true; 5471afa68edSGreg Clayton } 5481afa68edSGreg Clayton else 5491afa68edSGreg Clayton { 5501afa68edSGreg Clayton error.SetErrorStringWithFormat("KDP thread 0x%llx has no register context", kernel_thread_sp->GetID()); 5511afa68edSGreg Clayton } 5521afa68edSGreg Clayton } 5537925fbbaSGreg Clayton break; 5547925fbbaSGreg Clayton 5557925fbbaSGreg Clayton default: 55697d5cf05SGreg Clayton // The only valid thread resume states are listed above 5577925fbbaSGreg Clayton assert (!"invalid thread resume state"); 5587925fbbaSGreg Clayton break; 5594b1b8b3eSGreg Clayton } 5604b1b8b3eSGreg Clayton } 5617925fbbaSGreg Clayton 56297d5cf05SGreg Clayton if (resume) 56397d5cf05SGreg Clayton { 56497d5cf05SGreg Clayton if (log) 56597d5cf05SGreg Clayton log->Printf ("ProcessKDP::DoResume () sending resume"); 56697d5cf05SGreg Clayton 56797d5cf05SGreg Clayton if (m_comm.SendRequestResume ()) 5687925fbbaSGreg Clayton { 5697925fbbaSGreg Clayton m_async_broadcaster.BroadcastEvent (eBroadcastBitAsyncContinue); 5707925fbbaSGreg Clayton SetPrivateState(eStateRunning); 5717925fbbaSGreg Clayton } 5724b1b8b3eSGreg Clayton else 57307e66e3eSGreg Clayton error.SetErrorString ("KDP resume failed"); 5747925fbbaSGreg Clayton } 5757925fbbaSGreg Clayton else 5767925fbbaSGreg Clayton { 57797d5cf05SGreg Clayton error.SetErrorString ("kernel thread is suspended"); 5787925fbbaSGreg Clayton } 5797925fbbaSGreg Clayton 580f9765acdSGreg Clayton return error; 581f9765acdSGreg Clayton } 582f9765acdSGreg Clayton 58397d5cf05SGreg Clayton lldb::ThreadSP 584ba4e61d3SAndrew Kaylor ProcessKDP::GetKernelThread() 58597d5cf05SGreg Clayton { 58697d5cf05SGreg Clayton // KDP only tells us about one thread/core. Any other threads will usually 58797d5cf05SGreg Clayton // be the ones that are read from memory by the OS plug-ins. 588ba4e61d3SAndrew Kaylor 589ba4e61d3SAndrew Kaylor ThreadSP thread_sp (m_kernel_thread_wp.lock()); 59097d5cf05SGreg Clayton if (!thread_sp) 591ba4e61d3SAndrew Kaylor { 592ba4e61d3SAndrew Kaylor thread_sp.reset(new ThreadKDP (*this, g_kernel_tid)); 593ba4e61d3SAndrew Kaylor m_kernel_thread_wp = thread_sp; 594ba4e61d3SAndrew Kaylor } 59597d5cf05SGreg Clayton return thread_sp; 59697d5cf05SGreg Clayton } 59797d5cf05SGreg Clayton 59897d5cf05SGreg Clayton 59997d5cf05SGreg Clayton 60097d5cf05SGreg Clayton 6019fc13556SGreg Clayton bool 60256d9a1b3SGreg Clayton ProcessKDP::UpdateThreadList (ThreadList &old_thread_list, ThreadList &new_thread_list) 603f9765acdSGreg Clayton { 604f9765acdSGreg Clayton // locker will keep a mutex locked until it goes out of scope 6055160ce5cSGreg Clayton Log *log (ProcessKDPLog::GetLogIfAllCategoriesSet (KDP_LOG_THREAD)); 606f9765acdSGreg Clayton if (log && log->GetMask().Test(KDP_LOG_VERBOSE)) 607d01b2953SDaniel Malea log->Printf ("ProcessKDP::%s (pid = %" PRIu64 ")", __FUNCTION__, GetID()); 608f9765acdSGreg Clayton 60939da3efdSGreg Clayton // Even though there is a CPU mask, it doesn't mean we can see each CPU 610a868c13cSBruce Mitchener // individually, there is really only one. Lets call this thread 1. 611ba4e61d3SAndrew Kaylor ThreadSP thread_sp (old_thread_list.FindThreadByProtocolID(g_kernel_tid, false)); 612ba4e61d3SAndrew Kaylor if (!thread_sp) 613ba4e61d3SAndrew Kaylor thread_sp = GetKernelThread (); 614ba4e61d3SAndrew Kaylor new_thread_list.AddThread(thread_sp); 61597d5cf05SGreg Clayton 6169fc13556SGreg Clayton return new_thread_list.GetSize(false) > 0; 617f9765acdSGreg Clayton } 618f9765acdSGreg Clayton 619f9765acdSGreg Clayton void 620f9765acdSGreg Clayton ProcessKDP::RefreshStateAfterStop () 621f9765acdSGreg Clayton { 622f9765acdSGreg Clayton // Let all threads recover from stopping and do any clean up based 623f9765acdSGreg Clayton // on the previous thread state (if any). 624f9765acdSGreg Clayton m_thread_list.RefreshStateAfterStop(); 625f9765acdSGreg Clayton } 626f9765acdSGreg Clayton 627f9765acdSGreg Clayton Error 628f9765acdSGreg Clayton ProcessKDP::DoHalt (bool &caused_stop) 629f9765acdSGreg Clayton { 630f9765acdSGreg Clayton Error error; 631f9765acdSGreg Clayton 63297d5cf05SGreg Clayton if (m_comm.IsRunning()) 633f9765acdSGreg Clayton { 63497d5cf05SGreg Clayton if (m_destroy_in_process) 63597d5cf05SGreg Clayton { 63697d5cf05SGreg Clayton // If we are attemping to destroy, we need to not return an error to 63797d5cf05SGreg Clayton // Halt or DoDestroy won't get called. 63897d5cf05SGreg Clayton // We are also currently running, so send a process stopped event 63997d5cf05SGreg Clayton SetPrivateState (eStateStopped); 640f9765acdSGreg Clayton } 641f9765acdSGreg Clayton else 642f9765acdSGreg Clayton { 64397d5cf05SGreg Clayton error.SetErrorString ("KDP cannot interrupt a running kernel"); 644f9765acdSGreg Clayton } 645f9765acdSGreg Clayton } 646f9765acdSGreg Clayton return error; 647f9765acdSGreg Clayton } 648f9765acdSGreg Clayton 649f9765acdSGreg Clayton Error 650acff8950SJim Ingham ProcessKDP::DoDetach(bool keep_stopped) 651f9765acdSGreg Clayton { 652f9765acdSGreg Clayton Error error; 6535160ce5cSGreg Clayton Log *log (ProcessKDPLog::GetLogIfAllCategoriesSet(KDP_LOG_PROCESS)); 654f9765acdSGreg Clayton if (log) 655acff8950SJim Ingham log->Printf ("ProcessKDP::DoDetach(keep_stopped = %i)", keep_stopped); 656f9765acdSGreg Clayton 65797d5cf05SGreg Clayton if (m_comm.IsRunning()) 65897d5cf05SGreg Clayton { 65997d5cf05SGreg Clayton // We are running and we can't interrupt a running kernel, so we need 66097d5cf05SGreg Clayton // to just close the connection to the kernel and hope for the best 66197d5cf05SGreg Clayton } 66297d5cf05SGreg Clayton else 66397d5cf05SGreg Clayton { 664acff8950SJim Ingham // If we are going to keep the target stopped, then don't send the disconnect message. 665acff8950SJim Ingham if (!keep_stopped && m_comm.IsConnected()) 6663a29bdbeSGreg Clayton { 6676e0ff1a3SGreg Clayton const bool success = m_comm.SendRequestDisconnect(); 668f9765acdSGreg Clayton if (log) 669f9765acdSGreg Clayton { 6706e0ff1a3SGreg Clayton if (success) 6716e0ff1a3SGreg Clayton log->PutCString ("ProcessKDP::DoDetach() detach packet sent successfully"); 672f9765acdSGreg Clayton else 67377e82d1eSJim Ingham log->PutCString ("ProcessKDP::DoDetach() connection channel shutdown failed"); 674f9765acdSGreg Clayton } 6756e0ff1a3SGreg Clayton m_comm.Disconnect (); 6763a29bdbeSGreg Clayton } 67797d5cf05SGreg Clayton } 678f9765acdSGreg Clayton StopAsyncThread (); 67974d4193eSGreg Clayton m_comm.Clear(); 680f9765acdSGreg Clayton 681f9765acdSGreg Clayton SetPrivateState (eStateDetached); 682f9765acdSGreg Clayton ResumePrivateStateThread(); 683f9765acdSGreg Clayton 684f9765acdSGreg Clayton //KillDebugserverProcess (); 685f9765acdSGreg Clayton return error; 686f9765acdSGreg Clayton } 687f9765acdSGreg Clayton 688f9765acdSGreg Clayton Error 689f9765acdSGreg Clayton ProcessKDP::DoDestroy () 690f9765acdSGreg Clayton { 6917925fbbaSGreg Clayton // For KDP there really is no difference between destroy and detach 692acff8950SJim Ingham bool keep_stopped = false; 693acff8950SJim Ingham return DoDetach(keep_stopped); 694f9765acdSGreg Clayton } 695f9765acdSGreg Clayton 696f9765acdSGreg Clayton //------------------------------------------------------------------ 697f9765acdSGreg Clayton // Process Queries 698f9765acdSGreg Clayton //------------------------------------------------------------------ 699f9765acdSGreg Clayton 700f9765acdSGreg Clayton bool 701f9765acdSGreg Clayton ProcessKDP::IsAlive () 702f9765acdSGreg Clayton { 703a814f704SJason Molenda return m_comm.IsConnected() && Process::IsAlive(); 704f9765acdSGreg Clayton } 705f9765acdSGreg Clayton 706f9765acdSGreg Clayton //------------------------------------------------------------------ 707f9765acdSGreg Clayton // Process Memory 708f9765acdSGreg Clayton //------------------------------------------------------------------ 709f9765acdSGreg Clayton size_t 710f9765acdSGreg Clayton ProcessKDP::DoReadMemory (addr_t addr, void *buf, size_t size, Error &error) 711f9765acdSGreg Clayton { 7128eb32817SJason Molenda uint8_t *data_buffer = (uint8_t *) buf; 713a63d08c9SGreg Clayton if (m_comm.IsConnected()) 7148eb32817SJason Molenda { 7158eb32817SJason Molenda const size_t max_read_size = 512; 7168eb32817SJason Molenda size_t total_bytes_read = 0; 7178eb32817SJason Molenda 7188eb32817SJason Molenda // Read the requested amount of memory in 512 byte chunks 7198eb32817SJason Molenda while (total_bytes_read < size) 7208eb32817SJason Molenda { 7218eb32817SJason Molenda size_t bytes_to_read_this_request = size - total_bytes_read; 7228eb32817SJason Molenda if (bytes_to_read_this_request > max_read_size) 7238eb32817SJason Molenda { 7248eb32817SJason Molenda bytes_to_read_this_request = max_read_size; 7258eb32817SJason Molenda } 7268eb32817SJason Molenda size_t bytes_read = m_comm.SendRequestReadMemory (addr + total_bytes_read, 7278eb32817SJason Molenda data_buffer + total_bytes_read, 7288eb32817SJason Molenda bytes_to_read_this_request, error); 7298eb32817SJason Molenda total_bytes_read += bytes_read; 7308eb32817SJason Molenda if (error.Fail() || bytes_read == 0) 7318eb32817SJason Molenda { 7328eb32817SJason Molenda return total_bytes_read; 7338eb32817SJason Molenda } 7348eb32817SJason Molenda } 7358eb32817SJason Molenda 7368eb32817SJason Molenda return total_bytes_read; 7378eb32817SJason Molenda } 738a63d08c9SGreg Clayton error.SetErrorString ("not connected"); 739f9765acdSGreg Clayton return 0; 740f9765acdSGreg Clayton } 741f9765acdSGreg Clayton 742f9765acdSGreg Clayton size_t 743f9765acdSGreg Clayton ProcessKDP::DoWriteMemory (addr_t addr, const void *buf, size_t size, Error &error) 744f9765acdSGreg Clayton { 7457925fbbaSGreg Clayton if (m_comm.IsConnected()) 7467925fbbaSGreg Clayton return m_comm.SendRequestWriteMemory (addr, buf, size, error); 7477925fbbaSGreg Clayton error.SetErrorString ("not connected"); 748f9765acdSGreg Clayton return 0; 749f9765acdSGreg Clayton } 750f9765acdSGreg Clayton 751f9765acdSGreg Clayton lldb::addr_t 752f9765acdSGreg Clayton ProcessKDP::DoAllocateMemory (size_t size, uint32_t permissions, Error &error) 753f9765acdSGreg Clayton { 754f9765acdSGreg Clayton error.SetErrorString ("memory allocation not suppported in kdp remote debugging"); 755f9765acdSGreg Clayton return LLDB_INVALID_ADDRESS; 756f9765acdSGreg Clayton } 757f9765acdSGreg Clayton 758f9765acdSGreg Clayton Error 759f9765acdSGreg Clayton ProcessKDP::DoDeallocateMemory (lldb::addr_t addr) 760f9765acdSGreg Clayton { 761f9765acdSGreg Clayton Error error; 762f9765acdSGreg Clayton error.SetErrorString ("memory deallocation not suppported in kdp remote debugging"); 763f9765acdSGreg Clayton return error; 764f9765acdSGreg Clayton } 765f9765acdSGreg Clayton 766f9765acdSGreg Clayton Error 767299c0c1cSJim Ingham ProcessKDP::EnableBreakpointSite (BreakpointSite *bp_site) 768f9765acdSGreg Clayton { 76907e66e3eSGreg Clayton if (m_comm.LocalBreakpointsAreSupported ()) 77007e66e3eSGreg Clayton { 77107e66e3eSGreg Clayton Error error; 7725b88216dSGreg Clayton if (!bp_site->IsEnabled()) 7735b88216dSGreg Clayton { 7745b88216dSGreg Clayton if (m_comm.SendRequestBreakpoint(true, bp_site->GetLoadAddress())) 7755b88216dSGreg Clayton { 7765b88216dSGreg Clayton bp_site->SetEnabled(true); 7775b88216dSGreg Clayton bp_site->SetType (BreakpointSite::eExternal); 7785b88216dSGreg Clayton } 7795b88216dSGreg Clayton else 7805b88216dSGreg Clayton { 78107e66e3eSGreg Clayton error.SetErrorString ("KDP set breakpoint failed"); 7825b88216dSGreg Clayton } 7835b88216dSGreg Clayton } 78407e66e3eSGreg Clayton return error; 78507e66e3eSGreg Clayton } 786f9765acdSGreg Clayton return EnableSoftwareBreakpoint (bp_site); 787f9765acdSGreg Clayton } 788f9765acdSGreg Clayton 789f9765acdSGreg Clayton Error 790299c0c1cSJim Ingham ProcessKDP::DisableBreakpointSite (BreakpointSite *bp_site) 791f9765acdSGreg Clayton { 79207e66e3eSGreg Clayton if (m_comm.LocalBreakpointsAreSupported ()) 79307e66e3eSGreg Clayton { 79407e66e3eSGreg Clayton Error error; 7955b88216dSGreg Clayton if (bp_site->IsEnabled()) 7965b88216dSGreg Clayton { 7975b88216dSGreg Clayton BreakpointSite::Type bp_type = bp_site->GetType(); 7985b88216dSGreg Clayton if (bp_type == BreakpointSite::eExternal) 7995b88216dSGreg Clayton { 80097d5cf05SGreg Clayton if (m_destroy_in_process && m_comm.IsRunning()) 80197d5cf05SGreg Clayton { 80297d5cf05SGreg Clayton // We are trying to destroy our connection and we are running 80397d5cf05SGreg Clayton bp_site->SetEnabled(false); 80497d5cf05SGreg Clayton } 80597d5cf05SGreg Clayton else 80697d5cf05SGreg Clayton { 8075b88216dSGreg Clayton if (m_comm.SendRequestBreakpoint(false, bp_site->GetLoadAddress())) 8085b88216dSGreg Clayton bp_site->SetEnabled(false); 8095b88216dSGreg Clayton else 81007e66e3eSGreg Clayton error.SetErrorString ("KDP remove breakpoint failed"); 8115b88216dSGreg Clayton } 81297d5cf05SGreg Clayton } 8135b88216dSGreg Clayton else 8145b88216dSGreg Clayton { 8155b88216dSGreg Clayton error = DisableSoftwareBreakpoint (bp_site); 8165b88216dSGreg Clayton } 8175b88216dSGreg Clayton } 81807e66e3eSGreg Clayton return error; 81907e66e3eSGreg Clayton } 820f9765acdSGreg Clayton return DisableSoftwareBreakpoint (bp_site); 821f9765acdSGreg Clayton } 822f9765acdSGreg Clayton 823f9765acdSGreg Clayton Error 8241b5792e5SJim Ingham ProcessKDP::EnableWatchpoint (Watchpoint *wp, bool notify) 825f9765acdSGreg Clayton { 826f9765acdSGreg Clayton Error error; 827f9765acdSGreg Clayton error.SetErrorString ("watchpoints are not suppported in kdp remote debugging"); 828f9765acdSGreg Clayton return error; 829f9765acdSGreg Clayton } 830f9765acdSGreg Clayton 831f9765acdSGreg Clayton Error 8321b5792e5SJim Ingham ProcessKDP::DisableWatchpoint (Watchpoint *wp, bool notify) 833f9765acdSGreg Clayton { 834f9765acdSGreg Clayton Error error; 835f9765acdSGreg Clayton error.SetErrorString ("watchpoints are not suppported in kdp remote debugging"); 836f9765acdSGreg Clayton return error; 837f9765acdSGreg Clayton } 838f9765acdSGreg Clayton 839f9765acdSGreg Clayton void 840f9765acdSGreg Clayton ProcessKDP::Clear() 841f9765acdSGreg Clayton { 842f9765acdSGreg Clayton m_thread_list.Clear(); 843f9765acdSGreg Clayton } 844f9765acdSGreg Clayton 845f9765acdSGreg Clayton Error 846f9765acdSGreg Clayton ProcessKDP::DoSignal (int signo) 847f9765acdSGreg Clayton { 848f9765acdSGreg Clayton Error error; 849f9765acdSGreg Clayton error.SetErrorString ("sending signals is not suppported in kdp remote debugging"); 850f9765acdSGreg Clayton return error; 851f9765acdSGreg Clayton } 852f9765acdSGreg Clayton 853f9765acdSGreg Clayton void 854f9765acdSGreg Clayton ProcessKDP::Initialize() 855f9765acdSGreg Clayton { 856c8d69828SDavide Italiano static std::once_flag g_once_flag; 857f9765acdSGreg Clayton 858c8d69828SDavide Italiano std::call_once(g_once_flag, []() 859f9765acdSGreg Clayton { 860f9765acdSGreg Clayton PluginManager::RegisterPlugin (GetPluginNameStatic(), 861f9765acdSGreg Clayton GetPluginDescriptionStatic(), 8627f98240dSGreg Clayton CreateInstance, 863488c89edSBruce Mitchener DebuggerInitialize); 864f9765acdSGreg Clayton 865f9765acdSGreg Clayton Log::Callbacks log_callbacks = { 866f9765acdSGreg Clayton ProcessKDPLog::DisableLog, 867f9765acdSGreg Clayton ProcessKDPLog::EnableLog, 868f9765acdSGreg Clayton ProcessKDPLog::ListLogCategories 869f9765acdSGreg Clayton }; 870f9765acdSGreg Clayton 871f9765acdSGreg Clayton Log::RegisterLogChannel (ProcessKDP::GetPluginNameStatic(), log_callbacks); 872c8d69828SDavide Italiano }); 873f9765acdSGreg Clayton } 874f9765acdSGreg Clayton 8757f98240dSGreg Clayton void 8767f98240dSGreg Clayton ProcessKDP::DebuggerInitialize (lldb_private::Debugger &debugger) 8777f98240dSGreg Clayton { 8787f98240dSGreg Clayton if (!PluginManager::GetSettingForProcessPlugin(debugger, PluginProperties::GetSettingName())) 8797f98240dSGreg Clayton { 8807f98240dSGreg Clayton const bool is_global_setting = true; 8817f98240dSGreg Clayton PluginManager::CreateSettingForProcessPlugin (debugger, 8827f98240dSGreg Clayton GetGlobalPluginProperties()->GetValueProperties(), 8837f98240dSGreg Clayton ConstString ("Properties for the kdp-remote process plug-in."), 8847f98240dSGreg Clayton is_global_setting); 8857f98240dSGreg Clayton } 8867f98240dSGreg Clayton } 8877f98240dSGreg Clayton 888f9765acdSGreg Clayton bool 889f9765acdSGreg Clayton ProcessKDP::StartAsyncThread () 890f9765acdSGreg Clayton { 8915160ce5cSGreg Clayton Log *log (ProcessKDPLog::GetLogIfAllCategoriesSet(KDP_LOG_PROCESS)); 892f9765acdSGreg Clayton 893f9765acdSGreg Clayton if (log) 8947925fbbaSGreg Clayton log->Printf ("ProcessKDP::StartAsyncThread ()"); 895f9765acdSGreg Clayton 896acee96aeSZachary Turner if (m_async_thread.IsJoinable()) 8977925fbbaSGreg Clayton return true; 8987925fbbaSGreg Clayton 89939de3110SZachary Turner m_async_thread = ThreadLauncher::LaunchThread("<lldb.process.kdp-remote.async>", ProcessKDP::AsyncThread, this, NULL); 900acee96aeSZachary Turner return m_async_thread.IsJoinable(); 901f9765acdSGreg Clayton } 902f9765acdSGreg Clayton 903f9765acdSGreg Clayton void 904f9765acdSGreg Clayton ProcessKDP::StopAsyncThread () 905f9765acdSGreg Clayton { 9065160ce5cSGreg Clayton Log *log (ProcessKDPLog::GetLogIfAllCategoriesSet(KDP_LOG_PROCESS)); 907f9765acdSGreg Clayton 908f9765acdSGreg Clayton if (log) 9097925fbbaSGreg Clayton log->Printf ("ProcessKDP::StopAsyncThread ()"); 910f9765acdSGreg Clayton 911f9765acdSGreg Clayton m_async_broadcaster.BroadcastEvent (eBroadcastBitAsyncThreadShouldExit); 912f9765acdSGreg Clayton 913f9765acdSGreg Clayton // Stop the stdio thread 914acee96aeSZachary Turner if (m_async_thread.IsJoinable()) 91539de3110SZachary Turner m_async_thread.Join(nullptr); 916f9765acdSGreg Clayton } 917f9765acdSGreg Clayton 918f9765acdSGreg Clayton 919f9765acdSGreg Clayton void * 920f9765acdSGreg Clayton ProcessKDP::AsyncThread (void *arg) 921f9765acdSGreg Clayton { 922f9765acdSGreg Clayton ProcessKDP *process = (ProcessKDP*) arg; 923f9765acdSGreg Clayton 9247925fbbaSGreg Clayton const lldb::pid_t pid = process->GetID(); 9257925fbbaSGreg Clayton 9265160ce5cSGreg Clayton Log *log (ProcessKDPLog::GetLogIfAllCategoriesSet (KDP_LOG_PROCESS)); 927f9765acdSGreg Clayton if (log) 928d01b2953SDaniel Malea log->Printf ("ProcessKDP::AsyncThread (arg = %p, pid = %" PRIu64 ") thread starting...", arg, pid); 929f9765acdSGreg Clayton 930583bbb1dSJim Ingham ListenerSP listener_sp (Listener::MakeListener("ProcessKDP::AsyncThread")); 931f9765acdSGreg Clayton EventSP event_sp; 932f9765acdSGreg Clayton const uint32_t desired_event_mask = eBroadcastBitAsyncContinue | 933f9765acdSGreg Clayton eBroadcastBitAsyncThreadShouldExit; 934f9765acdSGreg Clayton 9357925fbbaSGreg Clayton 936583bbb1dSJim Ingham if (listener_sp->StartListeningForEvents (&process->m_async_broadcaster, desired_event_mask) == desired_event_mask) 937f9765acdSGreg Clayton { 938f9765acdSGreg Clayton bool done = false; 939f9765acdSGreg Clayton while (!done) 940f9765acdSGreg Clayton { 941f9765acdSGreg Clayton if (log) 942d01b2953SDaniel Malea log->Printf ("ProcessKDP::AsyncThread (pid = %" PRIu64 ") listener.WaitForEvent (NULL, event_sp)...", 9437925fbbaSGreg Clayton pid); 944*27279febSTodd Fiala if (listener_sp->WaitForEvent(std::chrono::microseconds(0), event_sp)) 945f9765acdSGreg Clayton { 9467925fbbaSGreg Clayton uint32_t event_type = event_sp->GetType(); 947f9765acdSGreg Clayton if (log) 948d01b2953SDaniel Malea log->Printf ("ProcessKDP::AsyncThread (pid = %" PRIu64 ") Got an event of type: %d...", 9497925fbbaSGreg Clayton pid, 9507925fbbaSGreg Clayton event_type); 951f9765acdSGreg Clayton 9527925fbbaSGreg Clayton // When we are running, poll for 1 second to try and get an exception 9537925fbbaSGreg Clayton // to indicate the process has stopped. If we don't get one, check to 9547925fbbaSGreg Clayton // make sure no one asked us to exit 9557925fbbaSGreg Clayton bool is_running = false; 9567925fbbaSGreg Clayton DataExtractor exc_reply_packet; 9577925fbbaSGreg Clayton do 9587925fbbaSGreg Clayton { 959f9765acdSGreg Clayton switch (event_type) 960f9765acdSGreg Clayton { 961f9765acdSGreg Clayton case eBroadcastBitAsyncContinue: 962f9765acdSGreg Clayton { 9637925fbbaSGreg Clayton is_running = true; 9647925fbbaSGreg Clayton if (process->m_comm.WaitForPacketWithTimeoutMicroSeconds (exc_reply_packet, 1 * USEC_PER_SEC)) 965f9765acdSGreg Clayton { 966ba4e61d3SAndrew Kaylor ThreadSP thread_sp (process->GetKernelThread()); 9671afa68edSGreg Clayton if (thread_sp) 9681afa68edSGreg Clayton { 9691afa68edSGreg Clayton lldb::RegisterContextSP reg_ctx_sp (thread_sp->GetRegisterContext()); 9701afa68edSGreg Clayton if (reg_ctx_sp) 9711afa68edSGreg Clayton reg_ctx_sp->InvalidateAllRegisters(); 97297d5cf05SGreg Clayton static_cast<ThreadKDP *>(thread_sp.get())->SetStopInfoFrom_KDP_EXCEPTION (exc_reply_packet); 9731afa68edSGreg Clayton } 97497d5cf05SGreg Clayton 9757925fbbaSGreg Clayton // TODO: parse the stop reply packet 9767925fbbaSGreg Clayton is_running = false; 9777925fbbaSGreg Clayton process->SetPrivateState(eStateStopped); 9787925fbbaSGreg Clayton } 9797925fbbaSGreg Clayton else 9807925fbbaSGreg Clayton { 9817925fbbaSGreg Clayton // Check to see if we are supposed to exit. There is no way to 9827925fbbaSGreg Clayton // interrupt a running kernel, so all we can do is wait for an 9837925fbbaSGreg Clayton // exception or detach... 984583bbb1dSJim Ingham if (listener_sp->GetNextEvent(event_sp)) 9857925fbbaSGreg Clayton { 9867925fbbaSGreg Clayton // We got an event, go through the loop again 9877925fbbaSGreg Clayton event_type = event_sp->GetType(); 9887925fbbaSGreg Clayton } 989f9765acdSGreg Clayton } 990f9765acdSGreg Clayton } 991f9765acdSGreg Clayton break; 992f9765acdSGreg Clayton 993f9765acdSGreg Clayton case eBroadcastBitAsyncThreadShouldExit: 994f9765acdSGreg Clayton if (log) 995d01b2953SDaniel Malea log->Printf ("ProcessKDP::AsyncThread (pid = %" PRIu64 ") got eBroadcastBitAsyncThreadShouldExit...", 9967925fbbaSGreg Clayton pid); 997f9765acdSGreg Clayton done = true; 9987925fbbaSGreg Clayton is_running = false; 999f9765acdSGreg Clayton break; 1000f9765acdSGreg Clayton 1001f9765acdSGreg Clayton default: 1002f9765acdSGreg Clayton if (log) 1003d01b2953SDaniel Malea log->Printf ("ProcessKDP::AsyncThread (pid = %" PRIu64 ") got unknown event 0x%8.8x", 10047925fbbaSGreg Clayton pid, 10057925fbbaSGreg Clayton event_type); 1006f9765acdSGreg Clayton done = true; 10077925fbbaSGreg Clayton is_running = false; 1008f9765acdSGreg Clayton break; 1009f9765acdSGreg Clayton } 10107925fbbaSGreg Clayton } while (is_running); 1011f9765acdSGreg Clayton } 1012f9765acdSGreg Clayton else 1013f9765acdSGreg Clayton { 1014f9765acdSGreg Clayton if (log) 1015d01b2953SDaniel Malea log->Printf ("ProcessKDP::AsyncThread (pid = %" PRIu64 ") listener.WaitForEvent (NULL, event_sp) => false", 10167925fbbaSGreg Clayton pid); 1017f9765acdSGreg Clayton done = true; 1018f9765acdSGreg Clayton } 1019f9765acdSGreg Clayton } 1020f9765acdSGreg Clayton } 1021f9765acdSGreg Clayton 1022f9765acdSGreg Clayton if (log) 1023d01b2953SDaniel Malea log->Printf ("ProcessKDP::AsyncThread (arg = %p, pid = %" PRIu64 ") thread exiting...", 10247925fbbaSGreg Clayton arg, 10257925fbbaSGreg Clayton pid); 1026f9765acdSGreg Clayton 102739de3110SZachary Turner process->m_async_thread.Reset(); 1028f9765acdSGreg Clayton return NULL; 1029f9765acdSGreg Clayton } 1030f9765acdSGreg Clayton 1031f9765acdSGreg Clayton 10321d19a2f2SGreg Clayton class CommandObjectProcessKDPPacketSend : public CommandObjectParsed 10331d19a2f2SGreg Clayton { 10341d19a2f2SGreg Clayton private: 10351d19a2f2SGreg Clayton 10361d19a2f2SGreg Clayton OptionGroupOptions m_option_group; 10371d19a2f2SGreg Clayton OptionGroupUInt64 m_command_byte; 10381d19a2f2SGreg Clayton OptionGroupString m_packet_data; 10391d19a2f2SGreg Clayton 10401d19a2f2SGreg Clayton virtual Options * 10411d19a2f2SGreg Clayton GetOptions () 10421d19a2f2SGreg Clayton { 10431d19a2f2SGreg Clayton return &m_option_group; 10441d19a2f2SGreg Clayton } 10451d19a2f2SGreg Clayton 10461d19a2f2SGreg Clayton 10471d19a2f2SGreg Clayton public: 10481d19a2f2SGreg Clayton CommandObjectProcessKDPPacketSend(CommandInterpreter &interpreter) : 10491d19a2f2SGreg Clayton CommandObjectParsed (interpreter, 10501d19a2f2SGreg Clayton "process plugin packet send", 10511d19a2f2SGreg Clayton "Send a custom packet through the KDP protocol by specifying the command byte and the packet payload data. A packet will be sent with a correct header and payload, and the raw result bytes will be displayed as a string value. ", 10521d19a2f2SGreg Clayton NULL), 10531d19a2f2SGreg Clayton m_option_group (interpreter), 10541d19a2f2SGreg Clayton m_command_byte(LLDB_OPT_SET_1, true , "command", 'c', 0, eArgTypeNone, "Specify the command byte to use when sending the KDP request packet.", 0), 10551d19a2f2SGreg Clayton m_packet_data (LLDB_OPT_SET_1, false, "payload", 'p', 0, eArgTypeNone, "Specify packet payload bytes as a hex ASCII string with no spaces or hex prefixes.", NULL) 10561d19a2f2SGreg Clayton { 10571d19a2f2SGreg Clayton m_option_group.Append (&m_command_byte, LLDB_OPT_SET_ALL, LLDB_OPT_SET_1); 10581d19a2f2SGreg Clayton m_option_group.Append (&m_packet_data , LLDB_OPT_SET_ALL, LLDB_OPT_SET_1); 10591d19a2f2SGreg Clayton m_option_group.Finalize(); 10601d19a2f2SGreg Clayton } 10611d19a2f2SGreg Clayton 10621d19a2f2SGreg Clayton ~CommandObjectProcessKDPPacketSend () 10631d19a2f2SGreg Clayton { 10641d19a2f2SGreg Clayton } 10651d19a2f2SGreg Clayton 10661d19a2f2SGreg Clayton bool 10671d19a2f2SGreg Clayton DoExecute (Args& command, CommandReturnObject &result) 10681d19a2f2SGreg Clayton { 10691d19a2f2SGreg Clayton const size_t argc = command.GetArgumentCount(); 10701d19a2f2SGreg Clayton if (argc == 0) 10711d19a2f2SGreg Clayton { 10721d19a2f2SGreg Clayton if (!m_command_byte.GetOptionValue().OptionWasSet()) 10731d19a2f2SGreg Clayton { 10741d19a2f2SGreg Clayton result.AppendError ("the --command option must be set to a valid command byte"); 10751d19a2f2SGreg Clayton result.SetStatus (eReturnStatusFailed); 10761d19a2f2SGreg Clayton } 10771d19a2f2SGreg Clayton else 10781d19a2f2SGreg Clayton { 10791d19a2f2SGreg Clayton const uint64_t command_byte = m_command_byte.GetOptionValue().GetUInt64Value(0); 10801d19a2f2SGreg Clayton if (command_byte > 0 && command_byte <= UINT8_MAX) 10811d19a2f2SGreg Clayton { 10821d19a2f2SGreg Clayton ProcessKDP *process = (ProcessKDP *)m_interpreter.GetExecutionContext().GetProcessPtr(); 10831d19a2f2SGreg Clayton if (process) 10841d19a2f2SGreg Clayton { 10851d19a2f2SGreg Clayton const StateType state = process->GetState(); 10861d19a2f2SGreg Clayton 10871d19a2f2SGreg Clayton if (StateIsStoppedState (state, true)) 10881d19a2f2SGreg Clayton { 10891d19a2f2SGreg Clayton std::vector<uint8_t> payload_bytes; 10901d19a2f2SGreg Clayton const char *ascii_hex_bytes_cstr = m_packet_data.GetOptionValue().GetCurrentValue(); 10911d19a2f2SGreg Clayton if (ascii_hex_bytes_cstr && ascii_hex_bytes_cstr[0]) 10921d19a2f2SGreg Clayton { 10931d19a2f2SGreg Clayton StringExtractor extractor(ascii_hex_bytes_cstr); 10941d19a2f2SGreg Clayton const size_t ascii_hex_bytes_cstr_len = extractor.GetStringRef().size(); 10951d19a2f2SGreg Clayton if (ascii_hex_bytes_cstr_len & 1) 10961d19a2f2SGreg Clayton { 10971d19a2f2SGreg Clayton result.AppendErrorWithFormat ("payload data must contain an even number of ASCII hex characters: '%s'", ascii_hex_bytes_cstr); 10981d19a2f2SGreg Clayton result.SetStatus (eReturnStatusFailed); 10991d19a2f2SGreg Clayton return false; 11001d19a2f2SGreg Clayton } 11011d19a2f2SGreg Clayton payload_bytes.resize(ascii_hex_bytes_cstr_len/2); 11021d19a2f2SGreg Clayton if (extractor.GetHexBytes(&payload_bytes[0], payload_bytes.size(), '\xdd') != payload_bytes.size()) 11031d19a2f2SGreg Clayton { 11041d19a2f2SGreg Clayton result.AppendErrorWithFormat ("payload data must only contain ASCII hex characters (no spaces or hex prefixes): '%s'", ascii_hex_bytes_cstr); 11051d19a2f2SGreg Clayton result.SetStatus (eReturnStatusFailed); 11061d19a2f2SGreg Clayton return false; 11071d19a2f2SGreg Clayton } 11081d19a2f2SGreg Clayton } 11091d19a2f2SGreg Clayton Error error; 11101d19a2f2SGreg Clayton DataExtractor reply; 11111d19a2f2SGreg Clayton process->GetCommunication().SendRawRequest (command_byte, 11121d19a2f2SGreg Clayton payload_bytes.empty() ? NULL : payload_bytes.data(), 11131d19a2f2SGreg Clayton payload_bytes.size(), 11141d19a2f2SGreg Clayton reply, 11151d19a2f2SGreg Clayton error); 11161d19a2f2SGreg Clayton 11171d19a2f2SGreg Clayton if (error.Success()) 11181d19a2f2SGreg Clayton { 11191d19a2f2SGreg Clayton // Copy the binary bytes into a hex ASCII string for the result 11201d19a2f2SGreg Clayton StreamString packet; 11211d19a2f2SGreg Clayton packet.PutBytesAsRawHex8(reply.GetDataStart(), 11221d19a2f2SGreg Clayton reply.GetByteSize(), 11239ccb970fSBruce Mitchener endian::InlHostByteOrder(), 11249ccb970fSBruce Mitchener endian::InlHostByteOrder()); 11251d19a2f2SGreg Clayton result.AppendMessage(packet.GetString().c_str()); 11261d19a2f2SGreg Clayton result.SetStatus (eReturnStatusSuccessFinishResult); 11271d19a2f2SGreg Clayton return true; 11281d19a2f2SGreg Clayton } 11291d19a2f2SGreg Clayton else 11301d19a2f2SGreg Clayton { 11311d19a2f2SGreg Clayton const char *error_cstr = error.AsCString(); 11321d19a2f2SGreg Clayton if (error_cstr && error_cstr[0]) 11331d19a2f2SGreg Clayton result.AppendError (error_cstr); 11341d19a2f2SGreg Clayton else 11351d19a2f2SGreg Clayton result.AppendErrorWithFormat ("unknown error 0x%8.8x", error.GetError()); 11361d19a2f2SGreg Clayton result.SetStatus (eReturnStatusFailed); 11371d19a2f2SGreg Clayton return false; 11381d19a2f2SGreg Clayton } 11391d19a2f2SGreg Clayton } 11401d19a2f2SGreg Clayton else 11411d19a2f2SGreg Clayton { 11421d19a2f2SGreg Clayton result.AppendErrorWithFormat ("process must be stopped in order to send KDP packets, state is %s", StateAsCString (state)); 11431d19a2f2SGreg Clayton result.SetStatus (eReturnStatusFailed); 11441d19a2f2SGreg Clayton } 11451d19a2f2SGreg Clayton } 11461d19a2f2SGreg Clayton else 11471d19a2f2SGreg Clayton { 11481d19a2f2SGreg Clayton result.AppendError ("invalid process"); 11491d19a2f2SGreg Clayton result.SetStatus (eReturnStatusFailed); 11501d19a2f2SGreg Clayton } 11511d19a2f2SGreg Clayton } 11521d19a2f2SGreg Clayton else 11531d19a2f2SGreg Clayton { 1154d01b2953SDaniel Malea result.AppendErrorWithFormat ("invalid command byte 0x%" PRIx64 ", valid values are 1 - 255", command_byte); 11551d19a2f2SGreg Clayton result.SetStatus (eReturnStatusFailed); 11561d19a2f2SGreg Clayton } 11571d19a2f2SGreg Clayton } 11581d19a2f2SGreg Clayton } 11591d19a2f2SGreg Clayton else 11601d19a2f2SGreg Clayton { 11611d19a2f2SGreg Clayton result.AppendErrorWithFormat ("'%s' takes no arguments, only options.", m_cmd_name.c_str()); 11621d19a2f2SGreg Clayton result.SetStatus (eReturnStatusFailed); 11631d19a2f2SGreg Clayton } 11641d19a2f2SGreg Clayton return false; 11651d19a2f2SGreg Clayton } 11661d19a2f2SGreg Clayton }; 11671d19a2f2SGreg Clayton 11681d19a2f2SGreg Clayton class CommandObjectProcessKDPPacket : public CommandObjectMultiword 11691d19a2f2SGreg Clayton { 11701d19a2f2SGreg Clayton private: 11711d19a2f2SGreg Clayton 11721d19a2f2SGreg Clayton public: 11731d19a2f2SGreg Clayton CommandObjectProcessKDPPacket(CommandInterpreter &interpreter) : 11741d19a2f2SGreg Clayton CommandObjectMultiword (interpreter, 11751d19a2f2SGreg Clayton "process plugin packet", 11761d19a2f2SGreg Clayton "Commands that deal with KDP remote packets.", 11771d19a2f2SGreg Clayton NULL) 11781d19a2f2SGreg Clayton { 11791d19a2f2SGreg Clayton LoadSubCommand ("send", CommandObjectSP (new CommandObjectProcessKDPPacketSend (interpreter))); 11801d19a2f2SGreg Clayton } 11811d19a2f2SGreg Clayton 11821d19a2f2SGreg Clayton ~CommandObjectProcessKDPPacket () 11831d19a2f2SGreg Clayton { 11841d19a2f2SGreg Clayton } 11851d19a2f2SGreg Clayton }; 11861d19a2f2SGreg Clayton 11871d19a2f2SGreg Clayton class CommandObjectMultiwordProcessKDP : public CommandObjectMultiword 11881d19a2f2SGreg Clayton { 11891d19a2f2SGreg Clayton public: 11907428a18cSKate Stone CommandObjectMultiwordProcessKDP(CommandInterpreter &interpreter) 11917428a18cSKate Stone : CommandObjectMultiword(interpreter, "process plugin", "Commands for operating on a ProcessKDP process.", 11921d19a2f2SGreg Clayton "process plugin <subcommand> [<subcommand-options>]") 11931d19a2f2SGreg Clayton { 11941d19a2f2SGreg Clayton LoadSubCommand ("packet", CommandObjectSP (new CommandObjectProcessKDPPacket (interpreter))); 11951d19a2f2SGreg Clayton } 11961d19a2f2SGreg Clayton 11971d19a2f2SGreg Clayton ~CommandObjectMultiwordProcessKDP () 11981d19a2f2SGreg Clayton { 11991d19a2f2SGreg Clayton } 12001d19a2f2SGreg Clayton }; 12011d19a2f2SGreg Clayton 12021d19a2f2SGreg Clayton CommandObject * 12031d19a2f2SGreg Clayton ProcessKDP::GetPluginCommandObject() 12041d19a2f2SGreg Clayton { 12051d19a2f2SGreg Clayton if (!m_command_sp) 12061d19a2f2SGreg Clayton m_command_sp.reset (new CommandObjectMultiwordProcessKDP (GetTarget().GetDebugger().GetCommandInterpreter())); 12071d19a2f2SGreg Clayton return m_command_sp.get(); 12081d19a2f2SGreg Clayton } 12091d19a2f2SGreg Clayton 1210