From 271ce5cff2b5e9c1e4d4f236c731ceb263716681 Mon Sep 17 00:00:00 2001 From: Benoit Germain Date: Tue, 18 Jun 2024 14:06:07 +0200 Subject: Give a chance to config.allocator to provide a specific allocator for internal stuff --- CHANGES | 2 +- docs/index.html | 2 +- src/allocator.h | 2 ++ src/state.cpp | 19 ++----------------- src/universe.cpp | 35 ++++++++++++++++++++++++++++------- src/universe.h | 1 + 6 files changed, 35 insertions(+), 26 deletions(-) diff --git a/CHANGES b/CHANGES index 15e8d66..f2031f7 100644 --- a/CHANGES +++ b/CHANGES @@ -5,7 +5,7 @@ CHANGE 2: BGe 11-Jun-24 - Version is now 4.0.0 - Lanes configuration settings: - Boolean parameters only accept boolean values. - - allocator provider function is called with a string hint to distinguish lane and keeper states. + - allocator provider function is called with a string hint to distinguish internal allocations, lane and keeper states. - demote_full_userdata removed. Use __lanesconvert instead. - keepers_gc_threshold added. Controls when GC runs inside keepers. - nb_keepers changed to nb_user_keepers. limited to 100 keepers on top of the internal keeper used by the timer Linda. diff --git a/docs/index.html b/docs/index.html index 4a9fc35..ca91273 100644 --- a/docs/index.html +++ b/docs/index.html @@ -289,7 +289,7 @@ If nil, Lua states are created with lua_newstate() and reuse the allocator from the master state.
If "protected", The default allocator obtained from lua_getallocf() in the master state is wrapped inside a critical section and used in all newly created states.
- If a function, this function is called prior to creating the state, with a single string argument, either "keeper" or "lane". It should return a full userdata created as follows: + If a function, this function is called prior to creating the state, with a single string argument, either "internal", "keeper" or "lane". It should return a full userdata created as follows:
diff --git a/src/allocator.h b/src/allocator.h index e09ab38..0505251 100644 --- a/src/allocator.h +++ b/src/allocator.h @@ -1,5 +1,7 @@ #pragma once +#include "compat.h" + // ################################################################################################# namespace lanes { diff --git a/src/state.cpp b/src/state.cpp index 6d3910b..1975148 100644 --- a/src/state.cpp +++ b/src/state.cpp @@ -204,23 +204,8 @@ namespace state { // for some reason, LuaJIT 64 bits does not support creating a state with lua_newstate... return luaL_newstate(); } else { - if (U->provideAllocator != nullptr) { // we have a function we can call to obtain an allocator - STACK_CHECK_START_REL(from, 0); - lua_pushcclosure(from, U->provideAllocator, 0); // L: provideAllocator() - luaG_pushstring(from, hint); // L: provideAllocator() "" - lua_call(from, 1, 1); // L: result - lanes::AllocatorDefinition* const _def{ luaG_tofulluserdata(from, -1) }; - if (!_def || _def->version != lanes::AllocatorDefinition::kAllocatorVersion) { - raise_luaL_error(from, "Bad config.allocator function, must provide a valid AllocatorDefinition"); - } - lua_State* const _L{ lua_newstate(_def->allocF, _def->allocUD) }; - lua_pop(from, 1); // L: - STACK_CHECK(from, 0); - return _L; - } else { - // reuse the allocator provided when the master state was created - return lua_newstate(U->protectedAllocator.allocF, U->protectedAllocator.allocUD); - } + lanes::AllocatorDefinition const _def{ U->resolveAllocator(from, hint) }; + return lua_newstate(_def.allocF, _def.allocUD); } } ) diff --git a/src/universe.cpp b/src/universe.cpp index 770fccf..097c642 100644 --- a/src/universe.cpp +++ b/src/universe.cpp @@ -176,7 +176,7 @@ Universe::Universe() // Do I need to disable this when compiling for LuaJIT to prevent issues? void Universe::initializeAllocatorFunction(lua_State* const L_) { - // start by just grabbing whatever allocator was provided to lua_newstate + // start by just grabbing whatever allocator was provided to the master state protectedAllocator.initFrom(L_); STACK_CHECK_START_REL(L_, 1); // L_: settings switch (luaG_getfield(L_, -1, "allocator")) { // L_: settings allocator|nil|"protected" @@ -212,19 +212,17 @@ void Universe::initializeAllocatorFunction(lua_State* const L_) default: // should be filtered out in lanes.lua raise_luaL_error(L_, "Bad config.allocator type %s", luaG_typename(L_, -1).data()); } - lua_pop(L_, 1); // L_: settings + lua_pop(L_, 1); // L_: settings STACK_CHECK(L_, 1); std::ignore = luaG_getfield(L_, -1, "internal_allocator"); // L_: settings "libc"|"allocator" + LUA_ASSERT(L_, lua_isstring(L_, -1)); // should be the case due to lanes.lua parameter validation std::string_view const _allocator{ luaG_tostring(L_, -1) }; if (_allocator == "libc") { internalAllocator = lanes::AllocatorDefinition{ lanes::AllocatorDefinition::kAllocatorVersion, libc_lua_Alloc, nullptr }; - } else if (provideAllocator == luaG_provide_protected_allocator) { - // user wants mutex protection on the state's allocator. Use protection for our own allocations too, just in case. - internalAllocator = protectedAllocator.makeDefinition(); } else { - // no protection required, just use whatever we have as-is. - internalAllocator = protectedAllocator; + // use whatever the provider provides + internalAllocator = resolveAllocator(L_, "internal"); } lua_pop(L_, 1); // L_: settings STACK_CHECK(L_, 1); @@ -254,6 +252,29 @@ int Universe::InitializeFinalizer(lua_State* const L_) // ################################################################################################# +lanes::AllocatorDefinition Universe::resolveAllocator(lua_State* const L_, std::string_view const& hint_) const +{ + lanes::AllocatorDefinition _ret{ protectedAllocator }; + if (provideAllocator == nullptr) { + return _ret; + } + + STACK_CHECK_START_REL(L_, 0); // here, we have a function we can call to obtain an allocator + lua_pushcclosure(L_, provideAllocator, 0); // L_: provideAllocator() + luaG_pushstring(L_, hint_); // L_: provideAllocator() "" + lua_call(L_, 1, 1); // L_: result + lanes::AllocatorDefinition* const _def{ luaG_tofulluserdata(L_, -1) }; + if (!_def || _def->version != lanes::AllocatorDefinition::kAllocatorVersion) { + raise_luaL_error(L_, "Bad config.allocator function, must provide a valid AllocatorDefinition"); + } + _ret = *_def; + lua_pop(L_, 1); // L_: + STACK_CHECK(L_, 0); + return _ret; +} + +// ################################################################################################# + void Universe::terminateFreeRunningLanes(lua_State* const L_, lua_Duration const shutdownTimeout_, CancelOp const op_) { if (selfdestructFirst != SELFDESTRUCT_END) { diff --git a/src/universe.h b/src/universe.h index d890e5c..6ccdaca 100644 --- a/src/universe.h +++ b/src/universe.h @@ -135,6 +135,7 @@ class Universe [[nodiscard]] static inline Universe* Get(lua_State* L_); void initializeAllocatorFunction(lua_State* L_); static int InitializeFinalizer(lua_State* L_); + lanes::AllocatorDefinition resolveAllocator(lua_State* const L_, std::string_view const& hint_) const; static inline void Store(lua_State* L_, Universe* U_); void terminateFreeRunningLanes(lua_State* L_, lua_Duration shutdownTimeout_, CancelOp op_); }; -- cgit v1.2.3-55-g6feb