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