diff options
author | Nicolas Noble <pixel@nobis-crew.org> | 2013-08-02 15:53:08 -0700 |
---|---|---|
committer | Nicolas Noble <pixel@nobis-crew.org> | 2013-08-02 15:53:08 -0700 |
commit | 903974e7b3ceecb977449ac5ea34808de9501997 (patch) | |
tree | 1c61d4574712a95a106c8647084a95b91d529a5b /src | |
parent | f416d651f3d6551aa0efbcdb8b5838269de9bff3 (diff) |
Heavily revamped the C-to-Lua yielding mechanism. Now more generic.
Diffstat (limited to 'src')
-rw-r--r-- | src/BLua.cc | 201 | ||||
-rw-r--r-- | src/LuaTask.cc | 5 |
2 files changed, 138 insertions, 68 deletions
diff --git a/src/BLua.cc b/src/BLua.cc index 9a2140f..29f4a58 100644 --- a/src/BLua.cc +++ b/src/BLua.cc @@ -2,6 +2,7 @@ #include "BLua.h" #include "Printer.h" #include "Input.h" +#include "Buffer.h" extern "C" { #include <lualib.h> @@ -12,6 +13,16 @@ extern "C" { #define BUFFERSIZE 2048 #endif +namespace { + +class LuaYield : public Balau::GeneralException { + public: + LuaYield(Balau::Future<int> & f) : GeneralException(), m_f(f) { } + Balau::Future<int> m_f; +}; + +}; + namespace Balau { class LuaStatics { @@ -204,15 +215,24 @@ int Balau::LuaStatics::print(lua_State * __L) { return 0; } +static char s_signature = 'B'; + int Balau::LuaStatics::callwrap(lua_State * __L, lua_CFunction func) { Lua L(__L); + int r = 0; try { - return func(__L); + r = func(__L); } catch (LuaException & e) { L.error(String("LuaException: ") + e.getMsg()); } + catch (LuaYield & y) { + Future<int> * f = new Future<int>(y.m_f); + L.push((void *) f); + L.push((void *) &s_signature); + r = L.yield(L.gettop()); + } catch (Balau::GeneralException & e) { L.error(String("GeneralException: ") + e.getMsg()); } @@ -221,7 +241,7 @@ int Balau::LuaStatics::callwrap(lua_State * __L, lua_CFunction func) { // L.error("Unknown C++ exception"); // } - return 0; + return r; } int Balau::LuaStatics::collector(lua_State * __L) { @@ -312,7 +332,7 @@ void Balau::Lua::open_base() { push("print"); push(LuaStatics::print); settable(); - while (n < gettop()) remove(-1); + while (n < gettop()) pop(); push("mkdir"); push(LuaStatics::mkdir); settable(LUA_GLOBALSINDEX); @@ -336,7 +356,7 @@ void Balau::Lua::open_base() { void Balau::Lua::open_table() { int n = gettop(); luaopen_table(L); - while (n < gettop()) remove(-1); + while (n < gettop()) pop(); } void Balau::Lua::open_string() { @@ -345,37 +365,37 @@ void Balau::Lua::open_string() { push("iconv"); push(LuaStatics::iconv); settable(); - while (n < gettop()) remove(-1); + while (n < gettop()) pop(); } void Balau::Lua::open_math() { int n = gettop(); luaopen_math(L); - while (n < gettop()) remove(-1); + while (n < gettop()) pop(); } void Balau::Lua::open_debug() { int n = gettop(); luaopen_debug(L); - while (n < gettop()) remove(-1); + while (n < gettop()) pop(); } void Balau::Lua::open_jit() { int n = gettop(); luaopen_jit(L); - while (n < gettop()) remove(-1); + while (n < gettop()) pop(); } void Balau::Lua::open_ffi() { int n = gettop(); luaopen_ffi(L); - while (n < gettop()) remove(-1); + while (n < gettop()) pop(); } void Balau::Lua::open_bit() { int n = gettop(); luaopen_bit(L); - while (n < gettop()) remove(-1); + while (n < gettop()) pop(); } void Balau::Lua::declareFunc(const char * name, lua_CFunction f, int i) { @@ -564,7 +584,15 @@ void Balau::Lua::load(const String & s, bool docall) throw (GeneralException) { } void Balau::Lua::dumpvars(IO<Handle> h, const String & prefix, int i) { - Task::SimpleContext sc; + if (h.isA<Buffer>()) { + dumpvars_i(h, prefix, i); + } else { + Task::SimpleContext sc; + dumpvars_i(h, prefix, i); + } +} + +void Balau::Lua::dumpvars_i(IO<Handle> h, const String & prefix, int i) { h->writeString(prefix); h->writeString(" = {\n"); dumpvars_r(h, i); @@ -706,10 +734,22 @@ Balau::Lua Balau::Lua::thread(bool saveit) { bool Balau::Lua::resume(int nargs) throw (GeneralException) { int r; + if (resumeC()) { + if (yielded()) { + yieldC(); + return true; + } else { + return false; + } + } + r = lua_resume(L, nargs); - if ((r == 0) || (r == LUA_YIELD)) - return 0; + if (r == LUA_YIELD) + return yieldC(); + + if (r == 0) + return false; pushLuaContext(); showerror(); @@ -732,6 +772,90 @@ bool Balau::Lua::resume(int nargs) throw (GeneralException) { } } +bool Balau::Lua::resumeC() { + if (!yielded()) + return false; + + if (gettop() < 2) + return false; + + if (!islightuserdata(-1)) + return false; + + if (!islightuserdata(-2)) + return false; + + void * s = touserdata(); + if (s != &s_signature) + return false; + + pop(); + + Future<int> * p = (Future<int> *) touserdata(); + Future<int> f(*p); + delete p; + pop(); + + int nargs = 0; + bool yieldedAgain = false; + + try { + nargs = f.get(); + } + catch (LuaException & e) { + error(String("LuaException: ") + e.getMsg()); + } + catch (EAgain & e) { + Future<int> * p = new Future<int>(f); + p->m_evt = e.getEvent(); + push((void *) p); + push((void *) &s_signature); + yieldedAgain = true; + } + catch (LuaYield & y) { + Future<int> * p = new Future<int>(y.m_f); + push((void *) p); + push((void *) &s_signature); + yieldedAgain = true; + } + catch (Balau::GeneralException & e) { + error(String("GeneralException: ") + e.getMsg()); + } + if (!yieldedAgain) + resume(nargs); + return true; +} + +int Balau::Lua::yield(Future<int> f) throw (GeneralException) { + throw LuaYield(f); +} + +bool Balau::Lua::yieldC() throw (GeneralException) { + if (!yielded()) + return true; + + if (gettop() < 2) + return true; + + if (!islightuserdata(-1)) + return true; + + if (!islightuserdata(-2)) + return true; + + void * s = touserdata(); + if (s != &s_signature) + return true; + + Future<int> * p = (Future<int> *) touserdata(-2); + + if (p->m_ranOnce) + throw EAgain(p->m_evt); + + resumeC(); + return yieldC(); +} + void Balau::Lua::showstack(int level) { int n = lua_gettop(L); int i; @@ -923,54 +1047,3 @@ void Balau::LuaHelpersBase::validate(const lua_functypes_t & entry, bool method, } } } - -static char s_signature = 'B'; - -void Balau::LuaHelpersBase::pushContext(Lua & L, std::function<int(Lua & L)> context, Events::BaseEvent * evt) { - L.push(new LuaHelpersBase(context, evt)); - L.push((void *) &s_signature); -} - -bool Balau::LuaHelpersBase::resume(Lua & L) { - if (L.gettop() < 2) - return false; - - if (!L.islightuserdata()) - return false; - - char * sig = (char *) L.touserdata(); - if (sig != &s_signature) - return false; - - L.remove(-1); - - LuaHelpersBase * b = (LuaHelpersBase *) L.touserdata(); - L.remove(-1); - - int r = b->m_context(L); - - delete b; - - if (r < 0) - return true; - - L.resume(r); - - return true; -} - -Balau::Events::BaseEvent * Balau::LuaHelpersBase::getEvent(Lua & L) { - if (L.gettop() < 2) - return NULL; - - if (!L.islightuserdata()) - return NULL; - - char * sig = (char *) L.touserdata(); - if (sig != &s_signature) - return NULL; - - LuaHelpersBase * b = (LuaHelpersBase *) L.touserdata(-2); - - return b->m_evt; -} diff --git a/src/LuaTask.cc b/src/LuaTask.cc index fa94621..5c03521 100644 --- a/src/LuaTask.cc +++ b/src/LuaTask.cc @@ -65,7 +65,7 @@ void Balau::LuaTask::Do() { while(true) { try { if (L.yielded()) - LuaHelpersBase::resume(L); + L.resume(); else m_cell->run(L); } @@ -76,9 +76,6 @@ void Balau::LuaTask::Do() { m_cell->setError(); } if (L.yielded()) { - Events::BaseEvent * evt = LuaHelpersBase::getEvent(L); - IAssert(evt, "We need an event for now here."); - waitFor(evt); yield(); continue; } |