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 15f9765acdSGreg Clayton // Other libraries and framework includes 163a29bdbeSGreg Clayton #include "lldb/Core/ConnectionFileDescriptor.h" 1707e66e3eSGreg Clayton #include "lldb/Core/Debugger.h" 18f9765acdSGreg Clayton #include "lldb/Core/PluginManager.h" 191f746071SGreg Clayton #include "lldb/Core/Module.h" 204bd4e7e3SJason Molenda #include "lldb/Core/ModuleSpec.h" 21f9765acdSGreg Clayton #include "lldb/Core/State.h" 224bd4e7e3SJason Molenda #include "lldb/Core/UUID.h" 23f9765acdSGreg Clayton #include "lldb/Host/Host.h" 244bd4e7e3SJason Molenda #include "lldb/Host/Symbols.h" 251d19a2f2SGreg Clayton #include "lldb/Interpreter/CommandInterpreter.h" 261d19a2f2SGreg Clayton #include "lldb/Interpreter/CommandObject.h" 271d19a2f2SGreg Clayton #include "lldb/Interpreter/CommandObjectMultiword.h" 281d19a2f2SGreg Clayton #include "lldb/Interpreter/CommandReturnObject.h" 291d19a2f2SGreg Clayton #include "lldb/Interpreter/OptionGroupString.h" 301d19a2f2SGreg Clayton #include "lldb/Interpreter/OptionGroupUInt64.h" 311f746071SGreg Clayton #include "lldb/Symbol/ObjectFile.h" 327925fbbaSGreg Clayton #include "lldb/Target/RegisterContext.h" 3357508026SGreg Clayton #include "lldb/Target/Target.h" 34a63d08c9SGreg Clayton #include "lldb/Target/Thread.h" 35f9765acdSGreg Clayton 36f9765acdSGreg Clayton // Project includes 37f9765acdSGreg Clayton #include "ProcessKDP.h" 38f9765acdSGreg Clayton #include "ProcessKDPLog.h" 39a63d08c9SGreg Clayton #include "ThreadKDP.h" 405e8534efSJason Molenda #include "Plugins/DynamicLoader/Darwin-Kernel/DynamicLoaderDarwinKernel.h" 41840f12cfSJason Molenda #include "Plugins/DynamicLoader/Static/DynamicLoaderStatic.h" 421d19a2f2SGreg Clayton #include "Utility/StringExtractor.h" 43f9765acdSGreg Clayton 44f9765acdSGreg Clayton using namespace lldb; 45f9765acdSGreg Clayton using namespace lldb_private; 46f9765acdSGreg Clayton 47f9765acdSGreg Clayton const char * 48f9765acdSGreg Clayton ProcessKDP::GetPluginNameStatic() 49f9765acdSGreg Clayton { 50f9765acdSGreg Clayton return "kdp-remote"; 51f9765acdSGreg Clayton } 52f9765acdSGreg Clayton 53f9765acdSGreg Clayton const char * 54f9765acdSGreg Clayton ProcessKDP::GetPluginDescriptionStatic() 55f9765acdSGreg Clayton { 56f9765acdSGreg Clayton return "KDP Remote protocol based debugging plug-in for darwin kernel debugging."; 57f9765acdSGreg Clayton } 58f9765acdSGreg Clayton 59f9765acdSGreg Clayton void 60f9765acdSGreg Clayton ProcessKDP::Terminate() 61f9765acdSGreg Clayton { 62f9765acdSGreg Clayton PluginManager::UnregisterPlugin (ProcessKDP::CreateInstance); 63f9765acdSGreg Clayton } 64f9765acdSGreg Clayton 65f9765acdSGreg Clayton 66c3776bf2SGreg Clayton lldb::ProcessSP 67c3776bf2SGreg Clayton ProcessKDP::CreateInstance (Target &target, 68c3776bf2SGreg Clayton Listener &listener, 69c3776bf2SGreg Clayton const FileSpec *crash_file_path) 70f9765acdSGreg Clayton { 71c3776bf2SGreg Clayton lldb::ProcessSP process_sp; 72c3776bf2SGreg Clayton if (crash_file_path == NULL) 73c3776bf2SGreg Clayton process_sp.reset(new ProcessKDP (target, listener)); 74c3776bf2SGreg Clayton return process_sp; 75f9765acdSGreg Clayton } 76f9765acdSGreg Clayton 77f9765acdSGreg Clayton bool 783a29bdbeSGreg Clayton ProcessKDP::CanDebug(Target &target, bool plugin_specified_by_name) 79f9765acdSGreg Clayton { 80596ed24eSGreg Clayton if (plugin_specified_by_name) 81596ed24eSGreg Clayton return true; 82596ed24eSGreg Clayton 83f9765acdSGreg Clayton // For now we are just making sure the file exists for a given module 84aa149cbdSGreg Clayton Module *exe_module = target.GetExecutableModulePointer(); 85aa149cbdSGreg Clayton if (exe_module) 86f9765acdSGreg Clayton { 87f9765acdSGreg Clayton const llvm::Triple &triple_ref = target.GetArchitecture().GetTriple(); 8870512317SGreg Clayton switch (triple_ref.getOS()) 8970512317SGreg Clayton { 9070512317SGreg Clayton case llvm::Triple::Darwin: // Should use "macosx" for desktop and "ios" for iOS, but accept darwin just in case 9170512317SGreg Clayton case llvm::Triple::MacOSX: // For desktop targets 9270512317SGreg Clayton case llvm::Triple::IOS: // For arm targets 9370512317SGreg Clayton if (triple_ref.getVendor() == llvm::Triple::Apple) 94f9765acdSGreg Clayton { 95aa149cbdSGreg Clayton ObjectFile *exe_objfile = exe_module->GetObjectFile(); 96f9765acdSGreg Clayton if (exe_objfile->GetType() == ObjectFile::eTypeExecutable && 97f9765acdSGreg Clayton exe_objfile->GetStrata() == ObjectFile::eStrataKernel) 98f9765acdSGreg Clayton return true; 99f9765acdSGreg Clayton } 10070512317SGreg Clayton break; 10170512317SGreg Clayton 10270512317SGreg Clayton default: 10370512317SGreg Clayton break; 10470512317SGreg Clayton } 105f9765acdSGreg Clayton } 106596ed24eSGreg Clayton return false; 1073a29bdbeSGreg Clayton } 108f9765acdSGreg Clayton 109f9765acdSGreg Clayton //---------------------------------------------------------------------- 110f9765acdSGreg Clayton // ProcessKDP constructor 111f9765acdSGreg Clayton //---------------------------------------------------------------------- 112f9765acdSGreg Clayton ProcessKDP::ProcessKDP(Target& target, Listener &listener) : 113f9765acdSGreg Clayton Process (target, listener), 114f9765acdSGreg Clayton m_comm("lldb.process.kdp-remote.communication"), 1154bddaeb5SJim Ingham m_async_broadcaster (NULL, "lldb.process.kdp-remote.async-broadcaster"), 11697d5cf05SGreg Clayton m_async_thread (LLDB_INVALID_HOST_THREAD), 1175e8534efSJason Molenda m_dyld_plugin_name (), 1181d19a2f2SGreg Clayton m_kernel_load_addr (LLDB_INVALID_ADDRESS), 1191d19a2f2SGreg Clayton m_command_sp() 120f9765acdSGreg Clayton { 1217925fbbaSGreg Clayton m_async_broadcaster.SetEventName (eBroadcastBitAsyncThreadShouldExit, "async thread should exit"); 1227925fbbaSGreg Clayton m_async_broadcaster.SetEventName (eBroadcastBitAsyncContinue, "async thread continue"); 123f9765acdSGreg Clayton } 124f9765acdSGreg Clayton 125f9765acdSGreg Clayton //---------------------------------------------------------------------- 126f9765acdSGreg Clayton // Destructor 127f9765acdSGreg Clayton //---------------------------------------------------------------------- 128f9765acdSGreg Clayton ProcessKDP::~ProcessKDP() 129f9765acdSGreg Clayton { 130f9765acdSGreg Clayton Clear(); 131e24c4acfSGreg Clayton // We need to call finalize on the process before destroying ourselves 132e24c4acfSGreg Clayton // to make sure all of the broadcaster cleanup goes as planned. If we 133e24c4acfSGreg Clayton // destruct this class, then Process::~Process() might have problems 134e24c4acfSGreg Clayton // trying to fully destroy the broadcaster. 135e24c4acfSGreg Clayton Finalize(); 136f9765acdSGreg Clayton } 137f9765acdSGreg Clayton 138f9765acdSGreg Clayton //---------------------------------------------------------------------- 139f9765acdSGreg Clayton // PluginInterface 140f9765acdSGreg Clayton //---------------------------------------------------------------------- 141f9765acdSGreg Clayton const char * 142f9765acdSGreg Clayton ProcessKDP::GetPluginName() 143f9765acdSGreg Clayton { 144f9765acdSGreg Clayton return "Process debugging plug-in that uses the Darwin KDP remote protocol"; 145f9765acdSGreg Clayton } 146f9765acdSGreg Clayton 147f9765acdSGreg Clayton const char * 148f9765acdSGreg Clayton ProcessKDP::GetShortPluginName() 149f9765acdSGreg Clayton { 150f9765acdSGreg Clayton return GetPluginNameStatic(); 151f9765acdSGreg Clayton } 152f9765acdSGreg Clayton 153f9765acdSGreg Clayton uint32_t 154f9765acdSGreg Clayton ProcessKDP::GetPluginVersion() 155f9765acdSGreg Clayton { 156f9765acdSGreg Clayton return 1; 157f9765acdSGreg Clayton } 158f9765acdSGreg Clayton 159f9765acdSGreg Clayton Error 160f9765acdSGreg Clayton ProcessKDP::WillLaunch (Module* module) 161f9765acdSGreg Clayton { 162f9765acdSGreg Clayton Error error; 163f9765acdSGreg Clayton error.SetErrorString ("launching not supported in kdp-remote plug-in"); 164f9765acdSGreg Clayton return error; 165f9765acdSGreg Clayton } 166f9765acdSGreg Clayton 167f9765acdSGreg Clayton Error 168f9765acdSGreg Clayton ProcessKDP::WillAttachToProcessWithID (lldb::pid_t pid) 169f9765acdSGreg Clayton { 170f9765acdSGreg Clayton Error error; 171f9765acdSGreg Clayton error.SetErrorString ("attaching to a by process ID not supported in kdp-remote plug-in"); 172f9765acdSGreg Clayton return error; 173f9765acdSGreg Clayton } 174f9765acdSGreg Clayton 175f9765acdSGreg Clayton Error 176f9765acdSGreg Clayton ProcessKDP::WillAttachToProcessWithName (const char *process_name, bool wait_for_launch) 177f9765acdSGreg Clayton { 178f9765acdSGreg Clayton Error error; 179f9765acdSGreg Clayton error.SetErrorString ("attaching to a by process name not supported in kdp-remote plug-in"); 180f9765acdSGreg Clayton return error; 181f9765acdSGreg Clayton } 182f9765acdSGreg Clayton 183f9765acdSGreg Clayton Error 1844bd4e7e3SJason Molenda ProcessKDP::DoConnectRemote (Stream *strm, const char *remote_url) 185f9765acdSGreg Clayton { 186f9765acdSGreg Clayton Error error; 1873a29bdbeSGreg Clayton 1887925fbbaSGreg Clayton // Don't let any JIT happen when doing KDP as we can't allocate 1897925fbbaSGreg Clayton // memory and we don't want to be mucking with threads that might 1907925fbbaSGreg Clayton // already be handling exceptions 1917925fbbaSGreg Clayton SetCanJIT(false); 1927925fbbaSGreg Clayton 1933a29bdbeSGreg Clayton if (remote_url == NULL || remote_url[0] == '\0') 1947925fbbaSGreg Clayton { 1957925fbbaSGreg Clayton error.SetErrorStringWithFormat ("invalid connection URL '%s'", remote_url); 1967925fbbaSGreg Clayton return error; 1977925fbbaSGreg Clayton } 1983a29bdbeSGreg Clayton 1993a29bdbeSGreg Clayton std::auto_ptr<ConnectionFileDescriptor> conn_ap(new ConnectionFileDescriptor()); 2003a29bdbeSGreg Clayton if (conn_ap.get()) 2013a29bdbeSGreg Clayton { 2023a29bdbeSGreg Clayton // Only try once for now. 2033a29bdbeSGreg Clayton // TODO: check if we should be retrying? 2043a29bdbeSGreg Clayton const uint32_t max_retry_count = 1; 2053a29bdbeSGreg Clayton for (uint32_t retry_count = 0; retry_count < max_retry_count; ++retry_count) 2063a29bdbeSGreg Clayton { 2073a29bdbeSGreg Clayton if (conn_ap->Connect(remote_url, &error) == eConnectionStatusSuccess) 2083a29bdbeSGreg Clayton break; 2093a29bdbeSGreg Clayton usleep (100000); 2103a29bdbeSGreg Clayton } 2113a29bdbeSGreg Clayton } 2123a29bdbeSGreg Clayton 2133a29bdbeSGreg Clayton if (conn_ap->IsConnected()) 2143a29bdbeSGreg Clayton { 2153a29bdbeSGreg Clayton const uint16_t reply_port = conn_ap->GetReadPort (); 2163a29bdbeSGreg Clayton 2173a29bdbeSGreg Clayton if (reply_port != 0) 2183a29bdbeSGreg Clayton { 2193a29bdbeSGreg Clayton m_comm.SetConnection(conn_ap.release()); 2203a29bdbeSGreg Clayton 2213a29bdbeSGreg Clayton if (m_comm.SendRequestReattach(reply_port)) 2223a29bdbeSGreg Clayton { 2233a29bdbeSGreg Clayton if (m_comm.SendRequestConnect(reply_port, reply_port, "Greetings from LLDB...")) 2243a29bdbeSGreg Clayton { 2253a29bdbeSGreg Clayton m_comm.GetVersion(); 2263a29bdbeSGreg Clayton uint32_t cpu = m_comm.GetCPUType(); 2273a29bdbeSGreg Clayton uint32_t sub = m_comm.GetCPUSubtype(); 2283a29bdbeSGreg Clayton ArchSpec kernel_arch; 2293a29bdbeSGreg Clayton kernel_arch.SetArchitecture(eArchTypeMachO, cpu, sub); 2303a29bdbeSGreg Clayton m_target.SetArchitecture(kernel_arch); 2314bd4e7e3SJason Molenda 232840f12cfSJason Molenda /* Get the kernel's UUID and load address via KDP_KERNELVERSION packet. */ 233840f12cfSJason Molenda /* An EFI kdp session has neither UUID nor load address. */ 2344bd4e7e3SJason Molenda 2354bd4e7e3SJason Molenda UUID kernel_uuid = m_comm.GetUUID (); 2364bd4e7e3SJason Molenda addr_t kernel_load_addr = m_comm.GetLoadAddress (); 2374bd4e7e3SJason Molenda 238840f12cfSJason Molenda if (m_comm.RemoteIsEFI ()) 239840f12cfSJason Molenda { 240840f12cfSJason Molenda m_dyld_plugin_name = DynamicLoaderStatic::GetPluginNameStatic(); 241840f12cfSJason Molenda } 242840f12cfSJason Molenda else if (kernel_load_addr != LLDB_INVALID_ADDRESS) 2434bd4e7e3SJason Molenda { 2445e8534efSJason Molenda m_kernel_load_addr = kernel_load_addr; 2455e8534efSJason Molenda m_dyld_plugin_name = DynamicLoaderDarwinKernel::GetPluginNameStatic(); 2464bd4e7e3SJason Molenda } 2474bd4e7e3SJason Molenda 24897d5cf05SGreg Clayton // Set the thread ID 24997d5cf05SGreg Clayton UpdateThreadListIfNeeded (); 250a63d08c9SGreg Clayton SetID (1); 25156d9a1b3SGreg Clayton GetThreadList (); 252a63d08c9SGreg Clayton SetPrivateState (eStateStopped); 25307e66e3eSGreg Clayton StreamSP async_strm_sp(m_target.GetDebugger().GetAsyncOutputStream()); 25407e66e3eSGreg Clayton if (async_strm_sp) 25507e66e3eSGreg Clayton { 2565b88216dSGreg Clayton const char *cstr; 2575b88216dSGreg Clayton if ((cstr = m_comm.GetKernelVersion ()) != NULL) 25807e66e3eSGreg Clayton { 2595b88216dSGreg Clayton async_strm_sp->Printf ("Version: %s\n", cstr); 26007e66e3eSGreg Clayton async_strm_sp->Flush(); 26107e66e3eSGreg Clayton } 2625b88216dSGreg Clayton // if ((cstr = m_comm.GetImagePath ()) != NULL) 2635b88216dSGreg Clayton // { 2645b88216dSGreg Clayton // async_strm_sp->Printf ("Image Path: %s\n", cstr); 2655b88216dSGreg Clayton // async_strm_sp->Flush(); 2665b88216dSGreg Clayton // } 26707e66e3eSGreg Clayton } 2683a29bdbeSGreg Clayton } 26997d5cf05SGreg Clayton else 27097d5cf05SGreg Clayton { 27197d5cf05SGreg Clayton error.SetErrorString("KDP_REATTACH failed"); 27297d5cf05SGreg Clayton } 2733a29bdbeSGreg Clayton } 2743a29bdbeSGreg Clayton else 2753a29bdbeSGreg Clayton { 27697d5cf05SGreg Clayton error.SetErrorString("KDP_REATTACH failed"); 2773a29bdbeSGreg Clayton } 2783a29bdbeSGreg Clayton } 2793a29bdbeSGreg Clayton else 2803a29bdbeSGreg Clayton { 2813a29bdbeSGreg Clayton error.SetErrorString("invalid reply port from UDP connection"); 2823a29bdbeSGreg Clayton } 2833a29bdbeSGreg Clayton } 2843a29bdbeSGreg Clayton else 2853a29bdbeSGreg Clayton { 2863a29bdbeSGreg Clayton if (error.Success()) 2873a29bdbeSGreg Clayton error.SetErrorStringWithFormat ("failed to connect to '%s'", remote_url); 2883a29bdbeSGreg Clayton } 2893a29bdbeSGreg Clayton if (error.Fail()) 2903a29bdbeSGreg Clayton m_comm.Disconnect(); 2913a29bdbeSGreg Clayton 292f9765acdSGreg Clayton return error; 293f9765acdSGreg Clayton } 294f9765acdSGreg Clayton 295f9765acdSGreg Clayton //---------------------------------------------------------------------- 296f9765acdSGreg Clayton // Process Control 297f9765acdSGreg Clayton //---------------------------------------------------------------------- 298f9765acdSGreg Clayton Error 299982c9762SGreg Clayton ProcessKDP::DoLaunch (Module *exe_module, 300982c9762SGreg Clayton const ProcessLaunchInfo &launch_info) 301f9765acdSGreg Clayton { 302f9765acdSGreg Clayton Error error; 303f9765acdSGreg Clayton error.SetErrorString ("launching not supported in kdp-remote plug-in"); 304f9765acdSGreg Clayton return error; 305f9765acdSGreg Clayton } 306f9765acdSGreg Clayton 307f9765acdSGreg Clayton 308f9765acdSGreg Clayton Error 309f9765acdSGreg Clayton ProcessKDP::DoAttachToProcessWithID (lldb::pid_t attach_pid) 310f9765acdSGreg Clayton { 311f9765acdSGreg Clayton Error error; 312f9765acdSGreg Clayton error.SetErrorString ("attach to process by ID is not suppported in kdp remote debugging"); 313f9765acdSGreg Clayton return error; 314f9765acdSGreg Clayton } 315f9765acdSGreg Clayton 316f9765acdSGreg Clayton Error 31784647048SHan Ming Ong ProcessKDP::DoAttachToProcessWithID (lldb::pid_t attach_pid, const ProcessAttachInfo &attach_info) 31884647048SHan Ming Ong { 31984647048SHan Ming Ong Error error; 32084647048SHan Ming Ong error.SetErrorString ("attach to process by ID is not suppported in kdp remote debugging"); 32184647048SHan Ming Ong return error; 32284647048SHan Ming Ong } 32384647048SHan Ming Ong 32484647048SHan Ming Ong Error 32584647048SHan Ming Ong ProcessKDP::DoAttachToProcessWithName (const char *process_name, bool wait_for_launch, const ProcessAttachInfo &attach_info) 326f9765acdSGreg Clayton { 327f9765acdSGreg Clayton Error error; 328f9765acdSGreg Clayton error.SetErrorString ("attach to process by name is not suppported in kdp remote debugging"); 329f9765acdSGreg Clayton return error; 330f9765acdSGreg Clayton } 331f9765acdSGreg Clayton 332f9765acdSGreg Clayton 333f9765acdSGreg Clayton void 334f9765acdSGreg Clayton ProcessKDP::DidAttach () 335f9765acdSGreg Clayton { 3365160ce5cSGreg Clayton Log *log (ProcessKDPLog::GetLogIfAllCategoriesSet (KDP_LOG_PROCESS)); 337f9765acdSGreg Clayton if (log) 33854cb8f83SJohnny Chen log->Printf ("ProcessKDP::DidAttach()"); 339f9765acdSGreg Clayton if (GetID() != LLDB_INVALID_PROCESS_ID) 340f9765acdSGreg Clayton { 341f9765acdSGreg Clayton // TODO: figure out the register context that we will use 342f9765acdSGreg Clayton } 343f9765acdSGreg Clayton } 344f9765acdSGreg Clayton 3455e8534efSJason Molenda addr_t 3465e8534efSJason Molenda ProcessKDP::GetImageInfoAddress() 3475e8534efSJason Molenda { 3485e8534efSJason Molenda return m_kernel_load_addr; 3495e8534efSJason Molenda } 3505e8534efSJason Molenda 3515e8534efSJason Molenda lldb_private::DynamicLoader * 3525e8534efSJason Molenda ProcessKDP::GetDynamicLoader () 3535e8534efSJason Molenda { 3545e8534efSJason Molenda if (m_dyld_ap.get() == NULL) 3555e8534efSJason Molenda m_dyld_ap.reset (DynamicLoader::FindPlugin(this, m_dyld_plugin_name.empty() ? NULL : m_dyld_plugin_name.c_str())); 3565e8534efSJason Molenda return m_dyld_ap.get(); 3575e8534efSJason Molenda } 3585e8534efSJason Molenda 359f9765acdSGreg Clayton Error 360f9765acdSGreg Clayton ProcessKDP::WillResume () 361f9765acdSGreg Clayton { 362f9765acdSGreg Clayton return Error(); 363f9765acdSGreg Clayton } 364f9765acdSGreg Clayton 365f9765acdSGreg Clayton Error 366f9765acdSGreg Clayton ProcessKDP::DoResume () 367f9765acdSGreg Clayton { 368f9765acdSGreg Clayton Error error; 3695160ce5cSGreg Clayton Log *log (ProcessKDPLog::GetLogIfAllCategoriesSet (KDP_LOG_PROCESS)); 3707925fbbaSGreg Clayton // Only start the async thread if we try to do any process control 3717925fbbaSGreg Clayton if (!IS_VALID_LLDB_HOST_THREAD(m_async_thread)) 3727925fbbaSGreg Clayton StartAsyncThread (); 3737925fbbaSGreg Clayton 37497d5cf05SGreg Clayton bool resume = false; 3757925fbbaSGreg Clayton 37697d5cf05SGreg Clayton // With KDP there is only one thread we can tell what to do 37797d5cf05SGreg Clayton ThreadSP kernel_thread_sp (GetKernelThread(m_thread_list, m_thread_list)); 37897d5cf05SGreg Clayton if (kernel_thread_sp) 3794b1b8b3eSGreg Clayton { 38097d5cf05SGreg Clayton const StateType thread_resume_state = kernel_thread_sp->GetTemporaryResumeState(); 3817925fbbaSGreg Clayton switch (thread_resume_state) 3824b1b8b3eSGreg Clayton { 3837925fbbaSGreg Clayton case eStateSuspended: 3847925fbbaSGreg Clayton // Nothing to do here when a thread will stay suspended 3857925fbbaSGreg Clayton // we just leave the CPU mask bit set to zero for the thread 3867925fbbaSGreg Clayton break; 3877925fbbaSGreg Clayton 3887925fbbaSGreg Clayton case eStateStepping: 3891afa68edSGreg Clayton { 3901afa68edSGreg Clayton lldb::RegisterContextSP reg_ctx_sp (kernel_thread_sp->GetRegisterContext()); 3911afa68edSGreg Clayton 3921afa68edSGreg Clayton if (reg_ctx_sp) 3931afa68edSGreg Clayton { 3941afa68edSGreg Clayton reg_ctx_sp->HardwareSingleStep (true); 39597d5cf05SGreg Clayton resume = true; 3961afa68edSGreg Clayton } 3971afa68edSGreg Clayton else 3981afa68edSGreg Clayton { 3991afa68edSGreg Clayton error.SetErrorStringWithFormat("KDP thread 0x%llx has no register context", kernel_thread_sp->GetID()); 4001afa68edSGreg Clayton } 4011afa68edSGreg Clayton } 4027925fbbaSGreg Clayton break; 4037925fbbaSGreg Clayton 40497d5cf05SGreg Clayton case eStateRunning: 4051afa68edSGreg Clayton { 4061afa68edSGreg Clayton lldb::RegisterContextSP reg_ctx_sp (kernel_thread_sp->GetRegisterContext()); 4071afa68edSGreg Clayton 4081afa68edSGreg Clayton if (reg_ctx_sp) 4091afa68edSGreg Clayton { 4101afa68edSGreg Clayton reg_ctx_sp->HardwareSingleStep (false); 41197d5cf05SGreg Clayton resume = true; 4121afa68edSGreg Clayton } 4131afa68edSGreg Clayton else 4141afa68edSGreg Clayton { 4151afa68edSGreg Clayton error.SetErrorStringWithFormat("KDP thread 0x%llx has no register context", kernel_thread_sp->GetID()); 4161afa68edSGreg Clayton } 4171afa68edSGreg Clayton } 4187925fbbaSGreg Clayton break; 4197925fbbaSGreg Clayton 4207925fbbaSGreg Clayton default: 42197d5cf05SGreg Clayton // The only valid thread resume states are listed above 4227925fbbaSGreg Clayton assert (!"invalid thread resume state"); 4237925fbbaSGreg Clayton break; 4244b1b8b3eSGreg Clayton } 4254b1b8b3eSGreg Clayton } 4267925fbbaSGreg Clayton 42797d5cf05SGreg Clayton if (resume) 42897d5cf05SGreg Clayton { 42997d5cf05SGreg Clayton if (log) 43097d5cf05SGreg Clayton log->Printf ("ProcessKDP::DoResume () sending resume"); 43197d5cf05SGreg Clayton 43297d5cf05SGreg Clayton if (m_comm.SendRequestResume ()) 4337925fbbaSGreg Clayton { 4347925fbbaSGreg Clayton m_async_broadcaster.BroadcastEvent (eBroadcastBitAsyncContinue); 4357925fbbaSGreg Clayton SetPrivateState(eStateRunning); 4367925fbbaSGreg Clayton } 4374b1b8b3eSGreg Clayton else 43807e66e3eSGreg Clayton error.SetErrorString ("KDP resume failed"); 4397925fbbaSGreg Clayton } 4407925fbbaSGreg Clayton else 4417925fbbaSGreg Clayton { 44297d5cf05SGreg Clayton error.SetErrorString ("kernel thread is suspended"); 4437925fbbaSGreg Clayton } 4447925fbbaSGreg Clayton 445f9765acdSGreg Clayton return error; 446f9765acdSGreg Clayton } 447f9765acdSGreg Clayton 44897d5cf05SGreg Clayton lldb::ThreadSP 44997d5cf05SGreg Clayton ProcessKDP::GetKernelThread(ThreadList &old_thread_list, ThreadList &new_thread_list) 45097d5cf05SGreg Clayton { 45197d5cf05SGreg Clayton // KDP only tells us about one thread/core. Any other threads will usually 45297d5cf05SGreg Clayton // be the ones that are read from memory by the OS plug-ins. 45397d5cf05SGreg Clayton const lldb::tid_t kernel_tid = 1; 45497d5cf05SGreg Clayton ThreadSP thread_sp (old_thread_list.FindThreadByID (kernel_tid, false)); 45597d5cf05SGreg Clayton if (!thread_sp) 4564f465cffSJim Ingham thread_sp.reset(new ThreadKDP (*this, kernel_tid)); 45797d5cf05SGreg Clayton new_thread_list.AddThread(thread_sp); 45897d5cf05SGreg Clayton return thread_sp; 45997d5cf05SGreg Clayton } 46097d5cf05SGreg Clayton 46197d5cf05SGreg Clayton 46297d5cf05SGreg Clayton 46397d5cf05SGreg Clayton 4649fc13556SGreg Clayton bool 46556d9a1b3SGreg Clayton ProcessKDP::UpdateThreadList (ThreadList &old_thread_list, ThreadList &new_thread_list) 466f9765acdSGreg Clayton { 467f9765acdSGreg Clayton // locker will keep a mutex locked until it goes out of scope 4685160ce5cSGreg Clayton Log *log (ProcessKDPLog::GetLogIfAllCategoriesSet (KDP_LOG_THREAD)); 469f9765acdSGreg Clayton if (log && log->GetMask().Test(KDP_LOG_VERBOSE)) 470d01b2953SDaniel Malea log->Printf ("ProcessKDP::%s (pid = %" PRIu64 ")", __FUNCTION__, GetID()); 471f9765acdSGreg Clayton 472*39da3efdSGreg Clayton // Even though there is a CPU mask, it doesn't mean we can see each CPU 47397d5cf05SGreg Clayton // indivudually, there is really only one. Lets call this thread 1. 47497d5cf05SGreg Clayton GetKernelThread (old_thread_list, new_thread_list); 47597d5cf05SGreg Clayton 4769fc13556SGreg Clayton return new_thread_list.GetSize(false) > 0; 477f9765acdSGreg Clayton } 478f9765acdSGreg Clayton 479f9765acdSGreg Clayton void 480f9765acdSGreg Clayton ProcessKDP::RefreshStateAfterStop () 481f9765acdSGreg Clayton { 482f9765acdSGreg Clayton // Let all threads recover from stopping and do any clean up based 483f9765acdSGreg Clayton // on the previous thread state (if any). 484f9765acdSGreg Clayton m_thread_list.RefreshStateAfterStop(); 485f9765acdSGreg Clayton } 486f9765acdSGreg Clayton 487f9765acdSGreg Clayton Error 488f9765acdSGreg Clayton ProcessKDP::DoHalt (bool &caused_stop) 489f9765acdSGreg Clayton { 490f9765acdSGreg Clayton Error error; 491f9765acdSGreg Clayton 49297d5cf05SGreg Clayton if (m_comm.IsRunning()) 493f9765acdSGreg Clayton { 49497d5cf05SGreg Clayton if (m_destroy_in_process) 49597d5cf05SGreg Clayton { 49697d5cf05SGreg Clayton // If we are attemping to destroy, we need to not return an error to 49797d5cf05SGreg Clayton // Halt or DoDestroy won't get called. 49897d5cf05SGreg Clayton // We are also currently running, so send a process stopped event 49997d5cf05SGreg Clayton SetPrivateState (eStateStopped); 500f9765acdSGreg Clayton } 501f9765acdSGreg Clayton else 502f9765acdSGreg Clayton { 50397d5cf05SGreg Clayton error.SetErrorString ("KDP cannot interrupt a running kernel"); 504f9765acdSGreg Clayton } 505f9765acdSGreg Clayton } 506f9765acdSGreg Clayton return error; 507f9765acdSGreg Clayton } 508f9765acdSGreg Clayton 509f9765acdSGreg Clayton Error 510f9765acdSGreg Clayton ProcessKDP::DoDetach() 511f9765acdSGreg Clayton { 512f9765acdSGreg Clayton Error error; 5135160ce5cSGreg Clayton Log *log (ProcessKDPLog::GetLogIfAllCategoriesSet(KDP_LOG_PROCESS)); 514f9765acdSGreg Clayton if (log) 515f9765acdSGreg Clayton log->Printf ("ProcessKDP::DoDetach()"); 516f9765acdSGreg Clayton 51797d5cf05SGreg Clayton if (m_comm.IsRunning()) 51897d5cf05SGreg Clayton { 51997d5cf05SGreg Clayton // We are running and we can't interrupt a running kernel, so we need 52097d5cf05SGreg Clayton // to just close the connection to the kernel and hope for the best 52197d5cf05SGreg Clayton } 52297d5cf05SGreg Clayton else 52397d5cf05SGreg Clayton { 524f9765acdSGreg Clayton DisableAllBreakpointSites (); 525f9765acdSGreg Clayton 526f9765acdSGreg Clayton m_thread_list.DiscardThreadPlans(); 527f9765acdSGreg Clayton 5283a29bdbeSGreg Clayton if (m_comm.IsConnected()) 5293a29bdbeSGreg Clayton { 5303a29bdbeSGreg Clayton 5313a29bdbeSGreg Clayton m_comm.SendRequestDisconnect(); 5323a29bdbeSGreg Clayton 53357508026SGreg Clayton size_t response_size = m_comm.Disconnect (); 534f9765acdSGreg Clayton if (log) 535f9765acdSGreg Clayton { 536f9765acdSGreg Clayton if (response_size) 537f9765acdSGreg Clayton log->PutCString ("ProcessKDP::DoDetach() detach packet sent successfully"); 538f9765acdSGreg Clayton else 539f9765acdSGreg Clayton log->PutCString ("ProcessKDP::DoDetach() detach packet send failed"); 540f9765acdSGreg Clayton } 5413a29bdbeSGreg Clayton } 54297d5cf05SGreg Clayton } 543f9765acdSGreg Clayton StopAsyncThread (); 54474d4193eSGreg Clayton m_comm.Clear(); 545f9765acdSGreg Clayton 546f9765acdSGreg Clayton SetPrivateState (eStateDetached); 547f9765acdSGreg Clayton ResumePrivateStateThread(); 548f9765acdSGreg Clayton 549f9765acdSGreg Clayton //KillDebugserverProcess (); 550f9765acdSGreg Clayton return error; 551f9765acdSGreg Clayton } 552f9765acdSGreg Clayton 553f9765acdSGreg Clayton Error 554f9765acdSGreg Clayton ProcessKDP::DoDestroy () 555f9765acdSGreg Clayton { 5567925fbbaSGreg Clayton // For KDP there really is no difference between destroy and detach 5577925fbbaSGreg Clayton return DoDetach(); 558f9765acdSGreg Clayton } 559f9765acdSGreg Clayton 560f9765acdSGreg Clayton //------------------------------------------------------------------ 561f9765acdSGreg Clayton // Process Queries 562f9765acdSGreg Clayton //------------------------------------------------------------------ 563f9765acdSGreg Clayton 564f9765acdSGreg Clayton bool 565f9765acdSGreg Clayton ProcessKDP::IsAlive () 566f9765acdSGreg Clayton { 567f9765acdSGreg Clayton return m_comm.IsConnected() && m_private_state.GetValue() != eStateExited; 568f9765acdSGreg Clayton } 569f9765acdSGreg Clayton 570f9765acdSGreg Clayton //------------------------------------------------------------------ 571f9765acdSGreg Clayton // Process Memory 572f9765acdSGreg Clayton //------------------------------------------------------------------ 573f9765acdSGreg Clayton size_t 574f9765acdSGreg Clayton ProcessKDP::DoReadMemory (addr_t addr, void *buf, size_t size, Error &error) 575f9765acdSGreg Clayton { 576a63d08c9SGreg Clayton if (m_comm.IsConnected()) 577a63d08c9SGreg Clayton return m_comm.SendRequestReadMemory (addr, buf, size, error); 578a63d08c9SGreg Clayton error.SetErrorString ("not connected"); 579f9765acdSGreg Clayton return 0; 580f9765acdSGreg Clayton } 581f9765acdSGreg Clayton 582f9765acdSGreg Clayton size_t 583f9765acdSGreg Clayton ProcessKDP::DoWriteMemory (addr_t addr, const void *buf, size_t size, Error &error) 584f9765acdSGreg Clayton { 5857925fbbaSGreg Clayton if (m_comm.IsConnected()) 5867925fbbaSGreg Clayton return m_comm.SendRequestWriteMemory (addr, buf, size, error); 5877925fbbaSGreg Clayton error.SetErrorString ("not connected"); 588f9765acdSGreg Clayton return 0; 589f9765acdSGreg Clayton } 590f9765acdSGreg Clayton 591f9765acdSGreg Clayton lldb::addr_t 592f9765acdSGreg Clayton ProcessKDP::DoAllocateMemory (size_t size, uint32_t permissions, Error &error) 593f9765acdSGreg Clayton { 594f9765acdSGreg Clayton error.SetErrorString ("memory allocation not suppported in kdp remote debugging"); 595f9765acdSGreg Clayton return LLDB_INVALID_ADDRESS; 596f9765acdSGreg Clayton } 597f9765acdSGreg Clayton 598f9765acdSGreg Clayton Error 599f9765acdSGreg Clayton ProcessKDP::DoDeallocateMemory (lldb::addr_t addr) 600f9765acdSGreg Clayton { 601f9765acdSGreg Clayton Error error; 602f9765acdSGreg Clayton error.SetErrorString ("memory deallocation not suppported in kdp remote debugging"); 603f9765acdSGreg Clayton return error; 604f9765acdSGreg Clayton } 605f9765acdSGreg Clayton 606f9765acdSGreg Clayton Error 607299c0c1cSJim Ingham ProcessKDP::EnableBreakpointSite (BreakpointSite *bp_site) 608f9765acdSGreg Clayton { 60907e66e3eSGreg Clayton if (m_comm.LocalBreakpointsAreSupported ()) 61007e66e3eSGreg Clayton { 61107e66e3eSGreg Clayton Error error; 6125b88216dSGreg Clayton if (!bp_site->IsEnabled()) 6135b88216dSGreg Clayton { 6145b88216dSGreg Clayton if (m_comm.SendRequestBreakpoint(true, bp_site->GetLoadAddress())) 6155b88216dSGreg Clayton { 6165b88216dSGreg Clayton bp_site->SetEnabled(true); 6175b88216dSGreg Clayton bp_site->SetType (BreakpointSite::eExternal); 6185b88216dSGreg Clayton } 6195b88216dSGreg Clayton else 6205b88216dSGreg Clayton { 62107e66e3eSGreg Clayton error.SetErrorString ("KDP set breakpoint failed"); 6225b88216dSGreg Clayton } 6235b88216dSGreg Clayton } 62407e66e3eSGreg Clayton return error; 62507e66e3eSGreg Clayton } 626f9765acdSGreg Clayton return EnableSoftwareBreakpoint (bp_site); 627f9765acdSGreg Clayton } 628f9765acdSGreg Clayton 629f9765acdSGreg Clayton Error 630299c0c1cSJim Ingham ProcessKDP::DisableBreakpointSite (BreakpointSite *bp_site) 631f9765acdSGreg Clayton { 63207e66e3eSGreg Clayton if (m_comm.LocalBreakpointsAreSupported ()) 63307e66e3eSGreg Clayton { 63407e66e3eSGreg Clayton Error error; 6355b88216dSGreg Clayton if (bp_site->IsEnabled()) 6365b88216dSGreg Clayton { 6375b88216dSGreg Clayton BreakpointSite::Type bp_type = bp_site->GetType(); 6385b88216dSGreg Clayton if (bp_type == BreakpointSite::eExternal) 6395b88216dSGreg Clayton { 64097d5cf05SGreg Clayton if (m_destroy_in_process && m_comm.IsRunning()) 64197d5cf05SGreg Clayton { 64297d5cf05SGreg Clayton // We are trying to destroy our connection and we are running 64397d5cf05SGreg Clayton bp_site->SetEnabled(false); 64497d5cf05SGreg Clayton } 64597d5cf05SGreg Clayton else 64697d5cf05SGreg Clayton { 6475b88216dSGreg Clayton if (m_comm.SendRequestBreakpoint(false, bp_site->GetLoadAddress())) 6485b88216dSGreg Clayton bp_site->SetEnabled(false); 6495b88216dSGreg Clayton else 65007e66e3eSGreg Clayton error.SetErrorString ("KDP remove breakpoint failed"); 6515b88216dSGreg Clayton } 65297d5cf05SGreg Clayton } 6535b88216dSGreg Clayton else 6545b88216dSGreg Clayton { 6555b88216dSGreg Clayton error = DisableSoftwareBreakpoint (bp_site); 6565b88216dSGreg Clayton } 6575b88216dSGreg Clayton } 65807e66e3eSGreg Clayton return error; 65907e66e3eSGreg Clayton } 660f9765acdSGreg Clayton return DisableSoftwareBreakpoint (bp_site); 661f9765acdSGreg Clayton } 662f9765acdSGreg Clayton 663f9765acdSGreg Clayton Error 6641b5792e5SJim Ingham ProcessKDP::EnableWatchpoint (Watchpoint *wp, bool notify) 665f9765acdSGreg Clayton { 666f9765acdSGreg Clayton Error error; 667f9765acdSGreg Clayton error.SetErrorString ("watchpoints are not suppported in kdp remote debugging"); 668f9765acdSGreg Clayton return error; 669f9765acdSGreg Clayton } 670f9765acdSGreg Clayton 671f9765acdSGreg Clayton Error 6721b5792e5SJim Ingham ProcessKDP::DisableWatchpoint (Watchpoint *wp, bool notify) 673f9765acdSGreg Clayton { 674f9765acdSGreg Clayton Error error; 675f9765acdSGreg Clayton error.SetErrorString ("watchpoints are not suppported in kdp remote debugging"); 676f9765acdSGreg Clayton return error; 677f9765acdSGreg Clayton } 678f9765acdSGreg Clayton 679f9765acdSGreg Clayton void 680f9765acdSGreg Clayton ProcessKDP::Clear() 681f9765acdSGreg Clayton { 682f9765acdSGreg Clayton m_thread_list.Clear(); 683f9765acdSGreg Clayton } 684f9765acdSGreg Clayton 685f9765acdSGreg Clayton Error 686f9765acdSGreg Clayton ProcessKDP::DoSignal (int signo) 687f9765acdSGreg Clayton { 688f9765acdSGreg Clayton Error error; 689f9765acdSGreg Clayton error.SetErrorString ("sending signals is not suppported in kdp remote debugging"); 690f9765acdSGreg Clayton return error; 691f9765acdSGreg Clayton } 692f9765acdSGreg Clayton 693f9765acdSGreg Clayton void 694f9765acdSGreg Clayton ProcessKDP::Initialize() 695f9765acdSGreg Clayton { 696f9765acdSGreg Clayton static bool g_initialized = false; 697f9765acdSGreg Clayton 698f9765acdSGreg Clayton if (g_initialized == false) 699f9765acdSGreg Clayton { 700f9765acdSGreg Clayton g_initialized = true; 701f9765acdSGreg Clayton PluginManager::RegisterPlugin (GetPluginNameStatic(), 702f9765acdSGreg Clayton GetPluginDescriptionStatic(), 703f9765acdSGreg Clayton CreateInstance); 704f9765acdSGreg Clayton 705f9765acdSGreg Clayton Log::Callbacks log_callbacks = { 706f9765acdSGreg Clayton ProcessKDPLog::DisableLog, 707f9765acdSGreg Clayton ProcessKDPLog::EnableLog, 708f9765acdSGreg Clayton ProcessKDPLog::ListLogCategories 709f9765acdSGreg Clayton }; 710f9765acdSGreg Clayton 711f9765acdSGreg Clayton Log::RegisterLogChannel (ProcessKDP::GetPluginNameStatic(), log_callbacks); 712f9765acdSGreg Clayton } 713f9765acdSGreg Clayton } 714f9765acdSGreg Clayton 715f9765acdSGreg Clayton bool 716f9765acdSGreg Clayton ProcessKDP::StartAsyncThread () 717f9765acdSGreg Clayton { 7185160ce5cSGreg Clayton Log *log (ProcessKDPLog::GetLogIfAllCategoriesSet(KDP_LOG_PROCESS)); 719f9765acdSGreg Clayton 720f9765acdSGreg Clayton if (log) 7217925fbbaSGreg Clayton log->Printf ("ProcessKDP::StartAsyncThread ()"); 722f9765acdSGreg Clayton 7237925fbbaSGreg Clayton if (IS_VALID_LLDB_HOST_THREAD(m_async_thread)) 7247925fbbaSGreg Clayton return true; 7257925fbbaSGreg Clayton 726f9765acdSGreg Clayton m_async_thread = Host::ThreadCreate ("<lldb.process.kdp-remote.async>", ProcessKDP::AsyncThread, this, NULL); 727f9765acdSGreg Clayton return IS_VALID_LLDB_HOST_THREAD(m_async_thread); 728f9765acdSGreg Clayton } 729f9765acdSGreg Clayton 730f9765acdSGreg Clayton void 731f9765acdSGreg Clayton ProcessKDP::StopAsyncThread () 732f9765acdSGreg Clayton { 7335160ce5cSGreg Clayton Log *log (ProcessKDPLog::GetLogIfAllCategoriesSet(KDP_LOG_PROCESS)); 734f9765acdSGreg Clayton 735f9765acdSGreg Clayton if (log) 7367925fbbaSGreg Clayton log->Printf ("ProcessKDP::StopAsyncThread ()"); 737f9765acdSGreg Clayton 738f9765acdSGreg Clayton m_async_broadcaster.BroadcastEvent (eBroadcastBitAsyncThreadShouldExit); 739f9765acdSGreg Clayton 740f9765acdSGreg Clayton // Stop the stdio thread 741f9765acdSGreg Clayton if (IS_VALID_LLDB_HOST_THREAD(m_async_thread)) 742f9765acdSGreg Clayton { 743f9765acdSGreg Clayton Host::ThreadJoin (m_async_thread, NULL, NULL); 7447925fbbaSGreg Clayton m_async_thread = LLDB_INVALID_HOST_THREAD; 745f9765acdSGreg Clayton } 746f9765acdSGreg Clayton } 747f9765acdSGreg Clayton 748f9765acdSGreg Clayton 749f9765acdSGreg Clayton void * 750f9765acdSGreg Clayton ProcessKDP::AsyncThread (void *arg) 751f9765acdSGreg Clayton { 752f9765acdSGreg Clayton ProcessKDP *process = (ProcessKDP*) arg; 753f9765acdSGreg Clayton 7547925fbbaSGreg Clayton const lldb::pid_t pid = process->GetID(); 7557925fbbaSGreg Clayton 7565160ce5cSGreg Clayton Log *log (ProcessKDPLog::GetLogIfAllCategoriesSet (KDP_LOG_PROCESS)); 757f9765acdSGreg Clayton if (log) 758d01b2953SDaniel Malea log->Printf ("ProcessKDP::AsyncThread (arg = %p, pid = %" PRIu64 ") thread starting...", arg, pid); 759f9765acdSGreg Clayton 760f9765acdSGreg Clayton Listener listener ("ProcessKDP::AsyncThread"); 761f9765acdSGreg Clayton EventSP event_sp; 762f9765acdSGreg Clayton const uint32_t desired_event_mask = eBroadcastBitAsyncContinue | 763f9765acdSGreg Clayton eBroadcastBitAsyncThreadShouldExit; 764f9765acdSGreg Clayton 7657925fbbaSGreg Clayton 766f9765acdSGreg Clayton if (listener.StartListeningForEvents (&process->m_async_broadcaster, desired_event_mask) == desired_event_mask) 767f9765acdSGreg Clayton { 768f9765acdSGreg Clayton bool done = false; 769f9765acdSGreg Clayton while (!done) 770f9765acdSGreg Clayton { 771f9765acdSGreg Clayton if (log) 772d01b2953SDaniel Malea log->Printf ("ProcessKDP::AsyncThread (pid = %" PRIu64 ") listener.WaitForEvent (NULL, event_sp)...", 7737925fbbaSGreg Clayton pid); 774f9765acdSGreg Clayton if (listener.WaitForEvent (NULL, event_sp)) 775f9765acdSGreg Clayton { 7767925fbbaSGreg Clayton uint32_t event_type = event_sp->GetType(); 777f9765acdSGreg Clayton if (log) 778d01b2953SDaniel Malea log->Printf ("ProcessKDP::AsyncThread (pid = %" PRIu64 ") Got an event of type: %d...", 7797925fbbaSGreg Clayton pid, 7807925fbbaSGreg Clayton event_type); 781f9765acdSGreg Clayton 7827925fbbaSGreg Clayton // When we are running, poll for 1 second to try and get an exception 7837925fbbaSGreg Clayton // to indicate the process has stopped. If we don't get one, check to 7847925fbbaSGreg Clayton // make sure no one asked us to exit 7857925fbbaSGreg Clayton bool is_running = false; 7867925fbbaSGreg Clayton DataExtractor exc_reply_packet; 7877925fbbaSGreg Clayton do 7887925fbbaSGreg Clayton { 789f9765acdSGreg Clayton switch (event_type) 790f9765acdSGreg Clayton { 791f9765acdSGreg Clayton case eBroadcastBitAsyncContinue: 792f9765acdSGreg Clayton { 7937925fbbaSGreg Clayton is_running = true; 7947925fbbaSGreg Clayton if (process->m_comm.WaitForPacketWithTimeoutMicroSeconds (exc_reply_packet, 1 * USEC_PER_SEC)) 795f9765acdSGreg Clayton { 79697d5cf05SGreg Clayton ThreadSP thread_sp (process->GetKernelThread(process->GetThreadList(), process->GetThreadList())); 7971afa68edSGreg Clayton if (thread_sp) 7981afa68edSGreg Clayton { 7991afa68edSGreg Clayton lldb::RegisterContextSP reg_ctx_sp (thread_sp->GetRegisterContext()); 8001afa68edSGreg Clayton if (reg_ctx_sp) 8011afa68edSGreg Clayton reg_ctx_sp->InvalidateAllRegisters(); 80297d5cf05SGreg Clayton static_cast<ThreadKDP *>(thread_sp.get())->SetStopInfoFrom_KDP_EXCEPTION (exc_reply_packet); 8031afa68edSGreg Clayton } 80497d5cf05SGreg Clayton 8057925fbbaSGreg Clayton // TODO: parse the stop reply packet 8067925fbbaSGreg Clayton is_running = false; 8077925fbbaSGreg Clayton process->SetPrivateState(eStateStopped); 8087925fbbaSGreg Clayton } 8097925fbbaSGreg Clayton else 8107925fbbaSGreg Clayton { 8117925fbbaSGreg Clayton // Check to see if we are supposed to exit. There is no way to 8127925fbbaSGreg Clayton // interrupt a running kernel, so all we can do is wait for an 8137925fbbaSGreg Clayton // exception or detach... 8147925fbbaSGreg Clayton if (listener.GetNextEvent(event_sp)) 8157925fbbaSGreg Clayton { 8167925fbbaSGreg Clayton // We got an event, go through the loop again 8177925fbbaSGreg Clayton event_type = event_sp->GetType(); 8187925fbbaSGreg Clayton } 819f9765acdSGreg Clayton } 820f9765acdSGreg Clayton } 821f9765acdSGreg Clayton break; 822f9765acdSGreg Clayton 823f9765acdSGreg Clayton case eBroadcastBitAsyncThreadShouldExit: 824f9765acdSGreg Clayton if (log) 825d01b2953SDaniel Malea log->Printf ("ProcessKDP::AsyncThread (pid = %" PRIu64 ") got eBroadcastBitAsyncThreadShouldExit...", 8267925fbbaSGreg Clayton pid); 827f9765acdSGreg Clayton done = true; 8287925fbbaSGreg Clayton is_running = false; 829f9765acdSGreg Clayton break; 830f9765acdSGreg Clayton 831f9765acdSGreg Clayton default: 832f9765acdSGreg Clayton if (log) 833d01b2953SDaniel Malea log->Printf ("ProcessKDP::AsyncThread (pid = %" PRIu64 ") got unknown event 0x%8.8x", 8347925fbbaSGreg Clayton pid, 8357925fbbaSGreg Clayton event_type); 836f9765acdSGreg Clayton done = true; 8377925fbbaSGreg Clayton is_running = false; 838f9765acdSGreg Clayton break; 839f9765acdSGreg Clayton } 8407925fbbaSGreg Clayton } while (is_running); 841f9765acdSGreg Clayton } 842f9765acdSGreg Clayton else 843f9765acdSGreg Clayton { 844f9765acdSGreg Clayton if (log) 845d01b2953SDaniel Malea log->Printf ("ProcessKDP::AsyncThread (pid = %" PRIu64 ") listener.WaitForEvent (NULL, event_sp) => false", 8467925fbbaSGreg Clayton pid); 847f9765acdSGreg Clayton done = true; 848f9765acdSGreg Clayton } 849f9765acdSGreg Clayton } 850f9765acdSGreg Clayton } 851f9765acdSGreg Clayton 852f9765acdSGreg Clayton if (log) 853d01b2953SDaniel Malea log->Printf ("ProcessKDP::AsyncThread (arg = %p, pid = %" PRIu64 ") thread exiting...", 8547925fbbaSGreg Clayton arg, 8557925fbbaSGreg Clayton pid); 856f9765acdSGreg Clayton 857f9765acdSGreg Clayton process->m_async_thread = LLDB_INVALID_HOST_THREAD; 858f9765acdSGreg Clayton return NULL; 859f9765acdSGreg Clayton } 860f9765acdSGreg Clayton 861f9765acdSGreg Clayton 8621d19a2f2SGreg Clayton class CommandObjectProcessKDPPacketSend : public CommandObjectParsed 8631d19a2f2SGreg Clayton { 8641d19a2f2SGreg Clayton private: 8651d19a2f2SGreg Clayton 8661d19a2f2SGreg Clayton OptionGroupOptions m_option_group; 8671d19a2f2SGreg Clayton OptionGroupUInt64 m_command_byte; 8681d19a2f2SGreg Clayton OptionGroupString m_packet_data; 8691d19a2f2SGreg Clayton 8701d19a2f2SGreg Clayton virtual Options * 8711d19a2f2SGreg Clayton GetOptions () 8721d19a2f2SGreg Clayton { 8731d19a2f2SGreg Clayton return &m_option_group; 8741d19a2f2SGreg Clayton } 8751d19a2f2SGreg Clayton 8761d19a2f2SGreg Clayton 8771d19a2f2SGreg Clayton public: 8781d19a2f2SGreg Clayton CommandObjectProcessKDPPacketSend(CommandInterpreter &interpreter) : 8791d19a2f2SGreg Clayton CommandObjectParsed (interpreter, 8801d19a2f2SGreg Clayton "process plugin packet send", 8811d19a2f2SGreg 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. ", 8821d19a2f2SGreg Clayton NULL), 8831d19a2f2SGreg Clayton m_option_group (interpreter), 8841d19a2f2SGreg 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), 8851d19a2f2SGreg 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) 8861d19a2f2SGreg Clayton { 8871d19a2f2SGreg Clayton m_option_group.Append (&m_command_byte, LLDB_OPT_SET_ALL, LLDB_OPT_SET_1); 8881d19a2f2SGreg Clayton m_option_group.Append (&m_packet_data , LLDB_OPT_SET_ALL, LLDB_OPT_SET_1); 8891d19a2f2SGreg Clayton m_option_group.Finalize(); 8901d19a2f2SGreg Clayton } 8911d19a2f2SGreg Clayton 8921d19a2f2SGreg Clayton ~CommandObjectProcessKDPPacketSend () 8931d19a2f2SGreg Clayton { 8941d19a2f2SGreg Clayton } 8951d19a2f2SGreg Clayton 8961d19a2f2SGreg Clayton bool 8971d19a2f2SGreg Clayton DoExecute (Args& command, CommandReturnObject &result) 8981d19a2f2SGreg Clayton { 8991d19a2f2SGreg Clayton const size_t argc = command.GetArgumentCount(); 9001d19a2f2SGreg Clayton if (argc == 0) 9011d19a2f2SGreg Clayton { 9021d19a2f2SGreg Clayton if (!m_command_byte.GetOptionValue().OptionWasSet()) 9031d19a2f2SGreg Clayton { 9041d19a2f2SGreg Clayton result.AppendError ("the --command option must be set to a valid command byte"); 9051d19a2f2SGreg Clayton result.SetStatus (eReturnStatusFailed); 9061d19a2f2SGreg Clayton } 9071d19a2f2SGreg Clayton else 9081d19a2f2SGreg Clayton { 9091d19a2f2SGreg Clayton const uint64_t command_byte = m_command_byte.GetOptionValue().GetUInt64Value(0); 9101d19a2f2SGreg Clayton if (command_byte > 0 && command_byte <= UINT8_MAX) 9111d19a2f2SGreg Clayton { 9121d19a2f2SGreg Clayton ProcessKDP *process = (ProcessKDP *)m_interpreter.GetExecutionContext().GetProcessPtr(); 9131d19a2f2SGreg Clayton if (process) 9141d19a2f2SGreg Clayton { 9151d19a2f2SGreg Clayton const StateType state = process->GetState(); 9161d19a2f2SGreg Clayton 9171d19a2f2SGreg Clayton if (StateIsStoppedState (state, true)) 9181d19a2f2SGreg Clayton { 9191d19a2f2SGreg Clayton std::vector<uint8_t> payload_bytes; 9201d19a2f2SGreg Clayton const char *ascii_hex_bytes_cstr = m_packet_data.GetOptionValue().GetCurrentValue(); 9211d19a2f2SGreg Clayton if (ascii_hex_bytes_cstr && ascii_hex_bytes_cstr[0]) 9221d19a2f2SGreg Clayton { 9231d19a2f2SGreg Clayton StringExtractor extractor(ascii_hex_bytes_cstr); 9241d19a2f2SGreg Clayton const size_t ascii_hex_bytes_cstr_len = extractor.GetStringRef().size(); 9251d19a2f2SGreg Clayton if (ascii_hex_bytes_cstr_len & 1) 9261d19a2f2SGreg Clayton { 9271d19a2f2SGreg Clayton result.AppendErrorWithFormat ("payload data must contain an even number of ASCII hex characters: '%s'", ascii_hex_bytes_cstr); 9281d19a2f2SGreg Clayton result.SetStatus (eReturnStatusFailed); 9291d19a2f2SGreg Clayton return false; 9301d19a2f2SGreg Clayton } 9311d19a2f2SGreg Clayton payload_bytes.resize(ascii_hex_bytes_cstr_len/2); 9321d19a2f2SGreg Clayton if (extractor.GetHexBytes(&payload_bytes[0], payload_bytes.size(), '\xdd') != payload_bytes.size()) 9331d19a2f2SGreg Clayton { 9341d19a2f2SGreg Clayton result.AppendErrorWithFormat ("payload data must only contain ASCII hex characters (no spaces or hex prefixes): '%s'", ascii_hex_bytes_cstr); 9351d19a2f2SGreg Clayton result.SetStatus (eReturnStatusFailed); 9361d19a2f2SGreg Clayton return false; 9371d19a2f2SGreg Clayton } 9381d19a2f2SGreg Clayton } 9391d19a2f2SGreg Clayton Error error; 9401d19a2f2SGreg Clayton DataExtractor reply; 9411d19a2f2SGreg Clayton process->GetCommunication().SendRawRequest (command_byte, 9421d19a2f2SGreg Clayton payload_bytes.empty() ? NULL : payload_bytes.data(), 9431d19a2f2SGreg Clayton payload_bytes.size(), 9441d19a2f2SGreg Clayton reply, 9451d19a2f2SGreg Clayton error); 9461d19a2f2SGreg Clayton 9471d19a2f2SGreg Clayton if (error.Success()) 9481d19a2f2SGreg Clayton { 9491d19a2f2SGreg Clayton // Copy the binary bytes into a hex ASCII string for the result 9501d19a2f2SGreg Clayton StreamString packet; 9511d19a2f2SGreg Clayton packet.PutBytesAsRawHex8(reply.GetDataStart(), 9521d19a2f2SGreg Clayton reply.GetByteSize(), 9531d19a2f2SGreg Clayton lldb::endian::InlHostByteOrder(), 9541d19a2f2SGreg Clayton lldb::endian::InlHostByteOrder()); 9551d19a2f2SGreg Clayton result.AppendMessage(packet.GetString().c_str()); 9561d19a2f2SGreg Clayton result.SetStatus (eReturnStatusSuccessFinishResult); 9571d19a2f2SGreg Clayton return true; 9581d19a2f2SGreg Clayton } 9591d19a2f2SGreg Clayton else 9601d19a2f2SGreg Clayton { 9611d19a2f2SGreg Clayton const char *error_cstr = error.AsCString(); 9621d19a2f2SGreg Clayton if (error_cstr && error_cstr[0]) 9631d19a2f2SGreg Clayton result.AppendError (error_cstr); 9641d19a2f2SGreg Clayton else 9651d19a2f2SGreg Clayton result.AppendErrorWithFormat ("unknown error 0x%8.8x", error.GetError()); 9661d19a2f2SGreg Clayton result.SetStatus (eReturnStatusFailed); 9671d19a2f2SGreg Clayton return false; 9681d19a2f2SGreg Clayton } 9691d19a2f2SGreg Clayton } 9701d19a2f2SGreg Clayton else 9711d19a2f2SGreg Clayton { 9721d19a2f2SGreg Clayton result.AppendErrorWithFormat ("process must be stopped in order to send KDP packets, state is %s", StateAsCString (state)); 9731d19a2f2SGreg Clayton result.SetStatus (eReturnStatusFailed); 9741d19a2f2SGreg Clayton } 9751d19a2f2SGreg Clayton } 9761d19a2f2SGreg Clayton else 9771d19a2f2SGreg Clayton { 9781d19a2f2SGreg Clayton result.AppendError ("invalid process"); 9791d19a2f2SGreg Clayton result.SetStatus (eReturnStatusFailed); 9801d19a2f2SGreg Clayton } 9811d19a2f2SGreg Clayton } 9821d19a2f2SGreg Clayton else 9831d19a2f2SGreg Clayton { 984d01b2953SDaniel Malea result.AppendErrorWithFormat ("invalid command byte 0x%" PRIx64 ", valid values are 1 - 255", command_byte); 9851d19a2f2SGreg Clayton result.SetStatus (eReturnStatusFailed); 9861d19a2f2SGreg Clayton } 9871d19a2f2SGreg Clayton } 9881d19a2f2SGreg Clayton } 9891d19a2f2SGreg Clayton else 9901d19a2f2SGreg Clayton { 9911d19a2f2SGreg Clayton result.AppendErrorWithFormat ("'%s' takes no arguments, only options.", m_cmd_name.c_str()); 9921d19a2f2SGreg Clayton result.SetStatus (eReturnStatusFailed); 9931d19a2f2SGreg Clayton } 9941d19a2f2SGreg Clayton return false; 9951d19a2f2SGreg Clayton } 9961d19a2f2SGreg Clayton }; 9971d19a2f2SGreg Clayton 9981d19a2f2SGreg Clayton class CommandObjectProcessKDPPacket : public CommandObjectMultiword 9991d19a2f2SGreg Clayton { 10001d19a2f2SGreg Clayton private: 10011d19a2f2SGreg Clayton 10021d19a2f2SGreg Clayton public: 10031d19a2f2SGreg Clayton CommandObjectProcessKDPPacket(CommandInterpreter &interpreter) : 10041d19a2f2SGreg Clayton CommandObjectMultiword (interpreter, 10051d19a2f2SGreg Clayton "process plugin packet", 10061d19a2f2SGreg Clayton "Commands that deal with KDP remote packets.", 10071d19a2f2SGreg Clayton NULL) 10081d19a2f2SGreg Clayton { 10091d19a2f2SGreg Clayton LoadSubCommand ("send", CommandObjectSP (new CommandObjectProcessKDPPacketSend (interpreter))); 10101d19a2f2SGreg Clayton } 10111d19a2f2SGreg Clayton 10121d19a2f2SGreg Clayton ~CommandObjectProcessKDPPacket () 10131d19a2f2SGreg Clayton { 10141d19a2f2SGreg Clayton } 10151d19a2f2SGreg Clayton }; 10161d19a2f2SGreg Clayton 10171d19a2f2SGreg Clayton class CommandObjectMultiwordProcessKDP : public CommandObjectMultiword 10181d19a2f2SGreg Clayton { 10191d19a2f2SGreg Clayton public: 10201d19a2f2SGreg Clayton CommandObjectMultiwordProcessKDP (CommandInterpreter &interpreter) : 10211d19a2f2SGreg Clayton CommandObjectMultiword (interpreter, 10221d19a2f2SGreg Clayton "process plugin", 10231d19a2f2SGreg Clayton "A set of commands for operating on a ProcessKDP process.", 10241d19a2f2SGreg Clayton "process plugin <subcommand> [<subcommand-options>]") 10251d19a2f2SGreg Clayton { 10261d19a2f2SGreg Clayton LoadSubCommand ("packet", CommandObjectSP (new CommandObjectProcessKDPPacket (interpreter))); 10271d19a2f2SGreg Clayton } 10281d19a2f2SGreg Clayton 10291d19a2f2SGreg Clayton ~CommandObjectMultiwordProcessKDP () 10301d19a2f2SGreg Clayton { 10311d19a2f2SGreg Clayton } 10321d19a2f2SGreg Clayton }; 10331d19a2f2SGreg Clayton 10341d19a2f2SGreg Clayton CommandObject * 10351d19a2f2SGreg Clayton ProcessKDP::GetPluginCommandObject() 10361d19a2f2SGreg Clayton { 10371d19a2f2SGreg Clayton if (!m_command_sp) 10381d19a2f2SGreg Clayton m_command_sp.reset (new CommandObjectMultiwordProcessKDP (GetTarget().GetDebugger().GetCommandInterpreter())); 10391d19a2f2SGreg Clayton return m_command_sp.get(); 10401d19a2f2SGreg Clayton } 10411d19a2f2SGreg Clayton 1042