128613242SJonas Devlieghere //===-- Lua.cpp -----------------------------------------------------------===//
228613242SJonas Devlieghere //
328613242SJonas Devlieghere // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
428613242SJonas Devlieghere // See https://llvm.org/LICENSE.txt for license information.
528613242SJonas Devlieghere // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
628613242SJonas Devlieghere //
728613242SJonas Devlieghere //===----------------------------------------------------------------------===//
828613242SJonas Devlieghere
928613242SJonas Devlieghere #include "Lua.h"
10a52af6d3SPavel Labath #include "SWIGLuaBridge.h"
11572b9f46SJonas Devlieghere #include "lldb/Host/FileSystem.h"
12572b9f46SJonas Devlieghere #include "lldb/Utility/FileSpec.h"
13a0d7406aSPedro Tammela #include "llvm/Support/Error.h"
1428613242SJonas Devlieghere #include "llvm/Support/FormatVariadic.h"
1528613242SJonas Devlieghere
1628613242SJonas Devlieghere using namespace lldb_private;
1745c971f7SJonas Devlieghere using namespace lldb;
1828613242SJonas Devlieghere
lldb_print(lua_State * L)19ca175710SPedro Tammela static int lldb_print(lua_State *L) {
20ca175710SPedro Tammela int n = lua_gettop(L);
21ca175710SPedro Tammela lua_getglobal(L, "io");
22ca175710SPedro Tammela lua_getfield(L, -1, "stdout");
23ca175710SPedro Tammela lua_getfield(L, -1, "write");
24ca175710SPedro Tammela for (int i = 1; i <= n; i++) {
25ca175710SPedro Tammela lua_pushvalue(L, -1); // write()
26ca175710SPedro Tammela lua_pushvalue(L, -3); // io.stdout
27ca175710SPedro Tammela luaL_tolstring(L, i, nullptr);
28ca175710SPedro Tammela lua_pushstring(L, i != n ? "\t" : "\n");
29ca175710SPedro Tammela lua_call(L, 3, 0);
30ca175710SPedro Tammela }
31ca175710SPedro Tammela return 0;
32ca175710SPedro Tammela }
33ca175710SPedro Tammela
Lua()34ca175710SPedro Tammela Lua::Lua() : m_lua_state(luaL_newstate()) {
35ca175710SPedro Tammela assert(m_lua_state);
36ca175710SPedro Tammela luaL_openlibs(m_lua_state);
37ca175710SPedro Tammela luaopen_lldb(m_lua_state);
38ca175710SPedro Tammela lua_pushcfunction(m_lua_state, lldb_print);
39ca175710SPedro Tammela lua_setglobal(m_lua_state, "print");
40ca175710SPedro Tammela }
41ca175710SPedro Tammela
~Lua()42ca175710SPedro Tammela Lua::~Lua() {
43ca175710SPedro Tammela assert(m_lua_state);
44ca175710SPedro Tammela lua_close(m_lua_state);
45ca175710SPedro Tammela }
46ca175710SPedro Tammela
Run(llvm::StringRef buffer)4728613242SJonas Devlieghere llvm::Error Lua::Run(llvm::StringRef buffer) {
4828613242SJonas Devlieghere int error =
4928613242SJonas Devlieghere luaL_loadbuffer(m_lua_state, buffer.data(), buffer.size(), "buffer") ||
5028613242SJonas Devlieghere lua_pcall(m_lua_state, 0, 0, 0);
514b9fa3b7SPedro Tammela if (error == LUA_OK)
5228613242SJonas Devlieghere return llvm::Error::success();
5328613242SJonas Devlieghere
5428613242SJonas Devlieghere llvm::Error e = llvm::make_error<llvm::StringError>(
5528613242SJonas Devlieghere llvm::formatv("{0}\n", lua_tostring(m_lua_state, -1)),
5628613242SJonas Devlieghere llvm::inconvertibleErrorCode());
5728613242SJonas Devlieghere // Pop error message from the stack.
5828613242SJonas Devlieghere lua_pop(m_lua_state, 1);
5928613242SJonas Devlieghere return e;
6028613242SJonas Devlieghere }
61572b9f46SJonas Devlieghere
RegisterBreakpointCallback(void * baton,const char * body)62a0d7406aSPedro Tammela llvm::Error Lua::RegisterBreakpointCallback(void *baton, const char *body) {
63a0d7406aSPedro Tammela lua_pushlightuserdata(m_lua_state, baton);
64a0d7406aSPedro Tammela const char *fmt_str = "return function(frame, bp_loc, ...) {0} end";
65a0d7406aSPedro Tammela std::string func_str = llvm::formatv(fmt_str, body).str();
66a0d7406aSPedro Tammela if (luaL_dostring(m_lua_state, func_str.c_str()) != LUA_OK) {
67a0d7406aSPedro Tammela llvm::Error e = llvm::make_error<llvm::StringError>(
68280ae107SPedro Tammela llvm::formatv("{0}", lua_tostring(m_lua_state, -1)),
69a0d7406aSPedro Tammela llvm::inconvertibleErrorCode());
70a0d7406aSPedro Tammela // Pop error message from the stack.
71a0d7406aSPedro Tammela lua_pop(m_lua_state, 2);
72a0d7406aSPedro Tammela return e;
73a0d7406aSPedro Tammela }
74a0d7406aSPedro Tammela lua_settable(m_lua_state, LUA_REGISTRYINDEX);
75a0d7406aSPedro Tammela return llvm::Error::success();
76a0d7406aSPedro Tammela }
77a0d7406aSPedro Tammela
78a0d7406aSPedro Tammela llvm::Expected<bool>
CallBreakpointCallback(void * baton,lldb::StackFrameSP stop_frame_sp,lldb::BreakpointLocationSP bp_loc_sp,StructuredData::ObjectSP extra_args_sp)79a0d7406aSPedro Tammela Lua::CallBreakpointCallback(void *baton, lldb::StackFrameSP stop_frame_sp,
80532e4203SPedro Tammela lldb::BreakpointLocationSP bp_loc_sp,
81532e4203SPedro Tammela StructuredData::ObjectSP extra_args_sp) {
82532e4203SPedro Tammela
83a0d7406aSPedro Tammela lua_pushlightuserdata(m_lua_state, baton);
84a0d7406aSPedro Tammela lua_gettable(m_lua_state, LUA_REGISTRYINDEX);
85*82de8df2SPavel Labath StructuredDataImpl extra_args_impl(std::move(extra_args_sp));
86a0d7406aSPedro Tammela return LLDBSwigLuaBreakpointCallbackFunction(m_lua_state, stop_frame_sp,
87532e4203SPedro Tammela bp_loc_sp, extra_args_impl);
88a0d7406aSPedro Tammela }
89a0d7406aSPedro Tammela
RegisterWatchpointCallback(void * baton,const char * body)90e81ba283SSiger Yang llvm::Error Lua::RegisterWatchpointCallback(void *baton, const char *body) {
91e81ba283SSiger Yang lua_pushlightuserdata(m_lua_state, baton);
92e81ba283SSiger Yang const char *fmt_str = "return function(frame, wp, ...) {0} end";
93e81ba283SSiger Yang std::string func_str = llvm::formatv(fmt_str, body).str();
94e81ba283SSiger Yang if (luaL_dostring(m_lua_state, func_str.c_str()) != LUA_OK) {
95e81ba283SSiger Yang llvm::Error e = llvm::make_error<llvm::StringError>(
96e81ba283SSiger Yang llvm::formatv("{0}", lua_tostring(m_lua_state, -1)),
97e81ba283SSiger Yang llvm::inconvertibleErrorCode());
98e81ba283SSiger Yang // Pop error message from the stack.
99e81ba283SSiger Yang lua_pop(m_lua_state, 2);
100e81ba283SSiger Yang return e;
101e81ba283SSiger Yang }
102e81ba283SSiger Yang lua_settable(m_lua_state, LUA_REGISTRYINDEX);
103e81ba283SSiger Yang return llvm::Error::success();
104e81ba283SSiger Yang }
105e81ba283SSiger Yang
106e81ba283SSiger Yang llvm::Expected<bool>
CallWatchpointCallback(void * baton,lldb::StackFrameSP stop_frame_sp,lldb::WatchpointSP wp_sp)107e81ba283SSiger Yang Lua::CallWatchpointCallback(void *baton, lldb::StackFrameSP stop_frame_sp,
108e81ba283SSiger Yang lldb::WatchpointSP wp_sp) {
109e81ba283SSiger Yang
110e81ba283SSiger Yang lua_pushlightuserdata(m_lua_state, baton);
111e81ba283SSiger Yang lua_gettable(m_lua_state, LUA_REGISTRYINDEX);
112e81ba283SSiger Yang return LLDBSwigLuaWatchpointCallbackFunction(m_lua_state, stop_frame_sp,
113e81ba283SSiger Yang wp_sp);
114e81ba283SSiger Yang }
115e81ba283SSiger Yang
CheckSyntax(llvm::StringRef buffer)116d853bd7aSPedro Tammela llvm::Error Lua::CheckSyntax(llvm::StringRef buffer) {
117d853bd7aSPedro Tammela int error =
118d853bd7aSPedro Tammela luaL_loadbuffer(m_lua_state, buffer.data(), buffer.size(), "buffer");
119d853bd7aSPedro Tammela if (error == LUA_OK) {
120d853bd7aSPedro Tammela // Pop buffer
121d853bd7aSPedro Tammela lua_pop(m_lua_state, 1);
122d853bd7aSPedro Tammela return llvm::Error::success();
123d853bd7aSPedro Tammela }
124d853bd7aSPedro Tammela
125d853bd7aSPedro Tammela llvm::Error e = llvm::make_error<llvm::StringError>(
126d853bd7aSPedro Tammela llvm::formatv("{0}\n", lua_tostring(m_lua_state, -1)),
127d853bd7aSPedro Tammela llvm::inconvertibleErrorCode());
128d853bd7aSPedro Tammela // Pop error message from the stack.
129d853bd7aSPedro Tammela lua_pop(m_lua_state, 1);
130d853bd7aSPedro Tammela return e;
131d853bd7aSPedro Tammela }
132d853bd7aSPedro Tammela
LoadModule(llvm::StringRef filename)133572b9f46SJonas Devlieghere llvm::Error Lua::LoadModule(llvm::StringRef filename) {
134572b9f46SJonas Devlieghere FileSpec file(filename);
135572b9f46SJonas Devlieghere if (!FileSystem::Instance().Exists(file)) {
136572b9f46SJonas Devlieghere return llvm::make_error<llvm::StringError>("invalid path",
137572b9f46SJonas Devlieghere llvm::inconvertibleErrorCode());
138572b9f46SJonas Devlieghere }
139572b9f46SJonas Devlieghere
140572b9f46SJonas Devlieghere ConstString module_extension = file.GetFileNameExtension();
141572b9f46SJonas Devlieghere if (module_extension != ".lua") {
142572b9f46SJonas Devlieghere return llvm::make_error<llvm::StringError>("invalid extension",
143572b9f46SJonas Devlieghere llvm::inconvertibleErrorCode());
144572b9f46SJonas Devlieghere }
145572b9f46SJonas Devlieghere
146572b9f46SJonas Devlieghere int error = luaL_loadfile(m_lua_state, filename.data()) ||
147572b9f46SJonas Devlieghere lua_pcall(m_lua_state, 0, 1, 0);
1484b9fa3b7SPedro Tammela if (error != LUA_OK) {
149572b9f46SJonas Devlieghere llvm::Error e = llvm::make_error<llvm::StringError>(
150572b9f46SJonas Devlieghere llvm::formatv("{0}\n", lua_tostring(m_lua_state, -1)),
151572b9f46SJonas Devlieghere llvm::inconvertibleErrorCode());
152572b9f46SJonas Devlieghere // Pop error message from the stack.
153572b9f46SJonas Devlieghere lua_pop(m_lua_state, 1);
154572b9f46SJonas Devlieghere return e;
155572b9f46SJonas Devlieghere }
156572b9f46SJonas Devlieghere
157572b9f46SJonas Devlieghere ConstString module_name = file.GetFileNameStrippingExtension();
158572b9f46SJonas Devlieghere lua_setglobal(m_lua_state, module_name.GetCString());
159572b9f46SJonas Devlieghere return llvm::Error::success();
160572b9f46SJonas Devlieghere }
161fa1b4a96SJonas Devlieghere
ChangeIO(FILE * out,FILE * err)162fa1b4a96SJonas Devlieghere llvm::Error Lua::ChangeIO(FILE *out, FILE *err) {
163fa1b4a96SJonas Devlieghere assert(out != nullptr);
164fa1b4a96SJonas Devlieghere assert(err != nullptr);
165fa1b4a96SJonas Devlieghere
166fa1b4a96SJonas Devlieghere lua_getglobal(m_lua_state, "io");
167fa1b4a96SJonas Devlieghere
168fa1b4a96SJonas Devlieghere lua_getfield(m_lua_state, -1, "stdout");
169fa1b4a96SJonas Devlieghere if (luaL_Stream *s = static_cast<luaL_Stream *>(
170fa1b4a96SJonas Devlieghere luaL_testudata(m_lua_state, -1, LUA_FILEHANDLE))) {
171fa1b4a96SJonas Devlieghere s->f = out;
172fa1b4a96SJonas Devlieghere lua_pop(m_lua_state, 1);
173fa1b4a96SJonas Devlieghere } else {
174fa1b4a96SJonas Devlieghere lua_pop(m_lua_state, 2);
175fa1b4a96SJonas Devlieghere return llvm::make_error<llvm::StringError>("could not get stdout",
176fa1b4a96SJonas Devlieghere llvm::inconvertibleErrorCode());
177fa1b4a96SJonas Devlieghere }
178fa1b4a96SJonas Devlieghere
179be494adbSJonas Devlieghere lua_getfield(m_lua_state, -1, "stderr");
180fa1b4a96SJonas Devlieghere if (luaL_Stream *s = static_cast<luaL_Stream *>(
181fa1b4a96SJonas Devlieghere luaL_testudata(m_lua_state, -1, LUA_FILEHANDLE))) {
182fa1b4a96SJonas Devlieghere s->f = out;
183fa1b4a96SJonas Devlieghere lua_pop(m_lua_state, 1);
184fa1b4a96SJonas Devlieghere } else {
185fa1b4a96SJonas Devlieghere lua_pop(m_lua_state, 2);
186fa1b4a96SJonas Devlieghere return llvm::make_error<llvm::StringError>("could not get stderr",
187fa1b4a96SJonas Devlieghere llvm::inconvertibleErrorCode());
188fa1b4a96SJonas Devlieghere }
189fa1b4a96SJonas Devlieghere
190fa1b4a96SJonas Devlieghere lua_pop(m_lua_state, 1);
191fa1b4a96SJonas Devlieghere return llvm::Error::success();
192fa1b4a96SJonas Devlieghere }
193