aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorBenoit Germain <benoit.germain@ubisoft.com>2024-05-23 08:35:40 +0200
committerBenoit Germain <benoit.germain@ubisoft.com>2024-05-23 08:35:40 +0200
commit17ca8b54fdc0aa2df875fc7b80def4dd56959d7f (patch)
tree38cf24c341cd4165ccf58aac16eaea0725bb11ec
parentd468b693d79ddb416c45669e5ca8400cbdc34e85 (diff)
downloadlanes-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.html14
-rw-r--r--src/state.cpp56
-rw-r--r--tests/manual_register.lua6
-rw-r--r--tests/package.lua8
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
210static constexpr char const* kOnStateCreate{ "on_state_create" };
211
210void InitializeOnStateCreate(Universe* U_, lua_State* L_) 212void 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
266void CallOnStateCreate(Universe* U_, lua_State* L_, lua_State* from_, LookupMode mode_) 268void 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
2local register_func = true 2local register_func = true
3 3
4local RequireAModuleThatExportsGlobalFunctions = function() 4local 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))
15end 15end
16 16
17 17
18local lanes = require "lanes".configure{ with_timers = false, on_state_create = RequireAModuleThatExportsGlobalFunctions} 18local 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
21RequireAModuleThatExportsGlobalFunctions() 21RequireAModuleThatExportsGlobalFunctions("main")
22 22
23local GlobalFuncUpval = GlobalFunc 23local 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
3assert(nil == loaders[5]) 3assert(nil == loaders[5])
4 4
5local configure_loaders = function() 5local 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!")
13end 13end
14 14
15configure_loaders() 15configure_loaders("main")
16 16
17for k,v in pairs(loaders) do 17for k,v in pairs(loaders) do
18 print( k, type(v)) 18 print( k, type(v))
19end 19end
20 20
21lanes = require "lanes" 21lanes = require "lanes"
22lanes.configure{with_timers=false, on_state_create = configure_loaders} \ No newline at end of file 22lanes.configure{on_state_create = configure_loaders} \ No newline at end of file