From c21b52ad18bcabf084f8dba40d1a09d9e52051bd Mon Sep 17 00:00:00 2001 From: Benoit Germain Date: Mon, 15 Apr 2024 15:46:36 +0200 Subject: C++ migration: all linda operations go through a different lua_CFunction so as not to defeat function lookup --- Makefile | 105 +++++---- src/deep.cpp | 6 +- src/keeper.cpp | 2 +- src/linda.cpp | 734 +++++++++++++++++++++++++++++---------------------------- 4 files changed, 427 insertions(+), 420 deletions(-) diff --git a/Makefile b/Makefile index 79d7dd9..493489a 100644 --- a/Makefile +++ b/Makefile @@ -93,110 +93,113 @@ test: $(MAKE) rupval $(MAKE) timer -basic: tests/basic.lua $(_TARGET_SO) - $(_PREFIX) $(LUA) $< - -cancel: tests/cancel.lua $(_TARGET_SO) - $(_PREFIX) $(LUA) $< - -# -# This tries to show out a bug which happens in lane cleanup (multicore CPU's only) -# -REP_ARGS=-llanes -e "print'say aaa'; for i=1,10 do print(i) end" -repetitive: $(_TARGET_SO) - for i in 1 2 3 4 5 6 7 8 9 10 a b c d e f g h i j k l m n o p q r s t u v w x y z; \ - do $(_PREFIX) $(LUA) $(REP_ARGS); \ - done - -repetitive1: $(_TARGET_SO) - $(_PREFIX) $(LUA) $(REP_ARGS) - -fifo: tests/fifo.lua $(_TARGET_SO) +appendud: tests/appendud.lua $(_TARGET_SO) $(_PREFIX) $(LUA) $< -keeper: tests/keeper.lua $(_TARGET_SO) +atexit: tests/atexit.lua $(_TARGET_SO) $(_PREFIX) $(LUA) $< -fibonacci: tests/fibonacci.lua $(_TARGET_SO) +atomic: tests/atomic.lua $(_TARGET_SO) $(_PREFIX) $(LUA) $< -timer: tests/timer.lua $(_TARGET_SO) +basic: tests/basic.lua $(_TARGET_SO) $(_PREFIX) $(LUA) $< -atomic: tests/atomic.lua $(_TARGET_SO) +cancel: tests/cancel.lua $(_TARGET_SO) $(_PREFIX) $(LUA) $< cyclic: tests/cyclic.lua $(_TARGET_SO) $(_PREFIX) $(LUA) $< -recursive: tests/recursive.lua $(_TARGET_SO) - $(_PREFIX) $(LUA) $< - -hangtest: tests/hangtest.lua $(_TARGET_SO) +deadlock: tests/deadlock.lua $(_TARGET_SO) $(_PREFIX) $(LUA) $< ehynes: tests/ehynes.lua $(_TARGET_SO) $(_PREFIX) $(LUA) $< -#require: tests/require.lua $(_TARGET_SO) -# $(_PREFIX) $(LUA) $< +errhangtest: tests/errhangtest.lua $(_TARGET_SO) + $(_PREFIX) $(LUA) $< -objects: tests/objects.lua $(_TARGET_SO) +error-test: tests/error.lua $(_TARGET_SO) $(_PREFIX) $(LUA) $< -irayo_recursive: tests/irayo_recursive.lua $(_TARGET_SO) +fibonacci: tests/fibonacci.lua $(_TARGET_SO) $(_PREFIX) $(LUA) $< -irayo_closure: tests/irayo_closure.lua $(_TARGET_SO) +fifo: tests/fifo.lua $(_TARGET_SO) $(_PREFIX) $(LUA) $< finalizer: tests/finalizer.lua $(_TARGET_SO) $(_PREFIX) $(LUA) $< -errhangtest: tests/errhangtest.lua $(_TARGET_SO) +func_is_string: tests/func_is_string.lua $(_TARGET_SO) $(_PREFIX) $(LUA) $< -error-test: tests/error.lua $(_TARGET_SO) +hangtest: tests/hangtest.lua $(_TARGET_SO) $(_PREFIX) $(LUA) $< -appendud: tests/appendud.lua $(_TARGET_SO) +irayo_closure: tests/irayo_closure.lua $(_TARGET_SO) $(_PREFIX) $(LUA) $< -func_is_string: tests/func_is_string.lua $(_TARGET_SO) +irayo_recursive: tests/irayo_recursive.lua $(_TARGET_SO) $(_PREFIX) $(LUA) $< -linda_perf: tests/linda_perf.lua $(_TARGET_SO) +keeper: tests/keeper.lua $(_TARGET_SO) $(_PREFIX) $(LUA) $< -atexit: tests/atexit.lua $(_TARGET_SO) - $(_PREFIX) $(LUA) $< +launchtest: tests/launchtest.lua $(_TARGET_SO) + $(MAKE) _perftest ARGS="$< $(N)" -rupval: tests/rupval.lua $(_TARGET_SO) +linda_perf: tests/linda_perf.lua $(_TARGET_SO) $(_PREFIX) $(LUA) $< -package: tests/package.lua $(_TARGET_SO) +objects: tests/objects.lua $(_TARGET_SO) $(_PREFIX) $(LUA) $< -pingpong: tests/pingpong.lua $(_TARGET_SO) +package: tests/package.lua $(_TARGET_SO) $(_PREFIX) $(LUA) $< -#--- -perftest-plain: tests/perftest.lua $(_TARGET_SO) - $(MAKE) _perftest ARGS="$< $(N) -plain" - perftest: tests/perftest.lua $(_TARGET_SO) $(MAKE) _perftest ARGS="$< $(N)" +perftest-even: tests/perftest.lua $(_TARGET_SO) + $(MAKE) _perftest ARGS="$< $(N) -prio=-2" + perftest-odd: tests/perftest.lua $(_TARGET_SO) $(MAKE) _perftest ARGS="$< $(N) -prio=+2" -perftest-even: tests/perftest.lua $(_TARGET_SO) - $(MAKE) _perftest ARGS="$< $(N) -prio=-2" +perftest-plain: tests/perftest.lua $(_TARGET_SO) + $(MAKE) _perftest ARGS="$< $(N) -plain" -#--- -launchtest: tests/launchtest.lua $(_TARGET_SO) - $(MAKE) _perftest ARGS="$< $(N)" +pingpong: tests/pingpong.lua $(_TARGET_SO) + $(_PREFIX) $(LUA) $< +recursive: tests/recursive.lua $(_TARGET_SO) + $(_PREFIX) $(LUA) $< + +# +# This tries to show out a bug which happens in lane cleanup (multicore CPU's only) +# +REP_ARGS=-llanes -e "print'say aaa'; for i=1,10 do print(i) end" +repetitive: $(_TARGET_SO) + for i in 1 2 3 4 5 6 7 8 9 10 a b c d e f g h i j k l m n o p q r s t u v w x y z; \ + do $(_PREFIX) $(LUA) $(REP_ARGS); \ + done + +repetitive1: $(_TARGET_SO) + $(_PREFIX) $(LUA) $(REP_ARGS) + +#require: tests/require.lua $(_TARGET_SO) +# $(_PREFIX) $(LUA) $< + +rupval: tests/rupval.lua $(_TARGET_SO) + $(_PREFIX) $(LUA) $< + +timer: tests/timer.lua $(_TARGET_SO) + $(_PREFIX) $(LUA) $< + +#--- +#--- _perftest: $(_PREFIX) $(TIME) $(LUA) $(ARGS) diff --git a/src/deep.cpp b/src/deep.cpp index 780c86e..3b0e015 100644 --- a/src/deep.cpp +++ b/src/deep.cpp @@ -223,13 +223,13 @@ char const* DeepFactory::PushDeepProxy(Dest L, DeepPrelude* prelude, int nuv_, L if( lua_isnil( L, -1)) // // No metatable yet. { - int oldtop = lua_gettop( L); // DPC proxy nil - lua_pop( L, 1); // DPC proxy + lua_pop(L, 1); // DPC proxy + int const oldtop{ lua_gettop(L) }; // 1 - make one and register it if (mode_ != LookupMode::ToKeeper) { factory.createMetatable(L); // DPC proxy metatable - if (lua_gettop(L) - oldtop != 0 || !lua_istable(L, -1)) + if (lua_gettop(L) - oldtop != 1 || !lua_istable(L, -1)) { // factory didn't push exactly 1 value, or the value it pushed is not a table: ERROR! lua_settop( L, oldtop); // DPC proxy X diff --git a/src/keeper.cpp b/src/keeper.cpp index 36733e3..650789b 100644 --- a/src/keeper.cpp +++ b/src/keeper.cpp @@ -744,7 +744,7 @@ void init_keepers(Universe* U, lua_State* L) // ################################################################################################## -// should be called only when inside a keeper_acquire/keeper_release pair (see linda_protected_call) +// should be called only when inside a keeper_acquire/keeper_release pair (see Linda::ProtectedCall) Keeper* which_keeper(Keepers* keepers_, uintptr_t magic_) { int const nbKeepers{ keepers_->nb_keepers }; diff --git a/src/linda.cpp b/src/linda.cpp index 103f4ed..8005211 100644 --- a/src/linda.cpp +++ b/src/linda.cpp @@ -57,6 +57,8 @@ class LindaFactory : public DeepFactory // I'm not totally happy with having a global variable. But since it's stateless, it will do for the time being. static LindaFactory g_LindaFactory; +// ################################################################################################# + /* * Actual data is kept within a keeper state, which is hashed by the 'Linda' * pointer (which is same to all userdatas pointing to it). @@ -110,7 +112,9 @@ class Linda : public DeepPrelude // Deep userdata MUST start with this header } } - private: + static int ProtectedCall(lua_State* L, lua_CFunction f_); + + private : void setName(char const* name_, size_t len_) { @@ -203,7 +207,8 @@ static void check_key_types(lua_State* L, int start_, int end_) // ################################################################################################# -LUAG_FUNC(linda_protected_call) +// used to perform all linda operations that access keepers +int Linda::ProtectedCall(lua_State* L, lua_CFunction f_) { Linda* const linda{ ToLinda(L, 1) }; @@ -215,8 +220,8 @@ LUAG_FUNC(linda_protected_call) // if we didn't do anything wrong, the keeper stack should be clean ASSERT_L(lua_gettop(KL) == 0); - // retrieve the actual function to be called and move it before the arguments - lua_pushvalue(L, lua_upvalueindex(1)); + // push the function to be called and move it before the arguments + lua_pushcfunction(L, f_); lua_insert(L, 1); // do a protected call int const rc{ lua_pcall(L, lua_gettop(L) - 1, LUA_MULTRET, 0) }; @@ -248,62 +253,232 @@ LUAG_FUNC(linda_protected_call) */ LUAG_FUNC(linda_send) { - Linda* const linda{ ToLinda(L, 1) }; - std::chrono::time_point until{ std::chrono::time_point::max() }; - int key_i{ 2 }; // index of first key, if timeout not there - - if (lua_type(L, 2) == LUA_TNUMBER) // we don't want to use lua_isnumber() because of autocoercion + auto send = [](lua_State* L) { - lua_Duration const duration{ lua_tonumber(L, 2) }; - if (duration.count() >= 0.0) + Linda* const linda{ ToLinda(L, 1) }; + std::chrono::time_point until{ std::chrono::time_point::max() }; + int key_i{ 2 }; // index of first key, if timeout not there + + if (lua_type(L, 2) == LUA_TNUMBER) // we don't want to use lua_isnumber() because of autocoercion + { + lua_Duration const duration{ lua_tonumber(L, 2) }; + if (duration.count() >= 0.0) + { + until = std::chrono::steady_clock::now() + std::chrono::duration_cast(duration); + } + ++key_i; + } + else if (lua_isnil(L, 2)) // alternate explicit "infinite timeout" by passing nil before the key { - until = std::chrono::steady_clock::now() + std::chrono::duration_cast(duration); + ++key_i; } - ++key_i; - } - else if (lua_isnil(L, 2)) // alternate explicit "infinite timeout" by passing nil before the key - { - ++key_i; - } - bool const as_nil_sentinel{ NIL_SENTINEL.equals(L, key_i) }; // if not nullptr, send() will silently send a single nil if nothing is provided - if (as_nil_sentinel) - { - // the real key to send data to is after the NIL_SENTINEL marker - ++key_i; - } + bool const as_nil_sentinel{ NIL_SENTINEL.equals(L, key_i) }; // if not nullptr, send() will silently send a single nil if nothing is provided + if (as_nil_sentinel) + { + // the real key to send data to is after the NIL_SENTINEL marker + ++key_i; + } + + // make sure the key is of a valid type + check_key_types(L, key_i, key_i); + + STACK_GROW(L, 1); + + // make sure there is something to send + if (lua_gettop(L) == key_i) + { + if (as_nil_sentinel) + { + // send a single nil if nothing is provided + NIL_SENTINEL.pushKey(L); + } + else + { + return luaL_error(L, "no data to send"); + } + } + + // convert nils to some special non-nil sentinel in sent values + keeper_toggle_nil_sentinels(L, key_i + 1, LookupMode::ToKeeper); + bool ret{ false }; + CancelRequest cancel{ CancelRequest::None }; + KeeperCallResult pushed; + { + Lane* const lane{ LANE_POINTER_REGKEY.readLightUserDataValue(L) }; + Keeper* const K{ which_keeper(linda->U->keepers, linda->hashSeed()) }; + lua_State* const KL{ K ? K->L : nullptr }; + if (KL == nullptr) + return 0; + + STACK_CHECK_START_REL(KL, 0); + for (bool try_again{ true };;) + { + if (lane != nullptr) + { + cancel = lane->cancel_request; + } + cancel = (cancel != CancelRequest::None) ? cancel : linda->simulate_cancel; + // if user wants to cancel, or looped because of a timeout, the call returns without sending anything + if (!try_again || cancel != CancelRequest::None) + { + pushed.emplace(0); + break; + } + + STACK_CHECK(KL, 0); + pushed = keeper_call(linda->U, KL, KEEPER_API(send), L, linda, key_i); + if (!pushed.has_value()) + { + break; + } + ASSERT_L(pushed.value() == 1); + + ret = lua_toboolean(L, -1) ? true : false; + lua_pop(L, 1); - // make sure the key is of a valid type - check_key_types(L, key_i, key_i); + if (ret) + { + // Wake up ALL waiting threads + linda->m_write_happened.notify_all(); + break; + } - STACK_GROW(L, 1); + // instant timout to bypass the wait syscall + if (std::chrono::steady_clock::now() >= until) + { + break; /* no wait; instant timeout */ + } - // make sure there is something to send - if (lua_gettop(L) == key_i) + // storage limit hit, wait until timeout or signalled that we should try again + { + Lane::Status prev_status{ Lane::Error }; // prevent 'might be used uninitialized' warnings + if (lane != nullptr) + { + // change status of lane to "waiting" + prev_status = lane->m_status; // Running, most likely + ASSERT_L(prev_status == Lane::Running); // but check, just in case + lane->m_status = Lane::Waiting; + ASSERT_L(lane->m_waiting_on == nullptr); + lane->m_waiting_on = &linda->m_read_happened; + } + // could not send because no room: wait until some data was read before trying again, or until timeout is reached + std::unique_lock keeper_lock{ K->m_mutex, std::adopt_lock }; + std::cv_status const status{ linda->m_read_happened.wait_until(keeper_lock, until) }; + keeper_lock.release(); // we don't want to release the lock! + try_again = (status == std::cv_status::no_timeout); // detect spurious wakeups + if (lane != nullptr) + { + lane->m_waiting_on = nullptr; + lane->m_status = prev_status; + } + } + } + STACK_CHECK(KL, 0); + } + + if (!pushed.has_value()) + { + luaL_error(L, "tried to copy unsupported types"); // doesn't return + } + + switch (cancel) + { + case CancelRequest::Soft: + // if user wants to soft-cancel, the call returns lanes.cancel_error + CANCEL_ERROR.pushKey(L); + return 1; + + case CancelRequest::Hard: + // raise an error interrupting execution only in case of hard cancel + raise_cancel_error(L); // raises an error and doesn't return + + default: + lua_pushboolean(L, ret); // true (success) or false (timeout) + return 1; + } + }; + return Linda::ProtectedCall(L, send); +} + +// ################################################################################################# + +/* + * 2 modes of operation + * [val, key]= linda_receive( linda_ud, [timeout_secs_num=-1], key_num|str|bool|lightuserdata [, ...] ) + * Consumes a single value from the Linda, in any key. + * Returns: received value (which is consumed from the slot), and the key which had it + + * [val1, ... valCOUNT]= linda_receive( linda_ud, [timeout_secs_num=-1], linda.batched, key_num|str|bool|lightuserdata, min_COUNT[, max_COUNT]) + * Consumes between min_COUNT and max_COUNT values from the linda, from a single key. + * returns the actual consumed values, or nil if there weren't enough values to consume + * + */ +LUAG_FUNC(linda_receive) +{ + auto receive = [](lua_State* L) { - if (as_nil_sentinel) + Linda* const linda{ ToLinda(L, 1) }; + std::chrono::time_point until{ std::chrono::time_point::max() }; + int key_i{ 2 }; // index of first key, if timeout not there + + if (lua_type(L, 2) == LUA_TNUMBER) // we don't want to use lua_isnumber() because of autocoercion + { + lua_Duration const duration{ lua_tonumber(L, 2) }; + if (duration.count() >= 0.0) + { + until = std::chrono::steady_clock::now() + std::chrono::duration_cast(duration); + } + ++key_i; + } + else if (lua_isnil(L, 2)) // alternate explicit "infinite timeout" by passing nil before the key { - // send a single nil if nothing is provided - NIL_SENTINEL.pushKey(L); + ++key_i; + } + + keeper_api_t selected_keeper_receive{ nullptr }; + int expected_pushed_min{ 0 }, expected_pushed_max{ 0 }; + // are we in batched mode? + BATCH_SENTINEL.pushKey(L); + int const is_batched{ lua501_equal(L, key_i, -1) }; + lua_pop(L, 1); + if (is_batched) + { + // no need to pass linda.batched in the keeper state + ++key_i; + // make sure the keys are of a valid type + check_key_types(L, key_i, key_i); + // receive multiple values from a single slot + selected_keeper_receive = KEEPER_API(receive_batched); + // we expect a user-defined amount of return value + expected_pushed_min = (int) luaL_checkinteger(L, key_i + 1); + expected_pushed_max = (int) luaL_optinteger(L, key_i + 2, expected_pushed_min); + // don't forget to count the key in addition to the values + ++expected_pushed_min; + ++expected_pushed_max; + if (expected_pushed_min > expected_pushed_max) + { + return luaL_error(L, "batched min/max error"); + } } else { - return luaL_error(L, "no data to send"); + // make sure the keys are of a valid type + check_key_types(L, key_i, lua_gettop(L)); + // receive a single value, checking multiple slots + selected_keeper_receive = KEEPER_API(receive); + // we expect a single (value, key) pair of returned values + expected_pushed_min = expected_pushed_max = 2; } - } - // convert nils to some special non-nil sentinel in sent values - keeper_toggle_nil_sentinels(L, key_i + 1, LookupMode::ToKeeper); - bool ret{ false }; - CancelRequest cancel{ CancelRequest::None }; - KeeperCallResult pushed; - { Lane* const lane{ LANE_POINTER_REGKEY.readLightUserDataValue(L) }; Keeper* const K{ which_keeper(linda->U->keepers, linda->hashSeed()) }; lua_State* const KL{ K ? K->L : nullptr }; if (KL == nullptr) return 0; + CancelRequest cancel{ CancelRequest::None }; + KeeperCallResult pushed; STACK_CHECK_START_REL(KL, 0); for (bool try_again{ true };;) { @@ -319,31 +494,29 @@ LUAG_FUNC(linda_send) break; } - STACK_CHECK(KL, 0); - pushed = keeper_call(linda->U, KL, KEEPER_API(send), L, linda, key_i); + // all arguments of receive() but the first are passed to the keeper's receive function + pushed = keeper_call(linda->U, KL, selected_keeper_receive, L, linda, key_i); if (!pushed.has_value()) { break; } - ASSERT_L(pushed.value() == 1); - - ret = lua_toboolean(L, -1) ? true : false; - lua_pop(L, 1); - - if (ret) + if (pushed.value() > 0) { - // Wake up ALL waiting threads - linda->m_write_happened.notify_all(); + ASSERT_L(pushed.value() >= expected_pushed_min && pushed.value() <= expected_pushed_max); + // replace sentinels with real nils + keeper_toggle_nil_sentinels(L, lua_gettop(L) - pushed.value(), LookupMode::FromKeeper); + // To be done from within the 'K' locking area + // + linda->m_read_happened.notify_all(); break; } - // instant timout to bypass the wait syscall if (std::chrono::steady_clock::now() >= until) { - break; /* no wait; instant timeout */ + break; /* instant timeout */ } - // storage limit hit, wait until timeout or signalled that we should try again + // nothing received, wait until timeout or signalled that we should try again { Lane::Status prev_status{ Lane::Error }; // prevent 'might be used uninitialized' warnings if (lane != nullptr) @@ -353,11 +526,11 @@ LUAG_FUNC(linda_send) ASSERT_L(prev_status == Lane::Running); // but check, just in case lane->m_status = Lane::Waiting; ASSERT_L(lane->m_waiting_on == nullptr); - lane->m_waiting_on = &linda->m_read_happened; + lane->m_waiting_on = &linda->m_write_happened; } - // could not send because no room: wait until some data was read before trying again, or until timeout is reached + // not enough data to read: wakeup when data was sent, or when timeout is reached std::unique_lock keeper_lock{ K->m_mutex, std::adopt_lock }; - std::cv_status const status{ linda->m_read_happened.wait_until(keeper_lock, until) }; + std::cv_status const status{ linda->m_write_happened.wait_until(keeper_lock, until) }; keeper_lock.release(); // we don't want to release the lock! try_again = (status == std::cv_status::no_timeout); // detect spurious wakeups if (lane != nullptr) @@ -368,188 +541,28 @@ LUAG_FUNC(linda_send) } } STACK_CHECK(KL, 0); - } - - if (!pushed.has_value()) - { - luaL_error(L, "tried to copy unsupported types"); // doesn't return - } - - switch (cancel) - { - case CancelRequest::Soft: - // if user wants to soft-cancel, the call returns lanes.cancel_error - CANCEL_ERROR.pushKey(L); - return 1; - - case CancelRequest::Hard: - // raise an error interrupting execution only in case of hard cancel - raise_cancel_error(L); // raises an error and doesn't return - - default: - lua_pushboolean(L, ret); // true (success) or false (timeout) - return 1; - } -} - -// ################################################################################################# - -/* - * 2 modes of operation - * [val, key]= linda_receive( linda_ud, [timeout_secs_num=-1], key_num|str|bool|lightuserdata [, ...] ) - * Consumes a single value from the Linda, in any key. - * Returns: received value (which is consumed from the slot), and the key which had it - - * [val1, ... valCOUNT]= linda_receive( linda_ud, [timeout_secs_num=-1], linda.batched, key_num|str|bool|lightuserdata, min_COUNT[, max_COUNT]) - * Consumes between min_COUNT and max_COUNT values from the linda, from a single key. - * returns the actual consumed values, or nil if there weren't enough values to consume - * - */ -LUAG_FUNC(linda_receive) -{ - Linda* const linda{ ToLinda(L, 1) }; - std::chrono::time_point until{ std::chrono::time_point::max() }; - int key_i{ 2 }; // index of first key, if timeout not there - - if (lua_type(L, 2) == LUA_TNUMBER) // we don't want to use lua_isnumber() because of autocoercion - { - lua_Duration const duration{ lua_tonumber(L, 2) }; - if (duration.count() >= 0.0) - { - until = std::chrono::steady_clock::now() + std::chrono::duration_cast(duration); - } - ++key_i; - } - else if (lua_isnil(L, 2)) // alternate explicit "infinite timeout" by passing nil before the key - { - ++key_i; - } - - keeper_api_t selected_keeper_receive{ nullptr }; - int expected_pushed_min{ 0 }, expected_pushed_max{ 0 }; - // are we in batched mode? - BATCH_SENTINEL.pushKey(L); - int const is_batched{ lua501_equal(L, key_i, -1) }; - lua_pop(L, 1); - if (is_batched) - { - // no need to pass linda.batched in the keeper state - ++key_i; - // make sure the keys are of a valid type - check_key_types(L, key_i, key_i); - // receive multiple values from a single slot - selected_keeper_receive = KEEPER_API(receive_batched); - // we expect a user-defined amount of return value - expected_pushed_min = (int) luaL_checkinteger(L, key_i + 1); - expected_pushed_max = (int) luaL_optinteger(L, key_i + 2, expected_pushed_min); - // don't forget to count the key in addition to the values - ++expected_pushed_min; - ++expected_pushed_max; - if (expected_pushed_min > expected_pushed_max) - { - return luaL_error(L, "batched min/max error"); - } - } - else - { - // make sure the keys are of a valid type - check_key_types(L, key_i, lua_gettop(L)); - // receive a single value, checking multiple slots - selected_keeper_receive = KEEPER_API(receive); - // we expect a single (value, key) pair of returned values - expected_pushed_min = expected_pushed_max = 2; - } - - Lane* const lane{ LANE_POINTER_REGKEY.readLightUserDataValue(L) }; - Keeper* const K{ which_keeper(linda->U->keepers, linda->hashSeed()) }; - lua_State* const KL{ K ? K->L : nullptr }; - if (KL == nullptr) - return 0; - - CancelRequest cancel{ CancelRequest::None }; - KeeperCallResult pushed; - STACK_CHECK_START_REL(KL, 0); - for (bool try_again{ true };;) - { - if (lane != nullptr) - { - cancel = lane->cancel_request; - } - cancel = (cancel != CancelRequest::None) ? cancel : linda->simulate_cancel; - // if user wants to cancel, or looped because of a timeout, the call returns without sending anything - if (!try_again || cancel != CancelRequest::None) - { - pushed.emplace(0); - break; - } - // all arguments of receive() but the first are passed to the keeper's receive function - pushed = keeper_call(linda->U, KL, selected_keeper_receive, L, linda, key_i); if (!pushed.has_value()) { - break; - } - if (pushed.value() > 0) - { - ASSERT_L(pushed.value() >= expected_pushed_min && pushed.value() <= expected_pushed_max); - // replace sentinels with real nils - keeper_toggle_nil_sentinels(L, lua_gettop(L) - pushed.value(), LookupMode::FromKeeper); - // To be done from within the 'K' locking area - // - linda->m_read_happened.notify_all(); - break; + return luaL_error(L, "tried to copy unsupported types"); } - if (std::chrono::steady_clock::now() >= until) + switch (cancel) { - break; /* instant timeout */ - } + case CancelRequest::Soft: + // if user wants to soft-cancel, the call returns CANCEL_ERROR + CANCEL_ERROR.pushKey(L); + return 1; - // nothing received, wait until timeout or signalled that we should try again - { - Lane::Status prev_status{ Lane::Error }; // prevent 'might be used uninitialized' warnings - if (lane != nullptr) - { - // change status of lane to "waiting" - prev_status = lane->m_status; // Running, most likely - ASSERT_L(prev_status == Lane::Running); // but check, just in case - lane->m_status = Lane::Waiting; - ASSERT_L(lane->m_waiting_on == nullptr); - lane->m_waiting_on = &linda->m_write_happened; - } - // not enough data to read: wakeup when data was sent, or when timeout is reached - std::unique_lock keeper_lock{ K->m_mutex, std::adopt_lock }; - std::cv_status const status{ linda->m_write_happened.wait_until(keeper_lock, until) }; - keeper_lock.release(); // we don't want to release the lock! - try_again = (status == std::cv_status::no_timeout); // detect spurious wakeups - if (lane != nullptr) - { - lane->m_waiting_on = nullptr; - lane->m_status = prev_status; - } - } - } - STACK_CHECK(KL, 0); + case CancelRequest::Hard: + // raise an error interrupting execution only in case of hard cancel + raise_cancel_error(L); // raises an error and doesn't return - if (!pushed.has_value()) - { - return luaL_error(L, "tried to copy unsupported types"); - } - - switch (cancel) - { - case CancelRequest::Soft: - // if user wants to soft-cancel, the call returns CANCEL_ERROR - CANCEL_ERROR.pushKey(L); - return 1; - - case CancelRequest::Hard: - // raise an error interrupting execution only in case of hard cancel - raise_cancel_error(L); // raises an error and doesn't return - - default: - return pushed.value(); - } + default: + return pushed.value(); + } + }; + return Linda::ProtectedCall(L, receive); } // ################################################################################################# @@ -564,47 +577,51 @@ LUAG_FUNC(linda_receive) */ LUAG_FUNC(linda_set) { - Linda* const linda{ ToLinda(L, 1) }; - bool const has_value{ lua_gettop(L) > 2 }; - // make sure the key is of a valid type (throws an error if not the case) - check_key_types(L, 2, 2); - - Keeper* const K{ which_keeper(linda->U->keepers, linda->hashSeed()) }; - KeeperCallResult pushed; - if (linda->simulate_cancel == CancelRequest::None) + auto set = [](lua_State* L) { - if (has_value) - { - // convert nils to some special non-nil sentinel in sent values - keeper_toggle_nil_sentinels(L, 3, LookupMode::ToKeeper); - } - pushed = keeper_call(linda->U, K->L, KEEPER_API(set), L, linda, 2); - if (pushed.has_value()) // no error? - { - ASSERT_L(pushed.value() == 0 || pushed.value() == 1); + Linda* const linda{ ToLinda(L, 1) }; + bool const has_value{ lua_gettop(L) > 2 }; + // make sure the key is of a valid type (throws an error if not the case) + check_key_types(L, 2, 2); + Keeper* const K{ which_keeper(linda->U->keepers, linda->hashSeed()) }; + KeeperCallResult pushed; + if (linda->simulate_cancel == CancelRequest::None) + { if (has_value) { - // we put some data in the slot, tell readers that they should wake - linda->m_write_happened.notify_all(); // To be done from within the 'K' locking area + // convert nils to some special non-nil sentinel in sent values + keeper_toggle_nil_sentinels(L, 3, LookupMode::ToKeeper); } - if (pushed.value() == 1) + pushed = keeper_call(linda->U, K->L, KEEPER_API(set), L, linda, 2); + if (pushed.has_value()) // no error? { - // the key was full, but it is no longer the case, tell writers they should wake - ASSERT_L(lua_type(L, -1) == LUA_TBOOLEAN && lua_toboolean(L, -1) == 1); - linda->m_read_happened.notify_all(); // To be done from within the 'K' locking area + ASSERT_L(pushed.value() == 0 || pushed.value() == 1); + + if (has_value) + { + // we put some data in the slot, tell readers that they should wake + linda->m_write_happened.notify_all(); // To be done from within the 'K' locking area + } + if (pushed.value() == 1) + { + // the key was full, but it is no longer the case, tell writers they should wake + ASSERT_L(lua_type(L, -1) == LUA_TBOOLEAN && lua_toboolean(L, -1) == 1); + linda->m_read_happened.notify_all(); // To be done from within the 'K' locking area + } } } - } - else // linda is cancelled - { - // do nothing and return lanes.cancel_error - CANCEL_ERROR.pushKey(L); - pushed.emplace(1); - } + else // linda is cancelled + { + // do nothing and return lanes.cancel_error + CANCEL_ERROR.pushKey(L); + pushed.emplace(1); + } - // must trigger any error after keeper state has been released - return OptionalValue(pushed, L, "tried to copy unsupported types"); + // must trigger any error after keeper state has been released + return OptionalValue(pushed, L, "tried to copy unsupported types"); + }; + return Linda::ProtectedCall(L, set); } // ################################################################################################# @@ -616,13 +633,17 @@ LUAG_FUNC(linda_set) */ LUAG_FUNC(linda_count) { - Linda* const linda{ ToLinda(L, 1) }; - // make sure the keys are of a valid type - check_key_types(L, 2, lua_gettop(L)); + auto count = [](lua_State* L) + { + Linda* const linda{ ToLinda(L, 1) }; + // make sure the keys are of a valid type + check_key_types(L, 2, lua_gettop(L)); - Keeper* const K{ which_keeper(linda->U->keepers, linda->hashSeed()) }; - KeeperCallResult const pushed{ keeper_call(linda->U, K->L, KEEPER_API(count), L, linda, 2) }; - return OptionalValue(pushed, L, "tried to count an invalid key"); + Keeper* const K{ which_keeper(linda->U->keepers, linda->hashSeed()) }; + KeeperCallResult const pushed{ keeper_call(linda->U, K->L, KEEPER_API(count), L, linda, 2) }; + return OptionalValue(pushed, L, "tried to count an invalid key"); + }; + return Linda::ProtectedCall(L, count); } // ################################################################################################# @@ -634,31 +655,35 @@ LUAG_FUNC(linda_count) */ LUAG_FUNC(linda_get) { - Linda* const linda{ ToLinda(L, 1) }; - lua_Integer const count{ luaL_optinteger(L, 3, 1) }; - luaL_argcheck(L, count >= 1, 3, "count should be >= 1"); - luaL_argcheck(L, lua_gettop(L) <= 3, 4, "too many arguments"); - // make sure the key is of a valid type (throws an error if not the case) - check_key_types(L, 2, 2); - - KeeperCallResult pushed; - if (linda->simulate_cancel == CancelRequest::None) + auto get = [](lua_State* L) { - Keeper* const K{ which_keeper(linda->U->keepers, linda->hashSeed()) }; - pushed = keeper_call(linda->U, K->L, KEEPER_API(get), L, linda, 2); - if (pushed.value_or(0) > 0) + Linda* const linda{ ToLinda(L, 1) }; + lua_Integer const count{ luaL_optinteger(L, 3, 1) }; + luaL_argcheck(L, count >= 1, 3, "count should be >= 1"); + luaL_argcheck(L, lua_gettop(L) <= 3, 4, "too many arguments"); + // make sure the key is of a valid type (throws an error if not the case) + check_key_types(L, 2, 2); + + KeeperCallResult pushed; + if (linda->simulate_cancel == CancelRequest::None) { - keeper_toggle_nil_sentinels(L, lua_gettop(L) - pushed.value(), LookupMode::FromKeeper); + Keeper* const K{ which_keeper(linda->U->keepers, linda->hashSeed()) }; + pushed = keeper_call(linda->U, K->L, KEEPER_API(get), L, linda, 2); + if (pushed.value_or(0) > 0) + { + keeper_toggle_nil_sentinels(L, lua_gettop(L) - pushed.value(), LookupMode::FromKeeper); + } } - } - else // linda is cancelled - { - // do nothing and return lanes.cancel_error - CANCEL_ERROR.pushKey(L); - pushed.emplace(1); - } - // an error can be raised if we attempt to read an unregistered function - return OptionalValue(pushed, L, "tried to copy unsupported types"); + else // linda is cancelled + { + // do nothing and return lanes.cancel_error + CANCEL_ERROR.pushKey(L); + pushed.emplace(1); + } + // an error can be raised if we attempt to read an unregistered function + return OptionalValue(pushed, L, "tried to copy unsupported types"); + }; + return Linda::ProtectedCall(L, get); } // ################################################################################################# @@ -671,34 +696,38 @@ LUAG_FUNC(linda_get) */ LUAG_FUNC(linda_limit) { - Linda* const linda{ ToLinda(L, 1) }; - // make sure we got 3 arguments: the linda, a key and a limit - luaL_argcheck( L, lua_gettop( L) == 3, 2, "wrong number of arguments"); - // make sure we got a numeric limit - luaL_checknumber( L, 3); - // make sure the key is of a valid type - check_key_types( L, 2, 2); - - KeeperCallResult pushed; - if (linda->simulate_cancel == CancelRequest::None) + auto limit = [](lua_State* L) { - Keeper* const K{ which_keeper(linda->U->keepers, linda->hashSeed()) }; - pushed = keeper_call(linda->U, K->L, KEEPER_API(limit), L, linda, 2); - ASSERT_L( pushed.has_value() && (pushed.value() == 0 || pushed.value() == 1)); // no error, optional boolean value saying if we should wake blocked writer threads - if (pushed.value() == 1) + Linda* const linda{ ToLinda(L, 1) }; + // make sure we got 3 arguments: the linda, a key and a limit + luaL_argcheck( L, lua_gettop( L) == 3, 2, "wrong number of arguments"); + // make sure we got a numeric limit + luaL_checknumber( L, 3); + // make sure the key is of a valid type + check_key_types( L, 2, 2); + + KeeperCallResult pushed; + if (linda->simulate_cancel == CancelRequest::None) { - ASSERT_L( lua_type( L, -1) == LUA_TBOOLEAN && lua_toboolean( L, -1) == 1); - linda->m_read_happened.notify_all(); // To be done from within the 'K' locking area + Keeper* const K{ which_keeper(linda->U->keepers, linda->hashSeed()) }; + pushed = keeper_call(linda->U, K->L, KEEPER_API(limit), L, linda, 2); + ASSERT_L( pushed.has_value() && (pushed.value() == 0 || pushed.value() == 1)); // no error, optional boolean value saying if we should wake blocked writer threads + if (pushed.value() == 1) + { + ASSERT_L( lua_type( L, -1) == LUA_TBOOLEAN && lua_toboolean( L, -1) == 1); + linda->m_read_happened.notify_all(); // To be done from within the 'K' locking area + } } - } - else // linda is cancelled - { - // do nothing and return lanes.cancel_error - CANCEL_ERROR.pushKey(L); - pushed.emplace(1); - } - // propagate pushed boolean if any - return pushed.value(); + else // linda is cancelled + { + // do nothing and return lanes.cancel_error + CANCEL_ERROR.pushKey(L); + pushed.emplace(1); + } + // propagate pushed boolean if any + return pushed.value(); + }; + return Linda::ProtectedCall(L, limit); } // ################################################################################################# @@ -831,8 +860,12 @@ LUAG_FUNC(linda_concat) */ LUAG_FUNC(linda_dump) { - Linda* const linda{ ToLinda(L, 1) }; - return keeper_push_linda_storage(linda->U, Dest{ L }, linda, linda->hashSeed()); + auto dump = [](lua_State* L) + { + Linda* const linda{ ToLinda(L, 1) }; + return keeper_push_linda_storage(linda->U, Dest{ L }, linda, linda->hashSeed()); + }; + return Linda::ProtectedCall(L, dump); } // ################################################################################################# @@ -922,6 +955,23 @@ void LindaFactory::deleteDeepObjectInternal(lua_State* L, DeepPrelude* o_) const // ################################################################################################# +static luaL_Reg const s_LindaMT[] = +{ + { "__concat", LG_linda_concat }, + { "__tostring", LG_linda_tostring }, + { "__towatch", LG_linda_towatch }, // Decoda __towatch support + { "cancel", LG_linda_cancel }, + { "count", LG_linda_count }, + { "deep", LG_linda_deep }, + { "dump", LG_linda_dump }, + { "get", LG_linda_get }, + { "limit", LG_linda_limit }, + { "receive", LG_linda_receive }, + { "send", LG_linda_send }, + { "set", LG_linda_set }, + { nullptr, nullptr } +}; + void LindaFactory::createMetatable(lua_State* L) const { STACK_CHECK_START_REL(L, 0); @@ -934,54 +984,8 @@ void LindaFactory::createMetatable(lua_State* L) const lua_pushliteral(L, "Linda"); lua_setfield(L, -2, "__metatable"); - lua_pushcfunction(L, LG_linda_tostring); - lua_setfield(L, -2, "__tostring"); - - // Decoda __towatch support - lua_pushcfunction(L, LG_linda_towatch); - lua_setfield(L, -2, "__towatch"); - - lua_pushcfunction(L, LG_linda_concat); - lua_setfield(L, -2, "__concat"); - - // protected calls, to ensure associated keeper is always released even in case of error - // all function are the protected call wrapper, where the actual operation is provided as upvalue - // note that this kind of thing can break function lookup as we use the function pointer here and there - // TODO: change that and use different functions! - - lua_pushcfunction(L, LG_linda_send); - lua_pushcclosure(L, LG_linda_protected_call, 1); - lua_setfield(L, -2, "send"); - - lua_pushcfunction(L, LG_linda_receive); - lua_pushcclosure(L, LG_linda_protected_call, 1); - lua_setfield(L, -2, "receive"); - - lua_pushcfunction(L, LG_linda_limit); - lua_pushcclosure(L, LG_linda_protected_call, 1); - lua_setfield(L, -2, "limit"); - - lua_pushcfunction(L, LG_linda_set); - lua_pushcclosure(L, LG_linda_protected_call, 1); - lua_setfield(L, -2, "set"); - - lua_pushcfunction(L, LG_linda_count); - lua_pushcclosure(L, LG_linda_protected_call, 1); - lua_setfield(L, -2, "count"); - - lua_pushcfunction(L, LG_linda_get); - lua_pushcclosure(L, LG_linda_protected_call, 1); - lua_setfield(L, -2, "get"); - - lua_pushcfunction(L, LG_linda_cancel); - lua_setfield(L, -2, "cancel"); - - lua_pushcfunction(L, LG_linda_deep); - lua_setfield(L, -2, "deep"); - - lua_pushcfunction(L, LG_linda_dump); - lua_pushcclosure(L, LG_linda_protected_call, 1); - lua_setfield(L, -2, "dump"); + // the linda functions + luaL_setfuncs(L, s_LindaMT, 0); // some constants BATCH_SENTINEL.pushKey(L); -- cgit v1.2.3-55-g6feb