luau/CLI/Repl.cpp

956 lines
25 KiB
C++
Raw Normal View History

// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details
2022-02-11 18:43:14 +00:00
#include "Repl.h"
#include "Luau/Common.h"
#include "lua.h"
#include "lualib.h"
2022-10-13 23:59:53 +01:00
#include "Luau/CodeGen.h"
#include "Luau/Compiler.h"
#include "Luau/Parser.h"
2023-04-21 22:41:03 +01:00
#include "Luau/TimeTrace.h"
2022-07-21 21:36:41 +01:00
#include "Coverage.h"
#include "FileUtils.h"
2022-07-21 21:36:41 +01:00
#include "Flags.h"
#include "Profiler.h"
#include "Require.h"
2022-02-03 23:09:37 +00:00
#include "isocline.h"
#include <memory>
2024-11-15 19:37:29 +00:00
#include <string>
#include <string_view>
2021-11-12 02:12:39 +00:00
#ifdef _WIN32
#include <io.h>
#include <fcntl.h>
2022-08-04 22:27:28 +01:00
#define WIN32_LEAN_AND_MEAN
#include <windows.h>
2021-11-12 02:12:39 +00:00
#endif
2023-05-12 13:15:01 +01:00
#ifdef __linux__
#include <unistd.h>
#endif
2022-07-08 02:05:31 +01:00
#ifdef CALLGRIND
#include <valgrind/callgrind.h>
#endif
2022-05-06 00:52:48 +01:00
#include <locale.h>
2022-08-04 22:27:28 +01:00
#include <signal.h>
2022-05-06 00:52:48 +01:00
2022-01-14 16:06:31 +00:00
LUAU_FASTFLAG(DebugLuauTimeTracing)
2022-02-18 00:41:20 +00:00
constexpr int MaxTraversalLimit = 50;
2022-10-13 23:59:53 +01:00
static bool codegen = false;
2024-02-23 18:40:00 +00:00
static int program_argc = 0;
char** program_argv = nullptr;
2022-10-13 23:59:53 +01:00
2022-08-04 22:27:28 +01:00
// Ctrl-C handling
static void sigintCallback(lua_State* L, int gc)
{
if (gc >= 0)
return;
lua_callbacks(L)->interrupt = NULL;
lua_rawcheckstack(L, 1); // reserve space for error string
luaL_error(L, "Execution interrupted");
}
static lua_State* replState = NULL;
#ifdef _WIN32
BOOL WINAPI sigintHandler(DWORD signal)
{
if (signal == CTRL_C_EVENT && replState)
lua_callbacks(replState)->interrupt = &sigintCallback;
return TRUE;
}
#else
static void sigintHandler(int signum)
{
if (signum == SIGINT && replState)
lua_callbacks(replState)->interrupt = &sigintCallback;
}
#endif
2022-01-21 16:23:02 +00:00
struct GlobalOptions
{
int optimizationLevel = 1;
2022-02-11 18:43:14 +00:00
int debugLevel = 1;
2022-01-21 16:23:02 +00:00
} globalOptions;
static Luau::CompileOptions copts()
{
Luau::CompileOptions result = {};
2022-01-21 16:23:02 +00:00
result.optimizationLevel = globalOptions.optimizationLevel;
2022-02-11 18:43:14 +00:00
result.debugLevel = globalOptions.debugLevel;
2024-04-25 21:57:23 +01:00
result.typeInfoLevel = 1;
result.coverageLevel = coverageActive() ? 2 : 0;
return result;
}
static int lua_loadstring(lua_State* L)
{
size_t l = 0;
const char* s = luaL_checklstring(L, 1, &l);
const char* chunkname = luaL_optstring(L, 2, s);
lua_setsafeenv(L, LUA_ENVIRONINDEX, false);
std::string bytecode = Luau::compile(std::string(s, l), copts());
2021-11-18 22:21:07 +00:00
if (luau_load(L, chunkname, bytecode.data(), bytecode.size(), 0) == 0)
return 1;
lua_pushnil(L);
2022-08-04 22:27:28 +01:00
lua_insert(L, -2); // put before error message
return 2; // return nil plus error message
}
static int finishrequire(lua_State* L)
{
if (lua_isstring(L, -1))
lua_error(L);
return 1;
}
2024-11-15 19:37:29 +00:00
struct RuntimeRequireContext : public RequireResolver::RequireContext
{
// In the context of the REPL, source is the calling context's chunkname.
//
// These chunknames have certain prefixes that indicate context. These
// are used when displaying debug information (see luaO_chunkid).
//
// Generally, the '@' prefix is used for filepaths, and the '=' prefix is
// used for custom chunknames, such as =stdin.
explicit RuntimeRequireContext(std::string source)
: source(std::move(source))
{
}
std::string getPath() override
{
return source.substr(1);
}
bool isRequireAllowed() override
{
return isStdin() || (!source.empty() && source[0] == '@');
}
bool isStdin() override
{
return source == "=stdin";
}
std::string createNewIdentifer(const std::string& path) override
{
return "@" + path;
}
private:
std::string source;
};
struct RuntimeCacheManager : public RequireResolver::CacheManager
{
explicit RuntimeCacheManager(lua_State* L)
: L(L)
{
}
bool isCached(const std::string& path) override
{
luaL_findtable(L, LUA_REGISTRYINDEX, "_MODULES", 1);
lua_getfield(L, -1, path.c_str());
bool cached = !lua_isnil(L, -1);
lua_pop(L, 2);
if (cached)
cacheKey = path;
return cached;
}
std::string cacheKey;
private:
lua_State* L;
};
struct RuntimeErrorHandler : RequireResolver::ErrorHandler
{
explicit RuntimeErrorHandler(lua_State* L)
: L(L)
{
}
void reportError(const std::string message) override
{
luaL_errorL(L, "%s", message.c_str());
}
private:
lua_State* L;
};
static int lua_require(lua_State* L)
{
2024-04-12 11:44:40 +01:00
std::string name = luaL_checkstring(L, 1);
2024-11-15 19:37:29 +00:00
RequireResolver::ResolvedRequire resolvedRequire;
{
lua_Debug ar;
lua_getinfo(L, 1, "s", &ar);
RuntimeRequireContext requireContext{ar.source};
RuntimeCacheManager cacheManager{L};
RuntimeErrorHandler errorHandler{L};
RequireResolver resolver(std::move(name), requireContext, cacheManager, errorHandler);
resolvedRequire = resolver.resolveRequire(
[L, &cacheKey = cacheManager.cacheKey](const RequireResolver::ModuleStatus status)
{
lua_getfield(L, LUA_REGISTRYINDEX, "_MODULES");
if (status == RequireResolver::ModuleStatus::Cached)
lua_getfield(L, -1, cacheKey.c_str());
}
);
}
2022-07-21 21:36:41 +01:00
2024-04-12 11:44:40 +01:00
if (resolvedRequire.status == RequireResolver::ModuleStatus::Cached)
return finishrequire(L);
2024-04-12 11:44:40 +01:00
// module needs to run in a new thread, isolated from the rest
// note: we create ML on main thread so that it doesn't inherit environment of L
lua_State* GL = lua_mainthread(L);
lua_State* ML = lua_newthread(GL);
lua_xmove(GL, L, 1);
2024-04-12 11:44:40 +01:00
// new thread needs to have the globals sandboxed
luaL_sandboxthread(ML);
2024-04-12 11:44:40 +01:00
// now we can compile & run module on the new thread
std::string bytecode = Luau::compile(resolvedRequire.sourceCode, copts());
2024-11-15 19:37:29 +00:00
if (luau_load(ML, resolvedRequire.identifier.c_str(), bytecode.data(), bytecode.size(), 0) == 0)
{
2024-04-12 11:44:40 +01:00
if (codegen)
2024-05-10 17:17:09 +01:00
{
Luau::CodeGen::CompilationOptions nativeOptions;
Luau::CodeGen::compile(ML, -1, nativeOptions);
}
2024-04-12 11:44:40 +01:00
if (coverageActive())
coverageTrack(ML, -1);
2024-04-12 11:44:40 +01:00
int status = lua_resume(ML, L, 0);
if (status == 0)
{
2024-04-12 11:44:40 +01:00
if (lua_gettop(ML) == 0)
lua_pushstring(ML, "module must return a value");
else if (!lua_istable(ML, -1) && !lua_isfunction(ML, -1))
lua_pushstring(ML, "module must return a table or function");
}
2024-04-12 11:44:40 +01:00
else if (status == LUA_YIELD)
{
2024-04-12 11:44:40 +01:00
lua_pushstring(ML, "module can not yield");
}
2024-04-12 11:44:40 +01:00
else if (!lua_isstring(ML, -1))
{
2024-04-12 11:44:40 +01:00
lua_pushstring(ML, "unknown error while running module");
}
}
2024-04-12 11:44:40 +01:00
// there's now a return value on top of ML; L stack: _MODULES ML
lua_xmove(ML, L, 1);
lua_pushvalue(L, -1);
lua_setfield(L, -4, resolvedRequire.absolutePath.c_str());
// L stack: _MODULES ML result
return finishrequire(L);
}
static int lua_collectgarbage(lua_State* L)
{
const char* option = luaL_optstring(L, 1, "collect");
if (strcmp(option, "collect") == 0)
{
lua_gc(L, LUA_GCCOLLECT, 0);
return 0;
}
if (strcmp(option, "count") == 0)
{
int c = lua_gc(L, LUA_GCCOUNT, 0);
lua_pushnumber(L, c);
return 1;
}
luaL_error(L, "collectgarbage must be called with 'count' or 'collect'");
}
2022-07-08 02:05:31 +01:00
#ifdef CALLGRIND
static int lua_callgrind(lua_State* L)
{
const char* option = luaL_checkstring(L, 1);
if (strcmp(option, "running") == 0)
{
int r = RUNNING_ON_VALGRIND;
lua_pushboolean(L, r);
return 1;
}
if (strcmp(option, "zero") == 0)
{
CALLGRIND_ZERO_STATS;
return 0;
}
if (strcmp(option, "dump") == 0)
{
const char* name = luaL_checkstring(L, 2);
CALLGRIND_DUMP_STATS_AT(name);
return 0;
}
luaL_error(L, "callgrind must be called with one of 'running', 'zero', 'dump'");
}
#endif
2022-01-27 21:29:34 +00:00
void setupState(lua_State* L)
{
2022-10-13 23:59:53 +01:00
if (codegen)
Luau::CodeGen::create(L);
luaL_openlibs(L);
static const luaL_Reg funcs[] = {
{"loadstring", lua_loadstring},
{"require", lua_require},
{"collectgarbage", lua_collectgarbage},
2022-07-08 02:05:31 +01:00
#ifdef CALLGRIND
{"callgrind", lua_callgrind},
#endif
{NULL, NULL},
};
lua_pushvalue(L, LUA_GLOBALSINDEX);
luaL_register(L, NULL, funcs);
lua_pop(L, 1);
luaL_sandbox(L);
}
2024-02-23 18:40:00 +00:00
void setupArguments(lua_State* L, int argc, char** argv)
{
2024-05-31 18:46:33 +01:00
lua_checkstack(L, argc);
2024-02-23 18:40:00 +00:00
for (int i = 0; i < argc; ++i)
lua_pushstring(L, argv[i]);
}
2022-01-27 21:29:34 +00:00
std::string runCode(lua_State* L, const std::string& source)
{
2024-05-31 18:46:33 +01:00
lua_checkstack(L, LUA_MINSTACK);
std::string bytecode = Luau::compile(source, copts());
2021-11-18 22:21:07 +00:00
if (luau_load(L, "=stdin", bytecode.data(), bytecode.size(), 0) != 0)
{
size_t len;
const char* msg = lua_tolstring(L, -1, &len);
std::string error(msg, len);
lua_pop(L, 1);
return error;
}
lua_State* T = lua_newthread(L);
lua_pushvalue(L, -2);
lua_remove(L, -3);
lua_xmove(L, T, 1);
int status = lua_resume(T, NULL, 0);
if (status == 0)
{
int n = lua_gettop(T);
if (n)
{
luaL_checkstack(T, LUA_MINSTACK, "too many results to print");
2022-01-27 21:29:34 +00:00
lua_getglobal(T, "_PRETTYPRINT");
// If _PRETTYPRINT is nil, then use the standard print function instead
if (lua_isnil(T, -1))
{
lua_pop(T, 1);
lua_getglobal(T, "print");
}
lua_insert(T, 1);
lua_pcall(T, n, 0, 0);
}
2023-02-17 14:53:37 +00:00
lua_pop(L, 1);
return std::string();
}
else
{
std::string error;
if (status == LUA_YIELD)
{
error = "thread yielded unexpectedly";
}
else if (const char* str = lua_tostring(T, -1))
{
error = str;
}
error += "\nstack backtrace:\n";
error += lua_debugtrace(T);
2023-02-17 14:53:37 +00:00
lua_pop(L, 1);
return error;
}
}
2022-02-18 00:41:20 +00:00
// Replaces the top of the lua stack with the metatable __index for the value
// if it exists. Returns true iff __index exists.
static bool tryReplaceTopWithIndex(lua_State* L)
{
2022-02-18 00:41:20 +00:00
if (luaL_getmetafield(L, -1, "__index"))
{
2022-02-18 00:41:20 +00:00
// Remove the table leaving __index on the top of stack
lua_remove(L, -2);
return true;
}
return false;
}
2022-02-18 00:41:20 +00:00
// This function is similar to lua_gettable, but it avoids calling any
// lua callback functions (e.g. __index) which might modify the Lua VM state.
static void safeGetTable(lua_State* L, int tableIndex)
{
lua_pushvalue(L, tableIndex); // Duplicate the table
// The loop invariant is that the table to search is at -1
// and the key is at -2.
for (int loopCount = 0;; loopCount++)
{
lua_pushvalue(L, -2); // Duplicate the key
lua_rawget(L, -2); // Try to find the key
if (!lua_isnil(L, -1) || loopCount >= MaxTraversalLimit)
{
// Either the key has been found, and/or we have reached the max traversal limit
break;
}
else
{
2022-02-18 00:41:20 +00:00
lua_pop(L, 1); // Pop the nil result
if (!luaL_getmetafield(L, -1, "__index"))
{
lua_pushnil(L);
break;
}
else if (lua_istable(L, -1))
{
// Replace the current table being searched with __index table
lua_replace(L, -2);
}
else
{
lua_pop(L, 1); // Pop the value
lua_pushnil(L);
break;
}
}
}
lua_remove(L, -2); // Remove the table
lua_remove(L, -2); // Remove the original key
}
// completePartialMatches finds keys that match the specified 'prefix'
// Note: the table/object to be searched must be on the top of the Lua stack
2024-08-02 00:25:12 +01:00
static void completePartialMatches(
lua_State* L,
bool completeOnlyFunctions,
const std::string& editBuffer,
std::string_view prefix,
const AddCompletionCallback& addCompletionCallback
)
2022-02-18 00:41:20 +00:00
{
for (int i = 0; i < MaxTraversalLimit && lua_istable(L, -1); i++)
{
// table, key
lua_pushnil(L);
2022-02-18 00:41:20 +00:00
// Loop over all the keys in the current table
while (lua_next(L, -2) != 0)
{
if (lua_type(L, -2) == LUA_TSTRING)
{
2022-02-18 00:41:20 +00:00
// table, key, value
std::string_view key = lua_tostring(L, -2);
int valueType = lua_type(L, -1);
2022-01-21 16:23:02 +00:00
2022-02-18 00:41:20 +00:00
// If the last separator was a ':' (i.e. a method call) then only functions should be completed.
bool requiredValueType = (!completeOnlyFunctions || valueType == LUA_TFUNCTION);
2022-01-21 16:23:02 +00:00
2022-02-18 00:41:20 +00:00
if (!key.empty() && requiredValueType && Luau::startsWith(key, prefix))
{
std::string completedComponent(key.substr(prefix.size()));
std::string completion(editBuffer + completedComponent);
if (valueType == LUA_TFUNCTION)
2022-01-21 16:23:02 +00:00
{
2022-02-18 00:41:20 +00:00
// Add an opening paren for function calls by default.
completion += "(";
2022-01-21 16:23:02 +00:00
}
2022-02-18 00:41:20 +00:00
addCompletionCallback(completion, std::string(key));
2022-01-06 22:10:07 +00:00
}
}
2022-02-18 00:41:20 +00:00
lua_pop(L, 1);
}
// Replace the current table being searched with an __index table if one exists
if (!tryReplaceTopWithIndex(L))
{
break;
}
}
}
static void completeIndexer(lua_State* L, const std::string& editBuffer, const AddCompletionCallback& addCompletionCallback)
{
std::string_view lookup = editBuffer;
bool completeOnlyFunctions = false;
2024-05-31 18:46:33 +01:00
lua_checkstack(L, LUA_MINSTACK);
2022-02-18 00:41:20 +00:00
// Push the global variable table to begin the search
lua_pushvalue(L, LUA_GLOBALSINDEX);
for (;;)
{
size_t sep = lookup.find_first_of(".:");
std::string_view prefix = lookup.substr(0, sep);
2022-02-18 00:41:20 +00:00
if (sep == std::string_view::npos)
{
completePartialMatches(L, completeOnlyFunctions, editBuffer, prefix, addCompletionCallback);
break;
}
else
{
// find the key in the table
lua_pushlstring(L, prefix.data(), prefix.size());
2022-02-18 00:41:20 +00:00
safeGetTable(L, -2);
lua_remove(L, -2);
2022-02-18 00:41:20 +00:00
if (lua_istable(L, -1) || tryReplaceTopWithIndex(L))
2022-01-21 16:23:02 +00:00
{
2022-02-18 00:41:20 +00:00
completeOnlyFunctions = lookup[sep] == ':';
lookup.remove_prefix(sep + 1);
2022-01-21 16:23:02 +00:00
}
2022-02-18 00:41:20 +00:00
else
{
// Unable to search for keys, so stop searching
break;
2022-02-18 00:41:20 +00:00
}
}
}
lua_pop(L, 1);
}
2022-02-11 18:43:14 +00:00
void getCompletions(lua_State* L, const std::string& editBuffer, const AddCompletionCallback& addCompletionCallback)
{
completeIndexer(L, editBuffer, addCompletionCallback);
}
static void icGetCompletions(ic_completion_env_t* cenv, const char* editBuffer)
{
auto* L = reinterpret_cast<lua_State*>(ic_completion_arg(cenv));
2024-08-02 00:25:12 +01:00
getCompletions(
L,
std::string(editBuffer),
[cenv](const std::string& completion, const std::string& display)
{
ic_add_completion_ex(cenv, completion.data(), display.data(), nullptr);
}
);
2022-02-11 18:43:14 +00:00
}
2022-02-03 23:09:37 +00:00
static bool isMethodOrFunctionChar(const char* s, long len)
{
2022-02-03 23:09:37 +00:00
char c = *s;
return len == 1 && (isalnum(c) || c == '.' || c == ':' || c == '_');
}
static void completeRepl(ic_completion_env_t* cenv, const char* editBuffer)
{
2022-02-11 18:43:14 +00:00
ic_complete_word(cenv, editBuffer, icGetCompletions, isMethodOrFunctionChar);
}
2022-03-04 16:19:20 +00:00
static void loadHistory(const char* name)
2022-01-06 22:10:07 +00:00
{
2022-03-04 16:19:20 +00:00
std::string path;
2022-01-06 22:10:07 +00:00
2022-03-04 16:19:20 +00:00
if (const char* home = getenv("HOME"))
{
path = joinPaths(home, name);
}
else if (const char* userProfile = getenv("USERPROFILE"))
{
path = joinPaths(userProfile, name);
2022-01-06 22:10:07 +00:00
}
2022-03-04 16:19:20 +00:00
if (!path.empty())
ic_set_history(path.c_str(), -1 /* default entries (= 200) */);
}
2022-01-06 22:10:07 +00:00
2022-01-21 16:23:02 +00:00
static void runReplImpl(lua_State* L)
{
2022-02-03 23:09:37 +00:00
ic_set_default_completer(completeRepl, L);
2022-05-06 00:52:48 +01:00
// Reset the locale to C
setlocale(LC_ALL, "C");
2022-02-03 23:09:37 +00:00
// Make brace matching easier to see
ic_style_def("ic-bracematch", "teal");
// Prevent auto insertion of braces
ic_enable_brace_insertion(false);
2022-03-04 16:19:20 +00:00
// Loads history from the given file; isocline automatically saves the history on process exit
loadHistory(".luau_history");
std::string buffer;
for (;;)
{
2022-02-11 18:43:14 +00:00
const char* prompt = buffer.empty() ? "" : ">";
std::unique_ptr<char, void (*)(void*)> line(ic_readline(prompt), free);
2022-02-03 23:09:37 +00:00
if (!line)
break;
2022-02-11 18:43:14 +00:00
if (buffer.empty() && runCode(L, std::string("return ") + line.get()) == std::string())
{
2022-02-11 18:43:14 +00:00
ic_history_add(line.get());
continue;
}
2022-02-03 23:09:37 +00:00
if (!buffer.empty())
buffer += "\n";
2022-02-11 18:43:14 +00:00
buffer += line.get();
std::string error = runCode(L, buffer);
if (error.length() >= 5 && error.compare(error.length() - 5, 5, "<eof>") == 0)
{
continue;
}
if (error.length())
{
fprintf(stdout, "%s\n", error.c_str());
}
2022-02-03 23:09:37 +00:00
ic_history_add(buffer.c_str());
buffer.clear();
}
}
2022-01-21 16:23:02 +00:00
static void runRepl()
{
std::unique_ptr<lua_State, void (*)(lua_State*)> globalState(luaL_newstate(), lua_close);
lua_State* L = globalState.get();
setupState(L);
2022-08-04 22:27:28 +01:00
// setup Ctrl+C handling
replState = L;
#ifdef _WIN32
SetConsoleCtrlHandler(sigintHandler, TRUE);
#else
signal(SIGINT, sigintHandler);
#endif
2022-01-21 16:23:02 +00:00
luaL_sandboxthread(L);
runReplImpl(L);
}
// `repl` is used it indicate if a repl should be started after executing the file.
static bool runFile(const char* name, lua_State* GL, bool repl)
{
std::optional<std::string> source = readFile(name);
if (!source)
{
fprintf(stderr, "Error opening %s\n", name);
return false;
}
// module needs to run in a new thread, isolated from the rest
lua_State* L = lua_newthread(GL);
// new thread needs to have the globals sandboxed
luaL_sandboxthread(L);
std::string chunkname = "=" + std::string(name);
std::string bytecode = Luau::compile(*source, copts());
int status = 0;
2021-11-18 22:21:07 +00:00
if (luau_load(L, chunkname.c_str(), bytecode.data(), bytecode.size(), 0) == 0)
{
2022-10-13 23:59:53 +01:00
if (codegen)
2024-05-10 17:17:09 +01:00
{
Luau::CodeGen::CompilationOptions nativeOptions;
Luau::CodeGen::compile(L, -1, nativeOptions);
}
2022-10-13 23:59:53 +01:00
if (coverageActive())
coverageTrack(L, -1);
2024-02-23 18:40:00 +00:00
setupArguments(L, program_argc, program_argv);
status = lua_resume(L, NULL, program_argc);
}
else
{
status = LUA_ERRSYNTAX;
}
2021-11-18 22:21:07 +00:00
if (status != 0)
{
std::string error;
if (status == LUA_YIELD)
{
error = "thread yielded unexpectedly";
}
else if (const char* str = lua_tostring(L, -1))
{
error = str;
}
error += "\nstacktrace:\n";
error += lua_debugtrace(L);
fprintf(stderr, "%s", error.c_str());
}
2021-11-18 22:21:07 +00:00
2022-01-21 16:23:02 +00:00
if (repl)
{
runReplImpl(L);
}
2021-11-18 22:21:07 +00:00
lua_pop(GL, 1);
return status == 0;
}
static void displayHelp(const char* argv0)
{
2024-02-23 18:40:00 +00:00
printf("Usage: %s [options] [file list] [-a] [arg list]\n", argv0);
printf("\n");
2023-06-16 18:01:18 +01:00
printf("When file list is omitted, an interactive REPL is started instead.\n");
printf("\n");
printf("Available options:\n");
2022-01-21 16:23:02 +00:00
printf(" --coverage: collect code coverage while running the code and output results to coverage.out\n");
2022-01-14 16:06:31 +00:00
printf(" -h, --help: Display this usage message.\n");
2022-01-21 16:23:02 +00:00
printf(" -i, --interactive: Run an interactive REPL after executing the last script specified.\n");
2022-02-11 18:43:14 +00:00
printf(" -O<n>: compile with optimization level n (default 1, n should be between 0 and 2).\n");
printf(" -g<n>: compile with debug level n (default 1, n should be between 0 and 2).\n");
printf(" --profile[=N]: profile the code using N Hz sampling (default 10000) and output results to profile.out\n");
2022-01-14 16:06:31 +00:00
printf(" --timetrace: record compiler time tracing information into trace.json\n");
2022-10-13 23:59:53 +01:00
printf(" --codegen: execute code using native code generation\n");
2024-03-15 21:01:00 +00:00
printf(" --program-args,-a: declare start of arguments to be passed to the Luau program\n");
}
2022-01-06 22:10:07 +00:00
static int assertionHandler(const char* expr, const char* file, int line, const char* function)
{
printf("%s(%d): ASSERTION FAILED: %s\n", file, line, expr);
return 1;
}
2022-01-27 21:29:34 +00:00
int replMain(int argc, char** argv)
{
Luau::assertHandler() = assertionHandler;
2022-07-21 21:36:41 +01:00
setLuauFlagsDefault();
2023-07-14 16:57:16 +01:00
#ifdef _WIN32
SetConsoleOutputCP(CP_UTF8);
#endif
2022-01-14 16:06:31 +00:00
int profile = 0;
bool coverage = false;
2022-01-21 16:23:02 +00:00
bool interactive = false;
2023-05-12 13:15:01 +01:00
bool codegenPerf = false;
2024-02-23 18:40:00 +00:00
int program_args = argc;
2022-01-14 16:06:31 +00:00
2023-06-16 18:01:18 +01:00
for (int i = 1; i < argc; i++)
{
2022-01-14 16:06:31 +00:00
if (strcmp(argv[i], "-h") == 0 || strcmp(argv[i], "--help") == 0)
{
displayHelp(argv[0]);
return 0;
}
2022-01-21 16:23:02 +00:00
else if (strcmp(argv[i], "-i") == 0 || strcmp(argv[i], "--interactive") == 0)
{
interactive = true;
}
else if (strncmp(argv[i], "-O", 2) == 0)
{
int level = atoi(argv[i] + 2);
if (level < 0 || level > 2)
{
fprintf(stderr, "Error: Optimization level must be between 0 and 2 inclusive.\n");
return 1;
}
globalOptions.optimizationLevel = level;
}
2022-02-11 18:43:14 +00:00
else if (strncmp(argv[i], "-g", 2) == 0)
{
int level = atoi(argv[i] + 2);
if (level < 0 || level > 2)
{
fprintf(stderr, "Error: Debug level must be between 0 and 2 inclusive.\n");
return 1;
}
globalOptions.debugLevel = level;
}
2022-01-14 16:06:31 +00:00
else if (strcmp(argv[i], "--profile") == 0)
{
profile = 10000; // default to 10 KHz
}
else if (strncmp(argv[i], "--profile=", 10) == 0)
{
profile = atoi(argv[i] + 10);
}
2022-10-13 23:59:53 +01:00
else if (strcmp(argv[i], "--codegen") == 0)
{
codegen = true;
}
2023-05-12 13:15:01 +01:00
else if (strcmp(argv[i], "--codegen-perf") == 0)
{
codegen = true;
codegenPerf = true;
}
2022-01-14 16:06:31 +00:00
else if (strcmp(argv[i], "--coverage") == 0)
{
coverage = true;
}
else if (strcmp(argv[i], "--timetrace") == 0)
{
FFlag::DebugLuauTimeTracing.value = true;
}
2022-04-21 22:04:22 +01:00
else if (strncmp(argv[i], "--fflags=", 9) == 0)
{
2022-07-21 21:36:41 +01:00
setLuauFlags(argv[i] + 9);
2022-04-21 22:04:22 +01:00
}
2024-02-23 18:40:00 +00:00
else if (strcmp(argv[i], "--program-args") == 0 || strcmp(argv[i], "-a") == 0)
{
program_args = i + 1;
break;
}
2022-01-14 16:06:31 +00:00
else if (argv[i][0] == '-')
{
2022-01-21 16:23:02 +00:00
fprintf(stderr, "Error: Unrecognized option '%s'.\n\n", argv[i]);
2022-01-14 16:06:31 +00:00
displayHelp(argv[0]);
return 1;
}
}
2021-11-12 02:12:39 +00:00
2024-02-23 18:40:00 +00:00
program_argc = argc - program_args;
program_argv = &argv[program_args];
2022-07-21 21:36:41 +01:00
#if !defined(LUAU_ENABLE_TIME_TRACE)
if (FFlag::DebugLuauTimeTracing)
{
fprintf(stderr, "To run with --timetrace, Luau has to be built with LUAU_ENABLE_TIME_TRACE enabled\n");
return 1;
}
#endif
2023-05-12 13:15:01 +01:00
if (codegenPerf)
{
#if __linux__
char path[128];
snprintf(path, sizeof(path), "/tmp/perf-%d.map", getpid());
// note, there's no need to close the log explicitly as it will be closed when the process exits
FILE* codegenPerfLog = fopen(path, "w");
2024-08-02 00:25:12 +01:00
Luau::CodeGen::setPerfLog(
codegenPerfLog,
[](void* context, uintptr_t addr, unsigned size, const char* symbol)
{
fprintf(static_cast<FILE*>(context), "%016lx %08x %s\n", long(addr), size, symbol);
}
);
2023-05-12 13:15:01 +01:00
#else
fprintf(stderr, "--codegen-perf option is only supported on Linux\n");
return 1;
#endif
}
2023-06-16 18:01:18 +01:00
if (codegen && !Luau::CodeGen::isSupported())
2023-08-25 16:25:09 +01:00
fprintf(stderr, "Warning: Native code generation is not supported in current configuration\n");
2022-10-13 23:59:53 +01:00
2023-06-16 18:01:18 +01:00
const std::vector<std::string> files = getSourceFiles(argc, argv);
2023-06-16 18:01:18 +01:00
if (files.empty())
2022-01-14 16:06:31 +00:00
{
runRepl();
return 0;
}
2023-06-16 18:01:18 +01:00
else
{
std::unique_ptr<lua_State, void (*)(lua_State*)> globalState(luaL_newstate(), lua_close);
lua_State* L = globalState.get();
setupState(L);
if (profile)
profilerStart(L, profile);
if (coverage)
coverageInit(L);
2021-11-18 22:21:07 +00:00
int failed = 0;
2022-01-21 16:23:02 +00:00
for (size_t i = 0; i < files.size(); ++i)
{
bool isLastFile = i == files.size() - 1;
failed += !runFile(files[i].c_str(), L, interactive && isLastFile);
}
if (profile)
{
profilerStop();
profilerDump("profile.out");
}
if (coverage)
coverageDump("coverage.out");
2022-01-14 16:06:31 +00:00
return failed ? 1 : 0;
}
}