ability for functions that throw not bound by sol to catch and return lua errors that can be used by a handler function (for protected function).

This commit is contained in:
ThePhD 2016-03-02 20:45:52 -05:00
parent 6c070661a7
commit 923c774d5e
11 changed files with 65 additions and 42 deletions

View File

@ -131,7 +131,7 @@ script:
- echo "Configuration info:" - echo "Configuration info:"
- export_compiler_vars - export_compiler_vars
- ninja --version - ninja --version
- ./bootstrap.py --ci && ninja - ./bootstrap.py --ci && ninja && ninja examples
notifications: notifications:
email: email:

View File

@ -1,5 +1,6 @@
#include <sol.hpp> #include <sol.hpp>
#include <cassert> #include <cassert>
#include <iostream>
inline int my_add(int x, int y) { inline int my_add(int x, int y) {
return x + y; return x + y;
@ -46,17 +47,27 @@ int main() {
// calling a stateful lambda modifies the value // calling a stateful lambda modifies the value
lua.script("inc()"); lua.script("inc()");
assert(x == 10); assert(x == 10);
if (x == 10) {
// Do something based on this information
std::cout << "Yahoo!" << std::endl;
}
// this can be done as many times as you want // this can be done as many times as you want
lua.script("inc()\ninc()\ninc()"); lua.script("inc()\ninc()\ninc()");
assert(x == 40); assert(x == 40);
if (x == 40) {
// Do something based on this information
std::cout << "Yahoo!" << std::endl;
}
// retrieval of a function is done similarly // retrieval of a function is done similarly
// to other variables, using sol::function // to other variables, using sol::function
sol::function add = lua["my_add"]; sol::function add = lua["my_add"];
int value = add(10, 11); int value = add(10, 11);
assert(add.call<int>(10, 11) == 21); assert(add.call<int>(10, 11) == 21);
assert(value == 21); assert(value == 21);
if (value == 21) {
std::cout << "Woo, it's 21!" << std::endl;
}
// multi-return functions are supported using // multi-return functions are supported using
// std::tuple as the interface. // std::tuple as the interface.
@ -76,8 +87,7 @@ int main() {
// you can even overload functions // you can even overload functions
// just pass in the different functions // just pass in the different functions
// you want to pack into a single name: // you want to pack into a single name:
// make SURE they take different number of // make SURE they take different types!
// arguments/different types!
lua.set_function("func", sol::overload([](int x) { return x; }, make_string, my_add)); lua.set_function("func", sol::overload([](int x) { return x; }, make_string, my_add));

View File

@ -18,9 +18,6 @@ int main() {
"}"); "}");
auto t2 = lua.get<sol::table>("table2");
auto nestedTable = t2.get<sol::table>("nestedTable");
/* Shorter Syntax: */ /* Shorter Syntax: */
// using the values stored in table1 // using the values stored in table1
/*std::cout << (std::string)lua["table1"][1] << " " /*std::cout << (std::string)lua["table1"][1] << " "
@ -29,18 +26,19 @@ int main() {
// some retrieval of values from the nested table // some retrieval of values from the nested table
// the cleaner way of doing things // the cleaner way of doing things
// chain off the the get<>() / [] results // chain off the the get<>() / [] results
auto t2 = lua.get<sol::table>("table2");
auto nestedTable = t2.get<sol::table>("nestedTable");
// Alternatively:
//sol::table t2 = lua["table2"];
//sol::table nestedTable = t2["nestedTable"];
std::string x = lua["table2"]["nestedTable"]["key2"]; std::string x = lua["table2"]["nestedTable"]["key2"];
std::cout << "nested table: key1 : " << nestedTable.get<std::string>("key1") << ", key2: " std::cout << "nested table: key1 : " << nestedTable.get<std::string>("key1") << ", key2: "
<< x << x
<< '\n'; << '\n';
std::cout << "name of t2: " << t2.get<std::string>("name") << '\n'; std::cout << "name of t2: " << t2.get<std::string>("name") << '\n';
#ifndef _MSC_VER std::string t2name = t2["name"];
// VC++ has a bug in its implementation and std::cout << "name of t2: " << t2name << '\n';
// I've filed a bug against the compiler at Microsoft,
// but the following
// works on g++ and clang++
std::cout << "name of t2: " << (std::string)t2["name"] << '\n';
#endif // VC++ being a dumb
/* Longer Syntax: */ /* Longer Syntax: */
// using the values stored in table1 // using the values stored in table1

View File

@ -60,7 +60,7 @@ int main() {
"y = x:test(10)"); "y = x:test(10)");
auto y = lua.get<int>("y"); auto y = lua.get<int>("y");
assert(y == 14); std::cout << y << std::endl; // show 14
// if you want a class to have more than one constructor // if you want a class to have more than one constructor
// the way to do so is through set_userdata and creating // the way to do so is through set_userdata and creating

View File

@ -1,5 +1,5 @@
#include <sol.hpp> #include <sol.hpp>
#include <cassert> #include <iostream>
int main() { int main() {
sol::state lua; sol::state lua;
@ -26,9 +26,9 @@ int main() {
std::string y2; std::string y2;
std::tie(x2, y2) = lua.get<int, std::string>("x", "y"); std::tie(x2, y2) = lua.get<int, std::string>("x", "y");
// assert the values // show the values
assert(x == 10); std::cout << x << std::endl;
assert(y == "hello"); std::cout << y << std::endl;
assert(x2 == 10); std::cout << x2 << std::endl;
assert(y2 == "hello"); std::cout << y2 << std::endl;
} }

View File

@ -163,26 +163,33 @@ private:
int firstreturn = std::max(1, stacksize - static_cast<int>(n) - 1); int firstreturn = std::max(1, stacksize - static_cast<int>(n) - 1);
int returncount = 0; int returncount = 0;
call_status code = call_status::ok; call_status code = call_status::ok;
#ifdef SOL_NO_EXCEPTIONS #ifndef SOL_NO_EXCEPTIONS
code = static_cast<call_status>(luacall(n, LUA_MULTRET, h)); auto onexcept = [&](const char* error) {
int poststacksize = lua_gettop(lua_state()); h.stackindex = 0;
returncount = poststacksize - firstreturn; if (h.target.valid()) {
#else h.target.push();
stack::push(lua_state(), error);
lua_call(lua_state(), 1, 1);
}
else {
stack::push(lua_state(), error);
}
};
try { try {
#endif // No Exceptions
code = static_cast<call_status>(luacall(n, LUA_MULTRET, h)); code = static_cast<call_status>(luacall(n, LUA_MULTRET, h));
int poststacksize = lua_gettop(lua_state()); int poststacksize = lua_gettop(lua_state());
returncount = poststacksize - firstreturn; returncount = poststacksize - firstreturn;
#ifndef SOL_NO_EXCEPTIONS
} }
// Handle C++ errors thrown from C++ functions bound inside of lua // Handle C++ errors thrown from C++ functions bound inside of lua
catch (const char* error) { catch (const char* error) {
h.stackindex = 0; onexcept(error);
stack::push(lua_state(), error);
firstreturn = lua_gettop(lua_state()); firstreturn = lua_gettop(lua_state());
return protected_function_result(lua_state(), firstreturn, 0, 1, call_status::runtime); return protected_function_result(lua_state(), firstreturn, 0, 1, call_status::runtime);
} }
catch (const std::exception& error) { catch (const std::exception& error) {
h.stackindex = 0; onexcept(error.what());
stack::push(lua_state(), error.what());
firstreturn = lua_gettop(lua_state()); firstreturn = lua_gettop(lua_state());
return protected_function_result(lua_state(), firstreturn, 0, 1, call_status::runtime); return protected_function_result(lua_state(), firstreturn, 0, 1, call_status::runtime);
} }
@ -347,7 +354,7 @@ struct pusher<function_sig<Sigs...>> {
template<typename... Args> template<typename... Args>
static int push(lua_State* L, Args&&... args) { static int push(lua_State* L, Args&&... args) {
// Set will always place one thing (function) on the stack // Set will always place one thing (function) on the stack
set(L, std::forward<Args>(args)...); set<Sigs...>(L, std::forward<Args>(args)...);
return 1; return 1;
} }
}; };

View File

@ -35,8 +35,6 @@ public:
template<typename T> template<typename T>
decltype(auto) as() const { decltype(auto) as() const {
push(); push();
type actual = stack::get<type>(lua_state());
type_assert(lua_state(), -1, type_of<T>(), actual);
return stack::pop<T>(lua_state()); return stack::pop<T>(lua_state());
} }

View File

@ -313,10 +313,7 @@ struct getter<const char*> {
template<> template<>
struct getter<nil_t> { struct getter<nil_t> {
static nil_t get(lua_State* L, int index = -1) { static nil_t get(lua_State*, int = -1) {
if(lua_isnil(L, index) == 0) {
throw error("not nil");
}
return nil_t{ }; return nil_t{ };
} }
}; };
@ -869,7 +866,6 @@ inline int call_into_lua(types<Ret0, Ret...>, types<Args...> ta, Fx&& fx, lua_St
} }
inline call_syntax get_call_syntax(lua_State* L, const std::string& meta) { inline call_syntax get_call_syntax(lua_State* L, const std::string& meta) {
type metatype = stack::get<type>(L);
luaL_getmetatable(L, meta.c_str()); luaL_getmetatable(L, meta.c_str());
if (lua_compare(L, -1, -2, LUA_OPEQ) == 1) { if (lua_compare(L, -1, -2, LUA_OPEQ) == 1) {
lua_pop(L, 1); lua_pop(L, 1);

View File

@ -29,9 +29,13 @@
namespace sol { namespace sol {
namespace detail { namespace detail {
inline int atpanic(lua_State* L) { inline int atpanic(lua_State* L) {
#ifdef SOL_NO_EXCEPTIONS
(void)L;
#else
const char* message = lua_tostring(L, -1); const char* message = lua_tostring(L, -1);
std::string err = message ? message : "An unexpected error occurred and forced the lua state to call atpanic"; std::string err = message ? message : "An unexpected error occurred and forced the lua state to call atpanic";
throw error(err); throw error(err);
#endif
} }
} // detail } // detail

View File

@ -237,7 +237,7 @@ private:
// Insert bubble to keep with compile-time argument count (simpler and cheaper to do) // Insert bubble to keep with compile-time argument count (simpler and cheaper to do)
functions.push_back(nullptr); functions.push_back(nullptr);
constructfunc = function_detail::construct<T, Args...>; constructfunc = function_detail::construct<T, Args...>;
metafunctiontable.push_back({ functionnames.back().c_str(), constructfunc }); metafunctiontable.push_back({ name.c_str(), constructfunc });
} }
template<std::size_t N> template<std::size_t N>

View File

@ -1193,12 +1193,16 @@ TEST_CASE( "functions/function_result-protected_function_result", "Function resu
+ "end" + "end"
); );
lua.set("nontrampoline", (lua_CFunction)[](lua_State*) -> int { throw "x";});
sol::protected_function doom = lua[ "doom" ]; sol::protected_function doom = lua[ "doom" ];
sol::protected_function luadoom = lua[ "luadoom" ]; sol::protected_function luadoom = lua["luadoom"];
sol::function luahandler = lua[ "luahandler" ]; sol::protected_function nontrampoline = lua["nontrampoline"];
sol::function luahandler = lua["luahandler"];
sol::function cpphandler = lua[ "cpphandler" ]; sol::function cpphandler = lua[ "cpphandler" ];
doom.error_handler = luahandler; doom.error_handler = luahandler;
luadoom.error_handler = cpphandler; luadoom.error_handler = cpphandler;
nontrampoline.error_handler = cpphandler;
{ {
sol::protected_function_result result = doom(); sol::protected_function_result result = doom();
@ -1212,6 +1216,12 @@ TEST_CASE( "functions/function_result-protected_function_result", "Function resu
std::string errorstring = result; std::string errorstring = result;
REQUIRE(errorstring == handlederrormessage); REQUIRE(errorstring == handlederrormessage);
} }
{
sol::protected_function_result result = nontrampoline();
REQUIRE(!result.valid());
std::string errorstring = result;
REQUIRE(errorstring == handlederrormessage);
}
} }
TEST_CASE("functions/destructor-tests", "Show that proper copies / destruction happens") { TEST_CASE("functions/destructor-tests", "Show that proper copies / destruction happens") {
@ -1289,7 +1299,7 @@ TEST_CASE("functions/destructor-tests", "Show that proper copies / destruction h
REQUIRE(last_call == static_call); REQUIRE(last_call == static_call);
} }
REQUIRE(created == 1); REQUIRE(created == 1);
REQUIRE(destroyed == 2); REQUIRE(destroyed == 1);
} }
} }