xref: /openbsd-src/gnu/llvm/lldb/source/Plugins/Process/scripted/ScriptedProcess.cpp (revision f6aab3d83b51b91c24247ad2c2573574de475a82)
1be691f3bSpatrick //===-- ScriptedProcess.cpp -----------------------------------------------===//
2be691f3bSpatrick //
3be691f3bSpatrick // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4be691f3bSpatrick // See https://llvm.org/LICENSE.txt for license information.
5be691f3bSpatrick // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6be691f3bSpatrick //
7be691f3bSpatrick //===----------------------------------------------------------------------===//
8be691f3bSpatrick 
9be691f3bSpatrick #include "ScriptedProcess.h"
10be691f3bSpatrick 
11be691f3bSpatrick #include "lldb/Core/Debugger.h"
12be691f3bSpatrick #include "lldb/Core/Module.h"
13be691f3bSpatrick #include "lldb/Core/PluginManager.h"
14be691f3bSpatrick 
15be691f3bSpatrick #include "lldb/Host/OptionParser.h"
16be691f3bSpatrick #include "lldb/Host/ThreadLauncher.h"
17be691f3bSpatrick #include "lldb/Interpreter/CommandInterpreter.h"
18be691f3bSpatrick #include "lldb/Interpreter/OptionArgParser.h"
19be691f3bSpatrick #include "lldb/Interpreter/OptionGroupBoolean.h"
20be691f3bSpatrick #include "lldb/Interpreter/ScriptInterpreter.h"
21*f6aab3d8Srobert #include "lldb/Interpreter/ScriptedMetadata.h"
22be691f3bSpatrick #include "lldb/Target/MemoryRegionInfo.h"
23*f6aab3d8Srobert #include "lldb/Target/Queue.h"
24be691f3bSpatrick #include "lldb/Target/RegisterContext.h"
25*f6aab3d8Srobert #include "lldb/Utility/LLDBLog.h"
26be691f3bSpatrick #include "lldb/Utility/State.h"
27be691f3bSpatrick 
28be691f3bSpatrick #include <mutex>
29be691f3bSpatrick 
30be691f3bSpatrick LLDB_PLUGIN_DEFINE(ScriptedProcess)
31be691f3bSpatrick 
32be691f3bSpatrick using namespace lldb;
33be691f3bSpatrick using namespace lldb_private;
34be691f3bSpatrick 
GetPluginDescriptionStatic()35*f6aab3d8Srobert llvm::StringRef ScriptedProcess::GetPluginDescriptionStatic() {
36be691f3bSpatrick   return "Scripted Process plug-in.";
37be691f3bSpatrick }
38be691f3bSpatrick 
39be691f3bSpatrick static constexpr lldb::ScriptLanguage g_supported_script_languages[] = {
40be691f3bSpatrick     ScriptLanguage::eScriptLanguagePython,
41be691f3bSpatrick };
42be691f3bSpatrick 
IsScriptLanguageSupported(lldb::ScriptLanguage language)43be691f3bSpatrick bool ScriptedProcess::IsScriptLanguageSupported(lldb::ScriptLanguage language) {
44be691f3bSpatrick   llvm::ArrayRef<lldb::ScriptLanguage> supported_languages =
45*f6aab3d8Srobert       llvm::ArrayRef(g_supported_script_languages);
46be691f3bSpatrick 
47be691f3bSpatrick   return llvm::is_contained(supported_languages, language);
48be691f3bSpatrick }
49be691f3bSpatrick 
CheckInterpreterAndScriptObject() const50be691f3bSpatrick void ScriptedProcess::CheckInterpreterAndScriptObject() const {
51be691f3bSpatrick   lldbassert(m_interpreter && "Invalid Script Interpreter.");
52be691f3bSpatrick   lldbassert(m_script_object_sp && "Invalid Script Object.");
53be691f3bSpatrick }
54be691f3bSpatrick 
CreateInstance(lldb::TargetSP target_sp,lldb::ListenerSP listener_sp,const FileSpec * file,bool can_connect)55be691f3bSpatrick lldb::ProcessSP ScriptedProcess::CreateInstance(lldb::TargetSP target_sp,
56be691f3bSpatrick                                                 lldb::ListenerSP listener_sp,
57be691f3bSpatrick                                                 const FileSpec *file,
58be691f3bSpatrick                                                 bool can_connect) {
59be691f3bSpatrick   if (!target_sp ||
60be691f3bSpatrick       !IsScriptLanguageSupported(target_sp->GetDebugger().GetScriptLanguage()))
61be691f3bSpatrick     return nullptr;
62be691f3bSpatrick 
63*f6aab3d8Srobert   ScriptedMetadata scripted_metadata(target_sp->GetProcessLaunchInfo());
64be691f3bSpatrick 
65*f6aab3d8Srobert   Status error;
66*f6aab3d8Srobert   auto process_sp = std::shared_ptr<ScriptedProcess>(
67*f6aab3d8Srobert       new ScriptedProcess(target_sp, listener_sp, scripted_metadata, error));
68be691f3bSpatrick 
69be691f3bSpatrick   if (error.Fail() || !process_sp || !process_sp->m_script_object_sp ||
70be691f3bSpatrick       !process_sp->m_script_object_sp->IsValid()) {
71*f6aab3d8Srobert     LLDB_LOGF(GetLog(LLDBLog::Process), "%s", error.AsCString());
72be691f3bSpatrick     return nullptr;
73be691f3bSpatrick   }
74be691f3bSpatrick 
75be691f3bSpatrick   return process_sp;
76be691f3bSpatrick }
77be691f3bSpatrick 
CanDebug(lldb::TargetSP target_sp,bool plugin_specified_by_name)78be691f3bSpatrick bool ScriptedProcess::CanDebug(lldb::TargetSP target_sp,
79be691f3bSpatrick                                bool plugin_specified_by_name) {
80be691f3bSpatrick   return true;
81be691f3bSpatrick }
82be691f3bSpatrick 
ScriptedProcess(lldb::TargetSP target_sp,lldb::ListenerSP listener_sp,const ScriptedMetadata & scripted_metadata,Status & error)83*f6aab3d8Srobert ScriptedProcess::ScriptedProcess(lldb::TargetSP target_sp,
84*f6aab3d8Srobert                                  lldb::ListenerSP listener_sp,
85*f6aab3d8Srobert                                  const ScriptedMetadata &scripted_metadata,
86be691f3bSpatrick                                  Status &error)
87*f6aab3d8Srobert     : Process(target_sp, listener_sp), m_scripted_metadata(scripted_metadata) {
88be691f3bSpatrick 
89be691f3bSpatrick   if (!target_sp) {
90be691f3bSpatrick     error.SetErrorStringWithFormat("ScriptedProcess::%s () - ERROR: %s",
91be691f3bSpatrick                                    __FUNCTION__, "Invalid target");
92be691f3bSpatrick     return;
93be691f3bSpatrick   }
94be691f3bSpatrick 
95be691f3bSpatrick   m_interpreter = target_sp->GetDebugger().GetScriptInterpreter();
96be691f3bSpatrick 
97be691f3bSpatrick   if (!m_interpreter) {
98be691f3bSpatrick     error.SetErrorStringWithFormat("ScriptedProcess::%s () - ERROR: %s",
99be691f3bSpatrick                                    __FUNCTION__,
100be691f3bSpatrick                                    "Debugger has no Script Interpreter");
101be691f3bSpatrick     return;
102be691f3bSpatrick   }
103be691f3bSpatrick 
104*f6aab3d8Srobert   ExecutionContext exe_ctx(target_sp, /*get_process=*/false);
105*f6aab3d8Srobert 
106*f6aab3d8Srobert   StructuredData::GenericSP object_sp = GetInterface().CreatePluginObject(
107*f6aab3d8Srobert       m_scripted_metadata.GetClassName(), exe_ctx,
108*f6aab3d8Srobert       m_scripted_metadata.GetArgsSP());
109be691f3bSpatrick 
110be691f3bSpatrick   if (!object_sp || !object_sp->IsValid()) {
111be691f3bSpatrick     error.SetErrorStringWithFormat("ScriptedProcess::%s () - ERROR: %s",
112be691f3bSpatrick                                    __FUNCTION__,
113be691f3bSpatrick                                    "Failed to create valid script object");
114be691f3bSpatrick     return;
115be691f3bSpatrick   }
116be691f3bSpatrick 
117be691f3bSpatrick   m_script_object_sp = object_sp;
118be691f3bSpatrick }
119be691f3bSpatrick 
~ScriptedProcess()120be691f3bSpatrick ScriptedProcess::~ScriptedProcess() {
121be691f3bSpatrick   Clear();
122be691f3bSpatrick   // We need to call finalize on the process before destroying ourselves to
123be691f3bSpatrick   // make sure all of the broadcaster cleanup goes as planned. If we destruct
124be691f3bSpatrick   // this class, then Process::~Process() might have problems trying to fully
125be691f3bSpatrick   // destroy the broadcaster.
126be691f3bSpatrick   Finalize();
127be691f3bSpatrick }
128be691f3bSpatrick 
Initialize()129be691f3bSpatrick void ScriptedProcess::Initialize() {
130be691f3bSpatrick   static llvm::once_flag g_once_flag;
131be691f3bSpatrick 
132be691f3bSpatrick   llvm::call_once(g_once_flag, []() {
133be691f3bSpatrick     PluginManager::RegisterPlugin(GetPluginNameStatic(),
134be691f3bSpatrick                                   GetPluginDescriptionStatic(), CreateInstance);
135be691f3bSpatrick   });
136be691f3bSpatrick }
137be691f3bSpatrick 
Terminate()138be691f3bSpatrick void ScriptedProcess::Terminate() {
139be691f3bSpatrick   PluginManager::UnregisterPlugin(ScriptedProcess::CreateInstance);
140be691f3bSpatrick }
141be691f3bSpatrick 
DoLoadCore()142be691f3bSpatrick Status ScriptedProcess::DoLoadCore() {
143be691f3bSpatrick   ProcessLaunchInfo launch_info = GetTarget().GetProcessLaunchInfo();
144be691f3bSpatrick 
145be691f3bSpatrick   return DoLaunch(nullptr, launch_info);
146be691f3bSpatrick }
147be691f3bSpatrick 
DoLaunch(Module * exe_module,ProcessLaunchInfo & launch_info)148be691f3bSpatrick Status ScriptedProcess::DoLaunch(Module *exe_module,
149be691f3bSpatrick                                  ProcessLaunchInfo &launch_info) {
150be691f3bSpatrick   CheckInterpreterAndScriptObject();
151be691f3bSpatrick 
152be691f3bSpatrick   /* FIXME: This doesn't reflect how lldb actually launches a process.
153be691f3bSpatrick            In reality, it attaches to debugserver, then resume the process. */
154be691f3bSpatrick   Status error = GetInterface().Launch();
155be691f3bSpatrick   SetPrivateState(eStateRunning);
156be691f3bSpatrick 
157be691f3bSpatrick   if (error.Fail())
158be691f3bSpatrick     return error;
159be691f3bSpatrick 
160be691f3bSpatrick   // TODO: Fetch next state from stopped event queue then send stop event
161be691f3bSpatrick   //  const StateType state = SetThreadStopInfo(response);
162be691f3bSpatrick   //  if (state != eStateInvalid) {
163be691f3bSpatrick   //    SetPrivateState(state);
164be691f3bSpatrick 
165be691f3bSpatrick   SetPrivateState(eStateStopped);
166be691f3bSpatrick 
167be691f3bSpatrick   return {};
168be691f3bSpatrick }
169be691f3bSpatrick 
DidLaunch()170be691f3bSpatrick void ScriptedProcess::DidLaunch() {
171be691f3bSpatrick   CheckInterpreterAndScriptObject();
172be691f3bSpatrick   m_pid = GetInterface().GetProcessID();
173*f6aab3d8Srobert   GetLoadedDynamicLibrariesInfos();
174be691f3bSpatrick }
175be691f3bSpatrick 
DoResume()176be691f3bSpatrick Status ScriptedProcess::DoResume() {
177be691f3bSpatrick   CheckInterpreterAndScriptObject();
178be691f3bSpatrick 
179*f6aab3d8Srobert   Log *log = GetLog(LLDBLog::Process);
180be691f3bSpatrick   // FIXME: Fetch data from thread.
181be691f3bSpatrick   const StateType thread_resume_state = eStateRunning;
182be691f3bSpatrick   LLDB_LOGF(log, "ScriptedProcess::%s thread_resume_state = %s", __FUNCTION__,
183be691f3bSpatrick             StateAsCString(thread_resume_state));
184be691f3bSpatrick 
185be691f3bSpatrick   bool resume = (thread_resume_state == eStateRunning);
186be691f3bSpatrick   assert(thread_resume_state == eStateRunning && "invalid thread resume state");
187be691f3bSpatrick 
188be691f3bSpatrick   Status error;
189be691f3bSpatrick   if (resume) {
190be691f3bSpatrick     LLDB_LOGF(log, "ScriptedProcess::%s sending resume", __FUNCTION__);
191be691f3bSpatrick 
192be691f3bSpatrick     SetPrivateState(eStateRunning);
193be691f3bSpatrick     SetPrivateState(eStateStopped);
194be691f3bSpatrick     error = GetInterface().Resume();
195be691f3bSpatrick   }
196be691f3bSpatrick 
197be691f3bSpatrick   return error;
198be691f3bSpatrick }
199be691f3bSpatrick 
DoStop()200be691f3bSpatrick Status ScriptedProcess::DoStop() {
201be691f3bSpatrick   CheckInterpreterAndScriptObject();
202be691f3bSpatrick 
203*f6aab3d8Srobert   Log *log = GetLog(LLDBLog::Process);
204be691f3bSpatrick 
205be691f3bSpatrick   if (GetInterface().ShouldStop()) {
206be691f3bSpatrick     SetPrivateState(eStateStopped);
207be691f3bSpatrick     LLDB_LOGF(log, "ScriptedProcess::%s Immediate stop", __FUNCTION__);
208be691f3bSpatrick     return {};
209be691f3bSpatrick   }
210be691f3bSpatrick 
211be691f3bSpatrick   LLDB_LOGF(log, "ScriptedProcess::%s Delayed stop", __FUNCTION__);
212be691f3bSpatrick   return GetInterface().Stop();
213be691f3bSpatrick }
214be691f3bSpatrick 
DoDestroy()215be691f3bSpatrick Status ScriptedProcess::DoDestroy() { return Status(); }
216be691f3bSpatrick 
IsAlive()217be691f3bSpatrick bool ScriptedProcess::IsAlive() {
218be691f3bSpatrick   if (m_interpreter && m_script_object_sp)
219be691f3bSpatrick     return GetInterface().IsAlive();
220be691f3bSpatrick   return false;
221be691f3bSpatrick }
222be691f3bSpatrick 
DoReadMemory(lldb::addr_t addr,void * buf,size_t size,Status & error)223be691f3bSpatrick size_t ScriptedProcess::DoReadMemory(lldb::addr_t addr, void *buf, size_t size,
224be691f3bSpatrick                                      Status &error) {
225be691f3bSpatrick   if (!m_interpreter)
226*f6aab3d8Srobert     return ScriptedInterface::ErrorWithMessage<size_t>(
227*f6aab3d8Srobert         LLVM_PRETTY_FUNCTION, "No interpreter.", error);
228be691f3bSpatrick 
229be691f3bSpatrick   lldb::DataExtractorSP data_extractor_sp =
230be691f3bSpatrick       GetInterface().ReadMemoryAtAddress(addr, size, error);
231be691f3bSpatrick 
232*f6aab3d8Srobert   if (!data_extractor_sp || !data_extractor_sp->GetByteSize() || error.Fail())
233be691f3bSpatrick     return 0;
234be691f3bSpatrick 
235be691f3bSpatrick   offset_t bytes_copied = data_extractor_sp->CopyByteOrderedData(
236be691f3bSpatrick       0, data_extractor_sp->GetByteSize(), buf, size, GetByteOrder());
237be691f3bSpatrick 
238be691f3bSpatrick   if (!bytes_copied || bytes_copied == LLDB_INVALID_OFFSET)
239*f6aab3d8Srobert     return ScriptedInterface::ErrorWithMessage<size_t>(
240*f6aab3d8Srobert         LLVM_PRETTY_FUNCTION, "Failed to copy read memory to buffer.", error);
241be691f3bSpatrick 
242be691f3bSpatrick   return size;
243be691f3bSpatrick }
244be691f3bSpatrick 
GetArchitecture()245be691f3bSpatrick ArchSpec ScriptedProcess::GetArchitecture() {
246be691f3bSpatrick   return GetTarget().GetArchitecture();
247be691f3bSpatrick }
248be691f3bSpatrick 
DoGetMemoryRegionInfo(lldb::addr_t load_addr,MemoryRegionInfo & region)249*f6aab3d8Srobert Status ScriptedProcess::DoGetMemoryRegionInfo(lldb::addr_t load_addr,
250be691f3bSpatrick                                               MemoryRegionInfo &region) {
251*f6aab3d8Srobert   CheckInterpreterAndScriptObject();
252*f6aab3d8Srobert 
253*f6aab3d8Srobert   Status error;
254*f6aab3d8Srobert   if (auto region_or_err =
255*f6aab3d8Srobert           GetInterface().GetMemoryRegionContainingAddress(load_addr, error))
256*f6aab3d8Srobert     region = *region_or_err;
257*f6aab3d8Srobert 
258*f6aab3d8Srobert   return error;
259be691f3bSpatrick }
260be691f3bSpatrick 
GetMemoryRegions(MemoryRegionInfos & region_list)261be691f3bSpatrick Status ScriptedProcess::GetMemoryRegions(MemoryRegionInfos &region_list) {
262be691f3bSpatrick   CheckInterpreterAndScriptObject();
263be691f3bSpatrick 
264*f6aab3d8Srobert   Status error;
265be691f3bSpatrick   lldb::addr_t address = 0;
266be691f3bSpatrick 
267*f6aab3d8Srobert   while (auto region_or_err =
268*f6aab3d8Srobert              GetInterface().GetMemoryRegionContainingAddress(address, error)) {
269*f6aab3d8Srobert     if (error.Fail())
270*f6aab3d8Srobert       break;
271*f6aab3d8Srobert 
272*f6aab3d8Srobert     MemoryRegionInfo &mem_region = *region_or_err;
273*f6aab3d8Srobert     auto range = mem_region.GetRange();
274be691f3bSpatrick     address += range.GetRangeBase() + range.GetByteSize();
275*f6aab3d8Srobert     region_list.push_back(mem_region);
276be691f3bSpatrick   }
277be691f3bSpatrick 
278*f6aab3d8Srobert   return error;
279be691f3bSpatrick }
280be691f3bSpatrick 
Clear()281be691f3bSpatrick void ScriptedProcess::Clear() { Process::m_thread_list.Clear(); }
282be691f3bSpatrick 
DoUpdateThreadList(ThreadList & old_thread_list,ThreadList & new_thread_list)283be691f3bSpatrick bool ScriptedProcess::DoUpdateThreadList(ThreadList &old_thread_list,
284be691f3bSpatrick                                          ThreadList &new_thread_list) {
285be691f3bSpatrick   // TODO: Implement
286be691f3bSpatrick   // This is supposed to get the current set of threads, if any of them are in
287be691f3bSpatrick   // old_thread_list then they get copied to new_thread_list, and then any
288be691f3bSpatrick   // actually new threads will get added to new_thread_list.
289*f6aab3d8Srobert 
290*f6aab3d8Srobert   CheckInterpreterAndScriptObject();
291*f6aab3d8Srobert   m_thread_plans.ClearThreadCache();
292*f6aab3d8Srobert 
293*f6aab3d8Srobert   Status error;
294*f6aab3d8Srobert   ScriptLanguage language = m_interpreter->GetLanguage();
295*f6aab3d8Srobert 
296*f6aab3d8Srobert   if (language != eScriptLanguagePython)
297*f6aab3d8Srobert     return ScriptedInterface::ErrorWithMessage<bool>(
298*f6aab3d8Srobert         LLVM_PRETTY_FUNCTION,
299*f6aab3d8Srobert         llvm::Twine("ScriptInterpreter language (" +
300*f6aab3d8Srobert                     llvm::Twine(m_interpreter->LanguageToString(language)) +
301*f6aab3d8Srobert                     llvm::Twine(") not supported."))
302*f6aab3d8Srobert             .str(),
303*f6aab3d8Srobert         error);
304*f6aab3d8Srobert 
305*f6aab3d8Srobert   StructuredData::DictionarySP thread_info_sp = GetInterface().GetThreadsInfo();
306*f6aab3d8Srobert 
307*f6aab3d8Srobert   if (!thread_info_sp)
308*f6aab3d8Srobert     return ScriptedInterface::ErrorWithMessage<bool>(
309*f6aab3d8Srobert         LLVM_PRETTY_FUNCTION,
310*f6aab3d8Srobert         "Couldn't fetch thread list from Scripted Process.", error);
311*f6aab3d8Srobert 
312*f6aab3d8Srobert   // Because `StructuredData::Dictionary` uses a `std::map<ConstString,
313*f6aab3d8Srobert   // ObjectSP>` for storage, each item is sorted based on the key alphabetical
314*f6aab3d8Srobert   // order. Since `GetThreadsInfo` provides thread indices as the key element,
315*f6aab3d8Srobert   // thread info comes ordered alphabetically, instead of numerically, so we
316*f6aab3d8Srobert   // need to sort the thread indices before creating thread.
317*f6aab3d8Srobert 
318*f6aab3d8Srobert   StructuredData::ArraySP keys = thread_info_sp->GetKeys();
319*f6aab3d8Srobert 
320*f6aab3d8Srobert   std::map<size_t, StructuredData::ObjectSP> sorted_threads;
321*f6aab3d8Srobert   auto sort_keys = [&sorted_threads,
322*f6aab3d8Srobert                     &thread_info_sp](StructuredData::Object *item) -> bool {
323*f6aab3d8Srobert     if (!item)
324*f6aab3d8Srobert       return false;
325*f6aab3d8Srobert 
326*f6aab3d8Srobert     llvm::StringRef key = item->GetStringValue();
327*f6aab3d8Srobert     size_t idx = 0;
328*f6aab3d8Srobert 
329*f6aab3d8Srobert     // Make sure the provided index is actually an integer
330*f6aab3d8Srobert     if (!llvm::to_integer(key, idx))
331*f6aab3d8Srobert       return false;
332*f6aab3d8Srobert 
333*f6aab3d8Srobert     sorted_threads[idx] = thread_info_sp->GetValueForKey(key);
334*f6aab3d8Srobert     return true;
335*f6aab3d8Srobert   };
336*f6aab3d8Srobert 
337*f6aab3d8Srobert   size_t thread_count = thread_info_sp->GetSize();
338*f6aab3d8Srobert 
339*f6aab3d8Srobert   if (!keys->ForEach(sort_keys) || sorted_threads.size() != thread_count)
340*f6aab3d8Srobert     // Might be worth showing the unsorted thread list instead of return early.
341*f6aab3d8Srobert     return ScriptedInterface::ErrorWithMessage<bool>(
342*f6aab3d8Srobert         LLVM_PRETTY_FUNCTION, "Couldn't sort thread list.", error);
343*f6aab3d8Srobert 
344*f6aab3d8Srobert   auto create_scripted_thread =
345*f6aab3d8Srobert       [this, &error, &new_thread_list](
346*f6aab3d8Srobert           const std::pair<size_t, StructuredData::ObjectSP> pair) -> bool {
347*f6aab3d8Srobert     size_t idx = pair.first;
348*f6aab3d8Srobert     StructuredData::ObjectSP object_sp = pair.second;
349*f6aab3d8Srobert 
350*f6aab3d8Srobert     if (!object_sp)
351*f6aab3d8Srobert       return ScriptedInterface::ErrorWithMessage<bool>(
352*f6aab3d8Srobert           LLVM_PRETTY_FUNCTION, "Invalid thread info object", error);
353*f6aab3d8Srobert 
354*f6aab3d8Srobert     auto thread_or_error =
355*f6aab3d8Srobert         ScriptedThread::Create(*this, object_sp->GetAsGeneric());
356*f6aab3d8Srobert 
357*f6aab3d8Srobert     if (!thread_or_error)
358*f6aab3d8Srobert       return ScriptedInterface::ErrorWithMessage<bool>(
359*f6aab3d8Srobert           LLVM_PRETTY_FUNCTION, toString(thread_or_error.takeError()), error);
360*f6aab3d8Srobert 
361*f6aab3d8Srobert     ThreadSP thread_sp = thread_or_error.get();
362*f6aab3d8Srobert     lldbassert(thread_sp && "Couldn't initialize scripted thread.");
363*f6aab3d8Srobert 
364*f6aab3d8Srobert     RegisterContextSP reg_ctx_sp = thread_sp->GetRegisterContext();
365*f6aab3d8Srobert     if (!reg_ctx_sp)
366*f6aab3d8Srobert       return ScriptedInterface::ErrorWithMessage<bool>(
367*f6aab3d8Srobert           LLVM_PRETTY_FUNCTION,
368*f6aab3d8Srobert           llvm::Twine("Invalid Register Context for thread " + llvm::Twine(idx))
369*f6aab3d8Srobert               .str(),
370*f6aab3d8Srobert           error);
371*f6aab3d8Srobert 
372*f6aab3d8Srobert     new_thread_list.AddThread(thread_sp);
373*f6aab3d8Srobert 
374*f6aab3d8Srobert     return true;
375*f6aab3d8Srobert   };
376*f6aab3d8Srobert 
377*f6aab3d8Srobert   llvm::for_each(sorted_threads, create_scripted_thread);
378*f6aab3d8Srobert 
379be691f3bSpatrick   return new_thread_list.GetSize(false) > 0;
380be691f3bSpatrick }
381be691f3bSpatrick 
RefreshStateAfterStop()382*f6aab3d8Srobert void ScriptedProcess::RefreshStateAfterStop() {
383*f6aab3d8Srobert   // Let all threads recover from stopping and do any clean up based on the
384*f6aab3d8Srobert   // previous thread state (if any).
385*f6aab3d8Srobert   m_thread_list.RefreshStateAfterStop();
386*f6aab3d8Srobert }
387*f6aab3d8Srobert 
GetProcessInfo(ProcessInstanceInfo & info)388be691f3bSpatrick bool ScriptedProcess::GetProcessInfo(ProcessInstanceInfo &info) {
389be691f3bSpatrick   info.Clear();
390be691f3bSpatrick   info.SetProcessID(GetID());
391be691f3bSpatrick   info.SetArchitecture(GetArchitecture());
392be691f3bSpatrick   lldb::ModuleSP module_sp = GetTarget().GetExecutableModule();
393be691f3bSpatrick   if (module_sp) {
394be691f3bSpatrick     const bool add_exe_file_as_first_arg = false;
395be691f3bSpatrick     info.SetExecutableFile(GetTarget().GetExecutableModule()->GetFileSpec(),
396be691f3bSpatrick                            add_exe_file_as_first_arg);
397be691f3bSpatrick   }
398be691f3bSpatrick   return true;
399be691f3bSpatrick }
400be691f3bSpatrick 
401*f6aab3d8Srobert lldb_private::StructuredData::ObjectSP
GetLoadedDynamicLibrariesInfos()402*f6aab3d8Srobert ScriptedProcess::GetLoadedDynamicLibrariesInfos() {
403*f6aab3d8Srobert   CheckInterpreterAndScriptObject();
404*f6aab3d8Srobert 
405*f6aab3d8Srobert   Status error;
406*f6aab3d8Srobert   auto error_with_message = [&error](llvm::StringRef message) {
407*f6aab3d8Srobert     return ScriptedInterface::ErrorWithMessage<bool>(LLVM_PRETTY_FUNCTION,
408*f6aab3d8Srobert                                                      message.data(), error);
409*f6aab3d8Srobert   };
410*f6aab3d8Srobert 
411*f6aab3d8Srobert   StructuredData::ArraySP loaded_images_sp = GetInterface().GetLoadedImages();
412*f6aab3d8Srobert 
413*f6aab3d8Srobert   if (!loaded_images_sp || !loaded_images_sp->GetSize())
414*f6aab3d8Srobert     return ScriptedInterface::ErrorWithMessage<StructuredData::ObjectSP>(
415*f6aab3d8Srobert         LLVM_PRETTY_FUNCTION, "No loaded images.", error);
416*f6aab3d8Srobert 
417*f6aab3d8Srobert   ModuleList module_list;
418*f6aab3d8Srobert   Target &target = GetTarget();
419*f6aab3d8Srobert 
420*f6aab3d8Srobert   auto reload_image = [&target, &module_list, &error_with_message](
421*f6aab3d8Srobert                           StructuredData::Object *obj) -> bool {
422*f6aab3d8Srobert     StructuredData::Dictionary *dict = obj->GetAsDictionary();
423*f6aab3d8Srobert 
424*f6aab3d8Srobert     if (!dict)
425*f6aab3d8Srobert       return error_with_message("Couldn't cast image object into dictionary.");
426*f6aab3d8Srobert 
427*f6aab3d8Srobert     ModuleSpec module_spec;
428*f6aab3d8Srobert     llvm::StringRef value;
429*f6aab3d8Srobert 
430*f6aab3d8Srobert     bool has_path = dict->HasKey("path");
431*f6aab3d8Srobert     bool has_uuid = dict->HasKey("uuid");
432*f6aab3d8Srobert     if (!has_path && !has_uuid)
433*f6aab3d8Srobert       return error_with_message("Dictionary should have key 'path' or 'uuid'");
434*f6aab3d8Srobert     if (!dict->HasKey("load_addr"))
435*f6aab3d8Srobert       return error_with_message("Dictionary is missing key 'load_addr'");
436*f6aab3d8Srobert 
437*f6aab3d8Srobert     if (has_path) {
438*f6aab3d8Srobert       dict->GetValueForKeyAsString("path", value);
439*f6aab3d8Srobert       module_spec.GetFileSpec().SetPath(value);
440*f6aab3d8Srobert     }
441*f6aab3d8Srobert 
442*f6aab3d8Srobert     if (has_uuid) {
443*f6aab3d8Srobert       dict->GetValueForKeyAsString("uuid", value);
444*f6aab3d8Srobert       module_spec.GetUUID().SetFromStringRef(value);
445*f6aab3d8Srobert     }
446*f6aab3d8Srobert     module_spec.GetArchitecture() = target.GetArchitecture();
447*f6aab3d8Srobert 
448*f6aab3d8Srobert     ModuleSP module_sp =
449*f6aab3d8Srobert         target.GetOrCreateModule(module_spec, true /* notify */);
450*f6aab3d8Srobert 
451*f6aab3d8Srobert     if (!module_sp)
452*f6aab3d8Srobert       return error_with_message("Couldn't create or get module.");
453*f6aab3d8Srobert 
454*f6aab3d8Srobert     lldb::addr_t load_addr = LLDB_INVALID_ADDRESS;
455*f6aab3d8Srobert     lldb::addr_t slide = LLDB_INVALID_OFFSET;
456*f6aab3d8Srobert     dict->GetValueForKeyAsInteger("load_addr", load_addr);
457*f6aab3d8Srobert     dict->GetValueForKeyAsInteger("slide", slide);
458*f6aab3d8Srobert     if (load_addr == LLDB_INVALID_ADDRESS)
459*f6aab3d8Srobert       return error_with_message(
460*f6aab3d8Srobert           "Couldn't get valid load address or slide offset.");
461*f6aab3d8Srobert 
462*f6aab3d8Srobert     if (slide != LLDB_INVALID_OFFSET)
463*f6aab3d8Srobert       load_addr += slide;
464*f6aab3d8Srobert 
465*f6aab3d8Srobert     bool changed = false;
466*f6aab3d8Srobert     module_sp->SetLoadAddress(target, load_addr, false /*=value_is_offset*/,
467*f6aab3d8Srobert                               changed);
468*f6aab3d8Srobert 
469*f6aab3d8Srobert     if (!changed && !module_sp->GetObjectFile())
470*f6aab3d8Srobert       return error_with_message("Couldn't set the load address for module.");
471*f6aab3d8Srobert 
472*f6aab3d8Srobert     dict->GetValueForKeyAsString("path", value);
473*f6aab3d8Srobert     FileSpec objfile(value);
474*f6aab3d8Srobert     module_sp->SetFileSpecAndObjectName(objfile, objfile.GetFilename());
475*f6aab3d8Srobert 
476*f6aab3d8Srobert     return module_list.AppendIfNeeded(module_sp);
477*f6aab3d8Srobert   };
478*f6aab3d8Srobert 
479*f6aab3d8Srobert   if (!loaded_images_sp->ForEach(reload_image))
480*f6aab3d8Srobert     return ScriptedInterface::ErrorWithMessage<StructuredData::ObjectSP>(
481*f6aab3d8Srobert         LLVM_PRETTY_FUNCTION, "Couldn't reload all images.", error);
482*f6aab3d8Srobert 
483*f6aab3d8Srobert   target.ModulesDidLoad(module_list);
484*f6aab3d8Srobert 
485*f6aab3d8Srobert   return loaded_images_sp;
486*f6aab3d8Srobert }
487*f6aab3d8Srobert 
GetMetadata()488*f6aab3d8Srobert lldb_private::StructuredData::DictionarySP ScriptedProcess::GetMetadata() {
489*f6aab3d8Srobert   CheckInterpreterAndScriptObject();
490*f6aab3d8Srobert 
491*f6aab3d8Srobert   StructuredData::DictionarySP metadata_sp = GetInterface().GetMetadata();
492*f6aab3d8Srobert 
493*f6aab3d8Srobert   Status error;
494*f6aab3d8Srobert   if (!metadata_sp || !metadata_sp->GetSize())
495*f6aab3d8Srobert     return ScriptedInterface::ErrorWithMessage<StructuredData::DictionarySP>(
496*f6aab3d8Srobert         LLVM_PRETTY_FUNCTION, "No metadata.", error);
497*f6aab3d8Srobert 
498*f6aab3d8Srobert   return metadata_sp;
499*f6aab3d8Srobert }
500*f6aab3d8Srobert 
UpdateQueueListIfNeeded()501*f6aab3d8Srobert void ScriptedProcess::UpdateQueueListIfNeeded() {
502*f6aab3d8Srobert   CheckInterpreterAndScriptObject();
503*f6aab3d8Srobert   for (ThreadSP thread_sp : Threads()) {
504*f6aab3d8Srobert     if (const char *queue_name = thread_sp->GetQueueName()) {
505*f6aab3d8Srobert       QueueSP queue_sp = std::make_shared<Queue>(
506*f6aab3d8Srobert           m_process->shared_from_this(), thread_sp->GetQueueID(), queue_name);
507*f6aab3d8Srobert       m_queue_list.AddQueue(queue_sp);
508*f6aab3d8Srobert     }
509*f6aab3d8Srobert   }
510*f6aab3d8Srobert }
511*f6aab3d8Srobert 
GetInterface() const512be691f3bSpatrick ScriptedProcessInterface &ScriptedProcess::GetInterface() const {
513be691f3bSpatrick   return m_interpreter->GetScriptedProcessInterface();
514be691f3bSpatrick }
515