1312b43daSMed Ismail Bennani //===-- ScriptedProcess.cpp -----------------------------------------------===//
2312b43daSMed Ismail Bennani //
3312b43daSMed Ismail Bennani // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4312b43daSMed Ismail Bennani // See https://llvm.org/LICENSE.txt for license information.
5312b43daSMed Ismail Bennani // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6312b43daSMed Ismail Bennani //
7312b43daSMed Ismail Bennani //===----------------------------------------------------------------------===//
8312b43daSMed Ismail Bennani 
9312b43daSMed Ismail Bennani #include "ScriptedProcess.h"
10312b43daSMed Ismail Bennani 
11312b43daSMed Ismail Bennani #include "lldb/Core/Debugger.h"
12312b43daSMed Ismail Bennani #include "lldb/Core/Module.h"
13312b43daSMed Ismail Bennani #include "lldb/Core/PluginManager.h"
14312b43daSMed Ismail Bennani 
15312b43daSMed Ismail Bennani #include "lldb/Host/OptionParser.h"
16312b43daSMed Ismail Bennani #include "lldb/Host/ThreadLauncher.h"
17312b43daSMed Ismail Bennani #include "lldb/Interpreter/CommandInterpreter.h"
18312b43daSMed Ismail Bennani #include "lldb/Interpreter/OptionArgParser.h"
19312b43daSMed Ismail Bennani #include "lldb/Interpreter/OptionGroupBoolean.h"
20312b43daSMed Ismail Bennani #include "lldb/Interpreter/ScriptInterpreter.h"
21312b43daSMed Ismail Bennani #include "lldb/Target/MemoryRegionInfo.h"
22312b43daSMed Ismail Bennani #include "lldb/Target/RegisterContext.h"
23c34698a8SPavel Labath #include "lldb/Utility/LLDBLog.h"
24312b43daSMed Ismail Bennani #include "lldb/Utility/State.h"
25312b43daSMed Ismail Bennani 
26312b43daSMed Ismail Bennani #include <mutex>
27312b43daSMed Ismail Bennani 
28312b43daSMed Ismail Bennani LLDB_PLUGIN_DEFINE(ScriptedProcess)
29312b43daSMed Ismail Bennani 
30312b43daSMed Ismail Bennani using namespace lldb;
31312b43daSMed Ismail Bennani using namespace lldb_private;
32312b43daSMed Ismail Bennani 
335f4980f0SPavel Labath llvm::StringRef ScriptedProcess::GetPluginDescriptionStatic() {
34312b43daSMed Ismail Bennani   return "Scripted Process plug-in.";
35312b43daSMed Ismail Bennani }
36312b43daSMed Ismail Bennani 
37312b43daSMed Ismail Bennani static constexpr lldb::ScriptLanguage g_supported_script_languages[] = {
38312b43daSMed Ismail Bennani     ScriptLanguage::eScriptLanguagePython,
39312b43daSMed Ismail Bennani };
40312b43daSMed Ismail Bennani 
41312b43daSMed Ismail Bennani bool ScriptedProcess::IsScriptLanguageSupported(lldb::ScriptLanguage language) {
42312b43daSMed Ismail Bennani   llvm::ArrayRef<lldb::ScriptLanguage> supported_languages =
43312b43daSMed Ismail Bennani       llvm::makeArrayRef(g_supported_script_languages);
44312b43daSMed Ismail Bennani 
45312b43daSMed Ismail Bennani   return llvm::is_contained(supported_languages, language);
46312b43daSMed Ismail Bennani }
47312b43daSMed Ismail Bennani 
48312b43daSMed Ismail Bennani void ScriptedProcess::CheckInterpreterAndScriptObject() const {
49312b43daSMed Ismail Bennani   lldbassert(m_interpreter && "Invalid Script Interpreter.");
50312b43daSMed Ismail Bennani   lldbassert(m_script_object_sp && "Invalid Script Object.");
51312b43daSMed Ismail Bennani }
52312b43daSMed Ismail Bennani 
53312b43daSMed Ismail Bennani lldb::ProcessSP ScriptedProcess::CreateInstance(lldb::TargetSP target_sp,
54312b43daSMed Ismail Bennani                                                 lldb::ListenerSP listener_sp,
55312b43daSMed Ismail Bennani                                                 const FileSpec *file,
56312b43daSMed Ismail Bennani                                                 bool can_connect) {
57312b43daSMed Ismail Bennani   if (!target_sp ||
58312b43daSMed Ismail Bennani       !IsScriptLanguageSupported(target_sp->GetDebugger().GetScriptLanguage()))
59312b43daSMed Ismail Bennani     return nullptr;
60312b43daSMed Ismail Bennani 
61312b43daSMed Ismail Bennani   Status error;
62312b43daSMed Ismail Bennani   ScriptedProcess::ScriptedProcessInfo scripted_process_info(
63312b43daSMed Ismail Bennani       target_sp->GetProcessLaunchInfo());
64312b43daSMed Ismail Bennani 
65312b43daSMed Ismail Bennani   auto process_sp = std::make_shared<ScriptedProcess>(
66312b43daSMed Ismail Bennani       target_sp, listener_sp, scripted_process_info, error);
67312b43daSMed Ismail Bennani 
68312b43daSMed Ismail Bennani   if (error.Fail() || !process_sp || !process_sp->m_script_object_sp ||
69312b43daSMed Ismail Bennani       !process_sp->m_script_object_sp->IsValid()) {
70a007a6d8SPavel Labath     LLDB_LOGF(GetLog(LLDBLog::Process), "%s", error.AsCString());
71312b43daSMed Ismail Bennani     return nullptr;
72312b43daSMed Ismail Bennani   }
73312b43daSMed Ismail Bennani 
74312b43daSMed Ismail Bennani   return process_sp;
75312b43daSMed Ismail Bennani }
76312b43daSMed Ismail Bennani 
77312b43daSMed Ismail Bennani bool ScriptedProcess::CanDebug(lldb::TargetSP target_sp,
78312b43daSMed Ismail Bennani                                bool plugin_specified_by_name) {
79312b43daSMed Ismail Bennani   return true;
80312b43daSMed Ismail Bennani }
81312b43daSMed Ismail Bennani 
82312b43daSMed Ismail Bennani ScriptedProcess::ScriptedProcess(
83312b43daSMed Ismail Bennani     lldb::TargetSP target_sp, lldb::ListenerSP listener_sp,
84312b43daSMed Ismail Bennani     const ScriptedProcess::ScriptedProcessInfo &scripted_process_info,
85312b43daSMed Ismail Bennani     Status &error)
86312b43daSMed Ismail Bennani     : Process(target_sp, listener_sp),
87312b43daSMed Ismail Bennani       m_scripted_process_info(scripted_process_info) {
88312b43daSMed Ismail Bennani 
89312b43daSMed Ismail Bennani   if (!target_sp) {
90312b43daSMed Ismail Bennani     error.SetErrorStringWithFormat("ScriptedProcess::%s () - ERROR: %s",
91312b43daSMed Ismail Bennani                                    __FUNCTION__, "Invalid target");
92312b43daSMed Ismail Bennani     return;
93312b43daSMed Ismail Bennani   }
94312b43daSMed Ismail Bennani 
95312b43daSMed Ismail Bennani   m_interpreter = target_sp->GetDebugger().GetScriptInterpreter();
96312b43daSMed Ismail Bennani 
97312b43daSMed Ismail Bennani   if (!m_interpreter) {
98312b43daSMed Ismail Bennani     error.SetErrorStringWithFormat("ScriptedProcess::%s () - ERROR: %s",
99312b43daSMed Ismail Bennani                                    __FUNCTION__,
100312b43daSMed Ismail Bennani                                    "Debugger has no Script Interpreter");
101312b43daSMed Ismail Bennani     return;
102312b43daSMed Ismail Bennani   }
103312b43daSMed Ismail Bennani 
1043925204cSMed Ismail Bennani   ExecutionContext exe_ctx(target_sp, /*get_process=*/false);
1053925204cSMed Ismail Bennani 
1063925204cSMed Ismail Bennani   StructuredData::GenericSP object_sp = GetInterface().CreatePluginObject(
1073925204cSMed Ismail Bennani       m_scripted_process_info.GetClassName().c_str(), exe_ctx,
108ad0f7d3dSMed Ismail Bennani       m_scripted_process_info.GetArgsSP());
109312b43daSMed Ismail Bennani 
110312b43daSMed Ismail Bennani   if (!object_sp || !object_sp->IsValid()) {
111312b43daSMed Ismail Bennani     error.SetErrorStringWithFormat("ScriptedProcess::%s () - ERROR: %s",
112312b43daSMed Ismail Bennani                                    __FUNCTION__,
113312b43daSMed Ismail Bennani                                    "Failed to create valid script object");
114312b43daSMed Ismail Bennani     return;
115312b43daSMed Ismail Bennani   }
116312b43daSMed Ismail Bennani 
117312b43daSMed Ismail Bennani   m_script_object_sp = object_sp;
118312b43daSMed Ismail Bennani }
119312b43daSMed Ismail Bennani 
120312b43daSMed Ismail Bennani ScriptedProcess::~ScriptedProcess() {
121312b43daSMed Ismail Bennani   Clear();
122312b43daSMed Ismail Bennani   // We need to call finalize on the process before destroying ourselves to
123312b43daSMed Ismail Bennani   // make sure all of the broadcaster cleanup goes as planned. If we destruct
124312b43daSMed Ismail Bennani   // this class, then Process::~Process() might have problems trying to fully
125312b43daSMed Ismail Bennani   // destroy the broadcaster.
126312b43daSMed Ismail Bennani   Finalize();
127312b43daSMed Ismail Bennani }
128312b43daSMed Ismail Bennani 
129312b43daSMed Ismail Bennani void ScriptedProcess::Initialize() {
130312b43daSMed Ismail Bennani   static llvm::once_flag g_once_flag;
131312b43daSMed Ismail Bennani 
132312b43daSMed Ismail Bennani   llvm::call_once(g_once_flag, []() {
133312b43daSMed Ismail Bennani     PluginManager::RegisterPlugin(GetPluginNameStatic(),
134312b43daSMed Ismail Bennani                                   GetPluginDescriptionStatic(), CreateInstance);
135312b43daSMed Ismail Bennani   });
136312b43daSMed Ismail Bennani }
137312b43daSMed Ismail Bennani 
138312b43daSMed Ismail Bennani void ScriptedProcess::Terminate() {
139312b43daSMed Ismail Bennani   PluginManager::UnregisterPlugin(ScriptedProcess::CreateInstance);
140312b43daSMed Ismail Bennani }
141312b43daSMed Ismail Bennani 
142312b43daSMed Ismail Bennani Status ScriptedProcess::DoLoadCore() {
143312b43daSMed Ismail Bennani   ProcessLaunchInfo launch_info = GetTarget().GetProcessLaunchInfo();
144312b43daSMed Ismail Bennani 
145312b43daSMed Ismail Bennani   return DoLaunch(nullptr, launch_info);
146312b43daSMed Ismail Bennani }
147312b43daSMed Ismail Bennani 
148312b43daSMed Ismail Bennani Status ScriptedProcess::DoLaunch(Module *exe_module,
149312b43daSMed Ismail Bennani                                  ProcessLaunchInfo &launch_info) {
150312b43daSMed Ismail Bennani   CheckInterpreterAndScriptObject();
151312b43daSMed Ismail Bennani 
152312b43daSMed Ismail Bennani   /* FIXME: This doesn't reflect how lldb actually launches a process.
153312b43daSMed Ismail Bennani            In reality, it attaches to debugserver, then resume the process. */
154312b43daSMed Ismail Bennani   Status error = GetInterface().Launch();
155312b43daSMed Ismail Bennani   SetPrivateState(eStateRunning);
156312b43daSMed Ismail Bennani 
157312b43daSMed Ismail Bennani   if (error.Fail())
158312b43daSMed Ismail Bennani     return error;
159312b43daSMed Ismail Bennani 
160312b43daSMed Ismail Bennani   // TODO: Fetch next state from stopped event queue then send stop event
161312b43daSMed Ismail Bennani   //  const StateType state = SetThreadStopInfo(response);
162312b43daSMed Ismail Bennani   //  if (state != eStateInvalid) {
163312b43daSMed Ismail Bennani   //    SetPrivateState(state);
164312b43daSMed Ismail Bennani 
165312b43daSMed Ismail Bennani   SetPrivateState(eStateStopped);
166312b43daSMed Ismail Bennani 
167312b43daSMed Ismail Bennani   return {};
168312b43daSMed Ismail Bennani }
169312b43daSMed Ismail Bennani 
170312b43daSMed Ismail Bennani void ScriptedProcess::DidLaunch() {
171312b43daSMed Ismail Bennani   CheckInterpreterAndScriptObject();
172312b43daSMed Ismail Bennani   m_pid = GetInterface().GetProcessID();
173*680ca7f2SMed Ismail Bennani   GetLoadedDynamicLibrariesInfos();
174312b43daSMed Ismail Bennani }
175312b43daSMed Ismail Bennani 
176312b43daSMed Ismail Bennani Status ScriptedProcess::DoResume() {
177312b43daSMed Ismail Bennani   CheckInterpreterAndScriptObject();
178312b43daSMed Ismail Bennani 
179a007a6d8SPavel Labath   Log *log = GetLog(LLDBLog::Process);
180312b43daSMed Ismail Bennani   // FIXME: Fetch data from thread.
181312b43daSMed Ismail Bennani   const StateType thread_resume_state = eStateRunning;
182312b43daSMed Ismail Bennani   LLDB_LOGF(log, "ScriptedProcess::%s thread_resume_state = %s", __FUNCTION__,
183312b43daSMed Ismail Bennani             StateAsCString(thread_resume_state));
184312b43daSMed Ismail Bennani 
185312b43daSMed Ismail Bennani   bool resume = (thread_resume_state == eStateRunning);
186312b43daSMed Ismail Bennani   assert(thread_resume_state == eStateRunning && "invalid thread resume state");
187312b43daSMed Ismail Bennani 
188312b43daSMed Ismail Bennani   Status error;
189312b43daSMed Ismail Bennani   if (resume) {
190312b43daSMed Ismail Bennani     LLDB_LOGF(log, "ScriptedProcess::%s sending resume", __FUNCTION__);
191312b43daSMed Ismail Bennani 
192312b43daSMed Ismail Bennani     SetPrivateState(eStateRunning);
193312b43daSMed Ismail Bennani     SetPrivateState(eStateStopped);
194312b43daSMed Ismail Bennani     error = GetInterface().Resume();
195312b43daSMed Ismail Bennani   }
196312b43daSMed Ismail Bennani 
197312b43daSMed Ismail Bennani   return error;
198312b43daSMed Ismail Bennani }
199312b43daSMed Ismail Bennani 
200312b43daSMed Ismail Bennani Status ScriptedProcess::DoStop() {
201312b43daSMed Ismail Bennani   CheckInterpreterAndScriptObject();
202312b43daSMed Ismail Bennani 
203a007a6d8SPavel Labath   Log *log = GetLog(LLDBLog::Process);
204312b43daSMed Ismail Bennani 
205312b43daSMed Ismail Bennani   if (GetInterface().ShouldStop()) {
206312b43daSMed Ismail Bennani     SetPrivateState(eStateStopped);
207312b43daSMed Ismail Bennani     LLDB_LOGF(log, "ScriptedProcess::%s Immediate stop", __FUNCTION__);
208312b43daSMed Ismail Bennani     return {};
209312b43daSMed Ismail Bennani   }
210312b43daSMed Ismail Bennani 
211312b43daSMed Ismail Bennani   LLDB_LOGF(log, "ScriptedProcess::%s Delayed stop", __FUNCTION__);
212312b43daSMed Ismail Bennani   return GetInterface().Stop();
213312b43daSMed Ismail Bennani }
214312b43daSMed Ismail Bennani 
215312b43daSMed Ismail Bennani Status ScriptedProcess::DoDestroy() { return Status(); }
216312b43daSMed Ismail Bennani 
217312b43daSMed Ismail Bennani bool ScriptedProcess::IsAlive() {
218312b43daSMed Ismail Bennani   if (m_interpreter && m_script_object_sp)
219312b43daSMed Ismail Bennani     return GetInterface().IsAlive();
220312b43daSMed Ismail Bennani   return false;
221312b43daSMed Ismail Bennani }
222312b43daSMed Ismail Bennani 
223312b43daSMed Ismail Bennani size_t ScriptedProcess::DoReadMemory(lldb::addr_t addr, void *buf, size_t size,
224312b43daSMed Ismail Bennani                                      Status &error) {
225312b43daSMed Ismail Bennani   if (!m_interpreter)
22691bb1161SMed Ismail Bennani     return ScriptedInterface::ErrorWithMessage<size_t>(
22791bb1161SMed Ismail Bennani         LLVM_PRETTY_FUNCTION, "No interpreter.", error);
228312b43daSMed Ismail Bennani 
229312b43daSMed Ismail Bennani   lldb::DataExtractorSP data_extractor_sp =
230312b43daSMed Ismail Bennani       GetInterface().ReadMemoryAtAddress(addr, size, error);
231312b43daSMed Ismail Bennani 
232a758c9f7SMed Ismail Bennani   if (!data_extractor_sp || !data_extractor_sp->GetByteSize() || error.Fail())
233312b43daSMed Ismail Bennani     return 0;
234312b43daSMed Ismail Bennani 
235312b43daSMed Ismail Bennani   offset_t bytes_copied = data_extractor_sp->CopyByteOrderedData(
236312b43daSMed Ismail Bennani       0, data_extractor_sp->GetByteSize(), buf, size, GetByteOrder());
237312b43daSMed Ismail Bennani 
238312b43daSMed Ismail Bennani   if (!bytes_copied || bytes_copied == LLDB_INVALID_OFFSET)
23991bb1161SMed Ismail Bennani     return ScriptedInterface::ErrorWithMessage<size_t>(
24088a941baSMed Ismail Bennani         LLVM_PRETTY_FUNCTION, "Failed to copy read memory to buffer.", error);
241312b43daSMed Ismail Bennani 
242312b43daSMed Ismail Bennani   return size;
243312b43daSMed Ismail Bennani }
244312b43daSMed Ismail Bennani 
245312b43daSMed Ismail Bennani ArchSpec ScriptedProcess::GetArchitecture() {
246312b43daSMed Ismail Bennani   return GetTarget().GetArchitecture();
247312b43daSMed Ismail Bennani }
248312b43daSMed Ismail Bennani 
2492937b282SDavid Spickett Status ScriptedProcess::DoGetMemoryRegionInfo(lldb::addr_t load_addr,
250312b43daSMed Ismail Bennani                                               MemoryRegionInfo &region) {
251a758c9f7SMed Ismail Bennani   CheckInterpreterAndScriptObject();
252a758c9f7SMed Ismail Bennani 
253a758c9f7SMed Ismail Bennani   Status error;
254a758c9f7SMed Ismail Bennani   if (auto region_or_err =
255a758c9f7SMed Ismail Bennani           GetInterface().GetMemoryRegionContainingAddress(load_addr, error))
256a758c9f7SMed Ismail Bennani     region = *region_or_err;
257a758c9f7SMed Ismail Bennani 
258a758c9f7SMed Ismail Bennani   return error;
259312b43daSMed Ismail Bennani }
260312b43daSMed Ismail Bennani 
261312b43daSMed Ismail Bennani Status ScriptedProcess::GetMemoryRegions(MemoryRegionInfos &region_list) {
262312b43daSMed Ismail Bennani   CheckInterpreterAndScriptObject();
263312b43daSMed Ismail Bennani 
264a758c9f7SMed Ismail Bennani   Status error;
265312b43daSMed Ismail Bennani   lldb::addr_t address = 0;
266312b43daSMed Ismail Bennani 
267a758c9f7SMed Ismail Bennani   while (auto region_or_err =
268a758c9f7SMed Ismail Bennani              GetInterface().GetMemoryRegionContainingAddress(address, error)) {
269a758c9f7SMed Ismail Bennani     if (error.Fail())
270a758c9f7SMed Ismail Bennani       break;
271a758c9f7SMed Ismail Bennani 
272a758c9f7SMed Ismail Bennani     MemoryRegionInfo &mem_region = *region_or_err;
273a758c9f7SMed Ismail Bennani     auto range = mem_region.GetRange();
274312b43daSMed Ismail Bennani     address += range.GetRangeBase() + range.GetByteSize();
275a758c9f7SMed Ismail Bennani     region_list.push_back(mem_region);
276312b43daSMed Ismail Bennani   }
277312b43daSMed Ismail Bennani 
278a758c9f7SMed Ismail Bennani   return error;
279312b43daSMed Ismail Bennani }
280312b43daSMed Ismail Bennani 
281312b43daSMed Ismail Bennani void ScriptedProcess::Clear() { Process::m_thread_list.Clear(); }
282312b43daSMed Ismail Bennani 
283312b43daSMed Ismail Bennani bool ScriptedProcess::DoUpdateThreadList(ThreadList &old_thread_list,
284312b43daSMed Ismail Bennani                                          ThreadList &new_thread_list) {
285312b43daSMed Ismail Bennani   // TODO: Implement
286312b43daSMed Ismail Bennani   // This is supposed to get the current set of threads, if any of them are in
287312b43daSMed Ismail Bennani   // old_thread_list then they get copied to new_thread_list, and then any
288312b43daSMed Ismail Bennani   // actually new threads will get added to new_thread_list.
28959d8dd79SMed Ismail Bennani 
29059d8dd79SMed Ismail Bennani   CheckInterpreterAndScriptObject();
291676576b6SMed Ismail Bennani   m_thread_plans.ClearThreadCache();
29259d8dd79SMed Ismail Bennani 
29359d8dd79SMed Ismail Bennani   Status error;
29459d8dd79SMed Ismail Bennani   ScriptLanguage language = m_interpreter->GetLanguage();
29559d8dd79SMed Ismail Bennani 
29659d8dd79SMed Ismail Bennani   if (language != eScriptLanguagePython)
29791bb1161SMed Ismail Bennani     return ScriptedInterface::ErrorWithMessage<bool>(
29888a941baSMed Ismail Bennani         LLVM_PRETTY_FUNCTION,
29959d8dd79SMed Ismail Bennani         llvm::Twine("ScriptInterpreter language (" +
30059d8dd79SMed Ismail Bennani                     llvm::Twine(m_interpreter->LanguageToString(language)) +
30159d8dd79SMed Ismail Bennani                     llvm::Twine(") not supported."))
30259d8dd79SMed Ismail Bennani             .str(),
30359d8dd79SMed Ismail Bennani         error);
30459d8dd79SMed Ismail Bennani 
305d3e0f7e1SMed Ismail Bennani   StructuredData::DictionarySP thread_info_sp = GetInterface().GetThreadsInfo();
306d3e0f7e1SMed Ismail Bennani 
3077c54ffdcSMed Ismail Bennani   // FIXME: Need to sort the dictionary otherwise the thread ids won't match the
3087c54ffdcSMed Ismail Bennani   // thread indices.
3097c54ffdcSMed Ismail Bennani 
310d3e0f7e1SMed Ismail Bennani   if (!thread_info_sp)
31191bb1161SMed Ismail Bennani     return ScriptedInterface::ErrorWithMessage<bool>(
312d3e0f7e1SMed Ismail Bennani         LLVM_PRETTY_FUNCTION,
313d3e0f7e1SMed Ismail Bennani         "Couldn't fetch thread list from Scripted Process.", error);
314d3e0f7e1SMed Ismail Bennani 
315d3e0f7e1SMed Ismail Bennani   auto create_scripted_thread =
316d3e0f7e1SMed Ismail Bennani       [this, &old_thread_list, &error,
317d3e0f7e1SMed Ismail Bennani        &new_thread_list](ConstString key, StructuredData::Object *val) -> bool {
318d3e0f7e1SMed Ismail Bennani     if (!val)
31991bb1161SMed Ismail Bennani       return ScriptedInterface::ErrorWithMessage<bool>(
320d3e0f7e1SMed Ismail Bennani           LLVM_PRETTY_FUNCTION, "Invalid thread info object", error);
321d3e0f7e1SMed Ismail Bennani 
322d3e0f7e1SMed Ismail Bennani     lldb::tid_t tid = LLDB_INVALID_THREAD_ID;
323d3e0f7e1SMed Ismail Bennani     if (!llvm::to_integer(key.AsCString(), tid))
32491bb1161SMed Ismail Bennani       return ScriptedInterface::ErrorWithMessage<bool>(
32591bb1161SMed Ismail Bennani           LLVM_PRETTY_FUNCTION, "Invalid thread id", error);
326d3e0f7e1SMed Ismail Bennani 
327d3e0f7e1SMed Ismail Bennani     if (ThreadSP thread_sp =
328d3e0f7e1SMed Ismail Bennani             old_thread_list.FindThreadByID(tid, false /*=can_update*/)) {
329d3e0f7e1SMed Ismail Bennani       // If the thread was already in the old_thread_list,
330d3e0f7e1SMed Ismail Bennani       // just add it back to the new_thread_list.
331d3e0f7e1SMed Ismail Bennani       new_thread_list.AddThread(thread_sp);
332d3e0f7e1SMed Ismail Bennani       return true;
333d3e0f7e1SMed Ismail Bennani     }
334d3e0f7e1SMed Ismail Bennani 
33545148bfeSMed Ismail Bennani     auto thread_or_error = ScriptedThread::Create(*this, val->GetAsGeneric());
33659d8dd79SMed Ismail Bennani 
33745148bfeSMed Ismail Bennani     if (!thread_or_error)
33891bb1161SMed Ismail Bennani       return ScriptedInterface::ErrorWithMessage<bool>(
33945148bfeSMed Ismail Bennani           LLVM_PRETTY_FUNCTION, toString(thread_or_error.takeError()), error);
34045148bfeSMed Ismail Bennani 
34145148bfeSMed Ismail Bennani     ThreadSP thread_sp = thread_or_error.get();
34245148bfeSMed Ismail Bennani     lldbassert(thread_sp && "Couldn't initialize scripted thread.");
34359d8dd79SMed Ismail Bennani 
344caea440aSMed Ismail Bennani     RegisterContextSP reg_ctx_sp = thread_sp->GetRegisterContext();
345caea440aSMed Ismail Bennani     if (!reg_ctx_sp)
34691bb1161SMed Ismail Bennani       return ScriptedInterface::ErrorWithMessage<bool>(
347d3e0f7e1SMed Ismail Bennani           LLVM_PRETTY_FUNCTION,
348d3e0f7e1SMed Ismail Bennani           llvm::Twine("Invalid Register Context for thread " +
349d3e0f7e1SMed Ismail Bennani                       llvm::Twine(key.AsCString()))
350d3e0f7e1SMed Ismail Bennani               .str(),
351d3e0f7e1SMed Ismail Bennani           error);
352caea440aSMed Ismail Bennani 
35359d8dd79SMed Ismail Bennani     new_thread_list.AddThread(thread_sp);
35459d8dd79SMed Ismail Bennani 
355d3e0f7e1SMed Ismail Bennani     return true;
356d3e0f7e1SMed Ismail Bennani   };
357d3e0f7e1SMed Ismail Bennani 
358d3e0f7e1SMed Ismail Bennani   thread_info_sp->ForEach(create_scripted_thread);
359d3e0f7e1SMed Ismail Bennani 
360312b43daSMed Ismail Bennani   return new_thread_list.GetSize(false) > 0;
361312b43daSMed Ismail Bennani }
362312b43daSMed Ismail Bennani 
363676576b6SMed Ismail Bennani void ScriptedProcess::RefreshStateAfterStop() {
364676576b6SMed Ismail Bennani   // Let all threads recover from stopping and do any clean up based on the
365676576b6SMed Ismail Bennani   // previous thread state (if any).
36670665844SMed Ismail Bennani   m_thread_list.RefreshStateAfterStop();
367676576b6SMed Ismail Bennani }
368676576b6SMed Ismail Bennani 
369312b43daSMed Ismail Bennani bool ScriptedProcess::GetProcessInfo(ProcessInstanceInfo &info) {
370312b43daSMed Ismail Bennani   info.Clear();
371312b43daSMed Ismail Bennani   info.SetProcessID(GetID());
372312b43daSMed Ismail Bennani   info.SetArchitecture(GetArchitecture());
373312b43daSMed Ismail Bennani   lldb::ModuleSP module_sp = GetTarget().GetExecutableModule();
374312b43daSMed Ismail Bennani   if (module_sp) {
375312b43daSMed Ismail Bennani     const bool add_exe_file_as_first_arg = false;
376312b43daSMed Ismail Bennani     info.SetExecutableFile(GetTarget().GetExecutableModule()->GetFileSpec(),
377312b43daSMed Ismail Bennani                            add_exe_file_as_first_arg);
378312b43daSMed Ismail Bennani   }
379312b43daSMed Ismail Bennani   return true;
380312b43daSMed Ismail Bennani }
381312b43daSMed Ismail Bennani 
382*680ca7f2SMed Ismail Bennani lldb_private::StructuredData::ObjectSP
383*680ca7f2SMed Ismail Bennani ScriptedProcess::GetLoadedDynamicLibrariesInfos() {
384*680ca7f2SMed Ismail Bennani   CheckInterpreterAndScriptObject();
385*680ca7f2SMed Ismail Bennani 
386*680ca7f2SMed Ismail Bennani   Status error;
387*680ca7f2SMed Ismail Bennani   auto error_with_message = [&error](llvm::StringRef message) {
388*680ca7f2SMed Ismail Bennani     return ScriptedInterface::ErrorWithMessage<bool>(LLVM_PRETTY_FUNCTION,
389*680ca7f2SMed Ismail Bennani                                                      message.data(), error);
390*680ca7f2SMed Ismail Bennani   };
391*680ca7f2SMed Ismail Bennani 
392*680ca7f2SMed Ismail Bennani   StructuredData::ArraySP loaded_images_sp = GetInterface().GetLoadedImages();
393*680ca7f2SMed Ismail Bennani 
394*680ca7f2SMed Ismail Bennani   if (!loaded_images_sp || !loaded_images_sp->GetSize())
395*680ca7f2SMed Ismail Bennani     return GetInterface().ErrorWithMessage<StructuredData::ObjectSP>(
396*680ca7f2SMed Ismail Bennani         LLVM_PRETTY_FUNCTION, "No loaded images.", error);
397*680ca7f2SMed Ismail Bennani 
398*680ca7f2SMed Ismail Bennani   ModuleList module_list;
399*680ca7f2SMed Ismail Bennani   Target &target = GetTarget();
400*680ca7f2SMed Ismail Bennani 
401*680ca7f2SMed Ismail Bennani   auto reload_image = [&target, &module_list, &error_with_message](
402*680ca7f2SMed Ismail Bennani                           StructuredData::Object *obj) -> bool {
403*680ca7f2SMed Ismail Bennani     StructuredData::Dictionary *dict = obj->GetAsDictionary();
404*680ca7f2SMed Ismail Bennani 
405*680ca7f2SMed Ismail Bennani     if (!dict)
406*680ca7f2SMed Ismail Bennani       return error_with_message("Couldn't cast image object into dictionary.");
407*680ca7f2SMed Ismail Bennani 
408*680ca7f2SMed Ismail Bennani     ModuleSpec module_spec;
409*680ca7f2SMed Ismail Bennani     llvm::StringRef value;
410*680ca7f2SMed Ismail Bennani 
411*680ca7f2SMed Ismail Bennani     bool has_path = dict->HasKey("path");
412*680ca7f2SMed Ismail Bennani     bool has_uuid = dict->HasKey("uuid");
413*680ca7f2SMed Ismail Bennani     if (!has_path && !has_uuid)
414*680ca7f2SMed Ismail Bennani       return error_with_message("Dictionary should have key 'path' or 'uuid'");
415*680ca7f2SMed Ismail Bennani     if (!dict->HasKey("load_addr"))
416*680ca7f2SMed Ismail Bennani       return error_with_message("Dictionary is missing key 'load_addr'");
417*680ca7f2SMed Ismail Bennani 
418*680ca7f2SMed Ismail Bennani     if (has_path) {
419*680ca7f2SMed Ismail Bennani       dict->GetValueForKeyAsString("path", value);
420*680ca7f2SMed Ismail Bennani       module_spec.GetFileSpec().SetPath(value);
421*680ca7f2SMed Ismail Bennani     }
422*680ca7f2SMed Ismail Bennani 
423*680ca7f2SMed Ismail Bennani     if (has_uuid) {
424*680ca7f2SMed Ismail Bennani       dict->GetValueForKeyAsString("uuid", value);
425*680ca7f2SMed Ismail Bennani       module_spec.GetUUID().SetFromStringRef(value);
426*680ca7f2SMed Ismail Bennani     }
427*680ca7f2SMed Ismail Bennani     module_spec.GetArchitecture() = target.GetArchitecture();
428*680ca7f2SMed Ismail Bennani 
429*680ca7f2SMed Ismail Bennani     ModuleSP module_sp =
430*680ca7f2SMed Ismail Bennani         target.GetOrCreateModule(module_spec, true /* notify */);
431*680ca7f2SMed Ismail Bennani 
432*680ca7f2SMed Ismail Bennani     if (!module_sp)
433*680ca7f2SMed Ismail Bennani       return error_with_message("Couldn't create or get module.");
434*680ca7f2SMed Ismail Bennani 
435*680ca7f2SMed Ismail Bennani     lldb::addr_t load_addr = LLDB_INVALID_ADDRESS;
436*680ca7f2SMed Ismail Bennani     lldb::addr_t slide = LLDB_INVALID_OFFSET;
437*680ca7f2SMed Ismail Bennani     dict->GetValueForKeyAsInteger("load_addr", load_addr);
438*680ca7f2SMed Ismail Bennani     dict->GetValueForKeyAsInteger("slide", slide);
439*680ca7f2SMed Ismail Bennani     if (load_addr == LLDB_INVALID_ADDRESS)
440*680ca7f2SMed Ismail Bennani       return error_with_message(
441*680ca7f2SMed Ismail Bennani           "Couldn't get valid load address or slide offset.");
442*680ca7f2SMed Ismail Bennani 
443*680ca7f2SMed Ismail Bennani     if (slide != LLDB_INVALID_OFFSET)
444*680ca7f2SMed Ismail Bennani       load_addr += slide;
445*680ca7f2SMed Ismail Bennani 
446*680ca7f2SMed Ismail Bennani     bool changed = false;
447*680ca7f2SMed Ismail Bennani     module_sp->SetLoadAddress(target, load_addr, false /*=value_is_offset*/,
448*680ca7f2SMed Ismail Bennani                               changed);
449*680ca7f2SMed Ismail Bennani 
450*680ca7f2SMed Ismail Bennani     if (!changed && !module_sp->GetObjectFile())
451*680ca7f2SMed Ismail Bennani       return error_with_message("Couldn't set the load address for module.");
452*680ca7f2SMed Ismail Bennani 
453*680ca7f2SMed Ismail Bennani     dict->GetValueForKeyAsString("path", value);
454*680ca7f2SMed Ismail Bennani     FileSpec objfile(value);
455*680ca7f2SMed Ismail Bennani     module_sp->SetFileSpecAndObjectName(objfile, objfile.GetFilename());
456*680ca7f2SMed Ismail Bennani 
457*680ca7f2SMed Ismail Bennani     return module_list.AppendIfNeeded(module_sp);
458*680ca7f2SMed Ismail Bennani   };
459*680ca7f2SMed Ismail Bennani 
460*680ca7f2SMed Ismail Bennani   if (!loaded_images_sp->ForEach(reload_image))
461*680ca7f2SMed Ismail Bennani     return GetInterface().ErrorWithMessage<StructuredData::ObjectSP>(
462*680ca7f2SMed Ismail Bennani         LLVM_PRETTY_FUNCTION, "Couldn't reload all images.", error);
463*680ca7f2SMed Ismail Bennani 
464*680ca7f2SMed Ismail Bennani   target.ModulesDidLoad(module_list);
465*680ca7f2SMed Ismail Bennani 
466*680ca7f2SMed Ismail Bennani   return loaded_images_sp;
467*680ca7f2SMed Ismail Bennani }
468*680ca7f2SMed Ismail Bennani 
469312b43daSMed Ismail Bennani ScriptedProcessInterface &ScriptedProcess::GetInterface() const {
470312b43daSMed Ismail Bennani   return m_interpreter->GetScriptedProcessInterface();
471312b43daSMed Ismail Bennani }
472