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 471b7746e3SGreg Clayton static const lldb::tid_t g_kernel_tid = 1; 481b7746e3SGreg Clayton 49f9765acdSGreg Clayton const char * 50f9765acdSGreg Clayton ProcessKDP::GetPluginNameStatic() 51f9765acdSGreg Clayton { 52f9765acdSGreg Clayton return "kdp-remote"; 53f9765acdSGreg Clayton } 54f9765acdSGreg Clayton 55f9765acdSGreg Clayton const char * 56f9765acdSGreg Clayton ProcessKDP::GetPluginDescriptionStatic() 57f9765acdSGreg Clayton { 58f9765acdSGreg Clayton return "KDP Remote protocol based debugging plug-in for darwin kernel debugging."; 59f9765acdSGreg Clayton } 60f9765acdSGreg Clayton 61f9765acdSGreg Clayton void 62f9765acdSGreg Clayton ProcessKDP::Terminate() 63f9765acdSGreg Clayton { 64f9765acdSGreg Clayton PluginManager::UnregisterPlugin (ProcessKDP::CreateInstance); 65f9765acdSGreg Clayton } 66f9765acdSGreg Clayton 67f9765acdSGreg Clayton 68c3776bf2SGreg Clayton lldb::ProcessSP 69c3776bf2SGreg Clayton ProcessKDP::CreateInstance (Target &target, 70c3776bf2SGreg Clayton Listener &listener, 71c3776bf2SGreg Clayton const FileSpec *crash_file_path) 72f9765acdSGreg Clayton { 73c3776bf2SGreg Clayton lldb::ProcessSP process_sp; 74c3776bf2SGreg Clayton if (crash_file_path == NULL) 75c3776bf2SGreg Clayton process_sp.reset(new ProcessKDP (target, listener)); 76c3776bf2SGreg Clayton return process_sp; 77f9765acdSGreg Clayton } 78f9765acdSGreg Clayton 79f9765acdSGreg Clayton bool 803a29bdbeSGreg Clayton ProcessKDP::CanDebug(Target &target, bool plugin_specified_by_name) 81f9765acdSGreg Clayton { 82596ed24eSGreg Clayton if (plugin_specified_by_name) 83596ed24eSGreg Clayton return true; 84596ed24eSGreg Clayton 85f9765acdSGreg Clayton // For now we are just making sure the file exists for a given module 86aa149cbdSGreg Clayton Module *exe_module = target.GetExecutableModulePointer(); 87aa149cbdSGreg Clayton if (exe_module) 88f9765acdSGreg Clayton { 89f9765acdSGreg Clayton const llvm::Triple &triple_ref = target.GetArchitecture().GetTriple(); 9070512317SGreg Clayton switch (triple_ref.getOS()) 9170512317SGreg Clayton { 9270512317SGreg Clayton case llvm::Triple::Darwin: // Should use "macosx" for desktop and "ios" for iOS, but accept darwin just in case 9370512317SGreg Clayton case llvm::Triple::MacOSX: // For desktop targets 9470512317SGreg Clayton case llvm::Triple::IOS: // For arm targets 9570512317SGreg Clayton if (triple_ref.getVendor() == llvm::Triple::Apple) 96f9765acdSGreg Clayton { 97aa149cbdSGreg Clayton ObjectFile *exe_objfile = exe_module->GetObjectFile(); 98f9765acdSGreg Clayton if (exe_objfile->GetType() == ObjectFile::eTypeExecutable && 99f9765acdSGreg Clayton exe_objfile->GetStrata() == ObjectFile::eStrataKernel) 100f9765acdSGreg Clayton return true; 101f9765acdSGreg Clayton } 10270512317SGreg Clayton break; 10370512317SGreg Clayton 10470512317SGreg Clayton default: 10570512317SGreg Clayton break; 10670512317SGreg Clayton } 107f9765acdSGreg Clayton } 108596ed24eSGreg Clayton return false; 1093a29bdbeSGreg Clayton } 110f9765acdSGreg Clayton 111f9765acdSGreg Clayton //---------------------------------------------------------------------- 112f9765acdSGreg Clayton // ProcessKDP constructor 113f9765acdSGreg Clayton //---------------------------------------------------------------------- 114f9765acdSGreg Clayton ProcessKDP::ProcessKDP(Target& target, Listener &listener) : 115f9765acdSGreg Clayton Process (target, listener), 116f9765acdSGreg Clayton m_comm("lldb.process.kdp-remote.communication"), 1174bddaeb5SJim Ingham m_async_broadcaster (NULL, "lldb.process.kdp-remote.async-broadcaster"), 11897d5cf05SGreg Clayton m_async_thread (LLDB_INVALID_HOST_THREAD), 1195e8534efSJason Molenda m_dyld_plugin_name (), 1201d19a2f2SGreg Clayton m_kernel_load_addr (LLDB_INVALID_ADDRESS), 1211b7746e3SGreg Clayton m_command_sp(), 1221b7746e3SGreg Clayton m_kernel_thread_wp() 123f9765acdSGreg Clayton { 1247925fbbaSGreg Clayton m_async_broadcaster.SetEventName (eBroadcastBitAsyncThreadShouldExit, "async thread should exit"); 1257925fbbaSGreg Clayton m_async_broadcaster.SetEventName (eBroadcastBitAsyncContinue, "async thread continue"); 126f9765acdSGreg Clayton } 127f9765acdSGreg Clayton 128f9765acdSGreg Clayton //---------------------------------------------------------------------- 129f9765acdSGreg Clayton // Destructor 130f9765acdSGreg Clayton //---------------------------------------------------------------------- 131f9765acdSGreg Clayton ProcessKDP::~ProcessKDP() 132f9765acdSGreg Clayton { 133f9765acdSGreg Clayton Clear(); 134e24c4acfSGreg Clayton // We need to call finalize on the process before destroying ourselves 135e24c4acfSGreg Clayton // to make sure all of the broadcaster cleanup goes as planned. If we 136e24c4acfSGreg Clayton // destruct this class, then Process::~Process() might have problems 137e24c4acfSGreg Clayton // trying to fully destroy the broadcaster. 138e24c4acfSGreg Clayton Finalize(); 139f9765acdSGreg Clayton } 140f9765acdSGreg Clayton 141f9765acdSGreg Clayton //---------------------------------------------------------------------- 142f9765acdSGreg Clayton // PluginInterface 143f9765acdSGreg Clayton //---------------------------------------------------------------------- 144f9765acdSGreg Clayton const char * 145f9765acdSGreg Clayton ProcessKDP::GetPluginName() 146f9765acdSGreg Clayton { 147f9765acdSGreg Clayton return "Process debugging plug-in that uses the Darwin KDP remote protocol"; 148f9765acdSGreg Clayton } 149f9765acdSGreg Clayton 150f9765acdSGreg Clayton const char * 151f9765acdSGreg Clayton ProcessKDP::GetShortPluginName() 152f9765acdSGreg Clayton { 153f9765acdSGreg Clayton return GetPluginNameStatic(); 154f9765acdSGreg Clayton } 155f9765acdSGreg Clayton 156f9765acdSGreg Clayton uint32_t 157f9765acdSGreg Clayton ProcessKDP::GetPluginVersion() 158f9765acdSGreg Clayton { 159f9765acdSGreg Clayton return 1; 160f9765acdSGreg Clayton } 161f9765acdSGreg Clayton 162f9765acdSGreg Clayton Error 163f9765acdSGreg Clayton ProcessKDP::WillLaunch (Module* module) 164f9765acdSGreg Clayton { 165f9765acdSGreg Clayton Error error; 166f9765acdSGreg Clayton error.SetErrorString ("launching not supported in kdp-remote plug-in"); 167f9765acdSGreg Clayton return error; 168f9765acdSGreg Clayton } 169f9765acdSGreg Clayton 170f9765acdSGreg Clayton Error 171f9765acdSGreg Clayton ProcessKDP::WillAttachToProcessWithID (lldb::pid_t pid) 172f9765acdSGreg Clayton { 173f9765acdSGreg Clayton Error error; 174f9765acdSGreg Clayton error.SetErrorString ("attaching to a by process ID not supported in kdp-remote plug-in"); 175f9765acdSGreg Clayton return error; 176f9765acdSGreg Clayton } 177f9765acdSGreg Clayton 178f9765acdSGreg Clayton Error 179f9765acdSGreg Clayton ProcessKDP::WillAttachToProcessWithName (const char *process_name, bool wait_for_launch) 180f9765acdSGreg Clayton { 181f9765acdSGreg Clayton Error error; 182f9765acdSGreg Clayton error.SetErrorString ("attaching to a by process name not supported in kdp-remote plug-in"); 183f9765acdSGreg Clayton return error; 184f9765acdSGreg Clayton } 185f9765acdSGreg Clayton 186f9765acdSGreg Clayton Error 1874bd4e7e3SJason Molenda ProcessKDP::DoConnectRemote (Stream *strm, const char *remote_url) 188f9765acdSGreg Clayton { 189f9765acdSGreg Clayton Error error; 1903a29bdbeSGreg Clayton 1917925fbbaSGreg Clayton // Don't let any JIT happen when doing KDP as we can't allocate 1927925fbbaSGreg Clayton // memory and we don't want to be mucking with threads that might 1937925fbbaSGreg Clayton // already be handling exceptions 1947925fbbaSGreg Clayton SetCanJIT(false); 1957925fbbaSGreg Clayton 1963a29bdbeSGreg Clayton if (remote_url == NULL || remote_url[0] == '\0') 1977925fbbaSGreg Clayton { 1987925fbbaSGreg Clayton error.SetErrorStringWithFormat ("invalid connection URL '%s'", remote_url); 1997925fbbaSGreg Clayton return error; 2007925fbbaSGreg Clayton } 2013a29bdbeSGreg Clayton 2027b0992d9SGreg Clayton std::unique_ptr<ConnectionFileDescriptor> conn_ap(new ConnectionFileDescriptor()); 2033a29bdbeSGreg Clayton if (conn_ap.get()) 2043a29bdbeSGreg Clayton { 2053a29bdbeSGreg Clayton // Only try once for now. 2063a29bdbeSGreg Clayton // TODO: check if we should be retrying? 2073a29bdbeSGreg Clayton const uint32_t max_retry_count = 1; 2083a29bdbeSGreg Clayton for (uint32_t retry_count = 0; retry_count < max_retry_count; ++retry_count) 2093a29bdbeSGreg Clayton { 2103a29bdbeSGreg Clayton if (conn_ap->Connect(remote_url, &error) == eConnectionStatusSuccess) 2113a29bdbeSGreg Clayton break; 2123a29bdbeSGreg Clayton usleep (100000); 2133a29bdbeSGreg Clayton } 2143a29bdbeSGreg Clayton } 2153a29bdbeSGreg Clayton 2163a29bdbeSGreg Clayton if (conn_ap->IsConnected()) 2173a29bdbeSGreg Clayton { 2183a29bdbeSGreg Clayton const uint16_t reply_port = conn_ap->GetReadPort (); 2193a29bdbeSGreg Clayton 2203a29bdbeSGreg Clayton if (reply_port != 0) 2213a29bdbeSGreg Clayton { 2223a29bdbeSGreg Clayton m_comm.SetConnection(conn_ap.release()); 2233a29bdbeSGreg Clayton 2243a29bdbeSGreg Clayton if (m_comm.SendRequestReattach(reply_port)) 2253a29bdbeSGreg Clayton { 2263a29bdbeSGreg Clayton if (m_comm.SendRequestConnect(reply_port, reply_port, "Greetings from LLDB...")) 2273a29bdbeSGreg Clayton { 2283a29bdbeSGreg Clayton m_comm.GetVersion(); 2293a29bdbeSGreg Clayton uint32_t cpu = m_comm.GetCPUType(); 2303a29bdbeSGreg Clayton uint32_t sub = m_comm.GetCPUSubtype(); 2313a29bdbeSGreg Clayton ArchSpec kernel_arch; 2323a29bdbeSGreg Clayton kernel_arch.SetArchitecture(eArchTypeMachO, cpu, sub); 2333a29bdbeSGreg Clayton m_target.SetArchitecture(kernel_arch); 2344bd4e7e3SJason Molenda 235840f12cfSJason Molenda /* Get the kernel's UUID and load address via KDP_KERNELVERSION packet. */ 236840f12cfSJason Molenda /* An EFI kdp session has neither UUID nor load address. */ 2374bd4e7e3SJason Molenda 2384bd4e7e3SJason Molenda UUID kernel_uuid = m_comm.GetUUID (); 2394bd4e7e3SJason Molenda addr_t kernel_load_addr = m_comm.GetLoadAddress (); 2404bd4e7e3SJason Molenda 241840f12cfSJason Molenda if (m_comm.RemoteIsEFI ()) 242840f12cfSJason Molenda { 243840f12cfSJason Molenda m_dyld_plugin_name = DynamicLoaderStatic::GetPluginNameStatic(); 244840f12cfSJason Molenda } 245840f12cfSJason Molenda else if (kernel_load_addr != LLDB_INVALID_ADDRESS) 2464bd4e7e3SJason Molenda { 2475e8534efSJason Molenda m_kernel_load_addr = kernel_load_addr; 2485e8534efSJason Molenda m_dyld_plugin_name = DynamicLoaderDarwinKernel::GetPluginNameStatic(); 2494bd4e7e3SJason Molenda } 2504bd4e7e3SJason Molenda 25197d5cf05SGreg Clayton // Set the thread ID 25297d5cf05SGreg Clayton UpdateThreadListIfNeeded (); 253a63d08c9SGreg Clayton SetID (1); 25456d9a1b3SGreg Clayton GetThreadList (); 255a63d08c9SGreg Clayton SetPrivateState (eStateStopped); 25607e66e3eSGreg Clayton StreamSP async_strm_sp(m_target.GetDebugger().GetAsyncOutputStream()); 25707e66e3eSGreg Clayton if (async_strm_sp) 25807e66e3eSGreg Clayton { 2595b88216dSGreg Clayton const char *cstr; 2605b88216dSGreg Clayton if ((cstr = m_comm.GetKernelVersion ()) != NULL) 26107e66e3eSGreg Clayton { 2625b88216dSGreg Clayton async_strm_sp->Printf ("Version: %s\n", cstr); 26307e66e3eSGreg Clayton async_strm_sp->Flush(); 26407e66e3eSGreg Clayton } 2655b88216dSGreg Clayton // if ((cstr = m_comm.GetImagePath ()) != NULL) 2665b88216dSGreg Clayton // { 2675b88216dSGreg Clayton // async_strm_sp->Printf ("Image Path: %s\n", cstr); 2685b88216dSGreg Clayton // async_strm_sp->Flush(); 2695b88216dSGreg Clayton // } 27007e66e3eSGreg Clayton } 2713a29bdbeSGreg Clayton } 27297d5cf05SGreg Clayton else 27397d5cf05SGreg Clayton { 27497d5cf05SGreg Clayton error.SetErrorString("KDP_REATTACH failed"); 27597d5cf05SGreg Clayton } 2763a29bdbeSGreg Clayton } 2773a29bdbeSGreg Clayton else 2783a29bdbeSGreg Clayton { 27997d5cf05SGreg Clayton error.SetErrorString("KDP_REATTACH failed"); 2803a29bdbeSGreg Clayton } 2813a29bdbeSGreg Clayton } 2823a29bdbeSGreg Clayton else 2833a29bdbeSGreg Clayton { 2843a29bdbeSGreg Clayton error.SetErrorString("invalid reply port from UDP connection"); 2853a29bdbeSGreg Clayton } 2863a29bdbeSGreg Clayton } 2873a29bdbeSGreg Clayton else 2883a29bdbeSGreg Clayton { 2893a29bdbeSGreg Clayton if (error.Success()) 2903a29bdbeSGreg Clayton error.SetErrorStringWithFormat ("failed to connect to '%s'", remote_url); 2913a29bdbeSGreg Clayton } 2923a29bdbeSGreg Clayton if (error.Fail()) 2933a29bdbeSGreg Clayton m_comm.Disconnect(); 2943a29bdbeSGreg Clayton 295f9765acdSGreg Clayton return error; 296f9765acdSGreg Clayton } 297f9765acdSGreg Clayton 298f9765acdSGreg Clayton //---------------------------------------------------------------------- 299f9765acdSGreg Clayton // Process Control 300f9765acdSGreg Clayton //---------------------------------------------------------------------- 301f9765acdSGreg Clayton Error 302982c9762SGreg Clayton ProcessKDP::DoLaunch (Module *exe_module, 303982c9762SGreg Clayton const ProcessLaunchInfo &launch_info) 304f9765acdSGreg Clayton { 305f9765acdSGreg Clayton Error error; 306f9765acdSGreg Clayton error.SetErrorString ("launching not supported in kdp-remote plug-in"); 307f9765acdSGreg Clayton return error; 308f9765acdSGreg Clayton } 309f9765acdSGreg Clayton 310f9765acdSGreg Clayton 311f9765acdSGreg Clayton Error 312f9765acdSGreg Clayton ProcessKDP::DoAttachToProcessWithID (lldb::pid_t attach_pid) 313f9765acdSGreg Clayton { 314f9765acdSGreg Clayton Error error; 315f9765acdSGreg Clayton error.SetErrorString ("attach to process by ID is not suppported in kdp remote debugging"); 316f9765acdSGreg Clayton return error; 317f9765acdSGreg Clayton } 318f9765acdSGreg Clayton 319f9765acdSGreg Clayton Error 32084647048SHan Ming Ong ProcessKDP::DoAttachToProcessWithID (lldb::pid_t attach_pid, const ProcessAttachInfo &attach_info) 32184647048SHan Ming Ong { 32284647048SHan Ming Ong Error error; 32384647048SHan Ming Ong error.SetErrorString ("attach to process by ID is not suppported in kdp remote debugging"); 32484647048SHan Ming Ong return error; 32584647048SHan Ming Ong } 32684647048SHan Ming Ong 32784647048SHan Ming Ong Error 32884647048SHan Ming Ong ProcessKDP::DoAttachToProcessWithName (const char *process_name, bool wait_for_launch, const ProcessAttachInfo &attach_info) 329f9765acdSGreg Clayton { 330f9765acdSGreg Clayton Error error; 331f9765acdSGreg Clayton error.SetErrorString ("attach to process by name is not suppported in kdp remote debugging"); 332f9765acdSGreg Clayton return error; 333f9765acdSGreg Clayton } 334f9765acdSGreg Clayton 335f9765acdSGreg Clayton 336f9765acdSGreg Clayton void 337f9765acdSGreg Clayton ProcessKDP::DidAttach () 338f9765acdSGreg Clayton { 3395160ce5cSGreg Clayton Log *log (ProcessKDPLog::GetLogIfAllCategoriesSet (KDP_LOG_PROCESS)); 340f9765acdSGreg Clayton if (log) 34154cb8f83SJohnny Chen log->Printf ("ProcessKDP::DidAttach()"); 342f9765acdSGreg Clayton if (GetID() != LLDB_INVALID_PROCESS_ID) 343f9765acdSGreg Clayton { 344f9765acdSGreg Clayton // TODO: figure out the register context that we will use 345f9765acdSGreg Clayton } 346f9765acdSGreg Clayton } 347f9765acdSGreg Clayton 3485e8534efSJason Molenda addr_t 3495e8534efSJason Molenda ProcessKDP::GetImageInfoAddress() 3505e8534efSJason Molenda { 3515e8534efSJason Molenda return m_kernel_load_addr; 3525e8534efSJason Molenda } 3535e8534efSJason Molenda 3545e8534efSJason Molenda lldb_private::DynamicLoader * 3555e8534efSJason Molenda ProcessKDP::GetDynamicLoader () 3565e8534efSJason Molenda { 3575e8534efSJason Molenda if (m_dyld_ap.get() == NULL) 3585e8534efSJason Molenda m_dyld_ap.reset (DynamicLoader::FindPlugin(this, m_dyld_plugin_name.empty() ? NULL : m_dyld_plugin_name.c_str())); 3595e8534efSJason Molenda return m_dyld_ap.get(); 3605e8534efSJason Molenda } 3615e8534efSJason Molenda 362f9765acdSGreg Clayton Error 363f9765acdSGreg Clayton ProcessKDP::WillResume () 364f9765acdSGreg Clayton { 365f9765acdSGreg Clayton return Error(); 366f9765acdSGreg Clayton } 367f9765acdSGreg Clayton 368f9765acdSGreg Clayton Error 369f9765acdSGreg Clayton ProcessKDP::DoResume () 370f9765acdSGreg Clayton { 371f9765acdSGreg Clayton Error error; 3725160ce5cSGreg Clayton Log *log (ProcessKDPLog::GetLogIfAllCategoriesSet (KDP_LOG_PROCESS)); 3737925fbbaSGreg Clayton // Only start the async thread if we try to do any process control 3747925fbbaSGreg Clayton if (!IS_VALID_LLDB_HOST_THREAD(m_async_thread)) 3757925fbbaSGreg Clayton StartAsyncThread (); 3767925fbbaSGreg Clayton 37797d5cf05SGreg Clayton bool resume = false; 3787925fbbaSGreg Clayton 37997d5cf05SGreg Clayton // With KDP there is only one thread we can tell what to do 3801b7746e3SGreg Clayton ThreadSP kernel_thread_sp (m_thread_list.FindThreadByProtocolID(g_kernel_tid)); 3811b7746e3SGreg Clayton 38297d5cf05SGreg Clayton if (kernel_thread_sp) 3834b1b8b3eSGreg Clayton { 38497d5cf05SGreg Clayton const StateType thread_resume_state = kernel_thread_sp->GetTemporaryResumeState(); 3857925fbbaSGreg Clayton switch (thread_resume_state) 3864b1b8b3eSGreg Clayton { 3877925fbbaSGreg Clayton case eStateSuspended: 3887925fbbaSGreg Clayton // Nothing to do here when a thread will stay suspended 3897925fbbaSGreg Clayton // we just leave the CPU mask bit set to zero for the thread 3907925fbbaSGreg Clayton break; 3917925fbbaSGreg Clayton 3927925fbbaSGreg Clayton case eStateStepping: 3931afa68edSGreg Clayton { 3941afa68edSGreg Clayton lldb::RegisterContextSP reg_ctx_sp (kernel_thread_sp->GetRegisterContext()); 3951afa68edSGreg Clayton 3961afa68edSGreg Clayton if (reg_ctx_sp) 3971afa68edSGreg Clayton { 3981afa68edSGreg Clayton reg_ctx_sp->HardwareSingleStep (true); 39997d5cf05SGreg Clayton resume = true; 4001afa68edSGreg Clayton } 4011afa68edSGreg Clayton else 4021afa68edSGreg Clayton { 4031afa68edSGreg Clayton error.SetErrorStringWithFormat("KDP thread 0x%llx has no register context", kernel_thread_sp->GetID()); 4041afa68edSGreg Clayton } 4051afa68edSGreg Clayton } 4067925fbbaSGreg Clayton break; 4077925fbbaSGreg Clayton 40897d5cf05SGreg Clayton case eStateRunning: 4091afa68edSGreg Clayton { 4101afa68edSGreg Clayton lldb::RegisterContextSP reg_ctx_sp (kernel_thread_sp->GetRegisterContext()); 4111afa68edSGreg Clayton 4121afa68edSGreg Clayton if (reg_ctx_sp) 4131afa68edSGreg Clayton { 4141afa68edSGreg Clayton reg_ctx_sp->HardwareSingleStep (false); 41597d5cf05SGreg Clayton resume = true; 4161afa68edSGreg Clayton } 4171afa68edSGreg Clayton else 4181afa68edSGreg Clayton { 4191afa68edSGreg Clayton error.SetErrorStringWithFormat("KDP thread 0x%llx has no register context", kernel_thread_sp->GetID()); 4201afa68edSGreg Clayton } 4211afa68edSGreg Clayton } 4227925fbbaSGreg Clayton break; 4237925fbbaSGreg Clayton 4247925fbbaSGreg Clayton default: 42597d5cf05SGreg Clayton // The only valid thread resume states are listed above 4267925fbbaSGreg Clayton assert (!"invalid thread resume state"); 4277925fbbaSGreg Clayton break; 4284b1b8b3eSGreg Clayton } 4294b1b8b3eSGreg Clayton } 4307925fbbaSGreg Clayton 43197d5cf05SGreg Clayton if (resume) 43297d5cf05SGreg Clayton { 43397d5cf05SGreg Clayton if (log) 43497d5cf05SGreg Clayton log->Printf ("ProcessKDP::DoResume () sending resume"); 43597d5cf05SGreg Clayton 43697d5cf05SGreg Clayton if (m_comm.SendRequestResume ()) 4377925fbbaSGreg Clayton { 4387925fbbaSGreg Clayton m_async_broadcaster.BroadcastEvent (eBroadcastBitAsyncContinue); 4397925fbbaSGreg Clayton SetPrivateState(eStateRunning); 4407925fbbaSGreg Clayton } 4414b1b8b3eSGreg Clayton else 44207e66e3eSGreg Clayton error.SetErrorString ("KDP resume failed"); 4437925fbbaSGreg Clayton } 4447925fbbaSGreg Clayton else 4457925fbbaSGreg Clayton { 44697d5cf05SGreg Clayton error.SetErrorString ("kernel thread is suspended"); 4477925fbbaSGreg Clayton } 4487925fbbaSGreg Clayton 449f9765acdSGreg Clayton return error; 450f9765acdSGreg Clayton } 451f9765acdSGreg Clayton 45297d5cf05SGreg Clayton lldb::ThreadSP 4531b7746e3SGreg Clayton ProcessKDP::GetKernelThread() 45497d5cf05SGreg Clayton { 45597d5cf05SGreg Clayton // KDP only tells us about one thread/core. Any other threads will usually 45697d5cf05SGreg Clayton // be the ones that are read from memory by the OS plug-ins. 4571b7746e3SGreg Clayton 4581b7746e3SGreg Clayton ThreadSP thread_sp (m_kernel_thread_wp.lock()); 45997d5cf05SGreg Clayton if (!thread_sp) 4601b7746e3SGreg Clayton { 4611b7746e3SGreg Clayton thread_sp.reset(new ThreadKDP (*this, g_kernel_tid)); 4621b7746e3SGreg Clayton m_kernel_thread_wp = thread_sp; 4631b7746e3SGreg Clayton } 46497d5cf05SGreg Clayton return thread_sp; 46597d5cf05SGreg Clayton } 46697d5cf05SGreg Clayton 46797d5cf05SGreg Clayton 46897d5cf05SGreg Clayton 46997d5cf05SGreg Clayton 4709fc13556SGreg Clayton bool 47156d9a1b3SGreg Clayton ProcessKDP::UpdateThreadList (ThreadList &old_thread_list, ThreadList &new_thread_list) 472f9765acdSGreg Clayton { 473f9765acdSGreg Clayton // locker will keep a mutex locked until it goes out of scope 4745160ce5cSGreg Clayton Log *log (ProcessKDPLog::GetLogIfAllCategoriesSet (KDP_LOG_THREAD)); 475f9765acdSGreg Clayton if (log && log->GetMask().Test(KDP_LOG_VERBOSE)) 476d01b2953SDaniel Malea log->Printf ("ProcessKDP::%s (pid = %" PRIu64 ")", __FUNCTION__, GetID()); 477f9765acdSGreg Clayton 47839da3efdSGreg Clayton // Even though there is a CPU mask, it doesn't mean we can see each CPU 47997d5cf05SGreg Clayton // indivudually, there is really only one. Lets call this thread 1. 480*513db4d9SJason Molenda ThreadSP thread_sp (old_thread_list.FindThreadByProtocolID(g_kernel_tid, false)); 4811b7746e3SGreg Clayton if (!thread_sp) 4821b7746e3SGreg Clayton thread_sp = GetKernelThread (); 4831b7746e3SGreg Clayton new_thread_list.AddThread(thread_sp); 48497d5cf05SGreg Clayton 4859fc13556SGreg Clayton return new_thread_list.GetSize(false) > 0; 486f9765acdSGreg Clayton } 487f9765acdSGreg Clayton 488f9765acdSGreg Clayton void 489f9765acdSGreg Clayton ProcessKDP::RefreshStateAfterStop () 490f9765acdSGreg Clayton { 491f9765acdSGreg Clayton // Let all threads recover from stopping and do any clean up based 492f9765acdSGreg Clayton // on the previous thread state (if any). 493f9765acdSGreg Clayton m_thread_list.RefreshStateAfterStop(); 494f9765acdSGreg Clayton } 495f9765acdSGreg Clayton 496f9765acdSGreg Clayton Error 497f9765acdSGreg Clayton ProcessKDP::DoHalt (bool &caused_stop) 498f9765acdSGreg Clayton { 499f9765acdSGreg Clayton Error error; 500f9765acdSGreg Clayton 50197d5cf05SGreg Clayton if (m_comm.IsRunning()) 502f9765acdSGreg Clayton { 50397d5cf05SGreg Clayton if (m_destroy_in_process) 50497d5cf05SGreg Clayton { 50597d5cf05SGreg Clayton // If we are attemping to destroy, we need to not return an error to 50697d5cf05SGreg Clayton // Halt or DoDestroy won't get called. 50797d5cf05SGreg Clayton // We are also currently running, so send a process stopped event 50897d5cf05SGreg Clayton SetPrivateState (eStateStopped); 509f9765acdSGreg Clayton } 510f9765acdSGreg Clayton else 511f9765acdSGreg Clayton { 51297d5cf05SGreg Clayton error.SetErrorString ("KDP cannot interrupt a running kernel"); 513f9765acdSGreg Clayton } 514f9765acdSGreg Clayton } 515f9765acdSGreg Clayton return error; 516f9765acdSGreg Clayton } 517f9765acdSGreg Clayton 518f9765acdSGreg Clayton Error 519acff8950SJim Ingham ProcessKDP::DoDetach(bool keep_stopped) 520f9765acdSGreg Clayton { 521f9765acdSGreg Clayton Error error; 5225160ce5cSGreg Clayton Log *log (ProcessKDPLog::GetLogIfAllCategoriesSet(KDP_LOG_PROCESS)); 523f9765acdSGreg Clayton if (log) 524acff8950SJim Ingham log->Printf ("ProcessKDP::DoDetach(keep_stopped = %i)", keep_stopped); 525f9765acdSGreg Clayton 52697d5cf05SGreg Clayton if (m_comm.IsRunning()) 52797d5cf05SGreg Clayton { 52897d5cf05SGreg Clayton // We are running and we can't interrupt a running kernel, so we need 52997d5cf05SGreg Clayton // to just close the connection to the kernel and hope for the best 53097d5cf05SGreg Clayton } 53197d5cf05SGreg Clayton else 53297d5cf05SGreg Clayton { 533f9765acdSGreg Clayton DisableAllBreakpointSites (); 534f9765acdSGreg Clayton 535f9765acdSGreg Clayton m_thread_list.DiscardThreadPlans(); 536f9765acdSGreg Clayton 537acff8950SJim Ingham // If we are going to keep the target stopped, then don't send the disconnect message. 538acff8950SJim Ingham if (!keep_stopped && m_comm.IsConnected()) 5393a29bdbeSGreg Clayton { 5403a29bdbeSGreg Clayton 5413a29bdbeSGreg Clayton m_comm.SendRequestDisconnect(); 5423a29bdbeSGreg Clayton 54357508026SGreg Clayton size_t response_size = m_comm.Disconnect (); 544f9765acdSGreg Clayton if (log) 545f9765acdSGreg Clayton { 546f9765acdSGreg Clayton if (response_size) 547f9765acdSGreg Clayton log->PutCString ("ProcessKDP::DoDetach() detach packet sent successfully"); 548f9765acdSGreg Clayton else 549f9765acdSGreg Clayton log->PutCString ("ProcessKDP::DoDetach() detach packet send failed"); 550f9765acdSGreg Clayton } 5513a29bdbeSGreg Clayton } 55297d5cf05SGreg Clayton } 553f9765acdSGreg Clayton StopAsyncThread (); 55474d4193eSGreg Clayton m_comm.Clear(); 555f9765acdSGreg Clayton 556f9765acdSGreg Clayton SetPrivateState (eStateDetached); 557f9765acdSGreg Clayton ResumePrivateStateThread(); 558f9765acdSGreg Clayton 559f9765acdSGreg Clayton //KillDebugserverProcess (); 560f9765acdSGreg Clayton return error; 561f9765acdSGreg Clayton } 562f9765acdSGreg Clayton 563f9765acdSGreg Clayton Error 564f9765acdSGreg Clayton ProcessKDP::DoDestroy () 565f9765acdSGreg Clayton { 5667925fbbaSGreg Clayton // For KDP there really is no difference between destroy and detach 567acff8950SJim Ingham bool keep_stopped = false; 568acff8950SJim Ingham return DoDetach(keep_stopped); 569f9765acdSGreg Clayton } 570f9765acdSGreg Clayton 571f9765acdSGreg Clayton //------------------------------------------------------------------ 572f9765acdSGreg Clayton // Process Queries 573f9765acdSGreg Clayton //------------------------------------------------------------------ 574f9765acdSGreg Clayton 575f9765acdSGreg Clayton bool 576f9765acdSGreg Clayton ProcessKDP::IsAlive () 577f9765acdSGreg Clayton { 578f9765acdSGreg Clayton return m_comm.IsConnected() && m_private_state.GetValue() != eStateExited; 579f9765acdSGreg Clayton } 580f9765acdSGreg Clayton 581f9765acdSGreg Clayton //------------------------------------------------------------------ 582f9765acdSGreg Clayton // Process Memory 583f9765acdSGreg Clayton //------------------------------------------------------------------ 584f9765acdSGreg Clayton size_t 585f9765acdSGreg Clayton ProcessKDP::DoReadMemory (addr_t addr, void *buf, size_t size, Error &error) 586f9765acdSGreg Clayton { 587a63d08c9SGreg Clayton if (m_comm.IsConnected()) 588a63d08c9SGreg Clayton return m_comm.SendRequestReadMemory (addr, buf, size, error); 589a63d08c9SGreg Clayton error.SetErrorString ("not connected"); 590f9765acdSGreg Clayton return 0; 591f9765acdSGreg Clayton } 592f9765acdSGreg Clayton 593f9765acdSGreg Clayton size_t 594f9765acdSGreg Clayton ProcessKDP::DoWriteMemory (addr_t addr, const void *buf, size_t size, Error &error) 595f9765acdSGreg Clayton { 5967925fbbaSGreg Clayton if (m_comm.IsConnected()) 5977925fbbaSGreg Clayton return m_comm.SendRequestWriteMemory (addr, buf, size, error); 5987925fbbaSGreg Clayton error.SetErrorString ("not connected"); 599f9765acdSGreg Clayton return 0; 600f9765acdSGreg Clayton } 601f9765acdSGreg Clayton 602f9765acdSGreg Clayton lldb::addr_t 603f9765acdSGreg Clayton ProcessKDP::DoAllocateMemory (size_t size, uint32_t permissions, Error &error) 604f9765acdSGreg Clayton { 605f9765acdSGreg Clayton error.SetErrorString ("memory allocation not suppported in kdp remote debugging"); 606f9765acdSGreg Clayton return LLDB_INVALID_ADDRESS; 607f9765acdSGreg Clayton } 608f9765acdSGreg Clayton 609f9765acdSGreg Clayton Error 610f9765acdSGreg Clayton ProcessKDP::DoDeallocateMemory (lldb::addr_t addr) 611f9765acdSGreg Clayton { 612f9765acdSGreg Clayton Error error; 613f9765acdSGreg Clayton error.SetErrorString ("memory deallocation not suppported in kdp remote debugging"); 614f9765acdSGreg Clayton return error; 615f9765acdSGreg Clayton } 616f9765acdSGreg Clayton 617f9765acdSGreg Clayton Error 618299c0c1cSJim Ingham ProcessKDP::EnableBreakpointSite (BreakpointSite *bp_site) 619f9765acdSGreg Clayton { 62007e66e3eSGreg Clayton if (m_comm.LocalBreakpointsAreSupported ()) 62107e66e3eSGreg Clayton { 62207e66e3eSGreg Clayton Error error; 6235b88216dSGreg Clayton if (!bp_site->IsEnabled()) 6245b88216dSGreg Clayton { 6255b88216dSGreg Clayton if (m_comm.SendRequestBreakpoint(true, bp_site->GetLoadAddress())) 6265b88216dSGreg Clayton { 6275b88216dSGreg Clayton bp_site->SetEnabled(true); 6285b88216dSGreg Clayton bp_site->SetType (BreakpointSite::eExternal); 6295b88216dSGreg Clayton } 6305b88216dSGreg Clayton else 6315b88216dSGreg Clayton { 63207e66e3eSGreg Clayton error.SetErrorString ("KDP set breakpoint failed"); 6335b88216dSGreg Clayton } 6345b88216dSGreg Clayton } 63507e66e3eSGreg Clayton return error; 63607e66e3eSGreg Clayton } 637f9765acdSGreg Clayton return EnableSoftwareBreakpoint (bp_site); 638f9765acdSGreg Clayton } 639f9765acdSGreg Clayton 640f9765acdSGreg Clayton Error 641299c0c1cSJim Ingham ProcessKDP::DisableBreakpointSite (BreakpointSite *bp_site) 642f9765acdSGreg Clayton { 64307e66e3eSGreg Clayton if (m_comm.LocalBreakpointsAreSupported ()) 64407e66e3eSGreg Clayton { 64507e66e3eSGreg Clayton Error error; 6465b88216dSGreg Clayton if (bp_site->IsEnabled()) 6475b88216dSGreg Clayton { 6485b88216dSGreg Clayton BreakpointSite::Type bp_type = bp_site->GetType(); 6495b88216dSGreg Clayton if (bp_type == BreakpointSite::eExternal) 6505b88216dSGreg Clayton { 65197d5cf05SGreg Clayton if (m_destroy_in_process && m_comm.IsRunning()) 65297d5cf05SGreg Clayton { 65397d5cf05SGreg Clayton // We are trying to destroy our connection and we are running 65497d5cf05SGreg Clayton bp_site->SetEnabled(false); 65597d5cf05SGreg Clayton } 65697d5cf05SGreg Clayton else 65797d5cf05SGreg Clayton { 6585b88216dSGreg Clayton if (m_comm.SendRequestBreakpoint(false, bp_site->GetLoadAddress())) 6595b88216dSGreg Clayton bp_site->SetEnabled(false); 6605b88216dSGreg Clayton else 66107e66e3eSGreg Clayton error.SetErrorString ("KDP remove breakpoint failed"); 6625b88216dSGreg Clayton } 66397d5cf05SGreg Clayton } 6645b88216dSGreg Clayton else 6655b88216dSGreg Clayton { 6665b88216dSGreg Clayton error = DisableSoftwareBreakpoint (bp_site); 6675b88216dSGreg Clayton } 6685b88216dSGreg Clayton } 66907e66e3eSGreg Clayton return error; 67007e66e3eSGreg Clayton } 671f9765acdSGreg Clayton return DisableSoftwareBreakpoint (bp_site); 672f9765acdSGreg Clayton } 673f9765acdSGreg Clayton 674f9765acdSGreg Clayton Error 6751b5792e5SJim Ingham ProcessKDP::EnableWatchpoint (Watchpoint *wp, bool notify) 676f9765acdSGreg Clayton { 677f9765acdSGreg Clayton Error error; 678f9765acdSGreg Clayton error.SetErrorString ("watchpoints are not suppported in kdp remote debugging"); 679f9765acdSGreg Clayton return error; 680f9765acdSGreg Clayton } 681f9765acdSGreg Clayton 682f9765acdSGreg Clayton Error 6831b5792e5SJim Ingham ProcessKDP::DisableWatchpoint (Watchpoint *wp, bool notify) 684f9765acdSGreg Clayton { 685f9765acdSGreg Clayton Error error; 686f9765acdSGreg Clayton error.SetErrorString ("watchpoints are not suppported in kdp remote debugging"); 687f9765acdSGreg Clayton return error; 688f9765acdSGreg Clayton } 689f9765acdSGreg Clayton 690f9765acdSGreg Clayton void 691f9765acdSGreg Clayton ProcessKDP::Clear() 692f9765acdSGreg Clayton { 693f9765acdSGreg Clayton m_thread_list.Clear(); 694f9765acdSGreg Clayton } 695f9765acdSGreg Clayton 696f9765acdSGreg Clayton Error 697f9765acdSGreg Clayton ProcessKDP::DoSignal (int signo) 698f9765acdSGreg Clayton { 699f9765acdSGreg Clayton Error error; 700f9765acdSGreg Clayton error.SetErrorString ("sending signals is not suppported in kdp remote debugging"); 701f9765acdSGreg Clayton return error; 702f9765acdSGreg Clayton } 703f9765acdSGreg Clayton 704f9765acdSGreg Clayton void 705f9765acdSGreg Clayton ProcessKDP::Initialize() 706f9765acdSGreg Clayton { 707f9765acdSGreg Clayton static bool g_initialized = false; 708f9765acdSGreg Clayton 709f9765acdSGreg Clayton if (g_initialized == false) 710f9765acdSGreg Clayton { 711f9765acdSGreg Clayton g_initialized = true; 712f9765acdSGreg Clayton PluginManager::RegisterPlugin (GetPluginNameStatic(), 713f9765acdSGreg Clayton GetPluginDescriptionStatic(), 714f9765acdSGreg Clayton CreateInstance); 715f9765acdSGreg Clayton 716f9765acdSGreg Clayton Log::Callbacks log_callbacks = { 717f9765acdSGreg Clayton ProcessKDPLog::DisableLog, 718f9765acdSGreg Clayton ProcessKDPLog::EnableLog, 719f9765acdSGreg Clayton ProcessKDPLog::ListLogCategories 720f9765acdSGreg Clayton }; 721f9765acdSGreg Clayton 722f9765acdSGreg Clayton Log::RegisterLogChannel (ProcessKDP::GetPluginNameStatic(), log_callbacks); 723f9765acdSGreg Clayton } 724f9765acdSGreg Clayton } 725f9765acdSGreg Clayton 726f9765acdSGreg Clayton bool 727f9765acdSGreg Clayton ProcessKDP::StartAsyncThread () 728f9765acdSGreg Clayton { 7295160ce5cSGreg Clayton Log *log (ProcessKDPLog::GetLogIfAllCategoriesSet(KDP_LOG_PROCESS)); 730f9765acdSGreg Clayton 731f9765acdSGreg Clayton if (log) 7327925fbbaSGreg Clayton log->Printf ("ProcessKDP::StartAsyncThread ()"); 733f9765acdSGreg Clayton 7347925fbbaSGreg Clayton if (IS_VALID_LLDB_HOST_THREAD(m_async_thread)) 7357925fbbaSGreg Clayton return true; 7367925fbbaSGreg Clayton 737f9765acdSGreg Clayton m_async_thread = Host::ThreadCreate ("<lldb.process.kdp-remote.async>", ProcessKDP::AsyncThread, this, NULL); 738f9765acdSGreg Clayton return IS_VALID_LLDB_HOST_THREAD(m_async_thread); 739f9765acdSGreg Clayton } 740f9765acdSGreg Clayton 741f9765acdSGreg Clayton void 742f9765acdSGreg Clayton ProcessKDP::StopAsyncThread () 743f9765acdSGreg Clayton { 7445160ce5cSGreg Clayton Log *log (ProcessKDPLog::GetLogIfAllCategoriesSet(KDP_LOG_PROCESS)); 745f9765acdSGreg Clayton 746f9765acdSGreg Clayton if (log) 7477925fbbaSGreg Clayton log->Printf ("ProcessKDP::StopAsyncThread ()"); 748f9765acdSGreg Clayton 749f9765acdSGreg Clayton m_async_broadcaster.BroadcastEvent (eBroadcastBitAsyncThreadShouldExit); 750f9765acdSGreg Clayton 751f9765acdSGreg Clayton // Stop the stdio thread 752f9765acdSGreg Clayton if (IS_VALID_LLDB_HOST_THREAD(m_async_thread)) 753f9765acdSGreg Clayton { 754f9765acdSGreg Clayton Host::ThreadJoin (m_async_thread, NULL, NULL); 7557925fbbaSGreg Clayton m_async_thread = LLDB_INVALID_HOST_THREAD; 756f9765acdSGreg Clayton } 757f9765acdSGreg Clayton } 758f9765acdSGreg Clayton 759f9765acdSGreg Clayton 760f9765acdSGreg Clayton void * 761f9765acdSGreg Clayton ProcessKDP::AsyncThread (void *arg) 762f9765acdSGreg Clayton { 763f9765acdSGreg Clayton ProcessKDP *process = (ProcessKDP*) arg; 764f9765acdSGreg Clayton 7657925fbbaSGreg Clayton const lldb::pid_t pid = process->GetID(); 7667925fbbaSGreg Clayton 7675160ce5cSGreg Clayton Log *log (ProcessKDPLog::GetLogIfAllCategoriesSet (KDP_LOG_PROCESS)); 768f9765acdSGreg Clayton if (log) 769d01b2953SDaniel Malea log->Printf ("ProcessKDP::AsyncThread (arg = %p, pid = %" PRIu64 ") thread starting...", arg, pid); 770f9765acdSGreg Clayton 771f9765acdSGreg Clayton Listener listener ("ProcessKDP::AsyncThread"); 772f9765acdSGreg Clayton EventSP event_sp; 773f9765acdSGreg Clayton const uint32_t desired_event_mask = eBroadcastBitAsyncContinue | 774f9765acdSGreg Clayton eBroadcastBitAsyncThreadShouldExit; 775f9765acdSGreg Clayton 7767925fbbaSGreg Clayton 777f9765acdSGreg Clayton if (listener.StartListeningForEvents (&process->m_async_broadcaster, desired_event_mask) == desired_event_mask) 778f9765acdSGreg Clayton { 779f9765acdSGreg Clayton bool done = false; 780f9765acdSGreg Clayton while (!done) 781f9765acdSGreg Clayton { 782f9765acdSGreg Clayton if (log) 783d01b2953SDaniel Malea log->Printf ("ProcessKDP::AsyncThread (pid = %" PRIu64 ") listener.WaitForEvent (NULL, event_sp)...", 7847925fbbaSGreg Clayton pid); 785f9765acdSGreg Clayton if (listener.WaitForEvent (NULL, event_sp)) 786f9765acdSGreg Clayton { 7877925fbbaSGreg Clayton uint32_t event_type = event_sp->GetType(); 788f9765acdSGreg Clayton if (log) 789d01b2953SDaniel Malea log->Printf ("ProcessKDP::AsyncThread (pid = %" PRIu64 ") Got an event of type: %d...", 7907925fbbaSGreg Clayton pid, 7917925fbbaSGreg Clayton event_type); 792f9765acdSGreg Clayton 7937925fbbaSGreg Clayton // When we are running, poll for 1 second to try and get an exception 7947925fbbaSGreg Clayton // to indicate the process has stopped. If we don't get one, check to 7957925fbbaSGreg Clayton // make sure no one asked us to exit 7967925fbbaSGreg Clayton bool is_running = false; 7977925fbbaSGreg Clayton DataExtractor exc_reply_packet; 7987925fbbaSGreg Clayton do 7997925fbbaSGreg Clayton { 800f9765acdSGreg Clayton switch (event_type) 801f9765acdSGreg Clayton { 802f9765acdSGreg Clayton case eBroadcastBitAsyncContinue: 803f9765acdSGreg Clayton { 8047925fbbaSGreg Clayton is_running = true; 8057925fbbaSGreg Clayton if (process->m_comm.WaitForPacketWithTimeoutMicroSeconds (exc_reply_packet, 1 * USEC_PER_SEC)) 806f9765acdSGreg Clayton { 8071b7746e3SGreg Clayton ThreadSP thread_sp (process->GetKernelThread()); 8081afa68edSGreg Clayton if (thread_sp) 8091afa68edSGreg Clayton { 8101afa68edSGreg Clayton lldb::RegisterContextSP reg_ctx_sp (thread_sp->GetRegisterContext()); 8111afa68edSGreg Clayton if (reg_ctx_sp) 8121afa68edSGreg Clayton reg_ctx_sp->InvalidateAllRegisters(); 81397d5cf05SGreg Clayton static_cast<ThreadKDP *>(thread_sp.get())->SetStopInfoFrom_KDP_EXCEPTION (exc_reply_packet); 8141afa68edSGreg Clayton } 81597d5cf05SGreg Clayton 8167925fbbaSGreg Clayton // TODO: parse the stop reply packet 8177925fbbaSGreg Clayton is_running = false; 8187925fbbaSGreg Clayton process->SetPrivateState(eStateStopped); 8197925fbbaSGreg Clayton } 8207925fbbaSGreg Clayton else 8217925fbbaSGreg Clayton { 8227925fbbaSGreg Clayton // Check to see if we are supposed to exit. There is no way to 8237925fbbaSGreg Clayton // interrupt a running kernel, so all we can do is wait for an 8247925fbbaSGreg Clayton // exception or detach... 8257925fbbaSGreg Clayton if (listener.GetNextEvent(event_sp)) 8267925fbbaSGreg Clayton { 8277925fbbaSGreg Clayton // We got an event, go through the loop again 8287925fbbaSGreg Clayton event_type = event_sp->GetType(); 8297925fbbaSGreg Clayton } 830f9765acdSGreg Clayton } 831f9765acdSGreg Clayton } 832f9765acdSGreg Clayton break; 833f9765acdSGreg Clayton 834f9765acdSGreg Clayton case eBroadcastBitAsyncThreadShouldExit: 835f9765acdSGreg Clayton if (log) 836d01b2953SDaniel Malea log->Printf ("ProcessKDP::AsyncThread (pid = %" PRIu64 ") got eBroadcastBitAsyncThreadShouldExit...", 8377925fbbaSGreg Clayton pid); 838f9765acdSGreg Clayton done = true; 8397925fbbaSGreg Clayton is_running = false; 840f9765acdSGreg Clayton break; 841f9765acdSGreg Clayton 842f9765acdSGreg Clayton default: 843f9765acdSGreg Clayton if (log) 844d01b2953SDaniel Malea log->Printf ("ProcessKDP::AsyncThread (pid = %" PRIu64 ") got unknown event 0x%8.8x", 8457925fbbaSGreg Clayton pid, 8467925fbbaSGreg Clayton event_type); 847f9765acdSGreg Clayton done = true; 8487925fbbaSGreg Clayton is_running = false; 849f9765acdSGreg Clayton break; 850f9765acdSGreg Clayton } 8517925fbbaSGreg Clayton } while (is_running); 852f9765acdSGreg Clayton } 853f9765acdSGreg Clayton else 854f9765acdSGreg Clayton { 855f9765acdSGreg Clayton if (log) 856d01b2953SDaniel Malea log->Printf ("ProcessKDP::AsyncThread (pid = %" PRIu64 ") listener.WaitForEvent (NULL, event_sp) => false", 8577925fbbaSGreg Clayton pid); 858f9765acdSGreg Clayton done = true; 859f9765acdSGreg Clayton } 860f9765acdSGreg Clayton } 861f9765acdSGreg Clayton } 862f9765acdSGreg Clayton 863f9765acdSGreg Clayton if (log) 864d01b2953SDaniel Malea log->Printf ("ProcessKDP::AsyncThread (arg = %p, pid = %" PRIu64 ") thread exiting...", 8657925fbbaSGreg Clayton arg, 8667925fbbaSGreg Clayton pid); 867f9765acdSGreg Clayton 868f9765acdSGreg Clayton process->m_async_thread = LLDB_INVALID_HOST_THREAD; 869f9765acdSGreg Clayton return NULL; 870f9765acdSGreg Clayton } 871f9765acdSGreg Clayton 872f9765acdSGreg Clayton 8731d19a2f2SGreg Clayton class CommandObjectProcessKDPPacketSend : public CommandObjectParsed 8741d19a2f2SGreg Clayton { 8751d19a2f2SGreg Clayton private: 8761d19a2f2SGreg Clayton 8771d19a2f2SGreg Clayton OptionGroupOptions m_option_group; 8781d19a2f2SGreg Clayton OptionGroupUInt64 m_command_byte; 8791d19a2f2SGreg Clayton OptionGroupString m_packet_data; 8801d19a2f2SGreg Clayton 8811d19a2f2SGreg Clayton virtual Options * 8821d19a2f2SGreg Clayton GetOptions () 8831d19a2f2SGreg Clayton { 8841d19a2f2SGreg Clayton return &m_option_group; 8851d19a2f2SGreg Clayton } 8861d19a2f2SGreg Clayton 8871d19a2f2SGreg Clayton 8881d19a2f2SGreg Clayton public: 8891d19a2f2SGreg Clayton CommandObjectProcessKDPPacketSend(CommandInterpreter &interpreter) : 8901d19a2f2SGreg Clayton CommandObjectParsed (interpreter, 8911d19a2f2SGreg Clayton "process plugin packet send", 8921d19a2f2SGreg 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. ", 8931d19a2f2SGreg Clayton NULL), 8941d19a2f2SGreg Clayton m_option_group (interpreter), 8951d19a2f2SGreg 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), 8961d19a2f2SGreg 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) 8971d19a2f2SGreg Clayton { 8981d19a2f2SGreg Clayton m_option_group.Append (&m_command_byte, LLDB_OPT_SET_ALL, LLDB_OPT_SET_1); 8991d19a2f2SGreg Clayton m_option_group.Append (&m_packet_data , LLDB_OPT_SET_ALL, LLDB_OPT_SET_1); 9001d19a2f2SGreg Clayton m_option_group.Finalize(); 9011d19a2f2SGreg Clayton } 9021d19a2f2SGreg Clayton 9031d19a2f2SGreg Clayton ~CommandObjectProcessKDPPacketSend () 9041d19a2f2SGreg Clayton { 9051d19a2f2SGreg Clayton } 9061d19a2f2SGreg Clayton 9071d19a2f2SGreg Clayton bool 9081d19a2f2SGreg Clayton DoExecute (Args& command, CommandReturnObject &result) 9091d19a2f2SGreg Clayton { 9101d19a2f2SGreg Clayton const size_t argc = command.GetArgumentCount(); 9111d19a2f2SGreg Clayton if (argc == 0) 9121d19a2f2SGreg Clayton { 9131d19a2f2SGreg Clayton if (!m_command_byte.GetOptionValue().OptionWasSet()) 9141d19a2f2SGreg Clayton { 9151d19a2f2SGreg Clayton result.AppendError ("the --command option must be set to a valid command byte"); 9161d19a2f2SGreg Clayton result.SetStatus (eReturnStatusFailed); 9171d19a2f2SGreg Clayton } 9181d19a2f2SGreg Clayton else 9191d19a2f2SGreg Clayton { 9201d19a2f2SGreg Clayton const uint64_t command_byte = m_command_byte.GetOptionValue().GetUInt64Value(0); 9211d19a2f2SGreg Clayton if (command_byte > 0 && command_byte <= UINT8_MAX) 9221d19a2f2SGreg Clayton { 9231d19a2f2SGreg Clayton ProcessKDP *process = (ProcessKDP *)m_interpreter.GetExecutionContext().GetProcessPtr(); 9241d19a2f2SGreg Clayton if (process) 9251d19a2f2SGreg Clayton { 9261d19a2f2SGreg Clayton const StateType state = process->GetState(); 9271d19a2f2SGreg Clayton 9281d19a2f2SGreg Clayton if (StateIsStoppedState (state, true)) 9291d19a2f2SGreg Clayton { 9301d19a2f2SGreg Clayton std::vector<uint8_t> payload_bytes; 9311d19a2f2SGreg Clayton const char *ascii_hex_bytes_cstr = m_packet_data.GetOptionValue().GetCurrentValue(); 9321d19a2f2SGreg Clayton if (ascii_hex_bytes_cstr && ascii_hex_bytes_cstr[0]) 9331d19a2f2SGreg Clayton { 9341d19a2f2SGreg Clayton StringExtractor extractor(ascii_hex_bytes_cstr); 9351d19a2f2SGreg Clayton const size_t ascii_hex_bytes_cstr_len = extractor.GetStringRef().size(); 9361d19a2f2SGreg Clayton if (ascii_hex_bytes_cstr_len & 1) 9371d19a2f2SGreg Clayton { 9381d19a2f2SGreg Clayton result.AppendErrorWithFormat ("payload data must contain an even number of ASCII hex characters: '%s'", ascii_hex_bytes_cstr); 9391d19a2f2SGreg Clayton result.SetStatus (eReturnStatusFailed); 9401d19a2f2SGreg Clayton return false; 9411d19a2f2SGreg Clayton } 9421d19a2f2SGreg Clayton payload_bytes.resize(ascii_hex_bytes_cstr_len/2); 9431d19a2f2SGreg Clayton if (extractor.GetHexBytes(&payload_bytes[0], payload_bytes.size(), '\xdd') != payload_bytes.size()) 9441d19a2f2SGreg Clayton { 9451d19a2f2SGreg Clayton result.AppendErrorWithFormat ("payload data must only contain ASCII hex characters (no spaces or hex prefixes): '%s'", ascii_hex_bytes_cstr); 9461d19a2f2SGreg Clayton result.SetStatus (eReturnStatusFailed); 9471d19a2f2SGreg Clayton return false; 9481d19a2f2SGreg Clayton } 9491d19a2f2SGreg Clayton } 9501d19a2f2SGreg Clayton Error error; 9511d19a2f2SGreg Clayton DataExtractor reply; 9521d19a2f2SGreg Clayton process->GetCommunication().SendRawRequest (command_byte, 9531d19a2f2SGreg Clayton payload_bytes.empty() ? NULL : payload_bytes.data(), 9541d19a2f2SGreg Clayton payload_bytes.size(), 9551d19a2f2SGreg Clayton reply, 9561d19a2f2SGreg Clayton error); 9571d19a2f2SGreg Clayton 9581d19a2f2SGreg Clayton if (error.Success()) 9591d19a2f2SGreg Clayton { 9601d19a2f2SGreg Clayton // Copy the binary bytes into a hex ASCII string for the result 9611d19a2f2SGreg Clayton StreamString packet; 9621d19a2f2SGreg Clayton packet.PutBytesAsRawHex8(reply.GetDataStart(), 9631d19a2f2SGreg Clayton reply.GetByteSize(), 9641d19a2f2SGreg Clayton lldb::endian::InlHostByteOrder(), 9651d19a2f2SGreg Clayton lldb::endian::InlHostByteOrder()); 9661d19a2f2SGreg Clayton result.AppendMessage(packet.GetString().c_str()); 9671d19a2f2SGreg Clayton result.SetStatus (eReturnStatusSuccessFinishResult); 9681d19a2f2SGreg Clayton return true; 9691d19a2f2SGreg Clayton } 9701d19a2f2SGreg Clayton else 9711d19a2f2SGreg Clayton { 9721d19a2f2SGreg Clayton const char *error_cstr = error.AsCString(); 9731d19a2f2SGreg Clayton if (error_cstr && error_cstr[0]) 9741d19a2f2SGreg Clayton result.AppendError (error_cstr); 9751d19a2f2SGreg Clayton else 9761d19a2f2SGreg Clayton result.AppendErrorWithFormat ("unknown error 0x%8.8x", error.GetError()); 9771d19a2f2SGreg Clayton result.SetStatus (eReturnStatusFailed); 9781d19a2f2SGreg Clayton return false; 9791d19a2f2SGreg Clayton } 9801d19a2f2SGreg Clayton } 9811d19a2f2SGreg Clayton else 9821d19a2f2SGreg Clayton { 9831d19a2f2SGreg Clayton result.AppendErrorWithFormat ("process must be stopped in order to send KDP packets, state is %s", StateAsCString (state)); 9841d19a2f2SGreg Clayton result.SetStatus (eReturnStatusFailed); 9851d19a2f2SGreg Clayton } 9861d19a2f2SGreg Clayton } 9871d19a2f2SGreg Clayton else 9881d19a2f2SGreg Clayton { 9891d19a2f2SGreg Clayton result.AppendError ("invalid process"); 9901d19a2f2SGreg Clayton result.SetStatus (eReturnStatusFailed); 9911d19a2f2SGreg Clayton } 9921d19a2f2SGreg Clayton } 9931d19a2f2SGreg Clayton else 9941d19a2f2SGreg Clayton { 995d01b2953SDaniel Malea result.AppendErrorWithFormat ("invalid command byte 0x%" PRIx64 ", valid values are 1 - 255", command_byte); 9961d19a2f2SGreg Clayton result.SetStatus (eReturnStatusFailed); 9971d19a2f2SGreg Clayton } 9981d19a2f2SGreg Clayton } 9991d19a2f2SGreg Clayton } 10001d19a2f2SGreg Clayton else 10011d19a2f2SGreg Clayton { 10021d19a2f2SGreg Clayton result.AppendErrorWithFormat ("'%s' takes no arguments, only options.", m_cmd_name.c_str()); 10031d19a2f2SGreg Clayton result.SetStatus (eReturnStatusFailed); 10041d19a2f2SGreg Clayton } 10051d19a2f2SGreg Clayton return false; 10061d19a2f2SGreg Clayton } 10071d19a2f2SGreg Clayton }; 10081d19a2f2SGreg Clayton 10091d19a2f2SGreg Clayton class CommandObjectProcessKDPPacket : public CommandObjectMultiword 10101d19a2f2SGreg Clayton { 10111d19a2f2SGreg Clayton private: 10121d19a2f2SGreg Clayton 10131d19a2f2SGreg Clayton public: 10141d19a2f2SGreg Clayton CommandObjectProcessKDPPacket(CommandInterpreter &interpreter) : 10151d19a2f2SGreg Clayton CommandObjectMultiword (interpreter, 10161d19a2f2SGreg Clayton "process plugin packet", 10171d19a2f2SGreg Clayton "Commands that deal with KDP remote packets.", 10181d19a2f2SGreg Clayton NULL) 10191d19a2f2SGreg Clayton { 10201d19a2f2SGreg Clayton LoadSubCommand ("send", CommandObjectSP (new CommandObjectProcessKDPPacketSend (interpreter))); 10211d19a2f2SGreg Clayton } 10221d19a2f2SGreg Clayton 10231d19a2f2SGreg Clayton ~CommandObjectProcessKDPPacket () 10241d19a2f2SGreg Clayton { 10251d19a2f2SGreg Clayton } 10261d19a2f2SGreg Clayton }; 10271d19a2f2SGreg Clayton 10281d19a2f2SGreg Clayton class CommandObjectMultiwordProcessKDP : public CommandObjectMultiword 10291d19a2f2SGreg Clayton { 10301d19a2f2SGreg Clayton public: 10311d19a2f2SGreg Clayton CommandObjectMultiwordProcessKDP (CommandInterpreter &interpreter) : 10321d19a2f2SGreg Clayton CommandObjectMultiword (interpreter, 10331d19a2f2SGreg Clayton "process plugin", 10341d19a2f2SGreg Clayton "A set of commands for operating on a ProcessKDP process.", 10351d19a2f2SGreg Clayton "process plugin <subcommand> [<subcommand-options>]") 10361d19a2f2SGreg Clayton { 10371d19a2f2SGreg Clayton LoadSubCommand ("packet", CommandObjectSP (new CommandObjectProcessKDPPacket (interpreter))); 10381d19a2f2SGreg Clayton } 10391d19a2f2SGreg Clayton 10401d19a2f2SGreg Clayton ~CommandObjectMultiwordProcessKDP () 10411d19a2f2SGreg Clayton { 10421d19a2f2SGreg Clayton } 10431d19a2f2SGreg Clayton }; 10441d19a2f2SGreg Clayton 10451d19a2f2SGreg Clayton CommandObject * 10461d19a2f2SGreg Clayton ProcessKDP::GetPluginCommandObject() 10471d19a2f2SGreg Clayton { 10481d19a2f2SGreg Clayton if (!m_command_sp) 10491d19a2f2SGreg Clayton m_command_sp.reset (new CommandObjectMultiwordProcessKDP (GetTarget().GetDebugger().GetCommandInterpreter())); 10501d19a2f2SGreg Clayton return m_command_sp.get(); 10511d19a2f2SGreg Clayton } 10521d19a2f2SGreg Clayton 1053