//===-- Lua.cpp -----------------------------------------------------------===// // // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. // See https://llvm.org/LICENSE.txt for license information. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception // //===----------------------------------------------------------------------===// #include "Lua.h" #include "lldb/Host/FileSystem.h" #include "lldb/Utility/FileSpec.h" #include "llvm/Support/Error.h" #include "llvm/Support/FormatVariadic.h" using namespace lldb_private; using namespace lldb; #pragma clang diagnostic push #pragma clang diagnostic ignored "-Wreturn-type-c-linkage" // Disable warning C4190: 'LLDBSwigPythonBreakpointCallbackFunction' has // C-linkage specified, but returns UDT 'llvm::Expected' which is // incompatible with C #if _MSC_VER #pragma warning (push) #pragma warning (disable : 4190) #endif extern "C" llvm::Expected LLDBSwigLuaBreakpointCallbackFunction(lua_State *L, lldb::StackFrameSP stop_frame_sp, lldb::BreakpointLocationSP bp_loc_sp); #if _MSC_VER #pragma warning (pop) #endif #pragma clang diagnostic pop static int lldb_print(lua_State *L) { int n = lua_gettop(L); lua_getglobal(L, "io"); lua_getfield(L, -1, "stdout"); lua_getfield(L, -1, "write"); for (int i = 1; i <= n; i++) { lua_pushvalue(L, -1); // write() lua_pushvalue(L, -3); // io.stdout luaL_tolstring(L, i, nullptr); lua_pushstring(L, i != n ? "\t" : "\n"); lua_call(L, 3, 0); } return 0; } Lua::Lua() : m_lua_state(luaL_newstate()) { assert(m_lua_state); luaL_openlibs(m_lua_state); luaopen_lldb(m_lua_state); lua_pushcfunction(m_lua_state, lldb_print); lua_setglobal(m_lua_state, "print"); } Lua::~Lua() { assert(m_lua_state); lua_close(m_lua_state); } llvm::Error Lua::Run(llvm::StringRef buffer) { int error = luaL_loadbuffer(m_lua_state, buffer.data(), buffer.size(), "buffer") || lua_pcall(m_lua_state, 0, 0, 0); if (error == LUA_OK) return llvm::Error::success(); llvm::Error e = llvm::make_error( llvm::formatv("{0}\n", lua_tostring(m_lua_state, -1)), llvm::inconvertibleErrorCode()); // Pop error message from the stack. lua_pop(m_lua_state, 1); return e; } llvm::Error Lua::RegisterBreakpointCallback(void *baton, const char *body) { lua_pushlightuserdata(m_lua_state, baton); const char *fmt_str = "return function(frame, bp_loc, ...) {0} end"; std::string func_str = llvm::formatv(fmt_str, body).str(); if (luaL_dostring(m_lua_state, func_str.c_str()) != LUA_OK) { llvm::Error e = llvm::make_error( llvm::formatv("{0}", lua_tostring(m_lua_state, -1)), llvm::inconvertibleErrorCode()); // Pop error message from the stack. lua_pop(m_lua_state, 2); return e; } lua_settable(m_lua_state, LUA_REGISTRYINDEX); return llvm::Error::success(); } llvm::Expected Lua::CallBreakpointCallback(void *baton, lldb::StackFrameSP stop_frame_sp, lldb::BreakpointLocationSP bp_loc_sp) { lua_pushlightuserdata(m_lua_state, baton); lua_gettable(m_lua_state, LUA_REGISTRYINDEX); return LLDBSwigLuaBreakpointCallbackFunction(m_lua_state, stop_frame_sp, bp_loc_sp); } llvm::Error Lua::LoadModule(llvm::StringRef filename) { FileSpec file(filename); if (!FileSystem::Instance().Exists(file)) { return llvm::make_error("invalid path", llvm::inconvertibleErrorCode()); } ConstString module_extension = file.GetFileNameExtension(); if (module_extension != ".lua") { return llvm::make_error("invalid extension", llvm::inconvertibleErrorCode()); } int error = luaL_loadfile(m_lua_state, filename.data()) || lua_pcall(m_lua_state, 0, 1, 0); if (error != LUA_OK) { llvm::Error e = llvm::make_error( llvm::formatv("{0}\n", lua_tostring(m_lua_state, -1)), llvm::inconvertibleErrorCode()); // Pop error message from the stack. lua_pop(m_lua_state, 1); return e; } ConstString module_name = file.GetFileNameStrippingExtension(); lua_setglobal(m_lua_state, module_name.GetCString()); return llvm::Error::success(); } llvm::Error Lua::ChangeIO(FILE *out, FILE *err) { assert(out != nullptr); assert(err != nullptr); lua_getglobal(m_lua_state, "io"); lua_getfield(m_lua_state, -1, "stdout"); if (luaL_Stream *s = static_cast( luaL_testudata(m_lua_state, -1, LUA_FILEHANDLE))) { s->f = out; lua_pop(m_lua_state, 1); } else { lua_pop(m_lua_state, 2); return llvm::make_error("could not get stdout", llvm::inconvertibleErrorCode()); } lua_getfield(m_lua_state, -1, "stderr"); if (luaL_Stream *s = static_cast( luaL_testudata(m_lua_state, -1, LUA_FILEHANDLE))) { s->f = out; lua_pop(m_lua_state, 1); } else { lua_pop(m_lua_state, 2); return llvm::make_error("could not get stderr", llvm::inconvertibleErrorCode()); } lua_pop(m_lua_state, 1); return llvm::Error::success(); }