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 } 245*a8ea4baeSJason Molenda else 246*a8ea4baeSJason Molenda { 247*a8ea4baeSJason Molenda if (kernel_load_addr != LLDB_INVALID_ADDRESS) 2484bd4e7e3SJason Molenda { 2495e8534efSJason Molenda m_kernel_load_addr = kernel_load_addr; 250*a8ea4baeSJason Molenda } 2515e8534efSJason Molenda m_dyld_plugin_name = DynamicLoaderDarwinKernel::GetPluginNameStatic(); 2524bd4e7e3SJason Molenda } 2534bd4e7e3SJason Molenda 25497d5cf05SGreg Clayton // Set the thread ID 25597d5cf05SGreg Clayton UpdateThreadListIfNeeded (); 256a63d08c9SGreg Clayton SetID (1); 25756d9a1b3SGreg Clayton GetThreadList (); 258a63d08c9SGreg Clayton SetPrivateState (eStateStopped); 25907e66e3eSGreg Clayton StreamSP async_strm_sp(m_target.GetDebugger().GetAsyncOutputStream()); 26007e66e3eSGreg Clayton if (async_strm_sp) 26107e66e3eSGreg Clayton { 2625b88216dSGreg Clayton const char *cstr; 2635b88216dSGreg Clayton if ((cstr = m_comm.GetKernelVersion ()) != NULL) 26407e66e3eSGreg Clayton { 2655b88216dSGreg Clayton async_strm_sp->Printf ("Version: %s\n", cstr); 26607e66e3eSGreg Clayton async_strm_sp->Flush(); 26707e66e3eSGreg Clayton } 2685b88216dSGreg Clayton // if ((cstr = m_comm.GetImagePath ()) != NULL) 2695b88216dSGreg Clayton // { 2705b88216dSGreg Clayton // async_strm_sp->Printf ("Image Path: %s\n", cstr); 2715b88216dSGreg Clayton // async_strm_sp->Flush(); 2725b88216dSGreg Clayton // } 27307e66e3eSGreg Clayton } 2743a29bdbeSGreg Clayton } 27597d5cf05SGreg Clayton else 27697d5cf05SGreg Clayton { 27797d5cf05SGreg Clayton error.SetErrorString("KDP_REATTACH failed"); 27897d5cf05SGreg Clayton } 2793a29bdbeSGreg Clayton } 2803a29bdbeSGreg Clayton else 2813a29bdbeSGreg Clayton { 28297d5cf05SGreg Clayton error.SetErrorString("KDP_REATTACH failed"); 2833a29bdbeSGreg Clayton } 2843a29bdbeSGreg Clayton } 2853a29bdbeSGreg Clayton else 2863a29bdbeSGreg Clayton { 2873a29bdbeSGreg Clayton error.SetErrorString("invalid reply port from UDP connection"); 2883a29bdbeSGreg Clayton } 2893a29bdbeSGreg Clayton } 2903a29bdbeSGreg Clayton else 2913a29bdbeSGreg Clayton { 2923a29bdbeSGreg Clayton if (error.Success()) 2933a29bdbeSGreg Clayton error.SetErrorStringWithFormat ("failed to connect to '%s'", remote_url); 2943a29bdbeSGreg Clayton } 2953a29bdbeSGreg Clayton if (error.Fail()) 2963a29bdbeSGreg Clayton m_comm.Disconnect(); 2973a29bdbeSGreg Clayton 298f9765acdSGreg Clayton return error; 299f9765acdSGreg Clayton } 300f9765acdSGreg Clayton 301f9765acdSGreg Clayton //---------------------------------------------------------------------- 302f9765acdSGreg Clayton // Process Control 303f9765acdSGreg Clayton //---------------------------------------------------------------------- 304f9765acdSGreg Clayton Error 305982c9762SGreg Clayton ProcessKDP::DoLaunch (Module *exe_module, 306982c9762SGreg Clayton const ProcessLaunchInfo &launch_info) 307f9765acdSGreg Clayton { 308f9765acdSGreg Clayton Error error; 309f9765acdSGreg Clayton error.SetErrorString ("launching not supported in kdp-remote plug-in"); 310f9765acdSGreg Clayton return error; 311f9765acdSGreg Clayton } 312f9765acdSGreg Clayton 313f9765acdSGreg Clayton 314f9765acdSGreg Clayton Error 315f9765acdSGreg Clayton ProcessKDP::DoAttachToProcessWithID (lldb::pid_t attach_pid) 316f9765acdSGreg Clayton { 317f9765acdSGreg Clayton Error error; 318f9765acdSGreg Clayton error.SetErrorString ("attach to process by ID is not suppported in kdp remote debugging"); 319f9765acdSGreg Clayton return error; 320f9765acdSGreg Clayton } 321f9765acdSGreg Clayton 322f9765acdSGreg Clayton Error 32384647048SHan Ming Ong ProcessKDP::DoAttachToProcessWithID (lldb::pid_t attach_pid, const ProcessAttachInfo &attach_info) 32484647048SHan Ming Ong { 32584647048SHan Ming Ong Error error; 32684647048SHan Ming Ong error.SetErrorString ("attach to process by ID is not suppported in kdp remote debugging"); 32784647048SHan Ming Ong return error; 32884647048SHan Ming Ong } 32984647048SHan Ming Ong 33084647048SHan Ming Ong Error 33184647048SHan Ming Ong ProcessKDP::DoAttachToProcessWithName (const char *process_name, bool wait_for_launch, const ProcessAttachInfo &attach_info) 332f9765acdSGreg Clayton { 333f9765acdSGreg Clayton Error error; 334f9765acdSGreg Clayton error.SetErrorString ("attach to process by name is not suppported in kdp remote debugging"); 335f9765acdSGreg Clayton return error; 336f9765acdSGreg Clayton } 337f9765acdSGreg Clayton 338f9765acdSGreg Clayton 339f9765acdSGreg Clayton void 340f9765acdSGreg Clayton ProcessKDP::DidAttach () 341f9765acdSGreg Clayton { 3425160ce5cSGreg Clayton Log *log (ProcessKDPLog::GetLogIfAllCategoriesSet (KDP_LOG_PROCESS)); 343f9765acdSGreg Clayton if (log) 34454cb8f83SJohnny Chen log->Printf ("ProcessKDP::DidAttach()"); 345f9765acdSGreg Clayton if (GetID() != LLDB_INVALID_PROCESS_ID) 346f9765acdSGreg Clayton { 347f9765acdSGreg Clayton // TODO: figure out the register context that we will use 348f9765acdSGreg Clayton } 349f9765acdSGreg Clayton } 350f9765acdSGreg Clayton 3515e8534efSJason Molenda addr_t 3525e8534efSJason Molenda ProcessKDP::GetImageInfoAddress() 3535e8534efSJason Molenda { 3545e8534efSJason Molenda return m_kernel_load_addr; 3555e8534efSJason Molenda } 3565e8534efSJason Molenda 3575e8534efSJason Molenda lldb_private::DynamicLoader * 3585e8534efSJason Molenda ProcessKDP::GetDynamicLoader () 3595e8534efSJason Molenda { 3605e8534efSJason Molenda if (m_dyld_ap.get() == NULL) 3615e8534efSJason Molenda m_dyld_ap.reset (DynamicLoader::FindPlugin(this, m_dyld_plugin_name.empty() ? NULL : m_dyld_plugin_name.c_str())); 3625e8534efSJason Molenda return m_dyld_ap.get(); 3635e8534efSJason Molenda } 3645e8534efSJason Molenda 365f9765acdSGreg Clayton Error 366f9765acdSGreg Clayton ProcessKDP::WillResume () 367f9765acdSGreg Clayton { 368f9765acdSGreg Clayton return Error(); 369f9765acdSGreg Clayton } 370f9765acdSGreg Clayton 371f9765acdSGreg Clayton Error 372f9765acdSGreg Clayton ProcessKDP::DoResume () 373f9765acdSGreg Clayton { 374f9765acdSGreg Clayton Error error; 3755160ce5cSGreg Clayton Log *log (ProcessKDPLog::GetLogIfAllCategoriesSet (KDP_LOG_PROCESS)); 3767925fbbaSGreg Clayton // Only start the async thread if we try to do any process control 3777925fbbaSGreg Clayton if (!IS_VALID_LLDB_HOST_THREAD(m_async_thread)) 3787925fbbaSGreg Clayton StartAsyncThread (); 3797925fbbaSGreg Clayton 38097d5cf05SGreg Clayton bool resume = false; 3817925fbbaSGreg Clayton 38297d5cf05SGreg Clayton // With KDP there is only one thread we can tell what to do 3831b7746e3SGreg Clayton ThreadSP kernel_thread_sp (m_thread_list.FindThreadByProtocolID(g_kernel_tid)); 3841b7746e3SGreg Clayton 38597d5cf05SGreg Clayton if (kernel_thread_sp) 3864b1b8b3eSGreg Clayton { 38797d5cf05SGreg Clayton const StateType thread_resume_state = kernel_thread_sp->GetTemporaryResumeState(); 3887925fbbaSGreg Clayton switch (thread_resume_state) 3894b1b8b3eSGreg Clayton { 3907925fbbaSGreg Clayton case eStateSuspended: 3917925fbbaSGreg Clayton // Nothing to do here when a thread will stay suspended 3927925fbbaSGreg Clayton // we just leave the CPU mask bit set to zero for the thread 3937925fbbaSGreg Clayton break; 3947925fbbaSGreg Clayton 3957925fbbaSGreg Clayton case eStateStepping: 3961afa68edSGreg Clayton { 3971afa68edSGreg Clayton lldb::RegisterContextSP reg_ctx_sp (kernel_thread_sp->GetRegisterContext()); 3981afa68edSGreg Clayton 3991afa68edSGreg Clayton if (reg_ctx_sp) 4001afa68edSGreg Clayton { 4011afa68edSGreg Clayton reg_ctx_sp->HardwareSingleStep (true); 40297d5cf05SGreg Clayton resume = true; 4031afa68edSGreg Clayton } 4041afa68edSGreg Clayton else 4051afa68edSGreg Clayton { 4061afa68edSGreg Clayton error.SetErrorStringWithFormat("KDP thread 0x%llx has no register context", kernel_thread_sp->GetID()); 4071afa68edSGreg Clayton } 4081afa68edSGreg Clayton } 4097925fbbaSGreg Clayton break; 4107925fbbaSGreg Clayton 41197d5cf05SGreg Clayton case eStateRunning: 4121afa68edSGreg Clayton { 4131afa68edSGreg Clayton lldb::RegisterContextSP reg_ctx_sp (kernel_thread_sp->GetRegisterContext()); 4141afa68edSGreg Clayton 4151afa68edSGreg Clayton if (reg_ctx_sp) 4161afa68edSGreg Clayton { 4171afa68edSGreg Clayton reg_ctx_sp->HardwareSingleStep (false); 41897d5cf05SGreg Clayton resume = true; 4191afa68edSGreg Clayton } 4201afa68edSGreg Clayton else 4211afa68edSGreg Clayton { 4221afa68edSGreg Clayton error.SetErrorStringWithFormat("KDP thread 0x%llx has no register context", kernel_thread_sp->GetID()); 4231afa68edSGreg Clayton } 4241afa68edSGreg Clayton } 4257925fbbaSGreg Clayton break; 4267925fbbaSGreg Clayton 4277925fbbaSGreg Clayton default: 42897d5cf05SGreg Clayton // The only valid thread resume states are listed above 4297925fbbaSGreg Clayton assert (!"invalid thread resume state"); 4307925fbbaSGreg Clayton break; 4314b1b8b3eSGreg Clayton } 4324b1b8b3eSGreg Clayton } 4337925fbbaSGreg Clayton 43497d5cf05SGreg Clayton if (resume) 43597d5cf05SGreg Clayton { 43697d5cf05SGreg Clayton if (log) 43797d5cf05SGreg Clayton log->Printf ("ProcessKDP::DoResume () sending resume"); 43897d5cf05SGreg Clayton 43997d5cf05SGreg Clayton if (m_comm.SendRequestResume ()) 4407925fbbaSGreg Clayton { 4417925fbbaSGreg Clayton m_async_broadcaster.BroadcastEvent (eBroadcastBitAsyncContinue); 4427925fbbaSGreg Clayton SetPrivateState(eStateRunning); 4437925fbbaSGreg Clayton } 4444b1b8b3eSGreg Clayton else 44507e66e3eSGreg Clayton error.SetErrorString ("KDP resume failed"); 4467925fbbaSGreg Clayton } 4477925fbbaSGreg Clayton else 4487925fbbaSGreg Clayton { 44997d5cf05SGreg Clayton error.SetErrorString ("kernel thread is suspended"); 4507925fbbaSGreg Clayton } 4517925fbbaSGreg Clayton 452f9765acdSGreg Clayton return error; 453f9765acdSGreg Clayton } 454f9765acdSGreg Clayton 45597d5cf05SGreg Clayton lldb::ThreadSP 4561b7746e3SGreg Clayton ProcessKDP::GetKernelThread() 45797d5cf05SGreg Clayton { 45897d5cf05SGreg Clayton // KDP only tells us about one thread/core. Any other threads will usually 45997d5cf05SGreg Clayton // be the ones that are read from memory by the OS plug-ins. 4601b7746e3SGreg Clayton 4611b7746e3SGreg Clayton ThreadSP thread_sp (m_kernel_thread_wp.lock()); 46297d5cf05SGreg Clayton if (!thread_sp) 4631b7746e3SGreg Clayton { 4641b7746e3SGreg Clayton thread_sp.reset(new ThreadKDP (*this, g_kernel_tid)); 4651b7746e3SGreg Clayton m_kernel_thread_wp = thread_sp; 4661b7746e3SGreg Clayton } 46797d5cf05SGreg Clayton return thread_sp; 46897d5cf05SGreg Clayton } 46997d5cf05SGreg Clayton 47097d5cf05SGreg Clayton 47197d5cf05SGreg Clayton 47297d5cf05SGreg Clayton 4739fc13556SGreg Clayton bool 47456d9a1b3SGreg Clayton ProcessKDP::UpdateThreadList (ThreadList &old_thread_list, ThreadList &new_thread_list) 475f9765acdSGreg Clayton { 476f9765acdSGreg Clayton // locker will keep a mutex locked until it goes out of scope 4775160ce5cSGreg Clayton Log *log (ProcessKDPLog::GetLogIfAllCategoriesSet (KDP_LOG_THREAD)); 478f9765acdSGreg Clayton if (log && log->GetMask().Test(KDP_LOG_VERBOSE)) 479d01b2953SDaniel Malea log->Printf ("ProcessKDP::%s (pid = %" PRIu64 ")", __FUNCTION__, GetID()); 480f9765acdSGreg Clayton 48139da3efdSGreg Clayton // Even though there is a CPU mask, it doesn't mean we can see each CPU 48297d5cf05SGreg Clayton // indivudually, there is really only one. Lets call this thread 1. 483513db4d9SJason Molenda ThreadSP thread_sp (old_thread_list.FindThreadByProtocolID(g_kernel_tid, false)); 4841b7746e3SGreg Clayton if (!thread_sp) 4851b7746e3SGreg Clayton thread_sp = GetKernelThread (); 4861b7746e3SGreg Clayton new_thread_list.AddThread(thread_sp); 48797d5cf05SGreg Clayton 4889fc13556SGreg Clayton return new_thread_list.GetSize(false) > 0; 489f9765acdSGreg Clayton } 490f9765acdSGreg Clayton 491f9765acdSGreg Clayton void 492f9765acdSGreg Clayton ProcessKDP::RefreshStateAfterStop () 493f9765acdSGreg Clayton { 494f9765acdSGreg Clayton // Let all threads recover from stopping and do any clean up based 495f9765acdSGreg Clayton // on the previous thread state (if any). 496f9765acdSGreg Clayton m_thread_list.RefreshStateAfterStop(); 497f9765acdSGreg Clayton } 498f9765acdSGreg Clayton 499f9765acdSGreg Clayton Error 500f9765acdSGreg Clayton ProcessKDP::DoHalt (bool &caused_stop) 501f9765acdSGreg Clayton { 502f9765acdSGreg Clayton Error error; 503f9765acdSGreg Clayton 50497d5cf05SGreg Clayton if (m_comm.IsRunning()) 505f9765acdSGreg Clayton { 50697d5cf05SGreg Clayton if (m_destroy_in_process) 50797d5cf05SGreg Clayton { 50897d5cf05SGreg Clayton // If we are attemping to destroy, we need to not return an error to 50997d5cf05SGreg Clayton // Halt or DoDestroy won't get called. 51097d5cf05SGreg Clayton // We are also currently running, so send a process stopped event 51197d5cf05SGreg Clayton SetPrivateState (eStateStopped); 512f9765acdSGreg Clayton } 513f9765acdSGreg Clayton else 514f9765acdSGreg Clayton { 51597d5cf05SGreg Clayton error.SetErrorString ("KDP cannot interrupt a running kernel"); 516f9765acdSGreg Clayton } 517f9765acdSGreg Clayton } 518f9765acdSGreg Clayton return error; 519f9765acdSGreg Clayton } 520f9765acdSGreg Clayton 521f9765acdSGreg Clayton Error 522acff8950SJim Ingham ProcessKDP::DoDetach(bool keep_stopped) 523f9765acdSGreg Clayton { 524f9765acdSGreg Clayton Error error; 5255160ce5cSGreg Clayton Log *log (ProcessKDPLog::GetLogIfAllCategoriesSet(KDP_LOG_PROCESS)); 526f9765acdSGreg Clayton if (log) 527acff8950SJim Ingham log->Printf ("ProcessKDP::DoDetach(keep_stopped = %i)", keep_stopped); 528f9765acdSGreg Clayton 52997d5cf05SGreg Clayton if (m_comm.IsRunning()) 53097d5cf05SGreg Clayton { 53197d5cf05SGreg Clayton // We are running and we can't interrupt a running kernel, so we need 53297d5cf05SGreg Clayton // to just close the connection to the kernel and hope for the best 53397d5cf05SGreg Clayton } 53497d5cf05SGreg Clayton else 53597d5cf05SGreg Clayton { 536f9765acdSGreg Clayton DisableAllBreakpointSites (); 537f9765acdSGreg Clayton 538f9765acdSGreg Clayton m_thread_list.DiscardThreadPlans(); 539f9765acdSGreg Clayton 540acff8950SJim Ingham // If we are going to keep the target stopped, then don't send the disconnect message. 541acff8950SJim Ingham if (!keep_stopped && m_comm.IsConnected()) 5423a29bdbeSGreg Clayton { 5433a29bdbeSGreg Clayton 5443a29bdbeSGreg Clayton m_comm.SendRequestDisconnect(); 5453a29bdbeSGreg Clayton 54657508026SGreg Clayton size_t response_size = m_comm.Disconnect (); 547f9765acdSGreg Clayton if (log) 548f9765acdSGreg Clayton { 549f9765acdSGreg Clayton if (response_size) 550f9765acdSGreg Clayton log->PutCString ("ProcessKDP::DoDetach() detach packet sent successfully"); 551f9765acdSGreg Clayton else 552f9765acdSGreg Clayton log->PutCString ("ProcessKDP::DoDetach() detach packet send failed"); 553f9765acdSGreg Clayton } 5543a29bdbeSGreg Clayton } 55597d5cf05SGreg Clayton } 556f9765acdSGreg Clayton StopAsyncThread (); 55774d4193eSGreg Clayton m_comm.Clear(); 558f9765acdSGreg Clayton 559f9765acdSGreg Clayton SetPrivateState (eStateDetached); 560f9765acdSGreg Clayton ResumePrivateStateThread(); 561f9765acdSGreg Clayton 562f9765acdSGreg Clayton //KillDebugserverProcess (); 563f9765acdSGreg Clayton return error; 564f9765acdSGreg Clayton } 565f9765acdSGreg Clayton 566f9765acdSGreg Clayton Error 567f9765acdSGreg Clayton ProcessKDP::DoDestroy () 568f9765acdSGreg Clayton { 5697925fbbaSGreg Clayton // For KDP there really is no difference between destroy and detach 570acff8950SJim Ingham bool keep_stopped = false; 571acff8950SJim Ingham return DoDetach(keep_stopped); 572f9765acdSGreg Clayton } 573f9765acdSGreg Clayton 574f9765acdSGreg Clayton //------------------------------------------------------------------ 575f9765acdSGreg Clayton // Process Queries 576f9765acdSGreg Clayton //------------------------------------------------------------------ 577f9765acdSGreg Clayton 578f9765acdSGreg Clayton bool 579f9765acdSGreg Clayton ProcessKDP::IsAlive () 580f9765acdSGreg Clayton { 581f9765acdSGreg Clayton return m_comm.IsConnected() && m_private_state.GetValue() != eStateExited; 582f9765acdSGreg Clayton } 583f9765acdSGreg Clayton 584f9765acdSGreg Clayton //------------------------------------------------------------------ 585f9765acdSGreg Clayton // Process Memory 586f9765acdSGreg Clayton //------------------------------------------------------------------ 587f9765acdSGreg Clayton size_t 588f9765acdSGreg Clayton ProcessKDP::DoReadMemory (addr_t addr, void *buf, size_t size, Error &error) 589f9765acdSGreg Clayton { 590a63d08c9SGreg Clayton if (m_comm.IsConnected()) 591a63d08c9SGreg Clayton return m_comm.SendRequestReadMemory (addr, buf, size, error); 592a63d08c9SGreg Clayton error.SetErrorString ("not connected"); 593f9765acdSGreg Clayton return 0; 594f9765acdSGreg Clayton } 595f9765acdSGreg Clayton 596f9765acdSGreg Clayton size_t 597f9765acdSGreg Clayton ProcessKDP::DoWriteMemory (addr_t addr, const void *buf, size_t size, Error &error) 598f9765acdSGreg Clayton { 5997925fbbaSGreg Clayton if (m_comm.IsConnected()) 6007925fbbaSGreg Clayton return m_comm.SendRequestWriteMemory (addr, buf, size, error); 6017925fbbaSGreg Clayton error.SetErrorString ("not connected"); 602f9765acdSGreg Clayton return 0; 603f9765acdSGreg Clayton } 604f9765acdSGreg Clayton 605f9765acdSGreg Clayton lldb::addr_t 606f9765acdSGreg Clayton ProcessKDP::DoAllocateMemory (size_t size, uint32_t permissions, Error &error) 607f9765acdSGreg Clayton { 608f9765acdSGreg Clayton error.SetErrorString ("memory allocation not suppported in kdp remote debugging"); 609f9765acdSGreg Clayton return LLDB_INVALID_ADDRESS; 610f9765acdSGreg Clayton } 611f9765acdSGreg Clayton 612f9765acdSGreg Clayton Error 613f9765acdSGreg Clayton ProcessKDP::DoDeallocateMemory (lldb::addr_t addr) 614f9765acdSGreg Clayton { 615f9765acdSGreg Clayton Error error; 616f9765acdSGreg Clayton error.SetErrorString ("memory deallocation not suppported in kdp remote debugging"); 617f9765acdSGreg Clayton return error; 618f9765acdSGreg Clayton } 619f9765acdSGreg Clayton 620f9765acdSGreg Clayton Error 621299c0c1cSJim Ingham ProcessKDP::EnableBreakpointSite (BreakpointSite *bp_site) 622f9765acdSGreg Clayton { 62307e66e3eSGreg Clayton if (m_comm.LocalBreakpointsAreSupported ()) 62407e66e3eSGreg Clayton { 62507e66e3eSGreg Clayton Error error; 6265b88216dSGreg Clayton if (!bp_site->IsEnabled()) 6275b88216dSGreg Clayton { 6285b88216dSGreg Clayton if (m_comm.SendRequestBreakpoint(true, bp_site->GetLoadAddress())) 6295b88216dSGreg Clayton { 6305b88216dSGreg Clayton bp_site->SetEnabled(true); 6315b88216dSGreg Clayton bp_site->SetType (BreakpointSite::eExternal); 6325b88216dSGreg Clayton } 6335b88216dSGreg Clayton else 6345b88216dSGreg Clayton { 63507e66e3eSGreg Clayton error.SetErrorString ("KDP set breakpoint failed"); 6365b88216dSGreg Clayton } 6375b88216dSGreg Clayton } 63807e66e3eSGreg Clayton return error; 63907e66e3eSGreg Clayton } 640f9765acdSGreg Clayton return EnableSoftwareBreakpoint (bp_site); 641f9765acdSGreg Clayton } 642f9765acdSGreg Clayton 643f9765acdSGreg Clayton Error 644299c0c1cSJim Ingham ProcessKDP::DisableBreakpointSite (BreakpointSite *bp_site) 645f9765acdSGreg Clayton { 64607e66e3eSGreg Clayton if (m_comm.LocalBreakpointsAreSupported ()) 64707e66e3eSGreg Clayton { 64807e66e3eSGreg Clayton Error error; 6495b88216dSGreg Clayton if (bp_site->IsEnabled()) 6505b88216dSGreg Clayton { 6515b88216dSGreg Clayton BreakpointSite::Type bp_type = bp_site->GetType(); 6525b88216dSGreg Clayton if (bp_type == BreakpointSite::eExternal) 6535b88216dSGreg Clayton { 65497d5cf05SGreg Clayton if (m_destroy_in_process && m_comm.IsRunning()) 65597d5cf05SGreg Clayton { 65697d5cf05SGreg Clayton // We are trying to destroy our connection and we are running 65797d5cf05SGreg Clayton bp_site->SetEnabled(false); 65897d5cf05SGreg Clayton } 65997d5cf05SGreg Clayton else 66097d5cf05SGreg Clayton { 6615b88216dSGreg Clayton if (m_comm.SendRequestBreakpoint(false, bp_site->GetLoadAddress())) 6625b88216dSGreg Clayton bp_site->SetEnabled(false); 6635b88216dSGreg Clayton else 66407e66e3eSGreg Clayton error.SetErrorString ("KDP remove breakpoint failed"); 6655b88216dSGreg Clayton } 66697d5cf05SGreg Clayton } 6675b88216dSGreg Clayton else 6685b88216dSGreg Clayton { 6695b88216dSGreg Clayton error = DisableSoftwareBreakpoint (bp_site); 6705b88216dSGreg Clayton } 6715b88216dSGreg Clayton } 67207e66e3eSGreg Clayton return error; 67307e66e3eSGreg Clayton } 674f9765acdSGreg Clayton return DisableSoftwareBreakpoint (bp_site); 675f9765acdSGreg Clayton } 676f9765acdSGreg Clayton 677f9765acdSGreg Clayton Error 6781b5792e5SJim Ingham ProcessKDP::EnableWatchpoint (Watchpoint *wp, bool notify) 679f9765acdSGreg Clayton { 680f9765acdSGreg Clayton Error error; 681f9765acdSGreg Clayton error.SetErrorString ("watchpoints are not suppported in kdp remote debugging"); 682f9765acdSGreg Clayton return error; 683f9765acdSGreg Clayton } 684f9765acdSGreg Clayton 685f9765acdSGreg Clayton Error 6861b5792e5SJim Ingham ProcessKDP::DisableWatchpoint (Watchpoint *wp, bool notify) 687f9765acdSGreg Clayton { 688f9765acdSGreg Clayton Error error; 689f9765acdSGreg Clayton error.SetErrorString ("watchpoints are not suppported in kdp remote debugging"); 690f9765acdSGreg Clayton return error; 691f9765acdSGreg Clayton } 692f9765acdSGreg Clayton 693f9765acdSGreg Clayton void 694f9765acdSGreg Clayton ProcessKDP::Clear() 695f9765acdSGreg Clayton { 696f9765acdSGreg Clayton m_thread_list.Clear(); 697f9765acdSGreg Clayton } 698f9765acdSGreg Clayton 699f9765acdSGreg Clayton Error 700f9765acdSGreg Clayton ProcessKDP::DoSignal (int signo) 701f9765acdSGreg Clayton { 702f9765acdSGreg Clayton Error error; 703f9765acdSGreg Clayton error.SetErrorString ("sending signals is not suppported in kdp remote debugging"); 704f9765acdSGreg Clayton return error; 705f9765acdSGreg Clayton } 706f9765acdSGreg Clayton 707f9765acdSGreg Clayton void 708f9765acdSGreg Clayton ProcessKDP::Initialize() 709f9765acdSGreg Clayton { 710f9765acdSGreg Clayton static bool g_initialized = false; 711f9765acdSGreg Clayton 712f9765acdSGreg Clayton if (g_initialized == false) 713f9765acdSGreg Clayton { 714f9765acdSGreg Clayton g_initialized = true; 715f9765acdSGreg Clayton PluginManager::RegisterPlugin (GetPluginNameStatic(), 716f9765acdSGreg Clayton GetPluginDescriptionStatic(), 717f9765acdSGreg Clayton CreateInstance); 718f9765acdSGreg Clayton 719f9765acdSGreg Clayton Log::Callbacks log_callbacks = { 720f9765acdSGreg Clayton ProcessKDPLog::DisableLog, 721f9765acdSGreg Clayton ProcessKDPLog::EnableLog, 722f9765acdSGreg Clayton ProcessKDPLog::ListLogCategories 723f9765acdSGreg Clayton }; 724f9765acdSGreg Clayton 725f9765acdSGreg Clayton Log::RegisterLogChannel (ProcessKDP::GetPluginNameStatic(), log_callbacks); 726f9765acdSGreg Clayton } 727f9765acdSGreg Clayton } 728f9765acdSGreg Clayton 729f9765acdSGreg Clayton bool 730f9765acdSGreg Clayton ProcessKDP::StartAsyncThread () 731f9765acdSGreg Clayton { 7325160ce5cSGreg Clayton Log *log (ProcessKDPLog::GetLogIfAllCategoriesSet(KDP_LOG_PROCESS)); 733f9765acdSGreg Clayton 734f9765acdSGreg Clayton if (log) 7357925fbbaSGreg Clayton log->Printf ("ProcessKDP::StartAsyncThread ()"); 736f9765acdSGreg Clayton 7377925fbbaSGreg Clayton if (IS_VALID_LLDB_HOST_THREAD(m_async_thread)) 7387925fbbaSGreg Clayton return true; 7397925fbbaSGreg Clayton 740f9765acdSGreg Clayton m_async_thread = Host::ThreadCreate ("<lldb.process.kdp-remote.async>", ProcessKDP::AsyncThread, this, NULL); 741f9765acdSGreg Clayton return IS_VALID_LLDB_HOST_THREAD(m_async_thread); 742f9765acdSGreg Clayton } 743f9765acdSGreg Clayton 744f9765acdSGreg Clayton void 745f9765acdSGreg Clayton ProcessKDP::StopAsyncThread () 746f9765acdSGreg Clayton { 7475160ce5cSGreg Clayton Log *log (ProcessKDPLog::GetLogIfAllCategoriesSet(KDP_LOG_PROCESS)); 748f9765acdSGreg Clayton 749f9765acdSGreg Clayton if (log) 7507925fbbaSGreg Clayton log->Printf ("ProcessKDP::StopAsyncThread ()"); 751f9765acdSGreg Clayton 752f9765acdSGreg Clayton m_async_broadcaster.BroadcastEvent (eBroadcastBitAsyncThreadShouldExit); 753f9765acdSGreg Clayton 754f9765acdSGreg Clayton // Stop the stdio thread 755f9765acdSGreg Clayton if (IS_VALID_LLDB_HOST_THREAD(m_async_thread)) 756f9765acdSGreg Clayton { 757f9765acdSGreg Clayton Host::ThreadJoin (m_async_thread, NULL, NULL); 7587925fbbaSGreg Clayton m_async_thread = LLDB_INVALID_HOST_THREAD; 759f9765acdSGreg Clayton } 760f9765acdSGreg Clayton } 761f9765acdSGreg Clayton 762f9765acdSGreg Clayton 763f9765acdSGreg Clayton void * 764f9765acdSGreg Clayton ProcessKDP::AsyncThread (void *arg) 765f9765acdSGreg Clayton { 766f9765acdSGreg Clayton ProcessKDP *process = (ProcessKDP*) arg; 767f9765acdSGreg Clayton 7687925fbbaSGreg Clayton const lldb::pid_t pid = process->GetID(); 7697925fbbaSGreg Clayton 7705160ce5cSGreg Clayton Log *log (ProcessKDPLog::GetLogIfAllCategoriesSet (KDP_LOG_PROCESS)); 771f9765acdSGreg Clayton if (log) 772d01b2953SDaniel Malea log->Printf ("ProcessKDP::AsyncThread (arg = %p, pid = %" PRIu64 ") thread starting...", arg, pid); 773f9765acdSGreg Clayton 774f9765acdSGreg Clayton Listener listener ("ProcessKDP::AsyncThread"); 775f9765acdSGreg Clayton EventSP event_sp; 776f9765acdSGreg Clayton const uint32_t desired_event_mask = eBroadcastBitAsyncContinue | 777f9765acdSGreg Clayton eBroadcastBitAsyncThreadShouldExit; 778f9765acdSGreg Clayton 7797925fbbaSGreg Clayton 780f9765acdSGreg Clayton if (listener.StartListeningForEvents (&process->m_async_broadcaster, desired_event_mask) == desired_event_mask) 781f9765acdSGreg Clayton { 782f9765acdSGreg Clayton bool done = false; 783f9765acdSGreg Clayton while (!done) 784f9765acdSGreg Clayton { 785f9765acdSGreg Clayton if (log) 786d01b2953SDaniel Malea log->Printf ("ProcessKDP::AsyncThread (pid = %" PRIu64 ") listener.WaitForEvent (NULL, event_sp)...", 7877925fbbaSGreg Clayton pid); 788f9765acdSGreg Clayton if (listener.WaitForEvent (NULL, event_sp)) 789f9765acdSGreg Clayton { 7907925fbbaSGreg Clayton uint32_t event_type = event_sp->GetType(); 791f9765acdSGreg Clayton if (log) 792d01b2953SDaniel Malea log->Printf ("ProcessKDP::AsyncThread (pid = %" PRIu64 ") Got an event of type: %d...", 7937925fbbaSGreg Clayton pid, 7947925fbbaSGreg Clayton event_type); 795f9765acdSGreg Clayton 7967925fbbaSGreg Clayton // When we are running, poll for 1 second to try and get an exception 7977925fbbaSGreg Clayton // to indicate the process has stopped. If we don't get one, check to 7987925fbbaSGreg Clayton // make sure no one asked us to exit 7997925fbbaSGreg Clayton bool is_running = false; 8007925fbbaSGreg Clayton DataExtractor exc_reply_packet; 8017925fbbaSGreg Clayton do 8027925fbbaSGreg Clayton { 803f9765acdSGreg Clayton switch (event_type) 804f9765acdSGreg Clayton { 805f9765acdSGreg Clayton case eBroadcastBitAsyncContinue: 806f9765acdSGreg Clayton { 8077925fbbaSGreg Clayton is_running = true; 8087925fbbaSGreg Clayton if (process->m_comm.WaitForPacketWithTimeoutMicroSeconds (exc_reply_packet, 1 * USEC_PER_SEC)) 809f9765acdSGreg Clayton { 8101b7746e3SGreg Clayton ThreadSP thread_sp (process->GetKernelThread()); 8111afa68edSGreg Clayton if (thread_sp) 8121afa68edSGreg Clayton { 8131afa68edSGreg Clayton lldb::RegisterContextSP reg_ctx_sp (thread_sp->GetRegisterContext()); 8141afa68edSGreg Clayton if (reg_ctx_sp) 8151afa68edSGreg Clayton reg_ctx_sp->InvalidateAllRegisters(); 81697d5cf05SGreg Clayton static_cast<ThreadKDP *>(thread_sp.get())->SetStopInfoFrom_KDP_EXCEPTION (exc_reply_packet); 8171afa68edSGreg Clayton } 81897d5cf05SGreg Clayton 8197925fbbaSGreg Clayton // TODO: parse the stop reply packet 8207925fbbaSGreg Clayton is_running = false; 8217925fbbaSGreg Clayton process->SetPrivateState(eStateStopped); 8227925fbbaSGreg Clayton } 8237925fbbaSGreg Clayton else 8247925fbbaSGreg Clayton { 8257925fbbaSGreg Clayton // Check to see if we are supposed to exit. There is no way to 8267925fbbaSGreg Clayton // interrupt a running kernel, so all we can do is wait for an 8277925fbbaSGreg Clayton // exception or detach... 8287925fbbaSGreg Clayton if (listener.GetNextEvent(event_sp)) 8297925fbbaSGreg Clayton { 8307925fbbaSGreg Clayton // We got an event, go through the loop again 8317925fbbaSGreg Clayton event_type = event_sp->GetType(); 8327925fbbaSGreg Clayton } 833f9765acdSGreg Clayton } 834f9765acdSGreg Clayton } 835f9765acdSGreg Clayton break; 836f9765acdSGreg Clayton 837f9765acdSGreg Clayton case eBroadcastBitAsyncThreadShouldExit: 838f9765acdSGreg Clayton if (log) 839d01b2953SDaniel Malea log->Printf ("ProcessKDP::AsyncThread (pid = %" PRIu64 ") got eBroadcastBitAsyncThreadShouldExit...", 8407925fbbaSGreg Clayton pid); 841f9765acdSGreg Clayton done = true; 8427925fbbaSGreg Clayton is_running = false; 843f9765acdSGreg Clayton break; 844f9765acdSGreg Clayton 845f9765acdSGreg Clayton default: 846f9765acdSGreg Clayton if (log) 847d01b2953SDaniel Malea log->Printf ("ProcessKDP::AsyncThread (pid = %" PRIu64 ") got unknown event 0x%8.8x", 8487925fbbaSGreg Clayton pid, 8497925fbbaSGreg Clayton event_type); 850f9765acdSGreg Clayton done = true; 8517925fbbaSGreg Clayton is_running = false; 852f9765acdSGreg Clayton break; 853f9765acdSGreg Clayton } 8547925fbbaSGreg Clayton } while (is_running); 855f9765acdSGreg Clayton } 856f9765acdSGreg Clayton else 857f9765acdSGreg Clayton { 858f9765acdSGreg Clayton if (log) 859d01b2953SDaniel Malea log->Printf ("ProcessKDP::AsyncThread (pid = %" PRIu64 ") listener.WaitForEvent (NULL, event_sp) => false", 8607925fbbaSGreg Clayton pid); 861f9765acdSGreg Clayton done = true; 862f9765acdSGreg Clayton } 863f9765acdSGreg Clayton } 864f9765acdSGreg Clayton } 865f9765acdSGreg Clayton 866f9765acdSGreg Clayton if (log) 867d01b2953SDaniel Malea log->Printf ("ProcessKDP::AsyncThread (arg = %p, pid = %" PRIu64 ") thread exiting...", 8687925fbbaSGreg Clayton arg, 8697925fbbaSGreg Clayton pid); 870f9765acdSGreg Clayton 871f9765acdSGreg Clayton process->m_async_thread = LLDB_INVALID_HOST_THREAD; 872f9765acdSGreg Clayton return NULL; 873f9765acdSGreg Clayton } 874f9765acdSGreg Clayton 875f9765acdSGreg Clayton 8761d19a2f2SGreg Clayton class CommandObjectProcessKDPPacketSend : public CommandObjectParsed 8771d19a2f2SGreg Clayton { 8781d19a2f2SGreg Clayton private: 8791d19a2f2SGreg Clayton 8801d19a2f2SGreg Clayton OptionGroupOptions m_option_group; 8811d19a2f2SGreg Clayton OptionGroupUInt64 m_command_byte; 8821d19a2f2SGreg Clayton OptionGroupString m_packet_data; 8831d19a2f2SGreg Clayton 8841d19a2f2SGreg Clayton virtual Options * 8851d19a2f2SGreg Clayton GetOptions () 8861d19a2f2SGreg Clayton { 8871d19a2f2SGreg Clayton return &m_option_group; 8881d19a2f2SGreg Clayton } 8891d19a2f2SGreg Clayton 8901d19a2f2SGreg Clayton 8911d19a2f2SGreg Clayton public: 8921d19a2f2SGreg Clayton CommandObjectProcessKDPPacketSend(CommandInterpreter &interpreter) : 8931d19a2f2SGreg Clayton CommandObjectParsed (interpreter, 8941d19a2f2SGreg Clayton "process plugin packet send", 8951d19a2f2SGreg 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. ", 8961d19a2f2SGreg Clayton NULL), 8971d19a2f2SGreg Clayton m_option_group (interpreter), 8981d19a2f2SGreg 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), 8991d19a2f2SGreg 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) 9001d19a2f2SGreg Clayton { 9011d19a2f2SGreg Clayton m_option_group.Append (&m_command_byte, LLDB_OPT_SET_ALL, LLDB_OPT_SET_1); 9021d19a2f2SGreg Clayton m_option_group.Append (&m_packet_data , LLDB_OPT_SET_ALL, LLDB_OPT_SET_1); 9031d19a2f2SGreg Clayton m_option_group.Finalize(); 9041d19a2f2SGreg Clayton } 9051d19a2f2SGreg Clayton 9061d19a2f2SGreg Clayton ~CommandObjectProcessKDPPacketSend () 9071d19a2f2SGreg Clayton { 9081d19a2f2SGreg Clayton } 9091d19a2f2SGreg Clayton 9101d19a2f2SGreg Clayton bool 9111d19a2f2SGreg Clayton DoExecute (Args& command, CommandReturnObject &result) 9121d19a2f2SGreg Clayton { 9131d19a2f2SGreg Clayton const size_t argc = command.GetArgumentCount(); 9141d19a2f2SGreg Clayton if (argc == 0) 9151d19a2f2SGreg Clayton { 9161d19a2f2SGreg Clayton if (!m_command_byte.GetOptionValue().OptionWasSet()) 9171d19a2f2SGreg Clayton { 9181d19a2f2SGreg Clayton result.AppendError ("the --command option must be set to a valid command byte"); 9191d19a2f2SGreg Clayton result.SetStatus (eReturnStatusFailed); 9201d19a2f2SGreg Clayton } 9211d19a2f2SGreg Clayton else 9221d19a2f2SGreg Clayton { 9231d19a2f2SGreg Clayton const uint64_t command_byte = m_command_byte.GetOptionValue().GetUInt64Value(0); 9241d19a2f2SGreg Clayton if (command_byte > 0 && command_byte <= UINT8_MAX) 9251d19a2f2SGreg Clayton { 9261d19a2f2SGreg Clayton ProcessKDP *process = (ProcessKDP *)m_interpreter.GetExecutionContext().GetProcessPtr(); 9271d19a2f2SGreg Clayton if (process) 9281d19a2f2SGreg Clayton { 9291d19a2f2SGreg Clayton const StateType state = process->GetState(); 9301d19a2f2SGreg Clayton 9311d19a2f2SGreg Clayton if (StateIsStoppedState (state, true)) 9321d19a2f2SGreg Clayton { 9331d19a2f2SGreg Clayton std::vector<uint8_t> payload_bytes; 9341d19a2f2SGreg Clayton const char *ascii_hex_bytes_cstr = m_packet_data.GetOptionValue().GetCurrentValue(); 9351d19a2f2SGreg Clayton if (ascii_hex_bytes_cstr && ascii_hex_bytes_cstr[0]) 9361d19a2f2SGreg Clayton { 9371d19a2f2SGreg Clayton StringExtractor extractor(ascii_hex_bytes_cstr); 9381d19a2f2SGreg Clayton const size_t ascii_hex_bytes_cstr_len = extractor.GetStringRef().size(); 9391d19a2f2SGreg Clayton if (ascii_hex_bytes_cstr_len & 1) 9401d19a2f2SGreg Clayton { 9411d19a2f2SGreg Clayton result.AppendErrorWithFormat ("payload data must contain an even number of ASCII hex characters: '%s'", ascii_hex_bytes_cstr); 9421d19a2f2SGreg Clayton result.SetStatus (eReturnStatusFailed); 9431d19a2f2SGreg Clayton return false; 9441d19a2f2SGreg Clayton } 9451d19a2f2SGreg Clayton payload_bytes.resize(ascii_hex_bytes_cstr_len/2); 9461d19a2f2SGreg Clayton if (extractor.GetHexBytes(&payload_bytes[0], payload_bytes.size(), '\xdd') != payload_bytes.size()) 9471d19a2f2SGreg Clayton { 9481d19a2f2SGreg Clayton result.AppendErrorWithFormat ("payload data must only contain ASCII hex characters (no spaces or hex prefixes): '%s'", ascii_hex_bytes_cstr); 9491d19a2f2SGreg Clayton result.SetStatus (eReturnStatusFailed); 9501d19a2f2SGreg Clayton return false; 9511d19a2f2SGreg Clayton } 9521d19a2f2SGreg Clayton } 9531d19a2f2SGreg Clayton Error error; 9541d19a2f2SGreg Clayton DataExtractor reply; 9551d19a2f2SGreg Clayton process->GetCommunication().SendRawRequest (command_byte, 9561d19a2f2SGreg Clayton payload_bytes.empty() ? NULL : payload_bytes.data(), 9571d19a2f2SGreg Clayton payload_bytes.size(), 9581d19a2f2SGreg Clayton reply, 9591d19a2f2SGreg Clayton error); 9601d19a2f2SGreg Clayton 9611d19a2f2SGreg Clayton if (error.Success()) 9621d19a2f2SGreg Clayton { 9631d19a2f2SGreg Clayton // Copy the binary bytes into a hex ASCII string for the result 9641d19a2f2SGreg Clayton StreamString packet; 9651d19a2f2SGreg Clayton packet.PutBytesAsRawHex8(reply.GetDataStart(), 9661d19a2f2SGreg Clayton reply.GetByteSize(), 9671d19a2f2SGreg Clayton lldb::endian::InlHostByteOrder(), 9681d19a2f2SGreg Clayton lldb::endian::InlHostByteOrder()); 9691d19a2f2SGreg Clayton result.AppendMessage(packet.GetString().c_str()); 9701d19a2f2SGreg Clayton result.SetStatus (eReturnStatusSuccessFinishResult); 9711d19a2f2SGreg Clayton return true; 9721d19a2f2SGreg Clayton } 9731d19a2f2SGreg Clayton else 9741d19a2f2SGreg Clayton { 9751d19a2f2SGreg Clayton const char *error_cstr = error.AsCString(); 9761d19a2f2SGreg Clayton if (error_cstr && error_cstr[0]) 9771d19a2f2SGreg Clayton result.AppendError (error_cstr); 9781d19a2f2SGreg Clayton else 9791d19a2f2SGreg Clayton result.AppendErrorWithFormat ("unknown error 0x%8.8x", error.GetError()); 9801d19a2f2SGreg Clayton result.SetStatus (eReturnStatusFailed); 9811d19a2f2SGreg Clayton return false; 9821d19a2f2SGreg Clayton } 9831d19a2f2SGreg Clayton } 9841d19a2f2SGreg Clayton else 9851d19a2f2SGreg Clayton { 9861d19a2f2SGreg Clayton result.AppendErrorWithFormat ("process must be stopped in order to send KDP packets, state is %s", StateAsCString (state)); 9871d19a2f2SGreg Clayton result.SetStatus (eReturnStatusFailed); 9881d19a2f2SGreg Clayton } 9891d19a2f2SGreg Clayton } 9901d19a2f2SGreg Clayton else 9911d19a2f2SGreg Clayton { 9921d19a2f2SGreg Clayton result.AppendError ("invalid process"); 9931d19a2f2SGreg Clayton result.SetStatus (eReturnStatusFailed); 9941d19a2f2SGreg Clayton } 9951d19a2f2SGreg Clayton } 9961d19a2f2SGreg Clayton else 9971d19a2f2SGreg Clayton { 998d01b2953SDaniel Malea result.AppendErrorWithFormat ("invalid command byte 0x%" PRIx64 ", valid values are 1 - 255", command_byte); 9991d19a2f2SGreg Clayton result.SetStatus (eReturnStatusFailed); 10001d19a2f2SGreg Clayton } 10011d19a2f2SGreg Clayton } 10021d19a2f2SGreg Clayton } 10031d19a2f2SGreg Clayton else 10041d19a2f2SGreg Clayton { 10051d19a2f2SGreg Clayton result.AppendErrorWithFormat ("'%s' takes no arguments, only options.", m_cmd_name.c_str()); 10061d19a2f2SGreg Clayton result.SetStatus (eReturnStatusFailed); 10071d19a2f2SGreg Clayton } 10081d19a2f2SGreg Clayton return false; 10091d19a2f2SGreg Clayton } 10101d19a2f2SGreg Clayton }; 10111d19a2f2SGreg Clayton 10121d19a2f2SGreg Clayton class CommandObjectProcessKDPPacket : public CommandObjectMultiword 10131d19a2f2SGreg Clayton { 10141d19a2f2SGreg Clayton private: 10151d19a2f2SGreg Clayton 10161d19a2f2SGreg Clayton public: 10171d19a2f2SGreg Clayton CommandObjectProcessKDPPacket(CommandInterpreter &interpreter) : 10181d19a2f2SGreg Clayton CommandObjectMultiword (interpreter, 10191d19a2f2SGreg Clayton "process plugin packet", 10201d19a2f2SGreg Clayton "Commands that deal with KDP remote packets.", 10211d19a2f2SGreg Clayton NULL) 10221d19a2f2SGreg Clayton { 10231d19a2f2SGreg Clayton LoadSubCommand ("send", CommandObjectSP (new CommandObjectProcessKDPPacketSend (interpreter))); 10241d19a2f2SGreg Clayton } 10251d19a2f2SGreg Clayton 10261d19a2f2SGreg Clayton ~CommandObjectProcessKDPPacket () 10271d19a2f2SGreg Clayton { 10281d19a2f2SGreg Clayton } 10291d19a2f2SGreg Clayton }; 10301d19a2f2SGreg Clayton 10311d19a2f2SGreg Clayton class CommandObjectMultiwordProcessKDP : public CommandObjectMultiword 10321d19a2f2SGreg Clayton { 10331d19a2f2SGreg Clayton public: 10341d19a2f2SGreg Clayton CommandObjectMultiwordProcessKDP (CommandInterpreter &interpreter) : 10351d19a2f2SGreg Clayton CommandObjectMultiword (interpreter, 10361d19a2f2SGreg Clayton "process plugin", 10371d19a2f2SGreg Clayton "A set of commands for operating on a ProcessKDP process.", 10381d19a2f2SGreg Clayton "process plugin <subcommand> [<subcommand-options>]") 10391d19a2f2SGreg Clayton { 10401d19a2f2SGreg Clayton LoadSubCommand ("packet", CommandObjectSP (new CommandObjectProcessKDPPacket (interpreter))); 10411d19a2f2SGreg Clayton } 10421d19a2f2SGreg Clayton 10431d19a2f2SGreg Clayton ~CommandObjectMultiwordProcessKDP () 10441d19a2f2SGreg Clayton { 10451d19a2f2SGreg Clayton } 10461d19a2f2SGreg Clayton }; 10471d19a2f2SGreg Clayton 10481d19a2f2SGreg Clayton CommandObject * 10491d19a2f2SGreg Clayton ProcessKDP::GetPluginCommandObject() 10501d19a2f2SGreg Clayton { 10511d19a2f2SGreg Clayton if (!m_command_sp) 10521d19a2f2SGreg Clayton m_command_sp.reset (new CommandObjectMultiwordProcessKDP (GetTarget().GetDebugger().GetCommandInterpreter())); 10531d19a2f2SGreg Clayton return m_command_sp.get(); 10541d19a2f2SGreg Clayton } 10551d19a2f2SGreg Clayton 1056