xref: /openbsd-src/gnu/llvm/lldb/source/Plugins/ScriptInterpreter/Lua/Lua.cpp (revision f6aab3d83b51b91c24247ad2c2573574de475a82)
1061da546Spatrick //===-- Lua.cpp -----------------------------------------------------------===//
2061da546Spatrick //
3061da546Spatrick // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4061da546Spatrick // See https://llvm.org/LICENSE.txt for license information.
5061da546Spatrick // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6061da546Spatrick //
7061da546Spatrick //===----------------------------------------------------------------------===//
8061da546Spatrick 
9061da546Spatrick #include "Lua.h"
10*f6aab3d8Srobert #include "SWIGLuaBridge.h"
11061da546Spatrick #include "lldb/Host/FileSystem.h"
12061da546Spatrick #include "lldb/Utility/FileSpec.h"
13be691f3bSpatrick #include "llvm/Support/Error.h"
14061da546Spatrick #include "llvm/Support/FormatVariadic.h"
15061da546Spatrick 
16061da546Spatrick using namespace lldb_private;
17061da546Spatrick using namespace lldb;
18061da546Spatrick 
lldb_print(lua_State * L)19be691f3bSpatrick static int lldb_print(lua_State *L) {
20be691f3bSpatrick   int n = lua_gettop(L);
21be691f3bSpatrick   lua_getglobal(L, "io");
22be691f3bSpatrick   lua_getfield(L, -1, "stdout");
23be691f3bSpatrick   lua_getfield(L, -1, "write");
24be691f3bSpatrick   for (int i = 1; i <= n; i++) {
25be691f3bSpatrick     lua_pushvalue(L, -1); // write()
26be691f3bSpatrick     lua_pushvalue(L, -3); // io.stdout
27be691f3bSpatrick     luaL_tolstring(L, i, nullptr);
28be691f3bSpatrick     lua_pushstring(L, i != n ? "\t" : "\n");
29be691f3bSpatrick     lua_call(L, 3, 0);
30be691f3bSpatrick   }
31be691f3bSpatrick   return 0;
32be691f3bSpatrick }
33be691f3bSpatrick 
Lua()34be691f3bSpatrick Lua::Lua() : m_lua_state(luaL_newstate()) {
35be691f3bSpatrick   assert(m_lua_state);
36be691f3bSpatrick   luaL_openlibs(m_lua_state);
37be691f3bSpatrick   luaopen_lldb(m_lua_state);
38be691f3bSpatrick   lua_pushcfunction(m_lua_state, lldb_print);
39be691f3bSpatrick   lua_setglobal(m_lua_state, "print");
40be691f3bSpatrick }
41be691f3bSpatrick 
~Lua()42be691f3bSpatrick Lua::~Lua() {
43be691f3bSpatrick   assert(m_lua_state);
44be691f3bSpatrick   lua_close(m_lua_state);
45be691f3bSpatrick }
46be691f3bSpatrick 
Run(llvm::StringRef buffer)47061da546Spatrick llvm::Error Lua::Run(llvm::StringRef buffer) {
48061da546Spatrick   int error =
49061da546Spatrick       luaL_loadbuffer(m_lua_state, buffer.data(), buffer.size(), "buffer") ||
50061da546Spatrick       lua_pcall(m_lua_state, 0, 0, 0);
51be691f3bSpatrick   if (error == LUA_OK)
52061da546Spatrick     return llvm::Error::success();
53061da546Spatrick 
54061da546Spatrick   llvm::Error e = llvm::make_error<llvm::StringError>(
55061da546Spatrick       llvm::formatv("{0}\n", lua_tostring(m_lua_state, -1)),
56061da546Spatrick       llvm::inconvertibleErrorCode());
57061da546Spatrick   // Pop error message from the stack.
58061da546Spatrick   lua_pop(m_lua_state, 1);
59061da546Spatrick   return e;
60061da546Spatrick }
61061da546Spatrick 
RegisterBreakpointCallback(void * baton,const char * body)62be691f3bSpatrick llvm::Error Lua::RegisterBreakpointCallback(void *baton, const char *body) {
63be691f3bSpatrick   lua_pushlightuserdata(m_lua_state, baton);
64be691f3bSpatrick   const char *fmt_str = "return function(frame, bp_loc, ...) {0} end";
65be691f3bSpatrick   std::string func_str = llvm::formatv(fmt_str, body).str();
66be691f3bSpatrick   if (luaL_dostring(m_lua_state, func_str.c_str()) != LUA_OK) {
67be691f3bSpatrick     llvm::Error e = llvm::make_error<llvm::StringError>(
68be691f3bSpatrick         llvm::formatv("{0}", lua_tostring(m_lua_state, -1)),
69be691f3bSpatrick         llvm::inconvertibleErrorCode());
70be691f3bSpatrick     // Pop error message from the stack.
71be691f3bSpatrick     lua_pop(m_lua_state, 2);
72be691f3bSpatrick     return e;
73be691f3bSpatrick   }
74be691f3bSpatrick   lua_settable(m_lua_state, LUA_REGISTRYINDEX);
75be691f3bSpatrick   return llvm::Error::success();
76be691f3bSpatrick }
77be691f3bSpatrick 
78be691f3bSpatrick llvm::Expected<bool>
CallBreakpointCallback(void * baton,lldb::StackFrameSP stop_frame_sp,lldb::BreakpointLocationSP bp_loc_sp,StructuredData::ObjectSP extra_args_sp)79be691f3bSpatrick Lua::CallBreakpointCallback(void *baton, lldb::StackFrameSP stop_frame_sp,
80be691f3bSpatrick                             lldb::BreakpointLocationSP bp_loc_sp,
81be691f3bSpatrick                             StructuredData::ObjectSP extra_args_sp) {
82be691f3bSpatrick 
83be691f3bSpatrick   lua_pushlightuserdata(m_lua_state, baton);
84be691f3bSpatrick   lua_gettable(m_lua_state, LUA_REGISTRYINDEX);
85*f6aab3d8Srobert   StructuredDataImpl extra_args_impl(std::move(extra_args_sp));
86be691f3bSpatrick   return LLDBSwigLuaBreakpointCallbackFunction(m_lua_state, stop_frame_sp,
87be691f3bSpatrick                                                bp_loc_sp, extra_args_impl);
88be691f3bSpatrick }
89be691f3bSpatrick 
RegisterWatchpointCallback(void * baton,const char * body)90be691f3bSpatrick llvm::Error Lua::RegisterWatchpointCallback(void *baton, const char *body) {
91be691f3bSpatrick   lua_pushlightuserdata(m_lua_state, baton);
92be691f3bSpatrick   const char *fmt_str = "return function(frame, wp, ...) {0} end";
93be691f3bSpatrick   std::string func_str = llvm::formatv(fmt_str, body).str();
94be691f3bSpatrick   if (luaL_dostring(m_lua_state, func_str.c_str()) != LUA_OK) {
95be691f3bSpatrick     llvm::Error e = llvm::make_error<llvm::StringError>(
96be691f3bSpatrick         llvm::formatv("{0}", lua_tostring(m_lua_state, -1)),
97be691f3bSpatrick         llvm::inconvertibleErrorCode());
98be691f3bSpatrick     // Pop error message from the stack.
99be691f3bSpatrick     lua_pop(m_lua_state, 2);
100be691f3bSpatrick     return e;
101be691f3bSpatrick   }
102be691f3bSpatrick   lua_settable(m_lua_state, LUA_REGISTRYINDEX);
103be691f3bSpatrick   return llvm::Error::success();
104be691f3bSpatrick }
105be691f3bSpatrick 
106be691f3bSpatrick llvm::Expected<bool>
CallWatchpointCallback(void * baton,lldb::StackFrameSP stop_frame_sp,lldb::WatchpointSP wp_sp)107be691f3bSpatrick Lua::CallWatchpointCallback(void *baton, lldb::StackFrameSP stop_frame_sp,
108be691f3bSpatrick                             lldb::WatchpointSP wp_sp) {
109be691f3bSpatrick 
110be691f3bSpatrick   lua_pushlightuserdata(m_lua_state, baton);
111be691f3bSpatrick   lua_gettable(m_lua_state, LUA_REGISTRYINDEX);
112be691f3bSpatrick   return LLDBSwigLuaWatchpointCallbackFunction(m_lua_state, stop_frame_sp,
113be691f3bSpatrick                                                wp_sp);
114be691f3bSpatrick }
115be691f3bSpatrick 
CheckSyntax(llvm::StringRef buffer)116be691f3bSpatrick llvm::Error Lua::CheckSyntax(llvm::StringRef buffer) {
117be691f3bSpatrick   int error =
118be691f3bSpatrick       luaL_loadbuffer(m_lua_state, buffer.data(), buffer.size(), "buffer");
119be691f3bSpatrick   if (error == LUA_OK) {
120be691f3bSpatrick     // Pop buffer
121be691f3bSpatrick     lua_pop(m_lua_state, 1);
122be691f3bSpatrick     return llvm::Error::success();
123be691f3bSpatrick   }
124be691f3bSpatrick 
125be691f3bSpatrick   llvm::Error e = llvm::make_error<llvm::StringError>(
126be691f3bSpatrick       llvm::formatv("{0}\n", lua_tostring(m_lua_state, -1)),
127be691f3bSpatrick       llvm::inconvertibleErrorCode());
128be691f3bSpatrick   // Pop error message from the stack.
129be691f3bSpatrick   lua_pop(m_lua_state, 1);
130be691f3bSpatrick   return e;
131be691f3bSpatrick }
132be691f3bSpatrick 
LoadModule(llvm::StringRef filename)133061da546Spatrick llvm::Error Lua::LoadModule(llvm::StringRef filename) {
134061da546Spatrick   FileSpec file(filename);
135061da546Spatrick   if (!FileSystem::Instance().Exists(file)) {
136061da546Spatrick     return llvm::make_error<llvm::StringError>("invalid path",
137061da546Spatrick                                                llvm::inconvertibleErrorCode());
138061da546Spatrick   }
139061da546Spatrick 
140061da546Spatrick   ConstString module_extension = file.GetFileNameExtension();
141061da546Spatrick   if (module_extension != ".lua") {
142061da546Spatrick     return llvm::make_error<llvm::StringError>("invalid extension",
143061da546Spatrick                                                llvm::inconvertibleErrorCode());
144061da546Spatrick   }
145061da546Spatrick 
146061da546Spatrick   int error = luaL_loadfile(m_lua_state, filename.data()) ||
147061da546Spatrick               lua_pcall(m_lua_state, 0, 1, 0);
148be691f3bSpatrick   if (error != LUA_OK) {
149061da546Spatrick     llvm::Error e = llvm::make_error<llvm::StringError>(
150061da546Spatrick         llvm::formatv("{0}\n", lua_tostring(m_lua_state, -1)),
151061da546Spatrick         llvm::inconvertibleErrorCode());
152061da546Spatrick     // Pop error message from the stack.
153061da546Spatrick     lua_pop(m_lua_state, 1);
154061da546Spatrick     return e;
155061da546Spatrick   }
156061da546Spatrick 
157061da546Spatrick   ConstString module_name = file.GetFileNameStrippingExtension();
158061da546Spatrick   lua_setglobal(m_lua_state, module_name.GetCString());
159061da546Spatrick   return llvm::Error::success();
160061da546Spatrick }
161dda28197Spatrick 
ChangeIO(FILE * out,FILE * err)162dda28197Spatrick llvm::Error Lua::ChangeIO(FILE *out, FILE *err) {
163dda28197Spatrick   assert(out != nullptr);
164dda28197Spatrick   assert(err != nullptr);
165dda28197Spatrick 
166dda28197Spatrick   lua_getglobal(m_lua_state, "io");
167dda28197Spatrick 
168dda28197Spatrick   lua_getfield(m_lua_state, -1, "stdout");
169dda28197Spatrick   if (luaL_Stream *s = static_cast<luaL_Stream *>(
170dda28197Spatrick           luaL_testudata(m_lua_state, -1, LUA_FILEHANDLE))) {
171dda28197Spatrick     s->f = out;
172dda28197Spatrick     lua_pop(m_lua_state, 1);
173dda28197Spatrick   } else {
174dda28197Spatrick     lua_pop(m_lua_state, 2);
175dda28197Spatrick     return llvm::make_error<llvm::StringError>("could not get stdout",
176dda28197Spatrick                                                llvm::inconvertibleErrorCode());
177dda28197Spatrick   }
178dda28197Spatrick 
179dda28197Spatrick   lua_getfield(m_lua_state, -1, "stderr");
180dda28197Spatrick   if (luaL_Stream *s = static_cast<luaL_Stream *>(
181dda28197Spatrick           luaL_testudata(m_lua_state, -1, LUA_FILEHANDLE))) {
182dda28197Spatrick     s->f = out;
183dda28197Spatrick     lua_pop(m_lua_state, 1);
184dda28197Spatrick   } else {
185dda28197Spatrick     lua_pop(m_lua_state, 2);
186dda28197Spatrick     return llvm::make_error<llvm::StringError>("could not get stderr",
187dda28197Spatrick                                                llvm::inconvertibleErrorCode());
188dda28197Spatrick   }
189dda28197Spatrick 
190dda28197Spatrick   lua_pop(m_lua_state, 1);
191dda28197Spatrick   return llvm::Error::success();
192dda28197Spatrick }
193