Handle LuaErrors in Lua -> C++ calls on LuaJIT

This commit is contained in:
ShadowNinja 2013-12-18 16:35:55 -05:00
parent 38d112033b
commit 49cec3f782
18 changed files with 113 additions and 134 deletions

View File

@ -12,6 +12,7 @@
#define CMAKE_USE_FREETYPE @USE_FREETYPE@ #define CMAKE_USE_FREETYPE @USE_FREETYPE@
#define CMAKE_STATIC_SHAREDIR "@SHAREDIR@" #define CMAKE_STATIC_SHAREDIR "@SHAREDIR@"
#define CMAKE_USE_LEVELDB @USE_LEVELDB@ #define CMAKE_USE_LEVELDB @USE_LEVELDB@
#define CMAKE_USE_LUAJIT @USE_LUAJIT@
#ifdef NDEBUG #ifdef NDEBUG
#define CMAKE_BUILD_TYPE "Release" #define CMAKE_BUILD_TYPE "Release"

View File

@ -14,6 +14,7 @@
#define USE_FREETYPE 0 #define USE_FREETYPE 0
#define STATIC_SHAREDIR "" #define STATIC_SHAREDIR ""
#define USE_LEVELDB 0 #define USE_LEVELDB 0
#define USE_LUAJIT 0
#ifdef USE_CMAKE_CONFIG_H #ifdef USE_CMAKE_CONFIG_H
#include "cmake_config.h" #include "cmake_config.h"
@ -33,6 +34,8 @@
#define STATIC_SHAREDIR CMAKE_STATIC_SHAREDIR #define STATIC_SHAREDIR CMAKE_STATIC_SHAREDIR
#undef USE_LEVELDB #undef USE_LEVELDB
#define USE_LEVELDB CMAKE_USE_LEVELDB #define USE_LEVELDB CMAKE_USE_LEVELDB
#undef USE_LUAJIT
#define USE_LUAJIT CMAKE_USE_LUAJIT
#endif #endif
#endif #endif

View File

