diff options
author | Benoit Germain <benoit.germain@ubisoft.com> | 2024-05-23 08:35:40 +0200 |
---|---|---|
committer | Benoit Germain <benoit.germain@ubisoft.com> | 2024-05-23 08:35:40 +0200 |
commit | 17ca8b54fdc0aa2df875fc7b80def4dd56959d7f (patch) | |
tree | 38cf24c341cd4165ccf58aac16eaea0725bb11ec | |
parent | d468b693d79ddb416c45669e5ca8400cbdc34e85 (diff) | |
download | lanes-17ca8b54fdc0aa2df875fc7b80def4dd56959d7f.tar.gz lanes-17ca8b54fdc0aa2df875fc7b80def4dd56959d7f.tar.bz2 lanes-17ca8b54fdc0aa2df875fc7b80def4dd56959d7f.zip |
on_state_create receives a string argument, "lane" or "keeper"
Diffstat (limited to '')
-rw-r--r-- | docs/index.html | 14 | ||||
-rw-r--r-- | src/state.cpp | 56 | ||||
-rw-r--r-- | tests/manual_register.lua | 6 | ||||
-rw-r--r-- | tests/package.lua | 8 |
4 files changed, 43 insertions, 41 deletions
diff --git a/docs/index.html b/docs/index.html index d0b06ae..47dc47c 100644 --- a/docs/index.html +++ b/docs/index.html | |||
@@ -400,20 +400,16 @@ | |||
400 | function/<tt>nil</tt> | 400 | function/<tt>nil</tt> |
401 | </td> | 401 | </td> |
402 | <td> | 402 | <td> |
403 | If provided, will be called in every created Lua state right after initializing the base libraries. | 403 | If provided, will be called in every created Lua state right after initializing the base libraries, with a single string argument, either <tt>"lane"</tt> or <tt>"keeper"</tt>.<br/> |
404 | <br /> | 404 | If it is a C function, a C closure will be reconstructed in the created state from the C pointer. Lanes will raise an error if the function has upvalues.<br/> |
405 | Keeper states will call it as well, but only if it is a C function (keeper states are not able to execute any user Lua code). | 405 | Keeper states will call it as well, but only if it is a C function (keeper states are not able to execute any user Lua code).<br/> |
406 | <br /> | ||
407 | Typical usage is twofold: | 406 | Typical usage is twofold: |
408 | <ul> | 407 | <ul> |
409 | <li>Tweak <tt>package.loaders</tt></li> | 408 | <li>Tweak <tt>package.loaders</tt></li> |
410 | <li>Load some additional C functions in the global space (of course only a C function will be able to do this).</li> | 409 | <li>Load some additional C functions in the global space (of course only a C function will be able to do this).</li> |
411 | </ul> | 410 | </ul> |
412 | That way, all changes in the state can be properly taken into account when building the function lookup database. Default is <tt>nil</tt>. | 411 | That way, all changes in the state can be properly taken into account when building the function lookup database. Default is <tt>nil</tt>.<br/> |
413 | <br /> | 412 | If <tt>on_state_create()</tt> is a Lua function, it will be transfered normally before the call.<br/> |
414 | If <tt>on_state_create()</tt> is a Lua function, it will be transfered normally before the call. | ||
415 | <br /> | ||
416 | If it is a C function, a C closure will be reconstructed in the created state from the C pointer. Lanes will raise an error if the function has upvalues. | ||
417 | </td> | 413 | </td> |
418 | </tr> | 414 | </tr> |
419 | 415 | ||
diff --git a/src/state.cpp b/src/state.cpp index 0175449..83b9e85 100644 --- a/src/state.cpp +++ b/src/state.cpp | |||
@@ -207,22 +207,24 @@ static void copy_one_time_settings(Universe* U_, SourceState L1_, DestState L2_) | |||
207 | 207 | ||
208 | // ################################################################################################# | 208 | // ################################################################################################# |
209 | 209 | ||
210 | static constexpr char const* kOnStateCreate{ "on_state_create" }; | ||
211 | |||
210 | void InitializeOnStateCreate(Universe* U_, lua_State* L_) | 212 | void InitializeOnStateCreate(Universe* U_, lua_State* L_) |
211 | { | 213 | { |
212 | STACK_CHECK_START_REL(L_, 1); // L_: settings | 214 | STACK_CHECK_START_REL(L_, 1); // L_: settings |
213 | if (luaG_getfield(L_, -1, "on_state_create") != LuaType::NIL) { // L_: settings on_state_create|nil | 215 | if (luaG_getfield(L_, -1, kOnStateCreate) != LuaType::NIL) { // L_: settings on_state_create|nil |
214 | // store C function pointer in an internal variable | 216 | // store C function pointer in an internal variable |
215 | U_->onStateCreateFunc = lua_tocfunction(L_, -1); // L_: settings on_state_create | 217 | U_->onStateCreateFunc = lua_tocfunction(L_, -1); // L_: settings on_state_create |
216 | if (U_->onStateCreateFunc != nullptr) { | 218 | if (U_->onStateCreateFunc != nullptr) { |
217 | // make sure the function doesn't have upvalues | 219 | // make sure the function doesn't have upvalues |
218 | char const* _upname{ lua_getupvalue(L_, -1, 1) }; // L_: settings on_state_create upval? | 220 | char const* _upname{ lua_getupvalue(L_, -1, 1) }; // L_: settings on_state_create upval? |
219 | if (_upname != nullptr) { // should be "" for C functions with upvalues if any | 221 | if (_upname != nullptr) { // should be "" for C functions with upvalues if any |
220 | raise_luaL_error(L_, "on_state_create shouldn't have upvalues"); | 222 | raise_luaL_error(L_, "%s shouldn't have upvalues", kOnStateCreate); |
221 | } | 223 | } |
222 | // remove this C function from the config table so that it doesn't cause problems | 224 | // remove this C function from the config table so that it doesn't cause problems |
223 | // when we transfer the config table in newly created Lua states | 225 | // when we transfer the config table in newly created Lua states |
224 | lua_pushnil(L_); // L_: settings on_state_create nil | 226 | lua_pushnil(L_); // L_: settings on_state_create nil |
225 | lua_setfield(L_, -3, "on_state_create"); // L_: settings on_state_create | 227 | lua_setfield(L_, -3, kOnStateCreate); // L_: settings on_state_create |
226 | } else { | 228 | } else { |
227 | // optim: store marker saying we have such a function in the config table | 229 | // optim: store marker saying we have such a function in the config table |
228 | U_->onStateCreateFunc = reinterpret_cast<lua_CFunction>(InitializeOnStateCreate); | 230 | U_->onStateCreateFunc = reinterpret_cast<lua_CFunction>(InitializeOnStateCreate); |
@@ -265,31 +267,35 @@ lua_State* create_state([[maybe_unused]] Universe* U_, lua_State* from_) | |||
265 | 267 | ||
266 | void CallOnStateCreate(Universe* U_, lua_State* L_, lua_State* from_, LookupMode mode_) | 268 | void CallOnStateCreate(Universe* U_, lua_State* L_, lua_State* from_, LookupMode mode_) |
267 | { | 269 | { |
268 | if (U_->onStateCreateFunc != nullptr) { | 270 | if (U_->onStateCreateFunc == nullptr) { |
269 | STACK_CHECK_START_REL(L_, 0); | 271 | return; |
270 | DEBUGSPEW_CODE(DebugSpew(U_) << "calling on_state_create()" << std::endl); | 272 | } |
271 | if (U_->onStateCreateFunc != reinterpret_cast<lua_CFunction>(InitializeOnStateCreate)) { | 273 | |
272 | // C function: recreate a closure in the new state, bypassing the lookup scheme | 274 | STACK_CHECK_START_REL(L_, 0); |
273 | lua_pushcfunction(L_, U_->onStateCreateFunc); // on_state_create() | 275 | DEBUGSPEW_CODE(DebugSpew(U_) << "calling on_state_create()" << std::endl); |
274 | } else { // Lua function located in the config table, copied when we opened "lanes.core" | 276 | if (U_->onStateCreateFunc != reinterpret_cast<lua_CFunction>(InitializeOnStateCreate)) { |
275 | if (mode_ != LookupMode::LaneBody) { | 277 | // C function: recreate a closure in the new state, bypassing the lookup scheme |
276 | // if attempting to call in a keeper state, do nothing because the function doesn't exist there | 278 | lua_pushcfunction(L_, U_->onStateCreateFunc); // on_state_create() |
277 | // this doesn't count as an error though | 279 | } else { // Lua function located in the config table, copied when we opened "lanes.core" |
278 | STACK_CHECK(L_, 0); | 280 | if (mode_ != LookupMode::LaneBody) { |
279 | return; | 281 | // if attempting to call in a keeper state, do nothing because the function doesn't exist there |
280 | } | 282 | // this doesn't count as an error though |
281 | kConfigRegKey.pushValue(L_); // L_: {} | 283 | STACK_CHECK(L_, 0); |
282 | STACK_CHECK(L_, 1); | 284 | return; |
283 | std::ignore = luaG_getfield(L_, -1, "on_state_create"); // L_: {} on_state_create() | ||
284 | lua_remove(L_, -2); // L_: on_state_create() | ||
285 | } | 285 | } |
286 | kConfigRegKey.pushValue(L_); // L_: {} | ||
286 | STACK_CHECK(L_, 1); | 287 | STACK_CHECK(L_, 1); |
287 | // capture error and raise it in caller state | 288 | std::ignore = luaG_getfield(L_, -1, kOnStateCreate); // L_: {} on_state_create() |
288 | if (lua_pcall(L_, 0, 0, 0) != LUA_OK) { | 289 | lua_remove(L_, -2); // L_: on_state_create() |
289 | raise_luaL_error(from_, "on_state_create failed: \"%s\"", lua_isstring(L_, -1) ? lua_tostring(L_, -1) : lua_typename(L_, lua_type(L_, -1))); | ||
290 | } | ||
291 | STACK_CHECK(L_, 0); | ||
292 | } | 290 | } |
291 | STACK_CHECK(L_, 1); | ||
292 | // capture error and raise it in caller state | ||
293 | std::string_view const _stateType{ mode_ == LookupMode::LaneBody ? "lane" : "keeper" }; | ||
294 | std::ignore = lua_pushstringview(L_, _stateType); // L_: on_state_create() "<type>" | ||
295 | if (lua_pcall(L_, 1, 0, 0) != LUA_OK) { | ||
296 | raise_luaL_error(from_, "%s failed: \"%s\"", kOnStateCreate, lua_isstring(L_, -1) ? lua_tostring(L_, -1) : lua_typename(L_, lua_type(L_, -1))); | ||
297 | } | ||
298 | STACK_CHECK(L_, 0); | ||
293 | } | 299 | } |
294 | 300 | ||
295 | // ################################################################################################# | 301 | // ################################################################################################# |
diff --git a/tests/manual_register.lua b/tests/manual_register.lua index 23b33c1..a8f2098 100644 --- a/tests/manual_register.lua +++ b/tests/manual_register.lua | |||
@@ -1,7 +1,7 @@ | |||
1 | 1 | ||
2 | local register_func = true | 2 | local register_func = true |
3 | 3 | ||
4 | local RequireAModuleThatExportsGlobalFunctions = function() | 4 | local RequireAModuleThatExportsGlobalFunctions = function(type_) |
5 | -- grab some module that exports C functions, this is good enough for our purpose | 5 | -- grab some module that exports C functions, this is good enough for our purpose |
6 | local lfs = require "lfs" | 6 | local lfs = require "lfs" |
7 | -- make one of these a global | 7 | -- make one of these a global |
@@ -11,14 +11,14 @@ local RequireAModuleThatExportsGlobalFunctions = function() | |||
11 | if register_func then | 11 | if register_func then |
12 | lanes.register( "GlobalFunc", GlobalFunc) | 12 | lanes.register( "GlobalFunc", GlobalFunc) |
13 | end | 13 | end |
14 | print("RequireAModuleThatExportsGlobalFunctions done:", lanes.nameof(GlobalFunc)) | 14 | print(type_, "RequireAModuleThatExportsGlobalFunctions done:", lanes.nameof(GlobalFunc)) |
15 | end | 15 | end |
16 | 16 | ||
17 | 17 | ||
18 | local lanes = require "lanes".configure{ with_timers = false, on_state_create = RequireAModuleThatExportsGlobalFunctions} | 18 | local lanes = require "lanes".configure{ with_timers = false, on_state_create = RequireAModuleThatExportsGlobalFunctions} |
19 | 19 | ||
20 | -- load some module that adds C functions to the global space | 20 | -- load some module that adds C functions to the global space |
21 | RequireAModuleThatExportsGlobalFunctions() | 21 | RequireAModuleThatExportsGlobalFunctions("main") |
22 | 22 | ||
23 | local GlobalFuncUpval = GlobalFunc | 23 | local GlobalFuncUpval = GlobalFunc |
24 | 24 | ||
diff --git a/tests/package.lua b/tests/package.lua index de11c9e..4e8b59f 100644 --- a/tests/package.lua +++ b/tests/package.lua | |||
@@ -2,21 +2,21 @@ local loaders = package.loaders or package.searchers | |||
2 | 2 | ||
3 | assert(nil == loaders[5]) | 3 | assert(nil == loaders[5]) |
4 | 4 | ||
5 | local configure_loaders = function() | 5 | local configure_loaders = function(type_) |
6 | table.insert(loaders, 4, function() end) | 6 | table.insert(loaders, 4, function() end) |
7 | assert(loaders[1]) | 7 | assert(loaders[1]) |
8 | assert(loaders[2]) | 8 | assert(loaders[2]) |
9 | assert(loaders[3]) | 9 | assert(loaders[3]) |
10 | assert(loaders[4]) | 10 | assert(loaders[4]) |
11 | assert(loaders[5]) | 11 | assert(loaders[5]) |
12 | print "loaders configured!" | 12 | print(type_, "loaders configured!") |
13 | end | 13 | end |
14 | 14 | ||
15 | configure_loaders() | 15 | configure_loaders("main") |
16 | 16 | ||
17 | for k,v in pairs(loaders) do | 17 | for k,v in pairs(loaders) do |
18 | print( k, type(v)) | 18 | print( k, type(v)) |
19 | end | 19 | end |
20 | 20 | ||
21 | lanes = require "lanes" | 21 | lanes = require "lanes" |
22 | lanes.configure{with_timers=false, on_state_create = configure_loaders} \ No newline at end of file | 22 | lanes.configure{on_state_create = configure_loaders} \ No newline at end of file |