/* ** Standard I/O (and system) library ** See Copyright Notice in lua.h */ #define FORBIDDEN_SYMBOL_EXCEPTION_setjmp #define FORBIDDEN_SYMBOL_EXCEPTION_longjmp #define FORBIDDEN_SYMBOL_EXCEPTION_fread #define FORBIDDEN_SYMBOL_EXCEPTION_fwrite #define FORBIDDEN_SYMBOL_EXCEPTION_fseek #define FORBIDDEN_SYMBOL_EXCEPTION_sprintf #define FORBIDDEN_SYMBOL_EXCEPTION_stderr #define FORBIDDEN_SYMBOL_EXCEPTION_stdin #define FORBIDDEN_SYMBOL_EXCEPTION_stdout #define FORBIDDEN_SYMBOL_EXCEPTION_exit #include "common/savefile.h" #include "common/fs.h" #include "common/system.h" #include "engines/grim/lua/lauxlib.h" #include "engines/grim/lua/lua.h" #include "engines/grim/lua/luadebug.h" #include "engines/grim/lua/lualib.h" #include "base/commandLine.h" #include "engines/grim/resource.h" namespace Grim { #define CLOSEDTAG 2 #define IOTAG 1 #define FIRSTARG 3 // 1st and 2nd are upvalues #define FINPUT "_INPUT" #define FOUTPUT "_OUTPUT" LuaFile *g_stderr; static int32 s_id = 0; Common::HashMap *g_files; LuaFile::LuaFile() : _in(nullptr), _out(nullptr), _stdin(false), _stdout(false), _stderr(false) { } LuaFile::~LuaFile() { close(); } void LuaFile::close() { delete _in; delete _out; _in = nullptr; _out = nullptr; _stdin = _stdout = _stderr = false; } bool LuaFile::isOpen() const { return _in || _out || _stdin || stdout || stderr; } uint32 LuaFile::read(void *buf, uint32 len) { if (_stdin) { return fread(buf, len, 1, stdin); } else if (_in) { return _in->read(buf, len); } else assert(0); return 0; } uint32 LuaFile::write(const char *buf, uint32 len) { if (_stdin) error("LuaFile::write() not allowed on stdin"); if (_in) error("LuaFile::write() not allowed on in"); if (_stdout) { return fwrite(buf, len, 1, stdout); } else if (_stderr) { return fwrite(buf, len, 1, stderr); } else if (_out) { return _out->write(buf, len); } else assert(0); return 0; } void LuaFile::seek(int32 pos, int whence) { if (_stdin) { fseek(stdin, pos, whence); } else if (_in) { _in->seek(pos, whence); } else assert(0); } static int32 gettag(int32 i) { return (int32)lua_getnumber(lua_getparam(i)); } static void pushresult(int32 i) { if (i) lua_pushuserdata(0); else { lua_pushnil(); lua_pushstring("File I/O error."); } } static int32 ishandler(lua_Object f) { if (lua_isuserdata(f)) { if (lua_tag(f) == gettag(CLOSEDTAG)) lua_error("cannot access a closed file"); return lua_tag(f) == gettag(IOTAG); } else return 0; } static LuaFile *getfile(int32 id) { if (g_files->contains(id)) { return (*g_files)[id]; } return nullptr; } static LuaFile *getfile(const char *name) { lua_Object f = lua_getglobal(name); if (!ishandler(f)) luaL_verror("global variable `%.50s' is not a file handle", name); return getfile(lua_getuserdata(f)); } static LuaFile *getfileparam(const char *name, int32 *arg) { lua_Object f = lua_getparam(*arg); if (ishandler(f)) { (*arg)++; return getfile(lua_getuserdata(f)); } else return getfile(name); } static void closefile(const char *name) { LuaFile *f = getfile(name); f->close(); lua_pushobject(lua_getglobal(name)); lua_settag(gettag(CLOSEDTAG)); } static void setfile(int32 id, const char *name, int32 tag) { lua_pushusertag(id, tag); lua_setglobal(name); } static void setreturn(int32 id, const char *name) { int32 tag = gettag(IOTAG); setfile(id, name, tag); lua_pushusertag(id, tag); } static int32 addfile(LuaFile *f) { ++s_id; (*g_files)[s_id] = f; return s_id; } static void io_readfrom() { lua_Object f = lua_getparam(FIRSTARG); if (f == LUA_NOOBJECT) { if (getfile(FINPUT) != getfile(1)) { closefile(FINPUT); setreturn(1, FINPUT); } } else if (lua_tag(f) == gettag(IOTAG)) { int32 id = lua_getuserdata(f); LuaFile *current = getfile(id); if (!current) { pushresult(0); return; } setreturn(id, FINPUT); } else { const char *s = luaL_check_string(FIRSTARG); Common::String fileName = Common::lastPathComponent(s, '\\'); Common::SeekableReadStream *inFile = nullptr; Common::SaveFileManager *saveFileMan = g_system->getSavefileManager(); inFile = saveFileMan->openForLoading(fileName); if (!inFile) { inFile = g_resourceloader->openNewStreamFile(s); } if (inFile) { LuaFile *current = new LuaFile(); current->_in = inFile; current->_filename = s; setreturn(addfile(current), FINPUT); } else { warning("liolib.cpp, io_readfrom(): Could not open file %s", s); pushresult(0); } } } static void io_writeto() { lua_Object f = lua_getparam(FIRSTARG); if (f == LUA_NOOBJECT) { if (getfile(FOUTPUT) != getfile(2)) { closefile(FOUTPUT); setreturn(2, FOUTPUT); } } else if (lua_tag(f) == gettag(IOTAG)) { int32 id = lua_getuserdata(f); LuaFile *current = getfile(id); if (!current->isOpen()) { pushresult(0); return; } setreturn(id, FOUTPUT); } else { Common::String s = Common::lastPathComponent(luaL_check_string(FIRSTARG), '\\'); LuaFile *current; Common::WriteStream *outFile = nullptr; Common::SaveFileManager *saveFileMan = g_system->getSavefileManager(); outFile = saveFileMan->openForSaving(s, false); if (!outFile) { pushresult(0); return; } current = new LuaFile(); current->_out = outFile; current->_filename = s; setreturn(addfile(current), FOUTPUT); } } static void io_appendto() { Common::String s = Common::lastPathComponent(luaL_check_string(FIRSTARG), '\\'); Common::SeekableReadStream *inFile = nullptr; Common::SaveFileManager *saveFileMan = g_system->getSavefileManager(); inFile = saveFileMan->openForLoading(s); if (!inFile) { pushresult(0); return; } int size = inFile->size(); byte *buf = new byte[size]; inFile->read(buf, size); delete inFile; Common::WriteStream *outFile = nullptr; outFile = saveFileMan->openForSaving(s); if (!outFile) pushresult(0); else { outFile->write(buf, size); LuaFile *current = new LuaFile(); current->_out = outFile; current->_filename = s; setreturn(addfile(current), FOUTPUT); } delete[] buf; } #define NEED_OTHER (EOF - 1) // just some flag different from EOF static void io_read() { int32 arg = FIRSTARG; LuaFile *f = (LuaFile *)getfileparam(FINPUT, &arg); char *buff; const char *p = luaL_opt_string(arg, "[^\n]*{\n}"); int inskip = 0; // to control {skips} int c = NEED_OTHER; luaL_resetbuffer(); while (*p) { if (*p == '{') { inskip++; p++; } else if (*p == '}') { if (inskip == 0) lua_error("unbalanced braces in read pattern"); inskip--; p++; } else { const char *ep; // get what is next int m; // match result if (c == NEED_OTHER) { char z; if (f->read(&z, 1) != 1) c = EOF; else c = z; } m = luaI_singlematch((c == EOF) ? 0 : (char)c, p, &ep); if (m) { if (inskip == 0) luaL_addchar(c); c = NEED_OTHER; } switch (*ep) { case '*': // repetition if (!m) p = ep + 1; // else stay in (repeat) the same item break; case '?': // optional p = ep + 1; // continues reading the pattern break; default: if (m) p = ep; // continues reading the pattern else goto break_while; // pattern fails } } } break_while: if (c >= 0) // not EOF nor NEED_OTHER? f->seek(-1, SEEK_CUR); luaL_addchar(0); buff = luaL_buffer(); if (*buff != 0 || *p == 0) // read something or did not fail? lua_pushstring(buff); } static void io_write() { int32 arg = FIRSTARG; LuaFile *f = (LuaFile *)getfileparam(FOUTPUT, &arg); int32 status = 1; const char *s; while ((s = luaL_opt_string(arg++, nullptr)) != nullptr) status = status && ((int32)f->write(s, strlen(s)) != EOF); pushresult(status); } static void io_date() { TimeDate t; char b[BUFSIZ]; g_system->getTimeAndDate(t); snprintf(b, BUFSIZ, "%02d.%02d.%d %02d:%02d.%02d", t.tm_mday, t.tm_mon + 1, 1900 + t.tm_year, t.tm_hour, t.tm_min, t.tm_sec); lua_pushstring(b); } static void io_exit() { lua_Object o = lua_getparam(1); exit((int)lua_isnumber(o) ? (int)lua_getnumber(o) : 1); } static void lua_printstack() { int32 level = 1; // skip level 0 (it's this function) lua_Object func; char buf[256]; while ((func = lua_stackedfunction(level++)) != LUA_NOOBJECT) { const char *name; int32 currentline; const char *filename; int32 linedefined; lua_funcinfo(func, &filename, &linedefined); snprintf(buf, 256, (level == 2) ? "Active Stack:\n\t" : "\t"); g_stderr->write(buf, strlen(buf)); switch (*lua_getobjname(func, &name)) { case 'g': snprintf(buf, 256, "function %s", name); break; case 't': snprintf(buf, 256, "`%s' tag method", name); break; default: { if (linedefined == 0) snprintf(buf, 256, "main of %s", filename); else if (linedefined < 0) snprintf(buf, 256, "%s", filename); else snprintf(buf, 256, "function (%s:%d)", filename, (int)linedefined); filename = nullptr; } } g_stderr->write(buf, strlen(buf)); if ((currentline = lua_currentline(func)) > 0) { snprintf(buf, 256, " at line %d", (int)currentline); g_stderr->write(buf, strlen(buf)); } if (filename) { snprintf(buf, 256, " [in file %s]", filename); g_stderr->write(buf, strlen(buf)); } snprintf(buf, 256, "\n"); g_stderr->write(buf, strlen(buf)); } } static void errorfb() { char buf[256]; snprintf(buf, 256, "lua: %s\n", lua_getstring(lua_getparam(1))); g_stderr->write(buf, strlen(buf)); lua_printstack(); } static struct luaL_reg iolib[] = { { "date", io_date }, { "exit", io_exit }, { "print_stack", errorfb } }; static struct luaL_reg iolibtag[] = { { "readfrom", io_readfrom }, { "writeto", io_writeto }, { "appendto", io_appendto }, { "read", io_read }, { "write", io_write } }; static void openwithtags() { int32 iotag = lua_newtag(); int32 closedtag = lua_newtag(); for (uint32 i = 0; i < sizeof(iolibtag) / sizeof(iolibtag[0]); i++) { // put both tags as upvalues for these functions lua_pushnumber(iotag); lua_pushnumber(closedtag); lua_pushCclosure(iolibtag[i].func, 2); lua_setglobal(iolibtag[i].name); } LuaFile* f; f = new LuaFile(); f->_stdin = true; setfile(addfile(f), FINPUT, iotag); f = new LuaFile(); f->_stdout = true; setfile(addfile(f), FOUTPUT, iotag); f = new LuaFile(); f->_stdin = true; setfile(addfile(f), "_STDIN", iotag); f = new LuaFile(); f->_stdout = true; setfile(addfile(f), "_STDOUT", iotag); g_stderr = new LuaFile(); g_stderr->_stderr = true; setfile(addfile(g_stderr), "_STDERR", iotag); } void lua_iolibopen() { g_files = new Common::HashMap(); luaL_openlib(iolib, (sizeof(iolib) / sizeof(iolib[0]))); luaL_addlibtolist(iolibtag, (sizeof(iolibtag) / sizeof(iolibtag[0]))); openwithtags(); lua_pushcfunction(errorfb); lua_seterrormethod(); } void lua_iolibclose() { for (Common::HashMap::iterator it = g_files->begin(); it != g_files->end(); ++it) { delete it->_value; } delete g_files; } } // end of namespace Grim