@ -21,132 +21,99 @@ with this program; if not, write to the Free Software Foundation, Inc.,
#define EXCEPTIONS_HEADER #define EXCEPTIONS_HEADER
#include <exception> #include <exception>
#include <string>
class BaseException : public std::exception class BaseException : public std::exception
{ {
public: public:
BaseException(const char *s) BaseException(const std::string s) throw()
{ {
m_s = s; m_s = s;
} }
~BaseException() throw() {}
virtual const char * what() const throw() virtual const char * what() const throw()
{ {
return m_s; return m_s.c_str();
} }
const char *m_s; protected:
std::string m_s;
}; };
class AsyncQueuedException : public BaseException class AsyncQueuedException : public BaseException {
{
public: public:
AsyncQueuedException(const char *s): AsyncQueuedException(std::string s): BaseException(s) {}
BaseException(s)
{}
}; };
class NotImplementedException : public BaseException class NotImplementedException : public BaseException {
{
public: public:
NotImplementedException(const char *s): NotImplementedException(std::string s): BaseException(s) {}
BaseException(s)
{}
}; };
class AlreadyExistsException : public BaseException class AlreadyExistsException : public BaseException {
{
public: public:
AlreadyExistsException(const char *s): AlreadyExistsException(std::string s): BaseException(s) {}
BaseException(s)
{}
}; };
class VersionMismatchException : public BaseException class VersionMismatchException : public BaseException {
{
public: public:
VersionMismatchException(const char *s): VersionMismatchException(std::string s): BaseException(s) {}
BaseException(s)
{}
}; };
class FileNotGoodException : public BaseException class FileNotGoodException : public BaseException {
{
public: public:
FileNotGoodException(const char *s): FileNotGoodException(std::string s): BaseException(s) {}
BaseException(s)
{}
}; };
class SerializationError : public BaseException class SerializationError : public BaseException {
{
public: public:
SerializationError(const char *s): SerializationError(std::string s): BaseException(s) {}
BaseException(s)
{}
}; };
class LoadError : public BaseException class LoadError : public BaseException {
{
public: public:
LoadError(const char *s): LoadError(std::string s): BaseException(s) {}
BaseException(s)
{}
}; };
class ContainerFullException : public BaseException class ContainerFullException : public BaseException {
{
public: public:
ContainerFullException(const char *s): ContainerFullException(std::string s): BaseException(s) {}
BaseException(s)
{}
}; };
class SettingNotFoundException : public BaseException class SettingNotFoundException : public BaseException {
{
public: public:
SettingNotFoundException(const char *s): SettingNotFoundException(std::string s): BaseException(s) {}
BaseException(s)
{}
}; };
class InvalidFilenameException : public BaseException class InvalidFilenameException : public BaseException {
{
public: public:
InvalidFilenameException(const char *s): InvalidFilenameException(std::string s): BaseException(s) {}
BaseException(s)
{}
}; };
class ProcessingLimitException : public BaseException class ProcessingLimitException : public BaseException {
{
public: public:
ProcessingLimitException(const char *s): ProcessingLimitException(std::string s): BaseException(s) {}
BaseException(s)
{}
}; };
class CommandLineError : public BaseException class CommandLineError : public BaseException {
{
public: public:
CommandLineError(const char *s): CommandLineError(std::string s): BaseException(s) {}
BaseException(s)
{}
}; };
class ItemNotFoundException : public BaseException class ItemNotFoundException : public BaseException {
{
public: public:
ItemNotFoundException(const char *s): ItemNotFoundException(std::string s): BaseException(s) {}
BaseException(s) };
{}
class ServerError : public BaseException {
public:
ServerError(std::string s): BaseException(s) {}
}; };
// Only used on Windows (SEH) // Only used on Windows (SEH)
class FatalSystemException : public BaseException class FatalSystemException : public BaseException {
{
public: public:
FatalSystemException(const char *s): FatalSystemException(std::string s): BaseException(s) {}
BaseException(s)
{}
}; };
/* /*
@ -159,7 +126,7 @@ public:
InvalidPositionException(): InvalidPositionException():
BaseException("Somebody tried to get/set something in a nonexistent position.") BaseException("Somebody tried to get/set something in a nonexistent position.")
{} {}
InvalidPositionException(const char *s): InvalidPositionException(std::string s):
BaseException(s) BaseException(s)
{} {}
}; };

View File

@ -3438,14 +3438,12 @@ void the_game(
L" running a different version of Minetest."; L" running a different version of Minetest.";
errorstream<<wide_to_narrow(error_message)<<std::endl; errorstream<<wide_to_narrow(error_message)<<std::endl;
} }
catch(ServerError &e) catch(ServerError &e) {
{
error_message = narrow_to_wide(e.what()); error_message = narrow_to_wide(e.what());
errorstream<<wide_to_narrow(error_message)<<std::endl; errorstream << "ServerError: " << e.what() << std::endl;
} }
catch(ModError &e) catch(ModError &e) {
{ errorstream << "ModError: " << e.what() << std::endl;
errorstream<<e.what()<<std::endl;
error_message = narrow_to_wide(e.what()) + wgettext("\nCheck debug.txt for details."); error_message = narrow_to_wide(e.what()) + wgettext("\nCheck debug.txt for details.");
} }

View File

@ -653,7 +653,7 @@ ItemStack read_item(lua_State* L, int index,Server* srv)
} }
else else
{ {
throw LuaError(L, "Expecting itemstack, itemstring, table or nil"); throw LuaError(NULL, "Expecting itemstack, itemstring, table or nil");
} }
} }

View File

@ -55,6 +55,18 @@ int script_error_handler(lua_State *L) {
return 1; return 1;
} }
int script_exception_wrapper(lua_State *L, lua_CFunction f)
{
try {
return f(L); // Call wrapped function and return result.
} catch (const char *s) { // Catch and convert exceptions.
lua_pushstring(L, s);
} catch (LuaError& e) {
lua_pushstring(L, e.what());
}
return lua_error(L); // Rethrow as a Lua error.
}
void script_error(lua_State *L) void script_error(lua_State *L)
{ {
const char *s = lua_tostring(L, -1); const char *s = lua_tostring(L, -1);

View File

@ -66,6 +66,7 @@ enum RunCallbacksMode
std::string script_get_backtrace(lua_State *L); std::string script_get_backtrace(lua_State *L);
int script_error_handler(lua_State *L); int script_error_handler(lua_State *L);
int script_exception_wrapper(lua_State *L, lua_CFunction f);
void script_error(lua_State *L); void script_error(lua_State *L);
void script_run_callbacks(lua_State *L, int nargs, void script_run_callbacks(lua_State *L, int nargs,
RunCallbacksMode mode); RunCallbacksMode mode);

View File

@ -23,10 +23,12 @@ with this program; if not, write to the Free Software Foundation, Inc.,
#include "common/c_internal.h" #include "common/c_internal.h"
#include "itemdef.h" #include "itemdef.h"
LuaError::LuaError(lua_State *L, const std::string &s) LuaError::LuaError(lua_State *L, const std::string &s) :
ServerError(s)
{ {
m_s = "LuaError: " + s; if (L) {
if (L) m_s += '\n' + script_get_backtrace(L); m_s += '\n' + script_get_backtrace(L);
}
} }
struct EnumString es_ItemType[] = struct EnumString es_ItemType[] =

View File

@ -26,6 +26,8 @@ extern "C" {
#include <iostream> #include <iostream>
#include "exceptions.h"
struct EnumString struct EnumString
{ {
int num; int num;
@ -50,7 +52,7 @@ public:
} }
}; };
class LuaError : public std::exception class LuaError : public ServerError
{ {
public: public:
LuaError(lua_State *L, const std::string &s); LuaError(lua_State *L, const std::string &s);
@ -61,7 +63,6 @@ public:
{ {
return m_s.c_str(); return m_s.c_str();
} }
std::string m_s;
}; };

View File

@ -29,6 +29,9 @@ with this program; if not, write to the Free Software Foundation, Inc.,
extern "C" { extern "C" {
#include "lualib.h" #include "lualib.h"
#if USE_LUAJIT
#include "luajit.h"
#endif
} }
#include <stdio.h> #include <stdio.h>
@ -73,6 +76,14 @@ ScriptApiBase::ScriptApiBase()
lua_pushlightuserdata(m_luastack, this); lua_pushlightuserdata(m_luastack, this);
lua_setfield(m_luastack, LUA_REGISTRYINDEX, "scriptapi"); lua_setfield(m_luastack, LUA_REGISTRYINDEX, "scriptapi");
// If we are using LuaJIT add a C++ wrapper function to catch
// exceptions thrown in Lua -> C++ calls
#if USE_LUAJIT
lua_pushlightuserdata(m_luastack, (void*) script_exception_wrapper);
luaJIT_setmode(m_luastack, -1, LUAJIT_MODE_WRAPCFUNC | LUAJIT_MODE_ON);
lua_pop(m_luastack, 1);
#endif
m_server = 0; m_server = 0;
m_environment = 0; m_environment = 0;
m_guiengine = 0; m_guiengine = 0;

View File

@ -54,7 +54,7 @@ int ScriptApiDetached::detached_inventory_AllowMove(
if(lua_pcall(L, 7, 1, errorhandler)) if(lua_pcall(L, 7, 1, errorhandler))
scriptError(); scriptError();
if(!lua_isnumber(L, -1)) if(!lua_isnumber(L, -1))
throw LuaError(L, "allow_move should return a number"); throw LuaError(NULL, "allow_move should return a number");
int ret = luaL_checkinteger(L, -1); int ret = luaL_checkinteger(L, -1);
lua_pop(L, 2); // Pop integer and error handler lua_pop(L, 2); // Pop integer and error handler
return ret; return ret;
@ -86,7 +86,7 @@ int ScriptApiDetached::detached_inventory_AllowPut(
if(lua_pcall(L, 5, 1, errorhandler)) if(lua_pcall(L, 5, 1, errorhandler))
scriptError(); scriptError();
if(!lua_isnumber(L, -1)) if(!lua_isnumber(L, -1))
throw LuaError(L, "allow_put should return a number"); throw LuaError(NULL, "allow_put should return a number");
int ret = luaL_checkinteger(L, -1); int ret = luaL_checkinteger(L, -1);
lua_pop(L, 2); // Pop integer and error handler lua_pop(L, 2); // Pop integer and error handler
return ret; return ret;
@ -118,7 +118,7 @@ int ScriptApiDetached::detached_inventory_AllowTake(
if(lua_pcall(L, 5, 1, errorhandler)) if(lua_pcall(L, 5, 1, errorhandler))
scriptError(); scriptError();
if(!lua_isnumber(L, -1)) if(!lua_isnumber(L, -1))
throw LuaError(L, "allow_take should return a number"); throw LuaError(NULL, "allow_take should return a number");
int ret = luaL_checkinteger(L, -1); int ret = luaL_checkinteger(L, -1);
lua_pop(L, 2); // Pop integer and error handler lua_pop(L, 2); // Pop integer and error handler
return ret; return ret;

View File

@ -61,7 +61,7 @@ int ScriptApiNodemeta::nodemeta_inventory_AllowMove(v3s16 p,
scriptError(); scriptError();
lua_remove(L, errorhandler); // Remove error handler lua_remove(L, errorhandler); // Remove error handler
if(!lua_isnumber(L, -1)) if(!lua_isnumber(L, -1))
throw LuaError(L, "allow_metadata_inventory_move should return a number"); throw LuaError(NULL, "allow_metadata_inventory_move should return a number");
int num = luaL_checkinteger(L, -1); int num = luaL_checkinteger(L, -1);
lua_pop(L, 1); // Pop integer lua_pop(L, 1); // Pop integer
return num; return num;
@ -99,7 +99,7 @@ int ScriptApiNodemeta::nodemeta_inventory_AllowPut(v3s16 p,
scriptError(); scriptError();
lua_remove(L, errorhandler); // Remove error handler lua_remove(L, errorhandler); // Remove error handler
if(!lua_isnumber(L, -1)) if(!lua_isnumber(L, -1))
throw LuaError(L, "allow_metadata_inventory_put should return a number"); throw LuaError(NULL, "allow_metadata_inventory_put should return a number");
int num = luaL_checkinteger(L, -1); int num = luaL_checkinteger(L, -1);
lua_pop(L, 1); // Pop integer lua_pop(L, 1); // Pop integer
return num; return num;
@ -137,7 +137,7 @@ int ScriptApiNodemeta::nodemeta_inventory_AllowTake(v3s16 p,
scriptError(); scriptError();
lua_remove(L, errorhandler); // Remove error handler lua_remove(L, errorhandler); // Remove error handler
if(!lua_isnumber(L, -1)) if(!lua_isnumber(L, -1))
throw LuaError(L, "allow_metadata_inventory_take should return a number"); throw LuaError(NULL, "allow_metadata_inventory_take should return a number");
int num = luaL_checkinteger(L, -1); int num = luaL_checkinteger(L, -1);
lua_pop(L, 1); // Pop integer lua_pop(L, 1); // Pop integer
return num; return num;

View File

@ -33,7 +33,7 @@ bool ScriptApiServer::getAuth(const std::string &playername,
getAuthHandler(); getAuthHandler();
lua_getfield(L, -1, "get_auth"); lua_getfield(L, -1, "get_auth");
if(lua_type(L, -1) != LUA_TFUNCTION) if(lua_type(L, -1) != LUA_TFUNCTION)
throw LuaError(L, "Authentication handler missing get_auth"); throw LuaError(NULL, "Authentication handler missing get_auth");
lua_pushstring(L, playername.c_str()); lua_pushstring(L, playername.c_str());
if(lua_pcall(L, 1, 1, errorhandler)) if(lua_pcall(L, 1, 1, errorhandler))
scriptError(); scriptError();
@ -48,13 +48,13 @@ bool ScriptApiServer::getAuth(const std::string &playername,
std::string password; std::string password;
bool found = getstringfield(L, -1, "password", password); bool found = getstringfield(L, -1, "password", password);
if(!found) if(!found)
throw LuaError(L, "Authentication handler didn't return password"); throw LuaError(NULL, "Authentication handler didn't return password");
if(dst_password) if(dst_password)
*dst_password = password; *dst_password = password;
lua_getfield(L, -1, "privileges"); lua_getfield(L, -1, "privileges");
if(!lua_istable(L, -1)) if(!lua_istable(L, -1))
throw LuaError(L, "Authentication handler didn't return privilege table"); throw LuaError(NULL, "Authentication handler didn't return privilege table");
if(dst_privs) if(dst_privs)
readPrivileges(-1, *dst_privs); readPrivileges(-1, *dst_privs);
lua_pop(L, 1); lua_pop(L, 1);
@ -74,7 +74,7 @@ void ScriptApiServer::getAuthHandler()
} }
lua_remove(L, -2); // Remove minetest lua_remove(L, -2); // Remove minetest
if(lua_type(L, -1) != LUA_TTABLE) if(lua_type(L, -1) != LUA_TTABLE)
throw LuaError(L, "Authentication handler table not valid"); throw LuaError(NULL, "Authentication handler table not valid");
} }
void ScriptApiServer::readPrivileges(int index, std::set<std::string> &result) void ScriptApiServer::readPrivileges(int index, std::set<std::string> &result)
@ -108,7 +108,7 @@ void ScriptApiServer::createAuth(const std::string &playername,
lua_getfield(L, -1, "create_auth"); lua_getfield(L, -1, "create_auth");
lua_remove(L, -2); // Remove auth handler lua_remove(L, -2); // Remove auth handler
if(lua_type(L, -1) != LUA_TFUNCTION) if(lua_type(L, -1) != LUA_TFUNCTION)
throw LuaError(L, "Authentication handler missing create_auth"); throw LuaError(NULL, "Authentication handler missing create_auth");
lua_pushstring(L, playername.c_str()); lua_pushstring(L, playername.c_str());
lua_pushstring(L, password.c_str()); lua_pushstring(L, password.c_str());
if(lua_pcall(L, 2, 0, errorhandler)) if(lua_pcall(L, 2, 0, errorhandler))
@ -128,7 +128,7 @@ bool ScriptApiServer::setPassword(const std::string &playername,
lua_getfield(L, -1, "set_password"); lua_getfield(L, -1, "set_password");
lua_remove(L, -2); // Remove auth handler lua_remove(L, -2); // Remove auth handler
if(lua_type(L, -1) != LUA_TFUNCTION) if(lua_type(L, -1) != LUA_TFUNCTION)
throw LuaError(L, "Authentication handler missing set_password"); throw LuaError(NULL, "Authentication handler missing set_password");
lua_pushstring(L, playername.c_str()); lua_pushstring(L, playername.c_str());
lua_pushstring(L, password.c_str()); lua_pushstring(L, password.c_str());
if(lua_pcall(L, 2, 1, errorhandler)) if(lua_pcall(L, 2, 1, errorhandler))

View File

@ -48,7 +48,7 @@ protected:
ScriptApiBase *scriptIface = getScriptApiBase(L); ScriptApiBase *scriptIface = getScriptApiBase(L);
T *scriptIfaceDowncast = dynamic_cast<T*>(scriptIface); T *scriptIfaceDowncast = dynamic_cast<T*>(scriptIface);
if (!scriptIfaceDowncast) { if (!scriptIfaceDowncast) {
throw LuaError(L, "Requested unavailable ScriptApi - core engine bug!"); throw LuaError(NULL, "Requested unavailable ScriptApi - core engine bug!");
} }
return scriptIfaceDowncast; return scriptIfaceDowncast;
} }

View File

@ -150,16 +150,16 @@ int ModApiCraft::l_register_craft(lua_State *L)
if(type == "shaped"){ if(type == "shaped"){
std::string output = getstringfield_default(L, table, "output", ""); std::string output = getstringfield_default(L, table, "output", "");
if(output == "") if(output == "")
throw LuaError(L, "Crafting definition is missing an output"); throw LuaError(NULL, "Crafting definition is missing an output");
int width = 0; int width = 0;
std::vector<std::string> recipe; std::vector<std::string> recipe;
lua_getfield(L, table, "recipe"); lua_getfield(L, table, "recipe");
if(lua_isnil(L, -1)) if(lua_isnil(L, -1))
throw LuaError(L, "Crafting definition is missing a recipe" throw LuaError(NULL, "Crafting definition is missing a recipe"
" (output=\"" + output + "\")"); " (output=\"" + output + "\")");
if(!readCraftRecipeShaped(L, -1, width, recipe)) if(!readCraftRecipeShaped(L, -1, width, recipe))
throw LuaError(L, "Invalid crafting recipe" throw LuaError(NULL, "Invalid crafting recipe"
" (output=\"" + output + "\")"); " (output=\"" + output + "\")");
CraftReplacements replacements; CraftReplacements replacements;
@ -167,7 +167,7 @@ int ModApiCraft::l_register_craft(lua_State *L)
if(!lua_isnil(L, -1)) if(!lua_isnil(L, -1))
{ {
if(!readCraftReplacements(L, -1, replacements)) if(!readCraftReplacements(L, -1, replacements))
throw LuaError(L, "Invalid replacements" throw LuaError(NULL, "Invalid replacements"
" (output=\"" + output + "\")"); " (output=\"" + output + "\")");
} }
@ -181,17 +181,17 @@ int ModApiCraft::l_register_craft(lua_State *L)
else if(type == "shapeless"){ else if(type == "shapeless"){
std::string output = getstringfield_default(L, table, "output", ""); std::string output = getstringfield_default(L, table, "output", "");
if(output == "") if(output == "")
throw LuaError(L, "Crafting definition (shapeless)" throw LuaError(NULL, "Crafting definition (shapeless)"
" is missing an output"); " is missing an output");
std::vector<std::string> recipe; std::vector<std::string> recipe;
lua_getfield(L, table, "recipe"); lua_getfield(L, table, "recipe");
if(lua_isnil(L, -1)) if(lua_isnil(L, -1))
throw LuaError(L, "Crafting definition (shapeless)" throw LuaError(NULL, "Crafting definition (shapeless)"
" is missing a recipe" " is missing a recipe"
" (output=\"" + output + "\")"); " (output=\"" + output + "\")");
if(!readCraftRecipeShapeless(L, -1, recipe)) if(!readCraftRecipeShapeless(L, -1, recipe))
throw LuaError(L, "Invalid crafting recipe" throw LuaError(NULL, "Invalid crafting recipe"
" (output=\"" + output + "\")"); " (output=\"" + output + "\")");
CraftReplacements replacements; CraftReplacements replacements;
@ -199,7 +199,7 @@ int ModApiCraft::l_register_craft(lua_State *L)
if(!lua_isnil(L, -1)) if(!lua_isnil(L, -1))
{ {
if(!readCraftReplacements(L, -1, replacements)) if(!readCraftReplacements(L, -1, replacements))
throw LuaError(L, "Invalid replacements" throw LuaError(NULL, "Invalid replacements"
" (output=\"" + output + "\")"); " (output=\"" + output + "\")");
} }
@ -224,12 +224,12 @@ int ModApiCraft::l_register_craft(lua_State *L)
else if(type == "cooking"){ else if(type == "cooking"){
std::string output = getstringfield_default(L, table, "output", ""); std::string output = getstringfield_default(L, table, "output", "");
if(output == "") if(output == "")
throw LuaError(L, "Crafting definition (cooking)" throw LuaError(NULL, "Crafting definition (cooking)"
" is missing an output"); " is missing an output");
std::string recipe = getstringfield_default(L, table, "recipe", ""); std::string recipe = getstringfield_default(L, table, "recipe", "");
if(recipe == "") if(recipe == "")
throw LuaError(L, "Crafting definition (cooking)" throw LuaError(NULL, "Crafting definition (cooking)"
" is missing a recipe" " is missing a recipe"
" (output=\"" + output + "\")"); " (output=\"" + output + "\")");
@ -240,7 +240,7 @@ int ModApiCraft::l_register_craft(lua_State *L)
if(!lua_isnil(L, -1)) if(!lua_isnil(L, -1))
{ {
if(!readCraftReplacements(L, -1, replacements)) if(!readCraftReplacements(L, -1, replacements))
throw LuaError(L, "Invalid replacements" throw LuaError(NULL, "Invalid replacements"
" (cooking output=\"" + output + "\")"); " (cooking output=\"" + output + "\")");
} }
@ -254,7 +254,7 @@ int ModApiCraft::l_register_craft(lua_State *L)
else if(type == "fuel"){ else if(type == "fuel"){
std::string recipe = getstringfield_default(L, table, "recipe", ""); std::string recipe = getstringfield_default(L, table, "recipe", "");
if(recipe == "") if(recipe == "")
throw LuaError(L, "Crafting definition (fuel)" throw LuaError(NULL, "Crafting definition (fuel)"
" is missing a recipe"); " is missing a recipe");
float burntime = getfloatfield_default(L, table, "burntime", 1.0); float burntime = getfloatfield_default(L, table, "burntime", 1.0);
@ -264,7 +264,7 @@ int ModApiCraft::l_register_craft(lua_State *L)
if(!lua_isnil(L, -1)) if(!lua_isnil(L, -1))
{ {
if(!readCraftReplacements(L, -1, replacements)) if(!readCraftReplacements(L, -1, replacements))
throw LuaError(L, "Invalid replacements" throw LuaError(NULL, "Invalid replacements"
" (fuel recipe=\"" + recipe + "\")"); " (fuel recipe=\"" + recipe + "\")");
} }
@ -274,7 +274,7 @@ int ModApiCraft::l_register_craft(lua_State *L)
} }
else else
{ {
throw LuaError(L, "Unknown crafting definition type: \"" + type + "\""); throw LuaError(NULL, "Unknown crafting definition type: \"" + type + "\"");
} }
lua_pop(L, 1); lua_pop(L, 1);

View File

@ -470,7 +470,7 @@ int ModApiItemMod::l_register_item_raw(lua_State *L)
name = lua_tostring(L, -1); name = lua_tostring(L, -1);
verbosestream<<"register_item_raw: "<<name<<std::endl; verbosestream<<"register_item_raw: "<<name<<std::endl;
} else { } else {
throw LuaError(L, "register_item_raw: name is not defined or not a string"); throw LuaError(NULL, "register_item_raw: name is not defined or not a string");
} }
// Check if on_use is defined // Check if on_use is defined
@ -500,7 +500,7 @@ int ModApiItemMod::l_register_item_raw(lua_State *L)
content_t id = ndef->set(f.name, f); content_t id = ndef->set(f.name, f);
if(id > MAX_REGISTERED_CONTENT){ if(id > MAX_REGISTERED_CONTENT){
throw LuaError(L, "Number of registerable nodes (" throw LuaError(NULL, "Number of registerable nodes ("
+ itos(MAX_REGISTERED_CONTENT+1) + itos(MAX_REGISTERED_CONTENT+1)
+ ") exceeded (" + name + ")"); + ") exceeded (" + name + ")");
} }

View File

@ -330,10 +330,10 @@ int LuaPseudoRandom::l_next(lua_State *L)
max = luaL_checkinteger(L, 3); max = luaL_checkinteger(L, 3);
if(max < min){ if(max < min){
errorstream<<"PseudoRandom.next(): max="<<max<<" min="<<min<<std::endl; errorstream<<"PseudoRandom.next(): max="<<max<<" min="<<min<<std::endl;
throw LuaError(L, "PseudoRandom.next(): max < min"); throw LuaError(NULL, "PseudoRandom.next(): max < min");
} }
if(max - min != 32767 && max - min > 32767/5) if(max - min != 32767 && max - min > 32767/5)
throw LuaError(L, "PseudoRandom.next() max-min is not 32767" throw LuaError(NULL, "PseudoRandom.next() max-min is not 32767"
" and is > 32768/5. This is disallowed due to" " and is > 32768/5. This is disallowed due to"
" the bad random distribution the" " the bad random distribution the"
" implementation would otherwise make."); " implementation would otherwise make.");

View File

@ -55,23 +55,6 @@ class ServerEnvironment;
struct SimpleSoundSpec; struct SimpleSoundSpec;
class ServerError : public std::exception
{
public:
ServerError(const std::string &s)
{
m_s = "ServerError: ";
m_s += s;
}
virtual ~ServerError() throw()
{}
virtual const char * what() const throw()
{
return m_s.c_str();
}
std::string m_s;
};
/* /*
Some random functions Some random functions
*/ */