xref: /llvm-project/lldb/source/Plugins/ScriptInterpreter/Lua/ScriptInterpreterLua.cpp (revision e81ba283131cf76ae62fa9b601a24d080578efaa)
180814287SRaphael Isemann //===-- ScriptInterpreterLua.cpp ------------------------------------------===//
267de8962SJonas Devlieghere //
367de8962SJonas Devlieghere // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
467de8962SJonas Devlieghere // See https://llvm.org/LICENSE.txt for license information.
567de8962SJonas Devlieghere // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
667de8962SJonas Devlieghere //
767de8962SJonas Devlieghere //===----------------------------------------------------------------------===//
867de8962SJonas Devlieghere 
967de8962SJonas Devlieghere #include "ScriptInterpreterLua.h"
1028613242SJonas Devlieghere #include "Lua.h"
11a0d7406aSPedro Tammela #include "lldb/Breakpoint/StoppointCallbackContext.h"
1267de8962SJonas Devlieghere #include "lldb/Core/Debugger.h"
1367de8962SJonas Devlieghere #include "lldb/Core/PluginManager.h"
1467de8962SJonas Devlieghere #include "lldb/Core/StreamFile.h"
1528613242SJonas Devlieghere #include "lldb/Interpreter/CommandReturnObject.h"
16a0d7406aSPedro Tammela #include "lldb/Target/ExecutionContext.h"
1767de8962SJonas Devlieghere #include "lldb/Utility/Stream.h"
1867de8962SJonas Devlieghere #include "lldb/Utility/StringList.h"
1928613242SJonas Devlieghere #include "lldb/Utility/Timer.h"
20d853bd7aSPedro Tammela #include "llvm/ADT/StringRef.h"
21ed8184b7SJonas Devlieghere #include "llvm/Support/FormatAdapters.h"
22a0d7406aSPedro Tammela #include <memory>
23d853bd7aSPedro Tammela #include <vector>
2467de8962SJonas Devlieghere 
2567de8962SJonas Devlieghere using namespace lldb;
2667de8962SJonas Devlieghere using namespace lldb_private;
2767de8962SJonas Devlieghere 
28bba9ba8dSJonas Devlieghere LLDB_PLUGIN_DEFINE(ScriptInterpreterLua)
29fbb4d1e4SJonas Devlieghere 
30d853bd7aSPedro Tammela enum ActiveIOHandler {
31d853bd7aSPedro Tammela   eIOHandlerNone,
32d853bd7aSPedro Tammela   eIOHandlerBreakpoint,
33d853bd7aSPedro Tammela   eIOHandlerWatchpoint
34d853bd7aSPedro Tammela };
35d853bd7aSPedro Tammela 
3628613242SJonas Devlieghere class IOHandlerLuaInterpreter : public IOHandlerDelegate,
3728613242SJonas Devlieghere                                 public IOHandlerEditline {
3828613242SJonas Devlieghere public:
394164be72SJonas Devlieghere   IOHandlerLuaInterpreter(Debugger &debugger,
40d853bd7aSPedro Tammela                           ScriptInterpreterLua &script_interpreter,
41d853bd7aSPedro Tammela                           ActiveIOHandler active_io_handler = eIOHandlerNone)
4228613242SJonas Devlieghere       : IOHandlerEditline(debugger, IOHandler::Type::LuaInterpreter, "lua",
4328613242SJonas Devlieghere                           ">>> ", "..> ", true, debugger.GetUseColor(), 0,
4428613242SJonas Devlieghere                           *this, nullptr),
45d853bd7aSPedro Tammela         m_script_interpreter(script_interpreter),
46d853bd7aSPedro Tammela         m_active_io_handler(active_io_handler) {
47fa1b4a96SJonas Devlieghere     llvm::cantFail(m_script_interpreter.GetLua().ChangeIO(
48fa1b4a96SJonas Devlieghere         debugger.GetOutputFile().GetStream(),
49fa1b4a96SJonas Devlieghere         debugger.GetErrorFile().GetStream()));
5045c971f7SJonas Devlieghere     llvm::cantFail(m_script_interpreter.EnterSession(debugger.GetID()));
5145c971f7SJonas Devlieghere   }
5245c971f7SJonas Devlieghere 
535ddd4fc5SJonas Devlieghere   ~IOHandlerLuaInterpreter() override {
5445c971f7SJonas Devlieghere     llvm::cantFail(m_script_interpreter.LeaveSession());
5545c971f7SJonas Devlieghere   }
5628613242SJonas Devlieghere 
57d853bd7aSPedro Tammela   void IOHandlerActivated(IOHandler &io_handler, bool interactive) override {
58d853bd7aSPedro Tammela     const char *instructions = nullptr;
59d853bd7aSPedro Tammela     switch (m_active_io_handler) {
60d853bd7aSPedro Tammela     case eIOHandlerNone:
61*e81ba283SSiger Yang       break;
62d853bd7aSPedro Tammela     case eIOHandlerWatchpoint:
63*e81ba283SSiger Yang       instructions = "Enter your Lua command(s). Type 'quit' to end.\n"
64*e81ba283SSiger Yang                      "The commands are compiled as the body of the following "
65*e81ba283SSiger Yang                      "Lua function\n"
66*e81ba283SSiger Yang                      "function (frame, wp) end\n";
67*e81ba283SSiger Yang       SetPrompt(llvm::StringRef("..> "));
68d853bd7aSPedro Tammela       break;
69d853bd7aSPedro Tammela     case eIOHandlerBreakpoint:
70d853bd7aSPedro Tammela       instructions = "Enter your Lua command(s). Type 'quit' to end.\n"
71d853bd7aSPedro Tammela                      "The commands are compiled as the body of the following "
72d853bd7aSPedro Tammela                      "Lua function\n"
73d853bd7aSPedro Tammela                      "function (frame, bp_loc, ...) end\n";
74d853bd7aSPedro Tammela       SetPrompt(llvm::StringRef("..> "));
75d853bd7aSPedro Tammela       break;
76d853bd7aSPedro Tammela     }
77d853bd7aSPedro Tammela     if (instructions == nullptr)
78d853bd7aSPedro Tammela       return;
79d853bd7aSPedro Tammela     if (interactive)
80d853bd7aSPedro Tammela       *io_handler.GetOutputStreamFileSP() << instructions;
81d853bd7aSPedro Tammela   }
82d853bd7aSPedro Tammela 
83d853bd7aSPedro Tammela   bool IOHandlerIsInputComplete(IOHandler &io_handler,
84d853bd7aSPedro Tammela                                 StringList &lines) override {
85d853bd7aSPedro Tammela     size_t last = lines.GetSize() - 1;
86d853bd7aSPedro Tammela     if (IsQuitCommand(lines.GetStringAtIndex(last))) {
87*e81ba283SSiger Yang       if (m_active_io_handler == eIOHandlerBreakpoint ||
88*e81ba283SSiger Yang           m_active_io_handler == eIOHandlerWatchpoint)
89d853bd7aSPedro Tammela         lines.DeleteStringAtIndex(last);
90d853bd7aSPedro Tammela       return true;
91d853bd7aSPedro Tammela     }
92d853bd7aSPedro Tammela     StreamString str;
93d853bd7aSPedro Tammela     lines.Join("\n", str);
94d853bd7aSPedro Tammela     if (llvm::Error E =
95d853bd7aSPedro Tammela             m_script_interpreter.GetLua().CheckSyntax(str.GetString())) {
96d853bd7aSPedro Tammela       std::string error_str = toString(std::move(E));
97d853bd7aSPedro Tammela       // Lua always errors out to incomplete code with '<eof>'
98d853bd7aSPedro Tammela       return error_str.find("<eof>") == std::string::npos;
99d853bd7aSPedro Tammela     }
100*e81ba283SSiger Yang     // The breakpoint and watchpoint handler only exits with a explicit 'quit'
101*e81ba283SSiger Yang     return m_active_io_handler != eIOHandlerBreakpoint &&
102*e81ba283SSiger Yang            m_active_io_handler != eIOHandlerWatchpoint;
103d853bd7aSPedro Tammela   }
104d853bd7aSPedro Tammela 
10528613242SJonas Devlieghere   void IOHandlerInputComplete(IOHandler &io_handler,
10628613242SJonas Devlieghere                               std::string &data) override {
107d853bd7aSPedro Tammela     switch (m_active_io_handler) {
108d853bd7aSPedro Tammela     case eIOHandlerBreakpoint: {
109cfb96d84SJim Ingham       auto *bp_options_vec =
110cfb96d84SJim Ingham           static_cast<std::vector<std::reference_wrapper<BreakpointOptions>> *>(
111d853bd7aSPedro Tammela               io_handler.GetUserData());
112cfb96d84SJim Ingham       for (BreakpointOptions &bp_options : *bp_options_vec) {
113d853bd7aSPedro Tammela         Status error = m_script_interpreter.SetBreakpointCommandCallback(
114d853bd7aSPedro Tammela             bp_options, data.c_str());
115d853bd7aSPedro Tammela         if (error.Fail())
116d853bd7aSPedro Tammela           *io_handler.GetErrorStreamFileSP() << error.AsCString() << '\n';
117d853bd7aSPedro Tammela       }
118d853bd7aSPedro Tammela       io_handler.SetIsDone(true);
119d853bd7aSPedro Tammela     } break;
120*e81ba283SSiger Yang     case eIOHandlerWatchpoint: {
121*e81ba283SSiger Yang       auto *wp_options =
122*e81ba283SSiger Yang           static_cast<WatchpointOptions *>(io_handler.GetUserData());
123*e81ba283SSiger Yang       m_script_interpreter.SetWatchpointCommandCallback(wp_options,
124*e81ba283SSiger Yang                                                         data.c_str());
125d853bd7aSPedro Tammela       io_handler.SetIsDone(true);
126*e81ba283SSiger Yang     } break;
127d853bd7aSPedro Tammela     case eIOHandlerNone:
128d853bd7aSPedro Tammela       if (IsQuitCommand(data)) {
1291728dec2SJonas Devlieghere         io_handler.SetIsDone(true);
1301728dec2SJonas Devlieghere         return;
1311728dec2SJonas Devlieghere       }
132d853bd7aSPedro Tammela       if (llvm::Error error = m_script_interpreter.GetLua().Run(data))
133d853bd7aSPedro Tammela         *io_handler.GetErrorStreamFileSP() << toString(std::move(error));
134d853bd7aSPedro Tammela       break;
13528613242SJonas Devlieghere     }
13628613242SJonas Devlieghere   }
13728613242SJonas Devlieghere 
13828613242SJonas Devlieghere private:
1394164be72SJonas Devlieghere   ScriptInterpreterLua &m_script_interpreter;
140d853bd7aSPedro Tammela   ActiveIOHandler m_active_io_handler;
141d853bd7aSPedro Tammela 
142d853bd7aSPedro Tammela   bool IsQuitCommand(llvm::StringRef cmd) { return cmd.rtrim() == "quit"; }
14328613242SJonas Devlieghere };
14428613242SJonas Devlieghere 
14567de8962SJonas Devlieghere ScriptInterpreterLua::ScriptInterpreterLua(Debugger &debugger)
1464164be72SJonas Devlieghere     : ScriptInterpreter(debugger, eScriptLanguageLua),
1474164be72SJonas Devlieghere       m_lua(std::make_unique<Lua>()) {}
14867de8962SJonas Devlieghere 
149fd2433e1SJonas Devlieghere ScriptInterpreterLua::~ScriptInterpreterLua() = default;
15067de8962SJonas Devlieghere 
15167de8962SJonas Devlieghere bool ScriptInterpreterLua::ExecuteOneLine(llvm::StringRef command,
15228613242SJonas Devlieghere                                           CommandReturnObject *result,
15328613242SJonas Devlieghere                                           const ExecuteScriptOptions &options) {
154ed8184b7SJonas Devlieghere   if (command.empty()) {
155ed8184b7SJonas Devlieghere     if (result)
156ed8184b7SJonas Devlieghere       result->AppendError("empty command passed to lua\n");
157ed8184b7SJonas Devlieghere     return false;
158ed8184b7SJonas Devlieghere   }
159ed8184b7SJonas Devlieghere 
160ed8184b7SJonas Devlieghere   llvm::Expected<std::unique_ptr<ScriptInterpreterIORedirect>>
161ed8184b7SJonas Devlieghere       io_redirect_or_error = ScriptInterpreterIORedirect::Create(
162ed8184b7SJonas Devlieghere           options.GetEnableIO(), m_debugger, result);
163ed8184b7SJonas Devlieghere   if (!io_redirect_or_error) {
164ed8184b7SJonas Devlieghere     if (result)
165ed8184b7SJonas Devlieghere       result->AppendErrorWithFormatv(
166ed8184b7SJonas Devlieghere           "failed to redirect I/O: {0}\n",
167ed8184b7SJonas Devlieghere           llvm::fmt_consume(io_redirect_or_error.takeError()));
168ed8184b7SJonas Devlieghere     else
169ed8184b7SJonas Devlieghere       llvm::consumeError(io_redirect_or_error.takeError());
170ed8184b7SJonas Devlieghere     return false;
171ed8184b7SJonas Devlieghere   }
172ed8184b7SJonas Devlieghere 
173ed8184b7SJonas Devlieghere   ScriptInterpreterIORedirect &io_redirect = **io_redirect_or_error;
174ed8184b7SJonas Devlieghere 
175ed8184b7SJonas Devlieghere   if (llvm::Error e =
176ed8184b7SJonas Devlieghere           m_lua->ChangeIO(io_redirect.GetOutputFile()->GetStream(),
177ed8184b7SJonas Devlieghere                           io_redirect.GetErrorFile()->GetStream())) {
178ed8184b7SJonas Devlieghere     result->AppendErrorWithFormatv("lua failed to redirect I/O: {0}\n",
179ed8184b7SJonas Devlieghere                                    llvm::toString(std::move(e)));
180ed8184b7SJonas Devlieghere     return false;
181ed8184b7SJonas Devlieghere   }
182ed8184b7SJonas Devlieghere 
1834164be72SJonas Devlieghere   if (llvm::Error e = m_lua->Run(command)) {
18428613242SJonas Devlieghere     result->AppendErrorWithFormatv(
18528613242SJonas Devlieghere         "lua failed attempting to evaluate '{0}': {1}\n", command,
18628613242SJonas Devlieghere         llvm::toString(std::move(e)));
18767de8962SJonas Devlieghere     return false;
18867de8962SJonas Devlieghere   }
189ed8184b7SJonas Devlieghere 
190ed8184b7SJonas Devlieghere   io_redirect.Flush();
19128613242SJonas Devlieghere   return true;
19228613242SJonas Devlieghere }
19367de8962SJonas Devlieghere 
19467de8962SJonas Devlieghere void ScriptInterpreterLua::ExecuteInterpreterLoop() {
1955c1c8443SJonas Devlieghere   LLDB_SCOPED_TIMER();
19628613242SJonas Devlieghere 
19728613242SJonas Devlieghere   // At the moment, the only time the debugger does not have an input file
19828613242SJonas Devlieghere   // handle is when this is called directly from lua, in which case it is
19928613242SJonas Devlieghere   // both dangerous and unnecessary (not to mention confusing) to try to embed
20028613242SJonas Devlieghere   // a running interpreter loop inside the already running lua interpreter
20128613242SJonas Devlieghere   // loop, so we won't do it.
2026e3faaebSJonas Devlieghere   if (!m_debugger.GetInputFile().IsValid())
20328613242SJonas Devlieghere     return;
20428613242SJonas Devlieghere 
2056e3faaebSJonas Devlieghere   IOHandlerSP io_handler_sp(new IOHandlerLuaInterpreter(m_debugger, *this));
2066e3faaebSJonas Devlieghere   m_debugger.RunIOHandlerAsync(io_handler_sp);
20767de8962SJonas Devlieghere }
20867de8962SJonas Devlieghere 
209572b9f46SJonas Devlieghere bool ScriptInterpreterLua::LoadScriptingModule(
210572b9f46SJonas Devlieghere     const char *filename, bool init_session, lldb_private::Status &error,
21100bb397bSJonas Devlieghere     StructuredData::ObjectSP *module_sp, FileSpec extra_search_dir) {
212572b9f46SJonas Devlieghere 
2131f80e515SJonas Devlieghere   FileSystem::Instance().Collect(filename);
214572b9f46SJonas Devlieghere   if (llvm::Error e = m_lua->LoadModule(filename)) {
215572b9f46SJonas Devlieghere     error.SetErrorStringWithFormatv("lua failed to import '{0}': {1}\n",
216572b9f46SJonas Devlieghere                                     filename, llvm::toString(std::move(e)));
217572b9f46SJonas Devlieghere     return false;
218572b9f46SJonas Devlieghere   }
219572b9f46SJonas Devlieghere   return true;
220572b9f46SJonas Devlieghere }
221572b9f46SJonas Devlieghere 
22267de8962SJonas Devlieghere void ScriptInterpreterLua::Initialize() {
22367de8962SJonas Devlieghere   static llvm::once_flag g_once_flag;
22467de8962SJonas Devlieghere 
22567de8962SJonas Devlieghere   llvm::call_once(g_once_flag, []() {
22667de8962SJonas Devlieghere     PluginManager::RegisterPlugin(GetPluginNameStatic(),
22767de8962SJonas Devlieghere                                   GetPluginDescriptionStatic(),
22867de8962SJonas Devlieghere                                   lldb::eScriptLanguageLua, CreateInstance);
22967de8962SJonas Devlieghere   });
23067de8962SJonas Devlieghere }
23167de8962SJonas Devlieghere 
23267de8962SJonas Devlieghere void ScriptInterpreterLua::Terminate() {}
23367de8962SJonas Devlieghere 
23445c971f7SJonas Devlieghere llvm::Error ScriptInterpreterLua::EnterSession(user_id_t debugger_id) {
23545c971f7SJonas Devlieghere   if (m_session_is_active)
23645c971f7SJonas Devlieghere     return llvm::Error::success();
23745c971f7SJonas Devlieghere 
23845c971f7SJonas Devlieghere   const char *fmt_str =
23945c971f7SJonas Devlieghere       "lldb.debugger = lldb.SBDebugger.FindDebuggerWithID({0}); "
24045c971f7SJonas Devlieghere       "lldb.target = lldb.debugger:GetSelectedTarget(); "
24145c971f7SJonas Devlieghere       "lldb.process = lldb.target:GetProcess(); "
24245c971f7SJonas Devlieghere       "lldb.thread = lldb.process:GetSelectedThread(); "
24345c971f7SJonas Devlieghere       "lldb.frame = lldb.thread:GetSelectedFrame()";
24445c971f7SJonas Devlieghere   return m_lua->Run(llvm::formatv(fmt_str, debugger_id).str());
24545c971f7SJonas Devlieghere }
24645c971f7SJonas Devlieghere 
24745c971f7SJonas Devlieghere llvm::Error ScriptInterpreterLua::LeaveSession() {
24845c971f7SJonas Devlieghere   if (!m_session_is_active)
24945c971f7SJonas Devlieghere     return llvm::Error::success();
25045c971f7SJonas Devlieghere 
25145c971f7SJonas Devlieghere   m_session_is_active = false;
25245c971f7SJonas Devlieghere 
25345c971f7SJonas Devlieghere   llvm::StringRef str = "lldb.debugger = nil; "
25445c971f7SJonas Devlieghere                         "lldb.target = nil; "
25545c971f7SJonas Devlieghere                         "lldb.process = nil; "
25645c971f7SJonas Devlieghere                         "lldb.thread = nil; "
25745c971f7SJonas Devlieghere                         "lldb.frame = nil";
25845c971f7SJonas Devlieghere   return m_lua->Run(str);
25945c971f7SJonas Devlieghere }
26045c971f7SJonas Devlieghere 
261a0d7406aSPedro Tammela bool ScriptInterpreterLua::BreakpointCallbackFunction(
262a0d7406aSPedro Tammela     void *baton, StoppointCallbackContext *context, user_id_t break_id,
263a0d7406aSPedro Tammela     user_id_t break_loc_id) {
264a0d7406aSPedro Tammela   assert(context);
265a0d7406aSPedro Tammela 
266a0d7406aSPedro Tammela   ExecutionContext exe_ctx(context->exe_ctx_ref);
267a0d7406aSPedro Tammela   Target *target = exe_ctx.GetTargetPtr();
268a0d7406aSPedro Tammela   if (target == nullptr)
269a0d7406aSPedro Tammela     return true;
270a0d7406aSPedro Tammela 
271a0d7406aSPedro Tammela   StackFrameSP stop_frame_sp(exe_ctx.GetFrameSP());
272a0d7406aSPedro Tammela   BreakpointSP breakpoint_sp = target->GetBreakpointByID(break_id);
273a0d7406aSPedro Tammela   BreakpointLocationSP bp_loc_sp(breakpoint_sp->FindLocationByID(break_loc_id));
274a0d7406aSPedro Tammela 
275a0d7406aSPedro Tammela   Debugger &debugger = target->GetDebugger();
276a0d7406aSPedro Tammela   ScriptInterpreterLua *lua_interpreter = static_cast<ScriptInterpreterLua *>(
277a0d7406aSPedro Tammela       debugger.GetScriptInterpreter(true, eScriptLanguageLua));
278a0d7406aSPedro Tammela   Lua &lua = lua_interpreter->GetLua();
279a0d7406aSPedro Tammela 
280532e4203SPedro Tammela   CommandDataLua *bp_option_data = static_cast<CommandDataLua *>(baton);
281532e4203SPedro Tammela   llvm::Expected<bool> BoolOrErr = lua.CallBreakpointCallback(
282532e4203SPedro Tammela       baton, stop_frame_sp, bp_loc_sp, bp_option_data->m_extra_args_sp);
283a0d7406aSPedro Tammela   if (llvm::Error E = BoolOrErr.takeError()) {
284a0d7406aSPedro Tammela     debugger.GetErrorStream() << toString(std::move(E));
285a0d7406aSPedro Tammela     return true;
286a0d7406aSPedro Tammela   }
287a0d7406aSPedro Tammela 
288a0d7406aSPedro Tammela   return *BoolOrErr;
289a0d7406aSPedro Tammela }
290a0d7406aSPedro Tammela 
291*e81ba283SSiger Yang bool ScriptInterpreterLua::WatchpointCallbackFunction(
292*e81ba283SSiger Yang     void *baton, StoppointCallbackContext *context, user_id_t watch_id) {
293*e81ba283SSiger Yang   assert(context);
294*e81ba283SSiger Yang 
295*e81ba283SSiger Yang   ExecutionContext exe_ctx(context->exe_ctx_ref);
296*e81ba283SSiger Yang   Target *target = exe_ctx.GetTargetPtr();
297*e81ba283SSiger Yang   if (target == nullptr)
298*e81ba283SSiger Yang     return true;
299*e81ba283SSiger Yang 
300*e81ba283SSiger Yang   StackFrameSP stop_frame_sp(exe_ctx.GetFrameSP());
301*e81ba283SSiger Yang   WatchpointSP wp_sp = target->GetWatchpointList().FindByID(watch_id);
302*e81ba283SSiger Yang 
303*e81ba283SSiger Yang   Debugger &debugger = target->GetDebugger();
304*e81ba283SSiger Yang   ScriptInterpreterLua *lua_interpreter = static_cast<ScriptInterpreterLua *>(
305*e81ba283SSiger Yang       debugger.GetScriptInterpreter(true, eScriptLanguageLua));
306*e81ba283SSiger Yang   Lua &lua = lua_interpreter->GetLua();
307*e81ba283SSiger Yang 
308*e81ba283SSiger Yang   llvm::Expected<bool> BoolOrErr =
309*e81ba283SSiger Yang       lua.CallWatchpointCallback(baton, stop_frame_sp, wp_sp);
310*e81ba283SSiger Yang   if (llvm::Error E = BoolOrErr.takeError()) {
311*e81ba283SSiger Yang     debugger.GetErrorStream() << toString(std::move(E));
312*e81ba283SSiger Yang     return true;
313*e81ba283SSiger Yang   }
314*e81ba283SSiger Yang 
315*e81ba283SSiger Yang   return *BoolOrErr;
316*e81ba283SSiger Yang }
317*e81ba283SSiger Yang 
318d853bd7aSPedro Tammela void ScriptInterpreterLua::CollectDataForBreakpointCommandCallback(
319cfb96d84SJim Ingham     std::vector<std::reference_wrapper<BreakpointOptions>> &bp_options_vec,
320d853bd7aSPedro Tammela     CommandReturnObject &result) {
321d853bd7aSPedro Tammela   IOHandlerSP io_handler_sp(
322d853bd7aSPedro Tammela       new IOHandlerLuaInterpreter(m_debugger, *this, eIOHandlerBreakpoint));
323d853bd7aSPedro Tammela   io_handler_sp->SetUserData(&bp_options_vec);
324d853bd7aSPedro Tammela   m_debugger.RunIOHandlerAsync(io_handler_sp);
325d853bd7aSPedro Tammela }
326d853bd7aSPedro Tammela 
327*e81ba283SSiger Yang void ScriptInterpreterLua::CollectDataForWatchpointCommandCallback(
328*e81ba283SSiger Yang     WatchpointOptions *wp_options, CommandReturnObject &result) {
329*e81ba283SSiger Yang   IOHandlerSP io_handler_sp(
330*e81ba283SSiger Yang       new IOHandlerLuaInterpreter(m_debugger, *this, eIOHandlerWatchpoint));
331*e81ba283SSiger Yang   io_handler_sp->SetUserData(wp_options);
332*e81ba283SSiger Yang   m_debugger.RunIOHandlerAsync(io_handler_sp);
333*e81ba283SSiger Yang }
334*e81ba283SSiger Yang 
335532e4203SPedro Tammela Status ScriptInterpreterLua::SetBreakpointCommandCallbackFunction(
336cfb96d84SJim Ingham     BreakpointOptions &bp_options, const char *function_name,
337532e4203SPedro Tammela     StructuredData::ObjectSP extra_args_sp) {
338532e4203SPedro Tammela   const char *fmt_str = "return {0}(frame, bp_loc, ...)";
339532e4203SPedro Tammela   std::string oneliner = llvm::formatv(fmt_str, function_name).str();
340532e4203SPedro Tammela   return RegisterBreakpointCallback(bp_options, oneliner.c_str(),
341532e4203SPedro Tammela                                     extra_args_sp);
342532e4203SPedro Tammela }
343532e4203SPedro Tammela 
344a0d7406aSPedro Tammela Status ScriptInterpreterLua::SetBreakpointCommandCallback(
345cfb96d84SJim Ingham     BreakpointOptions &bp_options, const char *command_body_text) {
346532e4203SPedro Tammela   return RegisterBreakpointCallback(bp_options, command_body_text, {});
347532e4203SPedro Tammela }
348532e4203SPedro Tammela 
349532e4203SPedro Tammela Status ScriptInterpreterLua::RegisterBreakpointCallback(
350cfb96d84SJim Ingham     BreakpointOptions &bp_options, const char *command_body_text,
351532e4203SPedro Tammela     StructuredData::ObjectSP extra_args_sp) {
352a0d7406aSPedro Tammela   Status error;
353532e4203SPedro Tammela   auto data_up = std::make_unique<CommandDataLua>(extra_args_sp);
354a0d7406aSPedro Tammela   error = m_lua->RegisterBreakpointCallback(data_up.get(), command_body_text);
355a0d7406aSPedro Tammela   if (error.Fail())
356a0d7406aSPedro Tammela     return error;
357a0d7406aSPedro Tammela   auto baton_sp =
358a0d7406aSPedro Tammela       std::make_shared<BreakpointOptions::CommandBaton>(std::move(data_up));
359cfb96d84SJim Ingham   bp_options.SetCallback(ScriptInterpreterLua::BreakpointCallbackFunction,
360a0d7406aSPedro Tammela                          baton_sp);
361a0d7406aSPedro Tammela   return error;
362a0d7406aSPedro Tammela }
363a0d7406aSPedro Tammela 
364*e81ba283SSiger Yang void ScriptInterpreterLua::SetWatchpointCommandCallback(
365*e81ba283SSiger Yang     WatchpointOptions *wp_options, const char *command_body_text) {
366*e81ba283SSiger Yang   RegisterWatchpointCallback(wp_options, command_body_text, {});
367*e81ba283SSiger Yang }
368*e81ba283SSiger Yang 
369*e81ba283SSiger Yang Status ScriptInterpreterLua::RegisterWatchpointCallback(
370*e81ba283SSiger Yang     WatchpointOptions *wp_options, const char *command_body_text,
371*e81ba283SSiger Yang     StructuredData::ObjectSP extra_args_sp) {
372*e81ba283SSiger Yang   Status error;
373*e81ba283SSiger Yang   auto data_up = std::make_unique<WatchpointOptions::CommandData>();
374*e81ba283SSiger Yang   error = m_lua->RegisterWatchpointCallback(data_up.get(), command_body_text);
375*e81ba283SSiger Yang   if (error.Fail())
376*e81ba283SSiger Yang     return error;
377*e81ba283SSiger Yang   auto baton_sp =
378*e81ba283SSiger Yang       std::make_shared<WatchpointOptions::CommandBaton>(std::move(data_up));
379*e81ba283SSiger Yang   wp_options->SetCallback(ScriptInterpreterLua::WatchpointCallbackFunction,
380*e81ba283SSiger Yang                           baton_sp);
381*e81ba283SSiger Yang   return error;
382*e81ba283SSiger Yang }
383*e81ba283SSiger Yang 
38467de8962SJonas Devlieghere lldb::ScriptInterpreterSP
38567de8962SJonas Devlieghere ScriptInterpreterLua::CreateInstance(Debugger &debugger) {
38667de8962SJonas Devlieghere   return std::make_shared<ScriptInterpreterLua>(debugger);
38767de8962SJonas Devlieghere }
38867de8962SJonas Devlieghere 
38967de8962SJonas Devlieghere lldb_private::ConstString ScriptInterpreterLua::GetPluginNameStatic() {
39067de8962SJonas Devlieghere   static ConstString g_name("script-lua");
39167de8962SJonas Devlieghere   return g_name;
39267de8962SJonas Devlieghere }
39367de8962SJonas Devlieghere 
39467de8962SJonas Devlieghere const char *ScriptInterpreterLua::GetPluginDescriptionStatic() {
39567de8962SJonas Devlieghere   return "Lua script interpreter";
39667de8962SJonas Devlieghere }
39767de8962SJonas Devlieghere 
39867de8962SJonas Devlieghere lldb_private::ConstString ScriptInterpreterLua::GetPluginName() {
39967de8962SJonas Devlieghere   return GetPluginNameStatic();
40067de8962SJonas Devlieghere }
40167de8962SJonas Devlieghere 
40267de8962SJonas Devlieghere uint32_t ScriptInterpreterLua::GetPluginVersion() { return 1; }
4034164be72SJonas Devlieghere 
4044164be72SJonas Devlieghere Lua &ScriptInterpreterLua::GetLua() { return *m_lua; }
405