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"
23312b43daSMed Ismail Bennani 
24312b43daSMed Ismail Bennani #include "lldb/Utility/Log.h"
25312b43daSMed Ismail Bennani #include "lldb/Utility/Logging.h"
26312b43daSMed Ismail Bennani #include "lldb/Utility/State.h"
27312b43daSMed Ismail Bennani 
28312b43daSMed Ismail Bennani #include <mutex>
29312b43daSMed Ismail Bennani 
30312b43daSMed Ismail Bennani LLDB_PLUGIN_DEFINE(ScriptedProcess)
31312b43daSMed Ismail Bennani 
32312b43daSMed Ismail Bennani using namespace lldb;
33312b43daSMed Ismail Bennani using namespace lldb_private;
34312b43daSMed Ismail Bennani 
35312b43daSMed Ismail Bennani ConstString ScriptedProcess::GetPluginNameStatic() {
36312b43daSMed Ismail Bennani   static ConstString g_name("ScriptedProcess");
37312b43daSMed Ismail Bennani   return g_name;
38312b43daSMed Ismail Bennani }
39312b43daSMed Ismail Bennani 
40312b43daSMed Ismail Bennani const char *ScriptedProcess::GetPluginDescriptionStatic() {
41312b43daSMed Ismail Bennani   return "Scripted Process plug-in.";
42312b43daSMed Ismail Bennani }
43312b43daSMed Ismail Bennani 
44312b43daSMed Ismail Bennani static constexpr lldb::ScriptLanguage g_supported_script_languages[] = {
45312b43daSMed Ismail Bennani     ScriptLanguage::eScriptLanguagePython,
46312b43daSMed Ismail Bennani };
47312b43daSMed Ismail Bennani 
48312b43daSMed Ismail Bennani bool ScriptedProcess::IsScriptLanguageSupported(lldb::ScriptLanguage language) {
49312b43daSMed Ismail Bennani   llvm::ArrayRef<lldb::ScriptLanguage> supported_languages =
50312b43daSMed Ismail Bennani       llvm::makeArrayRef(g_supported_script_languages);
51312b43daSMed Ismail Bennani 
52312b43daSMed Ismail Bennani   return llvm::is_contained(supported_languages, language);
53312b43daSMed Ismail Bennani }
54312b43daSMed Ismail Bennani 
55312b43daSMed Ismail Bennani void ScriptedProcess::CheckInterpreterAndScriptObject() const {
56312b43daSMed Ismail Bennani   lldbassert(m_interpreter && "Invalid Script Interpreter.");
57312b43daSMed Ismail Bennani   lldbassert(m_script_object_sp && "Invalid Script Object.");
58312b43daSMed Ismail Bennani }
59312b43daSMed Ismail Bennani 
60312b43daSMed Ismail Bennani lldb::ProcessSP ScriptedProcess::CreateInstance(lldb::TargetSP target_sp,
61312b43daSMed Ismail Bennani                                                 lldb::ListenerSP listener_sp,
62312b43daSMed Ismail Bennani                                                 const FileSpec *file,
63312b43daSMed Ismail Bennani                                                 bool can_connect) {
64312b43daSMed Ismail Bennani   if (!target_sp ||
65312b43daSMed Ismail Bennani       !IsScriptLanguageSupported(target_sp->GetDebugger().GetScriptLanguage()))
66312b43daSMed Ismail Bennani     return nullptr;
67312b43daSMed Ismail Bennani 
68312b43daSMed Ismail Bennani   Status error;
69312b43daSMed Ismail Bennani   ScriptedProcess::ScriptedProcessInfo scripted_process_info(
70312b43daSMed Ismail Bennani       target_sp->GetProcessLaunchInfo());
71312b43daSMed Ismail Bennani 
72312b43daSMed Ismail Bennani   auto process_sp = std::make_shared<ScriptedProcess>(
73312b43daSMed Ismail Bennani       target_sp, listener_sp, scripted_process_info, error);
74312b43daSMed Ismail Bennani 
75312b43daSMed Ismail Bennani   if (error.Fail() || !process_sp || !process_sp->m_script_object_sp ||
76312b43daSMed Ismail Bennani       !process_sp->m_script_object_sp->IsValid()) {
77312b43daSMed Ismail Bennani     LLDB_LOGF(GetLogIfAllCategoriesSet(LIBLLDB_LOG_PROCESS), "%s",
78312b43daSMed Ismail Bennani               error.AsCString());
79312b43daSMed Ismail Bennani     return nullptr;
80312b43daSMed Ismail Bennani   }
81312b43daSMed Ismail Bennani 
82312b43daSMed Ismail Bennani   return process_sp;
83312b43daSMed Ismail Bennani }
84312b43daSMed Ismail Bennani 
85312b43daSMed Ismail Bennani bool ScriptedProcess::CanDebug(lldb::TargetSP target_sp,
86312b43daSMed Ismail Bennani                                bool plugin_specified_by_name) {
87312b43daSMed Ismail Bennani   return true;
88312b43daSMed Ismail Bennani }
89312b43daSMed Ismail Bennani 
90312b43daSMed Ismail Bennani ScriptedProcess::ScriptedProcess(
91312b43daSMed Ismail Bennani     lldb::TargetSP target_sp, lldb::ListenerSP listener_sp,
92312b43daSMed Ismail Bennani     const ScriptedProcess::ScriptedProcessInfo &scripted_process_info,
93312b43daSMed Ismail Bennani     Status &error)
94312b43daSMed Ismail Bennani     : Process(target_sp, listener_sp),
95312b43daSMed Ismail Bennani       m_scripted_process_info(scripted_process_info) {
96312b43daSMed Ismail Bennani 
97312b43daSMed Ismail Bennani   if (!target_sp) {
98312b43daSMed Ismail Bennani     error.SetErrorStringWithFormat("ScriptedProcess::%s () - ERROR: %s",
99312b43daSMed Ismail Bennani                                    __FUNCTION__, "Invalid target");
100312b43daSMed Ismail Bennani     return;
101312b43daSMed Ismail Bennani   }
102312b43daSMed Ismail Bennani 
103312b43daSMed Ismail Bennani   m_interpreter = target_sp->GetDebugger().GetScriptInterpreter();
104312b43daSMed Ismail Bennani 
105312b43daSMed Ismail Bennani   if (!m_interpreter) {
106312b43daSMed Ismail Bennani     error.SetErrorStringWithFormat("ScriptedProcess::%s () - ERROR: %s",
107312b43daSMed Ismail Bennani                                    __FUNCTION__,
108312b43daSMed Ismail Bennani                                    "Debugger has no Script Interpreter");
109312b43daSMed Ismail Bennani     return;
110312b43daSMed Ismail Bennani   }
111312b43daSMed Ismail Bennani 
112*3925204cSMed Ismail Bennani   ExecutionContext exe_ctx(target_sp, /*get_process=*/false);
113*3925204cSMed Ismail Bennani 
114*3925204cSMed Ismail Bennani   StructuredData::GenericSP object_sp = GetInterface().CreatePluginObject(
115*3925204cSMed Ismail Bennani       m_scripted_process_info.GetClassName().c_str(), exe_ctx,
116312b43daSMed Ismail Bennani       m_scripted_process_info.GetDictionarySP());
117312b43daSMed Ismail Bennani 
118312b43daSMed Ismail Bennani   if (!object_sp || !object_sp->IsValid()) {
119312b43daSMed Ismail Bennani     error.SetErrorStringWithFormat("ScriptedProcess::%s () - ERROR: %s",
120312b43daSMed Ismail Bennani                                    __FUNCTION__,
121312b43daSMed Ismail Bennani                                    "Failed to create valid script object");
122312b43daSMed Ismail Bennani     return;
123312b43daSMed Ismail Bennani   }
124312b43daSMed Ismail Bennani 
125312b43daSMed Ismail Bennani   m_script_object_sp = object_sp;
126312b43daSMed Ismail Bennani }
127312b43daSMed Ismail Bennani 
128312b43daSMed Ismail Bennani ScriptedProcess::~ScriptedProcess() {
129312b43daSMed Ismail Bennani   Clear();
130312b43daSMed Ismail Bennani   // We need to call finalize on the process before destroying ourselves to
131312b43daSMed Ismail Bennani   // make sure all of the broadcaster cleanup goes as planned. If we destruct
132312b43daSMed Ismail Bennani   // this class, then Process::~Process() might have problems trying to fully
133312b43daSMed Ismail Bennani   // destroy the broadcaster.
134312b43daSMed Ismail Bennani   Finalize();
135312b43daSMed Ismail Bennani }
136312b43daSMed Ismail Bennani 
137312b43daSMed Ismail Bennani void ScriptedProcess::Initialize() {
138312b43daSMed Ismail Bennani   static llvm::once_flag g_once_flag;
139312b43daSMed Ismail Bennani 
140312b43daSMed Ismail Bennani   llvm::call_once(g_once_flag, []() {
141312b43daSMed Ismail Bennani     PluginManager::RegisterPlugin(GetPluginNameStatic(),
142312b43daSMed Ismail Bennani                                   GetPluginDescriptionStatic(), CreateInstance);
143312b43daSMed Ismail Bennani   });
144312b43daSMed Ismail Bennani }
145312b43daSMed Ismail Bennani 
146312b43daSMed Ismail Bennani void ScriptedProcess::Terminate() {
147312b43daSMed Ismail Bennani   PluginManager::UnregisterPlugin(ScriptedProcess::CreateInstance);
148312b43daSMed Ismail Bennani }
149312b43daSMed Ismail Bennani 
150312b43daSMed Ismail Bennani ConstString ScriptedProcess::GetPluginName() { return GetPluginNameStatic(); }
151312b43daSMed Ismail Bennani 
152312b43daSMed Ismail Bennani uint32_t ScriptedProcess::GetPluginVersion() { return 1; }
153312b43daSMed Ismail Bennani 
154312b43daSMed Ismail Bennani Status ScriptedProcess::DoLoadCore() {
155312b43daSMed Ismail Bennani   ProcessLaunchInfo launch_info = GetTarget().GetProcessLaunchInfo();
156312b43daSMed Ismail Bennani 
157312b43daSMed Ismail Bennani   return DoLaunch(nullptr, launch_info);
158312b43daSMed Ismail Bennani }
159312b43daSMed Ismail Bennani 
160312b43daSMed Ismail Bennani Status ScriptedProcess::DoLaunch(Module *exe_module,
161312b43daSMed Ismail Bennani                                  ProcessLaunchInfo &launch_info) {
162312b43daSMed Ismail Bennani   CheckInterpreterAndScriptObject();
163312b43daSMed Ismail Bennani 
164312b43daSMed Ismail Bennani   /* FIXME: This doesn't reflect how lldb actually launches a process.
165312b43daSMed Ismail Bennani            In reality, it attaches to debugserver, then resume the process. */
166312b43daSMed Ismail Bennani   Status error = GetInterface().Launch();
167312b43daSMed Ismail Bennani   SetPrivateState(eStateRunning);
168312b43daSMed Ismail Bennani 
169312b43daSMed Ismail Bennani   if (error.Fail())
170312b43daSMed Ismail Bennani     return error;
171312b43daSMed Ismail Bennani 
172312b43daSMed Ismail Bennani   // TODO: Fetch next state from stopped event queue then send stop event
173312b43daSMed Ismail Bennani   //  const StateType state = SetThreadStopInfo(response);
174312b43daSMed Ismail Bennani   //  if (state != eStateInvalid) {
175312b43daSMed Ismail Bennani   //    SetPrivateState(state);
176312b43daSMed Ismail Bennani 
177312b43daSMed Ismail Bennani   SetPrivateState(eStateStopped);
178312b43daSMed Ismail Bennani 
179312b43daSMed Ismail Bennani   UpdateThreadListIfNeeded();
180312b43daSMed Ismail Bennani   GetThreadList();
181312b43daSMed Ismail Bennani 
182312b43daSMed Ismail Bennani   return {};
183312b43daSMed Ismail Bennani }
184312b43daSMed Ismail Bennani 
185312b43daSMed Ismail Bennani void ScriptedProcess::DidLaunch() {
186312b43daSMed Ismail Bennani   CheckInterpreterAndScriptObject();
187312b43daSMed Ismail Bennani   m_pid = GetInterface().GetProcessID();
188312b43daSMed Ismail Bennani }
189312b43daSMed Ismail Bennani 
190312b43daSMed Ismail Bennani Status ScriptedProcess::DoResume() {
191312b43daSMed Ismail Bennani   CheckInterpreterAndScriptObject();
192312b43daSMed Ismail Bennani 
193312b43daSMed Ismail Bennani   Log *log(lldb_private::GetLogIfAllCategoriesSet(LIBLLDB_LOG_PROCESS));
194312b43daSMed Ismail Bennani   // FIXME: Fetch data from thread.
195312b43daSMed Ismail Bennani   const StateType thread_resume_state = eStateRunning;
196312b43daSMed Ismail Bennani   LLDB_LOGF(log, "ScriptedProcess::%s thread_resume_state = %s", __FUNCTION__,
197312b43daSMed Ismail Bennani             StateAsCString(thread_resume_state));
198312b43daSMed Ismail Bennani 
199312b43daSMed Ismail Bennani   bool resume = (thread_resume_state == eStateRunning);
200312b43daSMed Ismail Bennani   assert(thread_resume_state == eStateRunning && "invalid thread resume state");
201312b43daSMed Ismail Bennani 
202312b43daSMed Ismail Bennani   Status error;
203312b43daSMed Ismail Bennani   if (resume) {
204312b43daSMed Ismail Bennani     LLDB_LOGF(log, "ScriptedProcess::%s sending resume", __FUNCTION__);
205312b43daSMed Ismail Bennani 
206312b43daSMed Ismail Bennani     SetPrivateState(eStateRunning);
207312b43daSMed Ismail Bennani     SetPrivateState(eStateStopped);
208312b43daSMed Ismail Bennani     error = GetInterface().Resume();
209312b43daSMed Ismail Bennani   }
210312b43daSMed Ismail Bennani 
211312b43daSMed Ismail Bennani   return error;
212312b43daSMed Ismail Bennani }
213312b43daSMed Ismail Bennani 
214312b43daSMed Ismail Bennani Status ScriptedProcess::DoStop() {
215312b43daSMed Ismail Bennani   CheckInterpreterAndScriptObject();
216312b43daSMed Ismail Bennani 
217312b43daSMed Ismail Bennani   Log *log(lldb_private::GetLogIfAllCategoriesSet(LIBLLDB_LOG_PROCESS));
218312b43daSMed Ismail Bennani 
219312b43daSMed Ismail Bennani   if (GetInterface().ShouldStop()) {
220312b43daSMed Ismail Bennani     SetPrivateState(eStateStopped);
221312b43daSMed Ismail Bennani     LLDB_LOGF(log, "ScriptedProcess::%s Immediate stop", __FUNCTION__);
222312b43daSMed Ismail Bennani     return {};
223312b43daSMed Ismail Bennani   }
224312b43daSMed Ismail Bennani 
225312b43daSMed Ismail Bennani   LLDB_LOGF(log, "ScriptedProcess::%s Delayed stop", __FUNCTION__);
226312b43daSMed Ismail Bennani   return GetInterface().Stop();
227312b43daSMed Ismail Bennani }
228312b43daSMed Ismail Bennani 
229312b43daSMed Ismail Bennani Status ScriptedProcess::DoDestroy() { return Status(); }
230312b43daSMed Ismail Bennani 
231312b43daSMed Ismail Bennani bool ScriptedProcess::IsAlive() {
232312b43daSMed Ismail Bennani   if (m_interpreter && m_script_object_sp)
233312b43daSMed Ismail Bennani     return GetInterface().IsAlive();
234312b43daSMed Ismail Bennani   return false;
235312b43daSMed Ismail Bennani }
236312b43daSMed Ismail Bennani 
237312b43daSMed Ismail Bennani size_t ScriptedProcess::DoReadMemory(lldb::addr_t addr, void *buf, size_t size,
238312b43daSMed Ismail Bennani                                      Status &error) {
239312b43daSMed Ismail Bennani 
240312b43daSMed Ismail Bennani   auto error_with_message = [&error](llvm::StringRef message) {
241312b43daSMed Ismail Bennani     error.SetErrorString(message);
242312b43daSMed Ismail Bennani     return 0;
243312b43daSMed Ismail Bennani   };
244312b43daSMed Ismail Bennani 
245312b43daSMed Ismail Bennani   if (!m_interpreter)
246312b43daSMed Ismail Bennani     return error_with_message("No interpreter.");
247312b43daSMed Ismail Bennani 
248312b43daSMed Ismail Bennani   lldb::DataExtractorSP data_extractor_sp =
249312b43daSMed Ismail Bennani       GetInterface().ReadMemoryAtAddress(addr, size, error);
250312b43daSMed Ismail Bennani 
251312b43daSMed Ismail Bennani   if (!data_extractor_sp || error.Fail())
252312b43daSMed Ismail Bennani     return 0;
253312b43daSMed Ismail Bennani 
254312b43daSMed Ismail Bennani   offset_t bytes_copied = data_extractor_sp->CopyByteOrderedData(
255312b43daSMed Ismail Bennani       0, data_extractor_sp->GetByteSize(), buf, size, GetByteOrder());
256312b43daSMed Ismail Bennani 
257312b43daSMed Ismail Bennani   if (!bytes_copied || bytes_copied == LLDB_INVALID_OFFSET)
258312b43daSMed Ismail Bennani     return error_with_message("Failed to copy read memory to buffer.");
259312b43daSMed Ismail Bennani 
260312b43daSMed Ismail Bennani   return size;
261312b43daSMed Ismail Bennani }
262312b43daSMed Ismail Bennani 
263312b43daSMed Ismail Bennani ArchSpec ScriptedProcess::GetArchitecture() {
264312b43daSMed Ismail Bennani   return GetTarget().GetArchitecture();
265312b43daSMed Ismail Bennani }
266312b43daSMed Ismail Bennani 
267312b43daSMed Ismail Bennani Status ScriptedProcess::GetMemoryRegionInfo(lldb::addr_t load_addr,
268312b43daSMed Ismail Bennani                                             MemoryRegionInfo &region) {
269312b43daSMed Ismail Bennani   // TODO: Implement
270312b43daSMed Ismail Bennani   return Status();
271312b43daSMed Ismail Bennani }
272312b43daSMed Ismail Bennani 
273312b43daSMed Ismail Bennani Status ScriptedProcess::GetMemoryRegions(MemoryRegionInfos &region_list) {
274312b43daSMed Ismail Bennani   CheckInterpreterAndScriptObject();
275312b43daSMed Ismail Bennani 
276312b43daSMed Ismail Bennani   lldb::addr_t address = 0;
277312b43daSMed Ismail Bennani   lldb::MemoryRegionInfoSP mem_region_sp = nullptr;
278312b43daSMed Ismail Bennani 
279312b43daSMed Ismail Bennani   while ((mem_region_sp =
280312b43daSMed Ismail Bennani               GetInterface().GetMemoryRegionContainingAddress(address))) {
281312b43daSMed Ismail Bennani     auto range = mem_region_sp->GetRange();
282312b43daSMed Ismail Bennani     address += range.GetRangeBase() + range.GetByteSize();
283312b43daSMed Ismail Bennani     region_list.push_back(*mem_region_sp.get());
284312b43daSMed Ismail Bennani   }
285312b43daSMed Ismail Bennani 
286312b43daSMed Ismail Bennani   return {};
287312b43daSMed Ismail Bennani }
288312b43daSMed Ismail Bennani 
289312b43daSMed Ismail Bennani void ScriptedProcess::Clear() { Process::m_thread_list.Clear(); }
290312b43daSMed Ismail Bennani 
291312b43daSMed Ismail Bennani bool ScriptedProcess::DoUpdateThreadList(ThreadList &old_thread_list,
292312b43daSMed Ismail Bennani                                          ThreadList &new_thread_list) {
293312b43daSMed Ismail Bennani   // TODO: Implement
294312b43daSMed Ismail Bennani   // This is supposed to get the current set of threads, if any of them are in
295312b43daSMed Ismail Bennani   // old_thread_list then they get copied to new_thread_list, and then any
296312b43daSMed Ismail Bennani   // actually new threads will get added to new_thread_list.
297312b43daSMed Ismail Bennani   return new_thread_list.GetSize(false) > 0;
298312b43daSMed Ismail Bennani }
299312b43daSMed Ismail Bennani 
300312b43daSMed Ismail Bennani bool ScriptedProcess::GetProcessInfo(ProcessInstanceInfo &info) {
301312b43daSMed Ismail Bennani   info.Clear();
302312b43daSMed Ismail Bennani   info.SetProcessID(GetID());
303312b43daSMed Ismail Bennani   info.SetArchitecture(GetArchitecture());
304312b43daSMed Ismail Bennani   lldb::ModuleSP module_sp = GetTarget().GetExecutableModule();
305312b43daSMed Ismail Bennani   if (module_sp) {
306312b43daSMed Ismail Bennani     const bool add_exe_file_as_first_arg = false;
307312b43daSMed Ismail Bennani     info.SetExecutableFile(GetTarget().GetExecutableModule()->GetFileSpec(),
308312b43daSMed Ismail Bennani                            add_exe_file_as_first_arg);
309312b43daSMed Ismail Bennani   }
310312b43daSMed Ismail Bennani   return true;
311312b43daSMed Ismail Bennani }
312312b43daSMed Ismail Bennani 
313312b43daSMed Ismail Bennani ScriptedProcessInterface &ScriptedProcess::GetInterface() const {
314312b43daSMed Ismail Bennani   return m_interpreter->GetScriptedProcessInterface();
315312b43daSMed Ismail Bennani }
316