From 23f25f3a9d327153e4c16ca86d937ec334803a10 Mon Sep 17 00:00:00 2001 From: Benoit Germain Date: Tue, 9 Apr 2024 10:12:30 +0200 Subject: C++ migration: still more threading code cleanup. 'sudo' global moved in the Universe --- src/tools.h | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) (limited to 'src/tools.h') diff --git a/src/tools.h b/src/tools.h index c1a8534..7d9aaab 100644 --- a/src/tools.h +++ b/src/tools.h @@ -6,7 +6,7 @@ #include "macros_and_utils.h" // forwards -struct Universe; +class Universe; // ################################################################################################ -- cgit v1.2.3-55-g6feb From 95932749a53f46ae5901798a17e1f0c4c33ac6b2 Mon Sep 17 00:00:00 2001 From: Benoit Germain Date: Tue, 9 Apr 2024 15:49:18 +0200 Subject: C++ migration: use strong type safety for source and destination states in transfer functions --- CHANGES | 744 +----------------------------------------------- lanes-3.16.3-0.rockspec | 78 ----- lanes-4.0.0-0.rockspec | 78 +++++ src/cancel.cpp | 7 +- src/compat.cpp | 35 ++- src/deep.cpp | 40 +-- src/deep.h | 4 +- src/keeper.cpp | 16 +- src/keeper.h | 2 +- src/lanes.cpp | 20 +- src/lanes_private.h | 4 +- src/linda.cpp | 10 +- src/macros_and_utils.h | 18 +- src/state.cpp | 29 +- src/state.h | 3 +- src/threading.cpp | 12 +- src/threading.h | 1 - src/tools.cpp | 85 +++--- src/tools.h | 14 +- src/universe.cpp | 8 +- 20 files changed, 237 insertions(+), 971 deletions(-) delete mode 100644 lanes-3.16.3-0.rockspec create mode 100644 lanes-4.0.0-0.rockspec (limited to 'src/tools.h') diff --git a/CHANGES b/CHANGES index d93f5b4..3725674 100644 --- a/CHANGES +++ b/CHANGES @@ -1,746 +1,6 @@ CHANGES: -CHANGE 159: BGe 19-Mar-24 - * fix small internal issue with when hitting timeout on thread kill during thread_cancel() in pthread implementation - -CHANGE 158: BGe 22-Feb-24 - * naive luajit detection in PUC-Lua-based builds, and vice-versa to detect mismatches - * internal version bumped to 3.16.3 - -CHANGE 157: Mitalie 17-Aug-23 - * Prevent crash on linux as non-root - * internal version bumped to 3.16.2 - -CHANGE 156: BGe 9-Aug-23 - * new configuration option .internal_allocator to help LuaJIT users. - * internal version bumped to 3.16.1 - -CHANGE 155: BGe 28-Jul-23 - * tweaks to linux thread priority management: do nothing if not super-user. if super-user, do nothing if nothing is provided (instead of trying to force a prio when LINUX_SCHED_RR is defined). - -CHANGE 154: eligovision 1-Mar-22 - * Fix 3-parametrized __lanesclone - -CHANGE 153: BGe 17-Feb-22 - * NEVER use allocator obtained from lua_getallocf to allocate stuff manually when compiling for LuaJIT - -CHANGE 152: BGe 7-Feb-22 - * bumped version to 3.16.0 - * __lanesclone is now called only once with 3 parameters dest, source, size -> BREAKS CUSTOM DEEP USERDATA API - -CHANGE 151: BGe 7-Feb-22 - * bumped version to 3.15.2 - * Lanes no longer relies on malloc/free for internal allocations, but uses the primary alloc function from the master Lua state - -CHANGE 150: BGe 22-Sep-21 - * fix require() wrapper to return all values returned by original require() - -CHANGE 149: BGe 8-Jul-21 - * bumped version to 3.15.1 - * fix function transfer with lua_dump for Lua 5.4 failing for functions big enough to necessitate a buffer reallocation - -CHANGE 148: BGe 23-Jun-21 - * __lanesclone now receives the original as light userdata the first time it is called -> BREAKS CUSTOM DEEP USERDATA API - -CHANGE 147: BGe 16-Jun-21 - * changed lanes.threads() output so that several lanes with the same name don't clobber each other in the result table -> BREAKS API - * bumped version to 3.15 because of the API change - -CHANGE 146: BGe 26-Apr-19 - * lane:cancel() rework (see doc). - * opt.cancelstep is gone, hook is installed by lane:cancel() if requested - -CHANGE 145: BGe 28-Nov-18 - * more code refacto - * don't test __lanesignore for POD types (-> slightly faster when trasnfering lots of data) - -CHANGE 144: BGe 28-Nov-18 - * some code refacto - -CHANGE 143: BGe 27-Nov-18 - * Lua 5.4 support - * __lanesclone and lanes.nameof support userdata uservalue(s) - -CHANGE 142: BGe 26-Nov-18 - * Version is available in public header - -CHANGE 141: BGe 25-Nov-18 - * protect_allocator configure option is gone, long live allocator (more embedders-friendly) - -CHANGE 140: BGe 22-Nov-18 - * Raise an error instead of crashing when attempting to transfer a non-deep full userdata - -CHANGE 139: BGe 21-Nov-18 - * more DEBUGSPEW - -CHANGE 138: BGe 19-Nov-18 - * Registry access code utility macros - * CONFIG_REGKEY and LOOKUP_REGKEY are now lightuserdata instead of strings - * Stack checking debug macros improvements - -CHANGE 137: BGe 15-Nov-18 - * Deep userdata must embed DeepPrelude to save an allocation (also changes Deep protocol) - -CHANGE 136: BGe 15-Nov-18 - * split linda code in a separate file - * rockspec for version v3.13.0 - -CHANGE 135: BGe 11-Nov-18 - * fix a bunch of compilation warnings - -CHANGE 134: BGe 3-Dec-13 - * new API lanes.set_thread_affinity() - * set_debug_threadname implemented with win32 pthread - -CHANGE 133: BGe 8-Nov-18 - * Make sure any linda operation that can raise an error won't ever leave a mutex unreleased - * lane:join() now returns nil, "timeout" in case of timeout - -CHANGE 132: BGe 7-Nov-18 - * __lanesclone mechanism should actually work now - -CHANGE 131: BGe 7-Nov-18 - * Fix potential crash at application shutdown when deep userdata were created before Lanes is required - -CHANGE 130: BGe 2-Nov-18 - * always duplicate the config structure in new lanes even when no libraries are initialized by the generator - (fixes an internal error trying to call on_state_create in a lane without any libs loaded) - -CHANGE 129: BGe 2-Nov-18 - * Bumped version to 3.13 - * fix error when autodetecting protect_allocator when running under LuaJIT - -CHANGE 128: BGe 31-Oct-18 - * Better default value autodetection for protect_allocator setting - -CHANGE 127: BGe 30-Oct-18 - * restrict internal light userdata constants to 47 significant bits when building against LuaJIT-x64 - -CHANGE 126: Bge 29-Oct-18 - * Add deep user data cloning support - -CHANGE 125: BGe 25-Oct-18 - * Fix Lanes build by reorganizing types around a bit - -CHANGE 124: BGe 9-Jul-18 - * Fix a stack overflow when copying large tables with verbose_errors option enabled - * Support for integer formatting in verbose errors - -CHANGE 123: BGe 2-Aug-17 - * added support for user-provided __gc in deep userdata - * more complete deep userdata sample - -CHANGE 122: BGe 1-Aug-17 - * fix crash trying to use a deep-aware module while not requiring Lanes - * bumped version to 3.12 - -CHANGE 121: BGe 13-Jun-17 - * no longer internally assert when an error message is not a string - -CHANGE 120: BGe 5-Jun-17 - * new API function lanes.register( "name", module) to manually register a module table after it was required - * Transfering registered module tables will link the equivalent in the destination state instead of cloning it - * bumped version to 3.11 - -CHANGE 119: BGe 10-May-17 - * Fixed some compilation warnings - * Improved LuaJIT support - -CHANGE 118: trukanduk 21-Nov-16 - * bumped version to 3.10.1 - * objects with a metatable that contains __lanesignore are skipped during data transfers - -CHANGE 117: mpeterv 21-Nov-16 - * Fix an implicit number-to-string conversion - -CHANGE 116: BGe, mpeterv 27-Apr-15 - * bumped version to 3.10.0 - * segfault fixed in LG_lane_new - * Lua 5.3 support - -CHANGE 115: BGe 18-Sep-14 - * bumped version to 3.9.7 - * new function lanes.sleep() - -CHANGE 114: BGe 8-Jul-14 - * Postponed _G scan for function lookup database to after on_state_create invocation - * Fixed a crash when USE_DEBUG_SPEW == 1 - -CHANGE 113: BGe 17-Jun-14 - * bumped version to 3.9.6 - * separate deep userdata code in a dedicated file to allow external modules to implement Lanes-compatible deep userdata without requiring a binary dependency against the Lanes module - because of this linda_id function(eDO_metatable) must push 2 values on the stack: a metatable and a deep version string obtained from luaG_pushdeepversion() - -CHANGE 112 BGe 16-May-14 - * bumped version to 3.9.5 - * fix linda.__towatch to return non-nil when the linda is empty - * lanes.gen() error reporting improvements - -CHANGE 111 BGe 24-Apr-14 - * fixed linda:send() possibly returning an undefined value - -CHANGE 110 Stepets 20-Apr-14 - * fix LuaJIT detection issues - -CHANGE 109 BGe 03-Apr-14 - * moved some Lua-version compatibility code in separate source files - -CHANGE 108: BGe 20-Mar-14 - * bumped version to 3.9.4 - * set_finalizer throws an error if provided finalizer isn't a function - * fix error handling when the error doesn't generate an error handler call (IOW, all errors but LUA_ERRRUN) - * provide callstack if LUA_ERRRUN occurs inside a finalizer - -CHANGE 107: BGe 19-Mar-14 - * Make sure we don't mutex-wrap require() more than once, just in case - -CHANGE 106: BGe 17-Mar-14 - * Fixed crash when using protect_allocator option - -CHANGE 105: BGe 27-Feb-14 - * Bumped version to 3.9.3 - * new exposed variable linda.null that exposes the internal NIL_SENTINEL marker - * linda:send() interprets send key linda.null as authorization to silently send a single nil when not provided with anything to send - (useful when sending results of a function that can return nothing) - -CHANGE 104: BGe 25-Feb-14 - * Bumped version to 3.9.2 - * Internal rework: the whole Lanes engine now works "per universe" to allow concurrent Lanes execution in more than one embedded master state - * this universe is a full userdata created in the master state, selfdestruct_gc is the __gc for this userdata - * most of what was initialized only once is now per-universe - * Fixed potential crashes at desinit if problems occur during keeper states initialisation - * Fixed require() not always serialized properly - * Raise an error instead of crashing on deep userdata prelude memory allocation failure - * Added forgotten mutex desinitialisation at universe shutdown - -CHANGE 103: BGe 24-Feb-14 - * Fix lookup database table not being created when it should if Lanes is required in more than one Lua master state - -CHANGE 102: BGe 18-Feb-14 - * raise an error instead of dereferencing a NULL pointer on deep userdata creation and lane struct creation - -CHANGE 101: BGe 18-Feb-14 - * version 3.9.1 - * removed some keeper desinit legacy dead code - * keeper array is allocated with master state's alloc function instead of malloc()/free() - * prevent application crash when specifying a very large number of keepers in the configuration options - * any error occuring during one-time inits is raised outside the one-time mutex protected code region - -CHANGE 100: BGe 17-Feb-14 - * lanes.linda() accepts an optional integer group to give control on keeper state repartition - -CHANGE 99: BGe 17-Feb-14 - * version 3.9.0 - * keepers now require "package", receive package.path & package.cpath, and call on_state_create() if it is a C function - * changed the deep public API (improved deep idfunc signature, renamed luaG_deep_userdata to luaG_newdeepuserdata) - * if an error occurs while copying a deep userdata, don't raise inside the keeper state - * fixed situations where raised errors could lead to memory leaks (deep gc) - -CHANGE 98: BGe 13-Feb-14 - * version 3.8.5 - * linda:limit() returns lanes.cancel_error on a limited linda - * lanes.genlock() and lanes.genatomic() support cancelled lindas by returning lanes.cancel_error whenever appropriate - * fixed a possible Lua stack overflow when calling linda:dump() - * fixed cases where linda:send() and linda:receive() would not return lanes.cancel_error when they should - -CHANGE 97: BGe 10-Feb-14 - * version 3.8.4 - * new API linda:cancel("read"|"write"|"both"|"none") - * all linda operations return lanes.cancel_error on a cancelled linda - * raised an internal string length so that longer linda names are fully output before truncation applies when doing tostring( linda) - -CHANGE 96: BGe 24-Jan-14 - * another Lua stack overflow fix when sending complex function through lindas or as lane body - -CHANGE 95: BGe 22-Jan-14 - * version 3.8.3 - * fixed a possible Lua stack overflow when sending complex function through lindas or as lane body - * experimental: lanes.nameof() scans the registry if a regular search didn't yield anything interesting - * fixed lanes.nameof() misbehaving when encountering a LUA_TTHREAD object - -CHANGE 94: BGe 22-Jan-14 - * version 3.8.2 - * new lane launcher option gc_cb to set a callback that is invoked when a lane is garbage collected - * Fix more invalid memory accesses when fetching the name of a joined lane with lanes:threads() (because its lua_State is closed) - -CHANGE 93: BGe 20-Jan-14 - * slightly improve linda performance when the producer/consumer scenario leaves leave the key empty - -CHANGE 92: BGe 20-Jan-14 - * version 3.8.1 - * new function lane:get_debug_threadname() - * Fix invalid memory accesses when fetching the name of a joined lane with lanes:threads() (because its lua_State is closed) - * use luaL_newmetatable() to create the metatable for lane objects - * prevent malicious code from crashing by calling lane methods without passing the lane as first argument (raise an error instead) - * set_debug_threadname() is no longer registered in the function lookup databases because it holds a C pointer as upvalue and it might crash if used maliciously - -CHANGE 91: BGe 20-Jan-14 - * version 3.8.0 - * linda:set() accepts multiple values to set in the specified slot - * linda:get() accepts an optional count to peek several values at once - -CHANGE 90: BGe 16-Jan-14 - * version 3.7.8 - * lane:cancel() now accepts a boolean second argument when soft cancelling (negative timeout) to wake the thread if necessary - * if a blocked linda send() or receive() call is interrupted by a cancellation request, - it returns CANCEL_ERROR so that this case can be differentiated from a simple timeout - * fixed WIN32 THREAD_CREATE() wrong _beginthreadex() error detection - * fatal WIN32 threading errors retrieve and output the error description string with FormatMessage() - * fixed missing lanes.set_singlethreaded - * fixed perftest.lua - * added test/cancel.lua - -CHANGE 89: BGe 09-Jan-14 - * version 3.7.7 - * fix crash when calling linda:count() on unknown keys - * purge key storage with linda:set( key, nil) on an unlimited key to reduce memory usage with lots of keys - * linda:limit() wakes write-blocked threads if necessary when the new limit enables writes to occur again - * linda:set() wakes write-blocked threads if necessary if the operation created some room to write into - -CHANGE 88: BGe 06-Jan-14 - * version 3.7.6 - * if config.on_state_create() is a C function, call it by direct C closure reconstruction in newly created states - -CHANGE 87: BGe 20-Dec-13 - * version 3.7.5 - * fixed a crash that can occur at shutdown when an object stored inside a keeper state performs a linda operation on a linda making use of another keeper - * new setting demote_full_userdata to select between light userdata demotion or raising an error when attempting to transfer a non-deep full userdata - -CHANGE 86: BGe 3-Dec-13 - * version 3.7.4 - * internal refactoring of pthread priority management code - * new API lanes.set_thread_priority() - -CHANGE 85: BGe 28-Nov-13 - * version 3.7.3 - * set pthread thread cancel type to PTHREAD_CANCEL_ASYNCHRONOUS - * lane_h:cancel() accepts a 3rd timeout argument used when waiting for actual thread termination (hitting the timeout raises an error) - * added PROPAGATE_ALLOCF macro to select state creation mode (lua_newstate or luaL_newstate) - -CHANGE 84: BGe 18-Nov-13 - * Fix a deadlock when GCing during a linda operation. - * Fix a compilation warning about an unused variable - * Get rid of uintptr_t to remove dependency on stdint.h - * Fix internal error at lane creation when the generator doesn't open any base library - -CHANGE 83: BGe 16-Nov-13 - * version 3.7.2 - * Fixed function returned by lanes.genlock() not handling numeric keys properly when release lock - * Enable lanes.genlock() to attempt lock with an optional "try" mode - * make EnableCrashingOnCrashes a one-time operation - -CHANGE 82: BGe 13-Nov-13 - * Fix a case where an error could be raised inside a keeper state - -CHANGE 81: BGe 07-Nov-13 - * Make set_finalizer(), set_debug_threadname(), cancel_test() and set_error_reporting() transferable from lane to lane - * Improved some DEBUGSPEW output - -CHANGE 80: BGe 06-Nov-13 - * Fix a few compilation warnings about uninitialized variables - * Fixed a bad extern variable declaration resulting in multiple instances (crashes the Pelles-C build) - -CHANGE 79: BGe 04-Nov-13 - * Fix lanes.nameof() crashing when encountering a light userdata - -CHANGE 78: BGe 25-Oct-13 - * Fix windows build not exporting public 'deep' API - * Don't call on_state_create in keeper states, as it is no longer necessary - * Remove inclusion of stdint.h - * Fix windows build for WINVER > 0x400 - -CHANGE 77: BGe 22-Oct-13 - * version 3.7.1 - * errors inside finalizers generate a full stack just like any other error - -CHANGE 76: BGe 10-Oct-13 - * version 3.7.0 - * fix lanes.threads() not being available in a lane where lanes.configure() settings didn't contain track_lanes although the initial configure() call did. - -CHANGE 75: BGe 7-Oct-13 - * require "lanes".configure() sequence is only necessary at the first require "lanes". - -CHANGE 74: BGe 7-Oct-13 - * fix a crash at application shutdown where in some situations we could deinitialize the protected allocator mutex while a lane was still using it. - -CHANGE 73: BGe 4-Oct-13 - * fix timers broken by change 69 - -CHANGE 72: BGe 3-Oct-13 - * bugfix: no longer create a global named "lanes.core" inside lanes having "*" as library list - -CHANGE 71: BGe 30-Sept-13 - * version 3.6.6 - * properly handle cases when a Lua C module is a C function - -CHANGE 70: BGe 27-Step-13 - * no longer call core.configure with dummy params when requiring lanes more than once (fixes potential multithreading issues with LuaJIT allocator) - * activated EnableCrashingOnCrashes() is active on Win32 debug builds - * fixed some comments in code - -CHANGE 69: BGe 26-Sept-13 - * version 3.6.5 - * Reduce memory footprint, simplify module order setup in conjuction with Lanes, and send over native functions a bit faster as well - * Lanes no longer has to internally require modules inside the keeper states because they no longer need a lookup database - the lookup name is stored as-is and actually converted in the destination state - * optimisation: bypass cache when sending native functions over - * removed all the KEEPER_MODEL_LUA code, as it can no longer work anyway - -CHANGE 68: BGe 24-Sept-13 - * version 3.6.4 - * Fix possible application hang at shutdown if a keeper state referenced a linda. - -CHANGE 67: BGe 2-Aug-13 - * version 3.6.3 - * lane:cancel() only causes cancel_test() to return true but won't interrupt execution of the lane during linda operations - -CHANGE 66: BGe 31-Jul-13 - * more explicit errors when trying to transfer unknown source functions (with new configure option verbose_errors) - -CHANGE 65: BGe 23-Jul-13 - * default options wrap allocator around a mutex when run by LuaJIT - -CHANGE 64: BGe 20-Jul-13 - * WIN32 builds against pre-Vista versions no longer use PulseEvent to fix occasional hangs when a wake event is missed - -CHANGE 63: BGe 20-May-13 - * version 3.6.2 - * WIN32 builds use condition variables instead of PulseEvent() when available. - * first steps toward fixing make-vc.cmd - -CHANGE 62: BGe 05-Apr-13 - * version 3.6.1 - * function lookup database population keeps the 'smaller' name in case of multiple hits, to remove the no-LUA_COMPAT_ALL restriction on Lua5.2 builds - -CHANGE 61: BGe 14-Mar-13 - * version 3.6.0 - * protect_allocator is an API change -> version bump - * bugfix: allocator protection should be done once per primary Lua state, not once only the first time ever Lanes is required - -CHANGE 60: BGe 13-Mar-13 - * version 3.5.2 - * stricter validation of with_timers config option: validator was accepting any non-boolean value - * new configuration option protect_allocator for VMs with thread unsafe allocators (such as LuaJIT) - * removed some obsolete bits of dead code - -CHANGE 59: BGe 12-Feb-13 - * version 3.5.1 - * new lanes.h header and API call luaopen_lanes_embedded() for embedders - * "lanes.core" is an acceptable library in the generator libs argument - * library "*" wildcard also opens lanes.core - * tweaked code for Xbox 360 build - -CHANGE 58: BGe 30-Jan-13 - * version 3.5.0 - * new: API lanes.require(), use it instead of regular require() for modules that export C functions you need to send over. - * new: lanes no longer require 'lanes.core' by default in every created state. Use {required={"lanes.core"}} if you need to transfer lanes functions. - * internal: because of the above, reworked the timer implementation to remove upvalue-dependency on lanes.core - * new: API lanes.timer_lane, to be able to operate on timer lane if need be - * improved: if a module is a full userdata, scan its metatable for function database population - * improved: on_state_create can be a Lua function - * changed: on_state_create is called after the base libraries are loaded - * package[loaders|searchers] is no longer transfered as function naming depends on slot order - * internal: changed separator from '.' to '/' in lookup databases to be able to distinguish search levels and dot coming from module names - * added some mode debug spew - * updated tests to reflect the above changes - -CHANGE 57: BGe 28-Jan-13 - * More detailed DEBUG_SPEW logs - * A bit of code cosmetics - -CHANGE 56: BGe 25-Jan-13 - * version 3.4.4 - * bugfix: take into account the fact that "coroutine" is no longer part of base library in Lua 5.2 - * bugfix: if "bit32" was listed in the libraries, it wouldn't open (library list parsing failing on digits) - * bugfix: Use luaL_requiref() to open standard libraries in Lua 5.2 as we should - * bugfix: any Lua state created by Lanes reuses the allocator function of the originating state - * bugfix: don't call on_state_create() while GC is suspended during lua state initialization - -CHANGE 55: BGe 24-Jan-13 - * version 3.4.3 - * raise an error if lane generator libs specification contains a lib more than once - * bit32 is a valid lib name in the libs specification (silently ignored by the Lua 5.1 build) - * improved lanes.nameof to search inside table- and userdata- metatables for an object's name - * bugfix: fixed an unwarranted error when trying to discover a function name upon a failed transfer - * contents of package.[path,cpath,preload,loaders|searchers] are pulled *only once* inside keeper states at initialisation - * Lua function upvalues equal to the global environment aren't copied by value, but bound to the destination's global environment - especially useful for Lua 5.2 _ENV - * bugfix: fixed loading of base libraries that didn't create the global tables when built for Lua 5.2 - -CHANGE 54: BGe 10-Jan-13 - * version 3.4.2 - * Don't pull "package" settings in the timer lane - * removed a limitation preventing Lua functions with indirect recursive upvalue references from being transferable - -CHANGE 53: BGe 11-Dec-2012 - * version 3.4.1 - * new function lanes.timers(), returns a list of all active timers. - -CHANGE 52: BGe 03-Dec-2012 - * linda:send() and linda:receive() no longer triggers string->number autocoercion when checking for the optional timeout argument: - a string is always a linda slot, even if coercible. - -CHANGE 51: BGe 27-Nov-2012 - * linux flavors with older glibc use prctl instead of pthread_setname_np - * selfdestruct chain handling is now the same on all platforms - -CHANGE 50: BGe 22-Nov-2012 - * bugfix: linda:set() no longer clears the storage limit - -CHANGE 49: BGe 21-Nov-2012 - * fix application shutdown crash by not registering anything with atexit() - * rockspec for version v3.4.0 - -CHANGE 48: BGe 25-Sep-2012 - * version 3.4.0 - * new method linda:dump() that outputs the full contents of a linda as a table, also linked to __towatch for Decoda support - * linda:receive() API change! - * instead of [val, key], linda:receive( timeout, key) returns [key, val] - * instead of [val, [...]], linda:receive( timeout, linda.batched key) returns [key, val[, ...]] - this is to unify the return values of regular and batched mode, and to be able to tell when batched mode is interrupted by a lane cancellation - * fixed Lua 5.2 build to take into account the "loaders"->"searchers" name change in 'package' module. - * a bit of html cleanup and added some infos in the documentation regarding the Lanes internals - -CHANGE 47: BGe 13-Sep-2012 - * implemented set_debug_threadname() for pthread builds where possible - * refactored linda __tostring and __concat - -CHANGE 46: BGe 10-Sep-2012 - * version 3.3.0 - * lane.status can return "killed" if lane was forcefully killed with lanes:cancel() - * lane:join(): return nil, "killed" if called on a killed lane. - * lane[]: produces [1] = nil, [2] = "killed" if the lane was killed - * lane:join(): fixed an assertion in debug builds when joining a lane forcefully cancelled with lane:cancel( , true). - * indexing a lane with a string other than "join", "cancel" or "status" raises an error. - * fixed configure() to correctly apply defaults when they are missing from the provided settings - * added a shutdown_timeout to control the duration Lanes will wait for graceful termination of running lanes at application shutdown. Default is 0.25. - -CHANGE 45: BGe 21-Aug-2012 - * keeper internals implemented in C instead of Lua for better performances - * fixed arguments checks in linda:limit() and linda:set() - -CHANGE 44: BGe 13-Aug-2012 - * lanes code updated to build against Lua 5.1 and Lua 5.2 - * removed the search for MSVCR80.DLL when building for MinGW32 since it no longer seems to be necessary - -CHANGE 43: BGe 09-Aug-2012 - * fix possible crash at application shutdown when a race condition causes linda objects to be collected after the keeper states are cleaned up. - -CHANGE 42: BGe 06-Aug-2012 - * lanes.linda() accepts an optional name for debug purposes - -CHANGE 41: BGe 07-Jul-2012 - * lua51-lanes renamed lanes/core - * keeper state microcode is no longer embedded inside lanes.core, but located and loaded with package.loaders[2] - * changed rockspec build type from "make" to "builtin" - -CHANGE 40: BGe 26-Jun-2012 - * when a transfered function is not found in source, guess its name to help the user find out what's wrong - * new function lanes.nameof() - -CHANGE 39: BGe 23-Jun-2012 - * lanes.timer() accepts a first_secs=nil to stop a timer - * timer lane catches errors and prints them - * fixed some typos in manual - -CHANGE 38: BGe 11-Jun-2012 - * linda:receive() batched mode now accepts a max_count optional argument - -CHANGE 37: BGe 4-Jun-2012 (fix and idea courtesy of sonoro1234) - * fixed thread_cancel() not working when called without argument - * new lane-global function set_error_reporting() to enable more data detailed data provided by lane_error() - -CHANGE 36 BGe 26-Apr-2012 - * improved LuaJIT2 compatibility by handling "*" library set through luaL_openlibs() - -CHANGE 35 BGe 17-Feb-2012 - * changed lanes.configure signature to receive a table instead of individual parameters - * added support for an on_state_create callback called to load custom functions in a state in addition to the base libraries - -CHANGE 34 BGe 14-Nov-2011 - * removed packagepath and packagecpath options, replaced by a package table, whose fields path, cpath, loaders, preload are transfered - * code cleanup to facilitate transition between WIN32 and PTHREAD implementations - * tentative fix for desinit crashes when free running lanes are killed at process shutdown - -CHANGE 33 BGe 5-Nov-2011: Lanes version 3.0-beta - * process exit change: close everything at GC when main state closes, not when atexit() handlers are processed - * Lua 5.2-style module: - * module() is no longer used to implement lanes.lua - * a global "lanes" variable is no longer created when the module is required - * the Lanes module table is returned instead - * Lanes must be initialized before used: - * the first occurence of 'require "lanes"' produces a minimal interface that only contains a configure() function - * the remainder of the interface is made available once this function is called - * subsequent calls to configure() do nothing - * configure() controls the number of keeper states and the startup of timers - * LuaJIT 2 compatibility - * non-Lua functions are no longer copied by creating a C closure from a C pointer, but through 2-way lookup tables - * this means that if a lane function body pulls non-Lua functions, the lane generator description must contain the list of libraries and modules that exports them - * introduces a change in configuration .globals management: contents are copied *after* std libs are loaded - * new .required configuration entry to list modules that must be require()'ed before lane body is transferred - * lane:cancel() wakes up waiting lindas like what is done at lane shutdown - -CHANGE 32 BGe 14-May-2011 - * raise an error when linda:send() has nothing to send - -CHANGE 31 BGe 17-Apr-2011 - * linda uses a fast FIFO implementation to speed up data exchanges - * new linda:count() method - * new linda batched data read mode - * proper key type check in all linda methods - * fix setup-vc.cmd to support Visual Studio 2010 and Windows 7 64 bits - * bugfix: release keeper state mutex at desinit - -CHANGE 30 BGe 30-Mar-2011 - * linda honors __tostring and __concat - * new accessor linda:keys(), to retrieve the list of keys with pending data inside a linda - * new lanes options packagepath and packagecpath, in case one needs to set them differently than the default - -CHANGE 29 BGe 1-Mar-2011 - fixed potential crash at application shutdown when calling lua_close() on a killed thread's VM. - exposed cancel_test() in the lanes to enable manual testing for cancellation requests. - removed kludgy {globals={threadName}} support, replaced with a new function set_debug_threadname(). - -CHANGE 28 BGe 18-Feb-2011 - - moved keeper-related code in a separate source file - - keeper.lua is now embedded in text form instead of bytecode to improve LuaJIT2-compatibility - -CHANGE 27 BGe 17-Feb-2011 - - we know Lanes is loaded in the master state, so we don't force it - to be required in every lane too when a linda deep userdata is copied - - Refactor lane proxy implementation: it is now a full userdata instead - of a table, and its methods are implemented in C instead of Lua - * its metatable is no longer accessible - * writing to the proxy raises an error - * it is no longer possible to overwrite its join() and cancel() methods - - when a deep userdata idfunc requests a module to be required, manually - check that it is not loaded before requiring it instead of relying on - the require function's loop detection feature - - when a module must be required, raise an error if the 'require' function - is not found in the target state - -CHANGE 26 BGe 14-Feb-2011: - Fixed application hang-up because keeper state was not released in case of errors thrown by - inter-state data copy for unsupported types - -CHANGE 25 BGe 12-Feb-2011: - Changed idfunc signature and contract to clarify that fact it is not lua-callable - and to be able to require the module it was exported from in the target lanes - -CHANGE 24 DPtr 25-Jan-2011: - Changed lanes.c to export functions as a module rather than writing them directly to the globals table. - -CHANGE 23 DPtr 23-Jan-2011: - Fixed bug where reference to Linda object was dropped for a short time ( crashing if GC was run during that time ). - Changed the atexit code to trip the timer thread's write signal. - -CHANGE 22 DPtr 19-Jan-2011: - Changed luaG_push_proxy to cache deep userdata proxies. - -CHANGE 21 (bugfixes) BGe 3-Jan-2011: - Several fixes by Martin Krpan: - - linda_send was waiting on the wrong signal - - buildfix when using i586-mingw32msvc-gcc cross compiler - - lanes_h:cancel() returns a boolean as it should - - timers could get blocked sometimes because they were initialized with negative values - - prepare_timeout could generate an illegal setting - -CHANGE 20 BGe 3-Dec-2010: - Enable to specify a string as lane code instead of a function so that we dont use lua_dump, which - isn't supported by LuaJIT. - -CHANGE 19 BGe 2-Dec-2010: - No longer rely on global function 'tostring' to generate unique identifiers when caching data being transfered through la linda. Should fix a compatilibity issue with LuaJIT2. - -CHANGE 18 BGe 6-Oct-2010: - Fixed 'memory leak' in some situations where a free running lane is collected before application shutdown - A bit of code cleanup - -CHANGE 17 BGe 21-Sept-2010: - Fixed stupid compilation errors. - -CHANGE 16 PLM 24-Aug-2010: - Releasing memory at gc / atexit. - Finalizers actually get error strings. - Fixed missing argument propagation in lane:cancel - Added variable threadName sent trough globals-table. Set in s_lane, and in debuggers on windows. - Added argument checking for linda-objects, where missing them caused crashes. - -CHANGE 15 (minor) BGe 27-Jul-2010: - Version bump for a true upgrade release (2.0.4 package was only a renamed 2.0.3) - -CHANGE 14 (bug fix) BGe 09-Jul-2010: - Fixed lane status to be correctly returned as "waiting" when it should. - -CHANGE 13 (fix for multithreaded host apps) AKa 24-Jun-2009: - mentioned Lanes expects the host application to be singlethreaded, - and there are troubles if Lanes is used from multiple threads, opened by the host - (before requiring Lanes). This is true, and fix should now be in place. - -CHANGE 12 (bug fix on Windows, 2.0.3) AKa 25-Jan-2009: - Did CHANGE 9 the way it should be done. - -CHANGE 11 (new feature, 2.0.3) AKa 23-Jan-2009: - Finalizers ('set_finalizer()') for being able to do cleanup of a lane's - resources, whether it returned succesfully or via an error. - -CHANGE 10 (new feature, 2.0.3) AKa 23-Jan-2009: - Call stack showing where an error occurred is not merged with the error - message, but delivered as a separate stack table ({ "filename:line" [, ...] }). - Getting call stacks of errorred lanes is now possible. - -CHANGE 9 (bug fix on Windows) AKa 10-Dec-2008 (> 2.0.2): - Applied patch from Kriss Daniels to avoid issues on 'now_time()' in Win32 - (http://luaforge.net/forum/forum.php?thread_id=22704&forum_id=1781). - -CHANGE 8 (bug fix) AKa 26-Oct-2008: - Avoids occasional segfault at process exit (on multicore CPUs). Does this - by keeping track of "free running" threads (s.a. the time thread) and - cancelling them at process exit. - - Tested (2.0.2) on Linux 64,x86, OS X, WinXP. - -CHANGE 7 (bug fix) AKa 15-Oct-2008: - Recursive functions that use themselves as direct upvalue can now be - passed to other lanes, and used as a lane function. - -CHANGE 6 (bug fix) AKa 15-Oct-2008: - Added local caches of the following to src/lanes.lua (was otherwise getting - errors at least in 'tests/irayo_recursive.lua'). - - local assert= assert - local string_gmatch= assert( string.gmatch ) - local select= assert( select ) - local type= assert( type ) - local pairs= assert( pairs ) - local tostring= assert( tostring ) - local error= assert( error ) - local setmetatable= assert( setmetatable ) - local rawget= assert( rawget ) - - Thanks to Irayo for detecting and reporting this. - -CHANGE 5 (new feature): - Modifying Makefile so it's better suited to LuaRocks. - -CHANGE 4 (new feature): - Metatable copying, allowing Lua objects to be copied across lanes. - -CHANGE 3 (bug fix) AKa 5-Aug-2008: - The '__gc' method was not tied to thread userdata, at all. Caused memory - lifespan problems at least on OS X when threads were cancelled (EINVAL). - -CHANGE 2 (bug fix) AKa 5-Aug-2008: - Better calculation of timeouts, always making them absolute (even in Win32) - to allow for events that wake the lane up but don't read/write the Linda - key that it was observing. - -CHANGE 1 (bug fix) AKa 4-Aug-2008: - Signalling woke up only one waiting thread, not all. This caused i.e. - receive to not wake up if there was another thread waiting on the same - Linda object. - - PThread fix: using 'pthread_cond_broadcast()' instead of 'pthread_cond_signal()' - Win32 fix: using manual events and 'PulseEvent()' +CHANGE 1: BGe 9-Apr-24 + * reset changelog, next entry will list API changes since last C-implementation. (end) diff --git a/lanes-3.16.3-0.rockspec b/lanes-3.16.3-0.rockspec deleted file mode 100644 index 106a2d9..0000000 --- a/lanes-3.16.3-0.rockspec +++ /dev/null @@ -1,78 +0,0 @@ --- --- Lanes rockspec --- --- Ref: --- --- - -package = "Lanes" - -version = "3.16.3-0" - -source= { - url= "git+https://github.com/LuaLanes/lanes.git", - branch= "v3.16.3" -} - -description = { - summary= "Multithreading support for Lua", - detailed= [[ - Lua Lanes is a portable, message passing multithreading library - providing the possibility to run multiple Lua states in parallel. - ]], - license= "MIT/X11", - homepage="https://github.com/LuaLanes/lanes", - maintainer="Benoit Germain " -} - --- Q: What is the difference of "windows" and "win32"? Seems there is none; --- so should we list either one or both? --- -supported_platforms= { "win32", - "macosx", - "linux", - "freebsd", -- TBD: not tested - "msys", -- TBD: not supported by LuaRocks 1.0 (or is it?) -} - -dependencies= { - "lua >= 5.1", -- builds with either 5.1/LuaJIT, 5.2, 5.3 and 5.4 -} - -build = { - type = "builtin", - platforms = - { - linux = - { - modules = - { - ["lanes.core"] = - { - libraries = "pthread" - }, - } - } - }, - modules = - { - ["lanes.core"] = - { - sources = - { - "src/cancel.cpp", - "src/compat.cpp", - "src/deep.cpp", - "src/keeper.cpp", - "src/lanes.cpp", - "src/linda.cpp", - "src/tools.cpp", - "src/state.cpp", - "src/threading.cpp", - "src/universe.cpp" - }, - incdirs = { "src"}, - }, - lanes = "src/lanes.lua" - } -} diff --git a/lanes-4.0.0-0.rockspec b/lanes-4.0.0-0.rockspec new file mode 100644 index 0000000..4e1b370 --- /dev/null +++ b/lanes-4.0.0-0.rockspec @@ -0,0 +1,78 @@ +-- +-- Lanes rockspec +-- +-- Ref: +-- +-- + +package = "Lanes" + +version = "4.0.0-0" + +source= { + url= "git+https://github.com/LuaLanes/lanes.git", + branch= "v4.0.0" +} + +description = { + summary= "Multithreading support for Lua", + detailed= [[ + Lua Lanes is a portable, message passing multithreading library + providing the possibility to run multiple Lua states in parallel. + ]], + license= "MIT/X11", + homepage="https://github.com/LuaLanes/lanes", + maintainer="Benoit Germain " +} + +-- Q: What is the difference of "windows" and "win32"? Seems there is none; +-- so should we list either one or both? +-- +supported_platforms= { "win32", + "macosx", + "linux", + "freebsd", -- TBD: not tested + "msys", -- TBD: not supported by LuaRocks 1.0 (or is it?) +} + +dependencies= { + "lua >= 5.1", -- builds with either 5.1/LuaJIT, 5.2, 5.3 and 5.4 +} + +build = { + type = "builtin", + platforms = + { + linux = + { + modules = + { + ["lanes.core"] = + { + libraries = "pthread" + }, + } + } + }, + modules = + { + ["lanes.core"] = + { + sources = + { + "src/cancel.cpp", + "src/compat.cpp", + "src/deep.cpp", + "src/keeper.cpp", + "src/lanes.cpp", + "src/linda.cpp", + "src/tools.cpp", + "src/state.cpp", + "src/threading.cpp", + "src/universe.cpp" + }, + incdirs = { "src"}, + }, + lanes = "src/lanes.lua" + } +} diff --git a/src/cancel.cpp b/src/cancel.cpp index 437a6f0..e08e975 100644 --- a/src/cancel.cpp +++ b/src/cancel.cpp @@ -1,6 +1,6 @@ /* -- --- CANCEL.C +-- CANCEL.CPP -- -- Lane cancellation support -- @@ -9,7 +9,7 @@ --[[ =============================================================================== -Copyright (C) 2011-2019 Benoit Germain +Copyright (C) 2011-2024 Benoit Germain Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal @@ -35,9 +35,6 @@ THE SOFTWARE. #include "cancel.h" -// #include -//#include - #include "lanes_private.h" #include "threading.h" #include "tools.h" diff --git a/src/compat.cpp b/src/compat.cpp index 47fe37e..8acab25 100644 --- a/src/compat.cpp +++ b/src/compat.cpp @@ -1,6 +1,6 @@ /* * ############################################################################################### - * ######################################### Lua 5.1/5.2 ######################################### + * ####################################### Lua 5.1/5.2/5.3 ####################################### * ############################################################################################### */ #include "compat.h" @@ -9,7 +9,12 @@ /* ** Copied from Lua 5.2 loadlib.c */ +// ################################################################################################ +// ################################################################################################ #if LUA_VERSION_NUM == 501 +// ################################################################################################ +// ################################################################################################ + static int luaL_getsubtable (lua_State *L, int idx, const char *fname) { lua_getfield(L, idx, fname); @@ -26,6 +31,8 @@ static int luaL_getsubtable (lua_State *L, int idx, const char *fname) } } +// ################################################################################################ + void luaL_requiref (lua_State *L, const char *modname, lua_CFunction openf, int glb) { lua_pushcfunction(L, openf); @@ -43,24 +50,30 @@ void luaL_requiref (lua_State *L, const char *modname, lua_CFunction openf, int } #endif // LUA_VERSION_NUM +// ################################################################################################ +// ################################################################################################ #if LUA_VERSION_NUM < 504 +// ################################################################################################ +// ################################################################################################ void* lua_newuserdatauv( lua_State* L, size_t sz, int nuvalue) { ASSERT_L( nuvalue <= 1); - return lua_newuserdata( L, sz); + return lua_newuserdata(L, sz); } +// ################################################################################################ + // push on stack uservalue #n of full userdata at idx int lua_getiuservalue(lua_State* L, int idx, int n) { // full userdata can have only 1 uservalue before 5.4 if( n > 1) { - lua_pushnil( L); + lua_pushnil(L); return LUA_TNONE; } - lua_getuservalue( L, idx); + lua_getuservalue(L, idx); #if LUA_VERSION_NUM == 501 /* default environment is not a nil (see lua_getfenv) */ @@ -68,30 +81,32 @@ int lua_getiuservalue(lua_State* L, int idx, int n) if (lua_rawequal(L, -2, -1) || lua_rawequal(L, -2, LUA_GLOBALSINDEX)) { lua_pop(L, 2); - lua_pushnil( L); + lua_pushnil(L); return LUA_TNONE; } lua_pop(L, 1); /* remove package */ #endif - return lua_type( L, -1); + return lua_type(L, -1); } +// ################################################################################################ + // pop stack top, sets it a uservalue #n of full userdata at idx -int lua_setiuservalue( lua_State* L, int idx, int n) +int lua_setiuservalue(lua_State* L, int idx, int n) { if( n > 1 #if LUA_VERSION_NUM == 501 - || lua_type( L, -1) != LUA_TTABLE + || lua_type(L, -1) != LUA_TTABLE #endif ) { - lua_pop( L, 1); + lua_pop(L, 1); return 0; } - (void) lua_setuservalue( L, idx); + std::ignore = lua_setuservalue(L, idx); return 1; // I guess anything non-0 is ok } diff --git a/src/deep.cpp b/src/deep.cpp index ac2905e..55063b3 100644 --- a/src/deep.cpp +++ b/src/deep.cpp @@ -1,5 +1,5 @@ /* - * DEEP.C Copyright (c) 2017, Benoit Germain + * DEEP.CPP Copyright (c) 2024, Benoit Germain * * Deep userdata support, separate in its own source file to help integration * without enforcing a Lanes dependency @@ -9,7 +9,7 @@ =============================================================================== Copyright (C) 2002-10 Asko Kauppi - 2011-17 Benoit Germain + 2011-24 Benoit Germain Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal @@ -41,13 +41,6 @@ THE SOFTWARE. #include #include -#include -#include -#include -#include -#if !defined(__APPLE__) -#include -#endif /*-- Metatable copying --*/ @@ -73,7 +66,7 @@ static constexpr UniqueKey DEEP_PROXY_CACHE_KEY{ 0x05773d6fc26be106ull }; * Sets up [-1]<->[-2] two-way lookups, and ensures the lookup table exists. * Pops the both values off the stack. */ -static void set_deep_lookup( lua_State* L) +static void set_deep_lookup(lua_State* L) { STACK_GROW( L, 3); STACK_CHECK_START_REL(L, 2); // a b @@ -88,11 +81,13 @@ static void set_deep_lookup( lua_State* L) STACK_CHECK( L, 0); } +// ################################################################################################ + /* * Pops the key (metatable or idfunc) off the stack, and replaces with the * deep lookup value (idfunc/metatable/nil). */ -static void get_deep_lookup( lua_State* L) +static void get_deep_lookup(lua_State* L) { STACK_GROW( L, 1); STACK_CHECK_START_REL(L, 1); // a @@ -106,11 +101,13 @@ static void get_deep_lookup( lua_State* L) STACK_CHECK( L, 1); } +// ################################################################################################ + /* * Return the registered ID function for 'index' (deep userdata proxy), * or nullptr if 'index' is not a deep userdata proxy. */ -static inline luaG_IdFunction get_idfunc( lua_State* L, int index, LookupMode mode_) +static inline luaG_IdFunction get_idfunc(lua_State* L, int index, LookupMode mode_) { // when looking inside a keeper, we are 100% sure the object is a deep userdata if (mode_ == LookupMode::FromKeeper) @@ -142,8 +139,9 @@ static inline luaG_IdFunction get_idfunc( lua_State* L, int index, LookupMode mo } } +// ################################################################################################ -void free_deep_prelude( lua_State* L, DeepPrelude* prelude_) +void free_deep_prelude(lua_State* L, DeepPrelude* prelude_) { ASSERT_L(prelude_->idfunc); STACK_CHECK_START_REL(L, 0); @@ -154,6 +152,7 @@ void free_deep_prelude( lua_State* L, DeepPrelude* prelude_) STACK_CHECK(L, 0); } +// ################################################################################################ /* * void= mt.__gc( proxy_ud ) @@ -161,7 +160,7 @@ void free_deep_prelude( lua_State* L, DeepPrelude* prelude_) * End of life for a proxy object; reduce the deep reference count and clean it up if reaches 0. * */ -static int deep_userdata_gc( lua_State* L) +static int deep_userdata_gc(lua_State* L) { DeepPrelude** const proxy{ lua_tofulluserdata(L, 1) }; DeepPrelude* p = *proxy; @@ -193,6 +192,7 @@ static int deep_userdata_gc( lua_State* L) return 0; } +// ################################################################################################ /* * Push a proxy userdata on the stack. @@ -203,7 +203,7 @@ static int deep_userdata_gc( lua_State* L) * used in this Lua state (metatable, registring it). Otherwise, increments the * reference count. */ -char const* push_deep_proxy(lua_State* L, DeepPrelude* prelude, int nuv_, LookupMode mode_) +char const* push_deep_proxy(Dest L, DeepPrelude* prelude, int nuv_, LookupMode mode_) { // Check if a proxy already exists push_registry_subtable_mode( L, DEEP_PROXY_CACHE_KEY, "v"); // DPC @@ -278,7 +278,7 @@ char const* push_deep_proxy(lua_State* L, DeepPrelude* prelude, int nuv_, Lookup // this is needed because we must make sure the shared library is still loaded as long as we hold a pointer on the idfunc { int oldtop_module = lua_gettop( L); - modname = (char const*) prelude->idfunc( L, DeepOp::Module); // DPC proxy metatable + modname = (char const*) prelude->idfunc( L, DeepOp::Module); // DPC proxy metatable // make sure the function pushed nothing on the stack! if( lua_gettop( L) - oldtop_module != 0) { @@ -348,6 +348,8 @@ char const* push_deep_proxy(lua_State* L, DeepPrelude* prelude, int nuv_, Lookup return nullptr; } +// ################################################################################################ + /* * Create a deep userdata * @@ -370,7 +372,7 @@ char const* push_deep_proxy(lua_State* L, DeepPrelude* prelude, int nuv_, Lookup * * Returns: 'proxy' userdata for accessing the deep data via 'luaG_todeep()' */ -int luaG_newdeepuserdata( lua_State* L, luaG_IdFunction idfunc, int nuv_) +int luaG_newdeepuserdata(Dest L, luaG_IdFunction idfunc, int nuv_) { STACK_GROW( L, 1); STACK_CHECK_START_REL(L, 0); @@ -409,6 +411,7 @@ int luaG_newdeepuserdata( lua_State* L, luaG_IdFunction idfunc, int nuv_) return 1; } +// ################################################################################################ /* * Access deep userdata through a proxy. @@ -430,6 +433,7 @@ DeepPrelude* luaG_todeep(lua_State* L, luaG_IdFunction idfunc, int index) return *proxy; } +// ################################################################################################ /* * Copy deep userdata between two separate Lua states (from L to L2) @@ -438,7 +442,7 @@ DeepPrelude* luaG_todeep(lua_State* L, luaG_IdFunction idfunc, int index) * the id function of the copied value, or nullptr for non-deep userdata * (not copied) */ -bool copydeep(Universe* U, lua_State* L2, int L2_cache_i, lua_State* L, int i, LookupMode mode_, char const* upName_) +bool copydeep(Universe* U, Dest L2, int L2_cache_i, Source L, int i, LookupMode mode_, char const* upName_) { luaG_IdFunction const idfunc { get_idfunc(L, i, mode_) }; if (idfunc == nullptr) diff --git a/src/deep.h b/src/deep.h index c09af23..6cf33ec 100644 --- a/src/deep.h +++ b/src/deep.h @@ -54,8 +54,8 @@ struct DeepPrelude std::atomic m_refcount{ 0 }; }; -char const* push_deep_proxy(lua_State* L, DeepPrelude* prelude, int nuv_, LookupMode mode_); +char const* push_deep_proxy(Dest L, DeepPrelude* prelude, int nuv_, LookupMode mode_); void free_deep_prelude( lua_State* L, DeepPrelude* prelude_); -LANES_API int luaG_newdeepuserdata( lua_State* L, luaG_IdFunction idfunc, int nuv_); +LANES_API int luaG_newdeepuserdata(Dest L, luaG_IdFunction idfunc, int nuv_); LANES_API DeepPrelude* luaG_todeep(lua_State* L, luaG_IdFunction idfunc, int index); diff --git a/src/keeper.cpp b/src/keeper.cpp index 0aea18e..9718bda 100644 --- a/src/keeper.cpp +++ b/src/keeper.cpp @@ -14,7 +14,7 @@ --[[ =============================================================================== - Copyright (C) 2011-2023 Benoit Germain + Copyright (C) 2011-2024 Benoit Germain Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal @@ -207,10 +207,10 @@ static void push_table(lua_State* L, int idx_) // ################################################################################################## -int keeper_push_linda_storage(Universe* U, lua_State* L, void* ptr_, uintptr_t magic_) +int keeper_push_linda_storage(Universe* U, Dest L, void* ptr_, uintptr_t magic_) { Keeper* const K{ which_keeper(U->keepers, magic_) }; - lua_State* const KL{ K ? K->L : nullptr }; + Source const KL{ K ? K->L : nullptr }; if (KL == nullptr) return 0; STACK_GROW(KL, 4); @@ -412,7 +412,7 @@ int keepercall_limit(lua_State* L) // set the new limit fifo->limit = limit; // return 0 or 1 value - return lua_gettop( L); + return lua_gettop(L); } // ################################################################################################## @@ -485,7 +485,7 @@ int keepercall_set(lua_State* L) lua_insert(L, 3); // fifos key fifotbl [val [, ...]] fifo_push(L, fifo, count); // fifos key fifotbl } - return should_wake_writers ? (lua_pushboolean( L, 1), 1) : 0; + return should_wake_writers ? (lua_pushboolean(L, 1), 1) : 0; } // ################################################################################################## @@ -717,7 +717,7 @@ void init_keepers(Universe* U, lua_State* L) if (!lua_isnil(L, -1)) { // when copying with mode LookupMode::ToKeeper, error message is pushed at the top of the stack, not raised immediately - if (luaG_inter_copy_package(U, L, K, -1, LookupMode::ToKeeper)) + if (luaG_inter_copy_package(U, Source{ L }, Dest{ K }, -1, LookupMode::ToKeeper)) { // if something went wrong, the error message is at the top of the stack lua_remove(L, -2); // error_msg @@ -840,7 +840,7 @@ int keeper_call(Universe* U, lua_State* K, keeper_api_t func_, lua_State* L, voi lua_pushlightuserdata(K, linda); - if ((args == 0) || luaG_inter_copy(U, L, K, args, LookupMode::ToKeeper) == 0) // L->K + if ((args == 0) || luaG_inter_copy(U, Source{ L }, Dest{ K }, args, LookupMode::ToKeeper) == 0) // L->K { lua_call(K, 1 + args, LUA_MULTRET); retvals = lua_gettop(K) - Ktos; @@ -848,7 +848,7 @@ int keeper_call(Universe* U, lua_State* K, keeper_api_t func_, lua_State* L, voi // this may interrupt a lane, causing the destruction of the underlying OS thread // after this, another lane making use of this keeper can get an error code from the mutex-locking function // when attempting to grab the mutex again (WINVER <= 0x400 does this, but locks just fine, I don't know about pthread) - if ((retvals > 0) && luaG_inter_move(U, K, L, retvals, LookupMode::FromKeeper) != 0) // K->L + if ((retvals > 0) && luaG_inter_move(U, Source{ K }, Dest{ L }, retvals, LookupMode::FromKeeper) != 0) // K->L { retvals = -1; } diff --git a/src/keeper.h b/src/keeper.h index ba5a57b..89fa2ab 100644 --- a/src/keeper.h +++ b/src/keeper.h @@ -42,7 +42,7 @@ Keeper* which_keeper(Keepers* keepers_, uintptr_t magic_); Keeper* keeper_acquire(Keepers* keepers_, uintptr_t magic_); void keeper_release(Keeper* K_); void keeper_toggle_nil_sentinels(lua_State* L, int val_i_, LookupMode const mode_); -int keeper_push_linda_storage(Universe* U, lua_State* L, void* ptr_, uintptr_t magic_); +int keeper_push_linda_storage(Universe* U, Dest L, void* ptr_, uintptr_t magic_); using keeper_api_t = lua_CFunction; #define KEEPER_API(_op) keepercall_##_op diff --git a/src/lanes.cpp b/src/lanes.cpp index cf443f2..3d0c70d 100644 --- a/src/lanes.cpp +++ b/src/lanes.cpp @@ -1027,7 +1027,7 @@ LUAG_FUNC(lane_new) DEBUGSPEW_CODE(U->debugspew_indent_depth.fetch_add(1, std::memory_order_relaxed)); // populate with selected libraries at the same time - lua_State* const L2{ luaG_newstate(U, L, libs_str) }; // L // L2 + lua_State* const L2{ luaG_newstate(U, Source{ L }, libs_str) }; // L // L2 // 'lane' is allocated from heap, not Lua, since its life span may surpass the handle's (if free running thread) Lane* const lane{ new (U) Lane{ U, L2 } }; @@ -1134,7 +1134,7 @@ LUAG_FUNC(lane_new) { DEBUGSPEW_CODE(fprintf(stderr, INDENT_BEGIN "lane_new: update 'package'\n" INDENT_END)); // when copying with mode LookupMode::LaneBody, should raise an error in case of problem, not leave it one the stack - (void) luaG_inter_copy_package(U, L, L2, package_idx, LookupMode::LaneBody); + std::ignore = luaG_inter_copy_package(U, Source{ L }, Dest{ L2 }, package_idx, LookupMode::LaneBody); } // modules to require in the target lane *before* the function is transfered! @@ -1150,7 +1150,7 @@ LUAG_FUNC(lane_new) } lua_pushnil(L); // func libs priority globals package required gc_cb [... args ...] nil - while( lua_next(L, required_idx) != 0) // func libs priority globals package required gc_cb [... args ...] n "modname" + while (lua_next(L, required_idx) != 0) // func libs priority globals package required gc_cb [... args ...] n "modname" { if (lua_type(L, -1) != LUA_TSTRING || lua_type(L, -2) != LUA_TNUMBER || lua_tonumber(L, -2) != nbRequired) { @@ -1176,7 +1176,7 @@ LUAG_FUNC(lane_new) if (lua_pcall( L2, 1, 1, 0) != LUA_OK) // ret/errcode { // propagate error to main state if any - luaG_inter_move(U, L2, L, 1, LookupMode::LaneBody); // func libs priority globals package required gc_cb [... args ...] n "modname" error + luaG_inter_move(U, Source{ L2 }, Dest{ L }, 1, LookupMode::LaneBody); // func libs priority globals package required gc_cb [... args ...] n "modname" error raise_lua_error(L); } // after requiring the module, register the functions it exported in our name<->function database @@ -1209,7 +1209,7 @@ LUAG_FUNC(lane_new) lua_pushglobaltable(L2); // _G while( lua_next(L, globals_idx)) // func libs priority globals package required gc_cb [... args ...] k v { - luaG_inter_copy(U, L, L2, 2, LookupMode::LaneBody); // _G k v + luaG_inter_copy(U, Source{ L }, Dest{ L2 }, 2, LookupMode::LaneBody); // _G k v // assign it in L2's globals table lua_rawset(L2, -3); // _G lua_pop(L, 1); // func libs priority globals package required gc_cb [... args ...] k @@ -1227,7 +1227,7 @@ LUAG_FUNC(lane_new) DEBUGSPEW_CODE(fprintf( stderr, INDENT_BEGIN "lane_new: transfer lane body\n" INDENT_END)); DEBUGSPEW_CODE(U->debugspew_indent_depth.fetch_add(1, std::memory_order_relaxed)); lua_pushvalue(L, 1); // func libs priority globals package required gc_cb [... args ...] func - int const res{ luaG_inter_move(U, L, L2, 1, LookupMode::LaneBody) };// func libs priority globals package required gc_cb [... args ...] // func + int const res{ luaG_inter_move(U, Source{ L }, Dest{ L2 }, 1, LookupMode::LaneBody) }; // func libs priority globals package required gc_cb [... args ...] // func DEBUGSPEW_CODE(U->debugspew_indent_depth.fetch_sub(1, std::memory_order_relaxed)); if (res != 0) { @@ -1253,7 +1253,7 @@ LUAG_FUNC(lane_new) int res; DEBUGSPEW_CODE(fprintf( stderr, INDENT_BEGIN "lane_new: transfer lane arguments\n" INDENT_END)); DEBUGSPEW_CODE(U->debugspew_indent_depth.fetch_add(1, std::memory_order_relaxed)); - res = luaG_inter_move(U, L, L2, nargs, LookupMode::LaneBody); // func libs priority globals package required gc_cb // func [... args ...] + res = luaG_inter_move(U, Source{ L }, Dest{ L2 }, nargs, LookupMode::LaneBody); // func libs priority globals package required gc_cb // func [... args ...] DEBUGSPEW_CODE(U->debugspew_indent_depth.fetch_sub(1, std::memory_order_relaxed)); if (res != 0) { @@ -1418,7 +1418,7 @@ LUAG_FUNC(thread_join) case Lane::Done: { int const n{ lua_gettop(L2) }; // whole L2 stack - if ((n > 0) && (luaG_inter_move(U, L2, L, n, LookupMode::LaneBody) != 0)) + if ((n > 0) && (luaG_inter_move(U, Source{ L2 }, Dest{ L }, n, LookupMode::LaneBody) != 0)) { return luaL_error(L, "tried to copy unsupported types"); } @@ -1432,7 +1432,7 @@ LUAG_FUNC(thread_join) STACK_GROW(L, 3); lua_pushnil(L); // even when ERROR_FULL_STACK, if the error is not LUA_ERRRUN, the handler wasn't called, and we only have 1 error message on the stack ... - if (luaG_inter_move(U, L2, L, n, LookupMode::LaneBody) != 0) // nil "err" [trace] + if (luaG_inter_move(U, Source{ L2 }, Dest{ L }, n, LookupMode::LaneBody) != 0) // nil "err" [trace] { return luaL_error(L, "tried to copy unsupported types: %s", lua_tostring(L, -n)); } @@ -1832,7 +1832,7 @@ LUAG_FUNC(configure) STACK_CHECK(L, 2); { - char const* errmsg{ push_deep_proxy(L, U->timer_deep, 0, LookupMode::LaneBody) }; // settings M timer_deep + char const* errmsg{ push_deep_proxy(Dest{ L }, U->timer_deep, 0, LookupMode::LaneBody) }; // settings M timer_deep if (errmsg != nullptr) { return luaL_error(L, errmsg); diff --git a/src/lanes_private.h b/src/lanes_private.h index 3ed52fe..0fcbbfc 100644 --- a/src/lanes_private.h +++ b/src/lanes_private.h @@ -98,6 +98,6 @@ static constexpr UniqueKey LANE_POINTER_REGKEY{ 0xB3022205633743BCull }; // used // 'Lane' are malloc/free'd and the handle only carries a pointer. // This is not deep userdata since the handle's not portable among lanes. // -#define lua_toLane( L, i) (*((Lane**) luaL_checkudata( L, i, "Lane"))) +#define lua_toLane(L, i) (*((Lane**) luaL_checkudata( L, i, "Lane"))) -int push_thread_status( lua_State* L, Lane* s); +int push_thread_status(lua_State* L, Lane* s); diff --git a/src/linda.cpp b/src/linda.cpp index fb74abe..39977bc 100644 --- a/src/linda.cpp +++ b/src/linda.cpp @@ -804,7 +804,7 @@ LUAG_FUNC(linda_concat) LUAG_FUNC(linda_dump) { Linda* const linda{ lua_toLinda(L, 1) }; - return keeper_push_linda_storage(linda->U, L, linda, linda->hashSeed()); + return keeper_push_linda_storage(linda->U, Dest{ L }, linda, linda->hashSeed()); } // ################################################################################################# @@ -816,7 +816,7 @@ LUAG_FUNC(linda_dump) LUAG_FUNC(linda_towatch) { Linda* const linda{ lua_toLinda(L, 1) }; - int pushed{ keeper_push_linda_storage(linda->U, L, linda, linda->hashSeed()) }; + int pushed{ keeper_push_linda_storage(linda->U, Dest{ L }, linda, linda->hashSeed()) }; if (pushed == 0) { // if the linda is empty, don't return nil @@ -1009,11 +1009,11 @@ static void* linda_id( lua_State* L, DeepOp op_) */ LUAG_FUNC(linda) { - int const top = lua_gettop(L); + int const top{ lua_gettop(L) }; luaL_argcheck(L, top <= 2, top, "too many arguments"); if (top == 1) { - int const t = lua_type(L, 1); + int const t{ lua_type(L, 1) }; luaL_argcheck(L, t == LUA_TSTRING || t == LUA_TNUMBER, 1, "wrong parameter (should be a string or a number)"); } else if (top == 2) @@ -1021,5 +1021,5 @@ LUAG_FUNC(linda) luaL_checktype(L, 1, LUA_TSTRING); luaL_checktype(L, 2, LUA_TNUMBER); } - return luaG_newdeepuserdata(L, linda_id, 0); + return luaG_newdeepuserdata(Dest{ L }, linda_id, 0); } diff --git a/src/macros_and_utils.h b/src/macros_and_utils.h index 997b452..47ce90c 100644 --- a/src/macros_and_utils.h +++ b/src/macros_and_utils.h @@ -44,8 +44,8 @@ extern char const* debugspew_indent; #else // NDEBUG -#define _ASSERT_L( L, cond_) if( (cond_) == 0) { (void) luaL_error( L, "ASSERT failed: %s:%d '%s'", __FILE__, __LINE__, #cond_);} -#define STACK_DUMP( L) luaG_dump( L) +#define _ASSERT_L(L, cond_) if( (cond_) == 0) { (void) luaL_error(L, "ASSERT failed: %s:%d '%s'", __FILE__, __LINE__, #cond_);} +#define STACK_DUMP(L) luaG_dump(L) class StackChecker { @@ -172,3 +172,17 @@ T* lua_newuserdatauv(lua_State* L, int nuvalue_) } using lua_Duration = std::chrono::template duration; + +// ################################################################################################# + +template +struct Unique +{ + T m_val; + Unique() = default; + operator T() const { return m_val; } + explicit Unique(T b_) : m_val{ b_ } {} +}; + +using Source = Unique; +using Dest = Unique; diff --git a/src/state.cpp b/src/state.cpp index 512009a..6a9ada7 100644 --- a/src/state.cpp +++ b/src/state.cpp @@ -1,5 +1,5 @@ /* -* STATE.C +* STATE.CPP * * Lua tools to support Lanes. */ @@ -31,20 +31,11 @@ THE SOFTWARE. =============================================================================== */ -#include -#include -#include -#include -#include -#if !defined(__APPLE__) -#include -#endif // __APPLE__ - -#include "compat.h" -#include "macros_and_utils.h" -#include "universe.h" -#include "tools.h" +#include "state.h" + #include "lanes.h" +#include "tools.h" +#include "universe.h" // ################################################################################################ @@ -194,9 +185,9 @@ static void open1lib( DEBUGSPEW_PARAM_COMMA( Universe* U) lua_State* L, char con // just like lua_xmove, args are (from, to) -static void copy_one_time_settings( Universe* U, lua_State* L, lua_State* L2) +static void copy_one_time_settings(Universe* U, Source L, Dest L2) { - STACK_GROW( L, 2); + STACK_GROW(L, 2); STACK_CHECK_START_REL(L, 0); STACK_CHECK_START_REL(L2, 0); @@ -326,11 +317,11 @@ void call_on_state_create(Universe* U, lua_State* L, lua_State* from_, LookupMod * *NOT* called for keeper states! * */ -lua_State* luaG_newstate( Universe* U, lua_State* from_, char const* libs_) +lua_State* luaG_newstate(Universe* U, Source from_, char const* libs_) { - lua_State* L = create_state( U, from_); + Dest const L{ create_state(U, from_) }; - STACK_GROW( L, 2); + STACK_GROW(L, 2); STACK_CHECK_START_ABS(L, 0); // copy the universe as a light userdata (only the master state holds the full userdata) diff --git a/src/state.h b/src/state.h index 0e069da..2601f77 100644 --- a/src/state.h +++ b/src/state.h @@ -3,6 +3,7 @@ #include "macros_and_utils.h" // forwards +enum class LookupMode; class Universe; void serialize_require(DEBUGSPEW_PARAM_COMMA(Universe* U) lua_State* L); @@ -10,7 +11,7 @@ void serialize_require(DEBUGSPEW_PARAM_COMMA(Universe* U) lua_State* L); // ################################################################################################ lua_State* create_state(Universe* U, lua_State* from_); -lua_State* luaG_newstate(Universe* U, lua_State* _from, char const* libs); +lua_State* luaG_newstate(Universe* U, Source _from, char const* libs); // ################################################################################################ diff --git a/src/threading.cpp b/src/threading.cpp index 4d210d6..d278bb1 100644 --- a/src/threading.cpp +++ b/src/threading.cpp @@ -1,6 +1,6 @@ /* - * THREADING.C Copyright (c) 2007-08, Asko Kauppi - * Copyright (C) 2009-19, Benoit Germain + * THREADING.CPP Copyright (c) 2007-08, Asko Kauppi + * Copyright (C) 2009-24, Benoit Germain * * Lua Lanes OS threading specific code. * @@ -12,7 +12,7 @@ =============================================================================== Copyright (C) 2007-10 Asko Kauppi -Copyright (C) 2009-14, Benoit Germain +Copyright (C) 2009-24, Benoit Germain Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal @@ -47,12 +47,6 @@ THE SOFTWARE. #endif // __linux__ -#include -#include -#include -#include -#include - #include "threading.h" #if !defined( PLATFORM_XBOX) && !defined( PLATFORM_WIN32) && !defined( PLATFORM_POCKETPC) diff --git a/src/threading.h b/src/threading.h index f38b2de..fc35730 100644 --- a/src/threading.h +++ b/src/threading.h @@ -2,7 +2,6 @@ #include "platform.h" -#include #include #define THREADAPI_WINDOWS 1 diff --git a/src/tools.cpp b/src/tools.cpp index ac5f7c5..07f9ae6 100644 --- a/src/tools.cpp +++ b/src/tools.cpp @@ -31,24 +31,12 @@ THE SOFTWARE. =============================================================================== */ -#include -#include -#include -#include -#include -#if !defined(__APPLE__) -#include -#endif // __APPLE__ - #include "tools.h" -#include "compat.h" + #include "universe.h" -#include "keeper.h" -#include "lanes.h" -#include "uniquekey.h" // functions implemented in deep.c -extern bool copydeep(Universe* U, lua_State* L2, int L2_cache_i, lua_State* L, int i, LookupMode mode_, char const* upName_); +extern bool copydeep(Universe* U, Dest L2, int L2_cache_i, Source L, int i, LookupMode mode_, char const* upName_); extern void push_registry_subtable( lua_State* L, UniqueKey key_); DEBUGSPEW_CODE( char const* debugspew_indent = "----+----!----+----!----+----!----+----!----+----!----+----!----+----!----+"); @@ -554,10 +542,10 @@ static void populate_func_lookup_table_recur(DEBUGSPEW_PARAM_COMMA(Universe* U) /* * create a "fully.qualified.name" <-> function equivalence database */ -void populate_func_lookup_table( lua_State* L, int _i, char const* name_) +void populate_func_lookup_table(lua_State* L, int i_, char const* name_) { - int const ctx_base = lua_gettop( L) + 1; - int const in_base = lua_absindex( L, _i); + int const ctx_base = lua_gettop(L) + 1; + int const in_base = lua_absindex(L, i_); int start_depth = 0; DEBUGSPEW_CODE( Universe* U = universe_get( L)); DEBUGSPEW_CODE( fprintf( stderr, INDENT_BEGIN "%p: populate_func_lookup_table('%s')\n" INDENT_END, L, name_ ? name_ : "nullptr")); @@ -765,7 +753,7 @@ static char const* find_lookup_name(lua_State* L, int i, LookupMode mode_, char /* * Push a looked-up table, or nothing if we found nothing */ -static bool lookup_table(lua_State* L2, lua_State* L, int i, LookupMode mode_, char const* upName_) +static bool lookup_table(Dest L2, Source L, int i, LookupMode mode_, char const* upName_) { // get the name of the table we want to send size_t len; @@ -775,7 +763,7 @@ static bool lookup_table(lua_State* L2, lua_State* L, int i, LookupMode mode_, c return false; } // push the equivalent table in the destination's stack, retrieved from the lookup table - STACK_CHECK_START_REL(L2, 0); // L // L2 + STACK_CHECK_START_REL(L2, 0); // L // L2 STACK_GROW( L2, 3); // up to 3 slots are necessary on error switch( mode_) { @@ -785,34 +773,34 @@ static bool lookup_table(lua_State* L2, lua_State* L, int i, LookupMode mode_, c case LookupMode::ToKeeper: // push a sentinel closure that holds the lookup name as upvalue - lua_pushlstring( L2, fqn, len); // "f.q.n" - lua_pushcclosure( L2, table_lookup_sentinel, 1); // f + lua_pushlstring(L2, fqn, len); // "f.q.n" + lua_pushcclosure(L2, table_lookup_sentinel, 1); // f break; case LookupMode::LaneBody: case LookupMode::FromKeeper: - LOOKUP_REGKEY.pushValue(L2); // {} - STACK_CHECK( L2, 1); - ASSERT_L( lua_istable( L2, -1)); - lua_pushlstring( L2, fqn, len); // {} "f.q.n" - lua_rawget( L2, -2); // {} t + LOOKUP_REGKEY.pushValue(L2); // {} + STACK_CHECK(L2, 1); + ASSERT_L(lua_istable(L2, -1)); + lua_pushlstring(L2, fqn, len); // {} "f.q.n" + lua_rawget(L2, -2); // {} t // we accept destination lookup failures in the case of transfering the Lanes body function (this will result in the source table being cloned instead) // but not when we extract something out of a keeper, as there is nothing to clone! if (lua_isnil(L2, -1) && mode_ == LookupMode::LaneBody) { - lua_pop( L2, 2); // - STACK_CHECK( L2, 0); + lua_pop(L2, 2); // + STACK_CHECK(L2, 0); return false; } - else if( !lua_istable( L2, -1)) + else if( !lua_istable(L2, -1)) { char const* from, *to; - lua_getglobal( L, "decoda_name"); // ... t ... decoda_name - from = lua_tostring( L, -1); - lua_pop( L, 1); // ... t ... - lua_getglobal( L2, "decoda_name"); // {} t decoda_name + lua_getglobal(L, "decoda_name"); // ... t ... decoda_name + from = lua_tostring(L, -1); + lua_pop(L, 1); // ... t ... + lua_getglobal(L2, "decoda_name"); // {} t decoda_name to = lua_tostring( L2, -1); - lua_pop( L2, 1); // {} t + lua_pop(L2, 1); // {} t // when mode_ == LookupMode::FromKeeper, L is a keeper state and L2 is not, therefore L2 is the state where we want to raise the error (void) luaL_error( (mode_ == LookupMode::FromKeeper) ? L2 : L @@ -823,7 +811,7 @@ static bool lookup_table(lua_State* L2, lua_State* L, int i, LookupMode mode_, c ); return false; } - lua_remove( L2, -2); // t + lua_remove(L2, -2); // t break; } STACK_CHECK( L2, 1); @@ -1194,7 +1182,7 @@ static int buf_writer( lua_State* L, void const* b, size_t size, void* ud) // ################################################################################################# -static void copy_func(Universe* U, lua_State* L2, int L2_cache_i, lua_State* L, int i, LookupMode mode_, char const* upName_) +static void copy_func(Universe* U, Dest L2, int L2_cache_i, Source L, int i, LookupMode mode_, char const* upName_) { int n, needToPush; luaL_Buffer B; @@ -1348,7 +1336,7 @@ static void copy_func(Universe* U, lua_State* L2, int L2_cache_i, lua_State* L, * * Always pushes a function to 'L2'. */ -static void copy_cached_func(Universe* U, lua_State* L2, int L2_cache_i, lua_State* L, int i, LookupMode mode_, char const* upName_) +static void copy_cached_func(Universe* U, Dest L2, int L2_cache_i, Source L, int i, LookupMode mode_, char const* upName_) { FuncSubType funcSubType; /*lua_CFunction cfunc =*/ luaG_tocfunction( L, i, &funcSubType); // nullptr for LuaJIT-fast && bytecode functions @@ -1403,7 +1391,7 @@ static void copy_cached_func(Universe* U, lua_State* L2, int L2_cache_i, lua_Sta // ################################################################################################# -static bool push_cached_metatable(Universe* U, lua_State* L2, int L2_cache_i, lua_State* L, int i, LookupMode mode_, char const* upName_) +static bool push_cached_metatable(Universe* U, Dest L2, int L2_cache_i, Source L, int i, LookupMode mode_, char const* upName_) { STACK_CHECK_START_REL(L, 0); if( lua_getmetatable( L, i)) // ... mt @@ -1454,7 +1442,7 @@ static bool push_cached_metatable(Universe* U, lua_State* L2, int L2_cache_i, lu // ################################################################################################# -static void inter_copy_keyvaluepair(Universe* U, lua_State* L2, int L2_cache_i, lua_State* L, VT vt_, LookupMode mode_, char const* upName_) +static void inter_copy_keyvaluepair(Universe* U, Dest L2, int L2_cache_i, Source L, VT vt_, LookupMode mode_, char const* upName_) { int val_i = lua_gettop(L); int key_i = val_i - 1; @@ -1526,7 +1514,7 @@ static void inter_copy_keyvaluepair(Universe* U, lua_State* L2, int L2_cache_i, */ static constexpr UniqueKey CLONABLES_CACHE_KEY{ 0xD04EE018B3DEE8F5ull }; -static bool copyclone(Universe* U, lua_State* L2, int L2_cache_i, lua_State* L, int source_i_, LookupMode mode_, char const* upName_) +static bool copyclone(Universe* U, Dest L2, int L2_cache_i, Source L, int source_i_, LookupMode mode_, char const* upName_) { void* const source = lua_touserdata( L, source_i_); source_i_ = lua_absindex( L, source_i_); @@ -1641,7 +1629,7 @@ static bool copyclone(Universe* U, lua_State* L2, int L2_cache_i, lua_State* L, // ################################################################################################# -static bool inter_copy_userdata(Universe* U, lua_State* L2, int L2_cache_i, lua_State* L, int i, VT vt_, LookupMode mode_, char const* upName_) +static bool inter_copy_userdata(Universe* U, Dest L2, int L2_cache_i, Source L, int i, VT vt_, LookupMode mode_, char const* upName_) { STACK_CHECK_START_REL(L, 0); STACK_CHECK_START_REL(L2, 0); @@ -1691,7 +1679,7 @@ static bool inter_copy_userdata(Universe* U, lua_State* L2, int L2_cache_i, lua_ // ################################################################################################# -static bool inter_copy_function(Universe* U, lua_State* L2, int L2_cache_i, lua_State* L, int source_i_, VT vt_, LookupMode mode_, char const* upName_) +static bool inter_copy_function(Universe* U, Dest L2, int L2_cache_i, Source L, int source_i_, VT vt_, LookupMode mode_, char const* upName_) { if (vt_ == VT::KEY) { @@ -1786,7 +1774,7 @@ static bool inter_copy_function(Universe* U, lua_State* L2, int L2_cache_i, lua_ // ################################################################################################# -static bool inter_copy_table(Universe* U, lua_State* L2, int L2_cache_i, lua_State* L, int i, VT vt_, LookupMode mode_, char const* upName_) +static bool inter_copy_table(Universe* U, Dest L2, int L2_cache_i, Source L, int i, VT vt_, LookupMode mode_, char const* upName_) { if (vt_ == VT::KEY) { @@ -1858,7 +1846,7 @@ static bool inter_copy_table(Universe* U, lua_State* L2, int L2_cache_i, lua_Sta * * Returns true if value was pushed, false if its type is non-supported. */ -bool inter_copy_one(Universe* U, lua_State* L2, int L2_cache_i, lua_State* L, int i, VT vt_, LookupMode mode_, char const* upName_) +bool inter_copy_one(Universe* U, Dest L2, int L2_cache_i, Source L, int i, VT vt_, LookupMode mode_, char const* upName_) { bool ret{ true }; int val_type = lua_type( L, i); @@ -1974,6 +1962,8 @@ bool inter_copy_one(Universe* U, lua_State* L2, int L2_cache_i, lua_State* L, in return ret; } +// ################################################################################################# + /* * Akin to 'lua_xmove' but copies values between _any_ Lua states. * @@ -1981,7 +1971,7 @@ bool inter_copy_one(Universe* U, lua_State* L2, int L2_cache_i, lua_State* L, in * * Note: Parameters are in this order ('L' = from first) to be same as 'lua_xmove'. */ -int luaG_inter_copy(Universe* U, lua_State* L, lua_State* L2, int n, LookupMode mode_) +int luaG_inter_copy(Universe* U, Source L, Dest L2, int n, LookupMode mode_) { int top_L = lua_gettop(L); // ... {}n int top_L2 = lua_gettop(L2); // ... @@ -2043,15 +2033,16 @@ int luaG_inter_copy(Universe* U, lua_State* L, lua_State* L2, int n, LookupMode return -2; } +// ################################################################################################# -int luaG_inter_move(Universe* U, lua_State* L, lua_State* L2, int n, LookupMode mode_) +int luaG_inter_move(Universe* U, Source L, Dest L2, int n, LookupMode mode_) { int ret = luaG_inter_copy( U, L, L2, n, mode_); lua_pop( L, (int) n); return ret; } -int luaG_inter_copy_package(Universe* U, lua_State* L, lua_State* L2, int package_idx_, LookupMode mode_) +int luaG_inter_copy_package(Universe* U, Source L, Dest L2, int package_idx_, LookupMode mode_) { DEBUGSPEW_CODE(fprintf(stderr, INDENT_BEGIN "luaG_inter_copy_package()\n" INDENT_END)); DEBUGSPEW_CODE(U->debugspew_indent_depth.fetch_add(1, std::memory_order_relaxed)); diff --git a/src/tools.h b/src/tools.h index 7d9aaab..8e95a4f 100644 --- a/src/tools.h +++ b/src/tools.h @@ -25,19 +25,19 @@ enum class VT KEY, METATABLE }; -bool inter_copy_one(Universe* U, lua_State* L2, int L2_cache_i, lua_State* L, int i, VT vt_, LookupMode mode_, char const* upName_); +bool inter_copy_one(Universe* U, Dest L2, int L2_cache_i, Source L, int i, VT vt_, LookupMode mode_, char const* upName_); // ################################################################################################ -int luaG_inter_copy_package( Universe* U, lua_State* L, lua_State* L2, int package_idx_, LookupMode mode_); +int luaG_inter_copy_package(Universe* U, Source L, Dest L2, int package_idx_, LookupMode mode_); -int luaG_inter_copy(Universe* U, lua_State* L, lua_State* L2, int n, LookupMode mode_); -int luaG_inter_move(Universe* U, lua_State* L, lua_State* L2, int n, LookupMode mode_); +int luaG_inter_copy(Universe* U, Source L, Dest L2, int n, LookupMode mode_); +int luaG_inter_move(Universe* U, Source L, Dest L2, int n, LookupMode mode_); -int luaG_nameof( lua_State* L); +int luaG_nameof(lua_State* L); -void populate_func_lookup_table( lua_State* L, int _i, char const* _name); -void initialize_allocator_function( Universe* U, lua_State* L); +void populate_func_lookup_table(lua_State* L, int _i, char const* _name); +void initialize_allocator_function(Universe* U, lua_State* L); // ################################################################################################ diff --git a/src/universe.cpp b/src/universe.cpp index 290e547..4c53987 100644 --- a/src/universe.cpp +++ b/src/universe.cpp @@ -80,7 +80,7 @@ Universe* universe_create(lua_State* L) U->Universe::Universe(); STACK_CHECK_START_REL(L, 1); UNIVERSE_FULL_REGKEY.setValue(L, [](lua_State* L) { lua_pushvalue(L, -2); }); - UNIVERSE_LIGHT_REGKEY.setValue(L, [U](lua_State* L) { lua_pushlightuserdata( L, U); }); + UNIVERSE_LIGHT_REGKEY.setValue(L, [U](lua_State* L) { lua_pushlightuserdata(L, U); }); STACK_CHECK(L, 1); return U; } @@ -91,8 +91,8 @@ void universe_store(lua_State* L, Universe* U) { ASSERT_L(!U || universe_get(L) == nullptr); STACK_CHECK_START_REL(L, 0); - UNIVERSE_LIGHT_REGKEY.setValue(L, [U](lua_State* L) { U ? lua_pushlightuserdata( L, U) : lua_pushnil( L); }); - STACK_CHECK( L, 0); + UNIVERSE_LIGHT_REGKEY.setValue(L, [U](lua_State* L) { U ? lua_pushlightuserdata(L, U) : lua_pushnil(L); }); + STACK_CHECK(L, 0); } // ################################################################################################ @@ -101,6 +101,6 @@ Universe* universe_get(lua_State* L) { STACK_CHECK_START_REL(L, 0); Universe* const universe{ UNIVERSE_LIGHT_REGKEY.readLightUserDataValue(L) }; - STACK_CHECK( L, 0); + STACK_CHECK(L, 0); return universe; } -- cgit v1.2.3-55-g6feb From aa9b3594e11498ca0977d87a9d0875d7fc107253 Mon Sep 17 00:00:00 2001 From: Benoit Germain Date: Tue, 9 Apr 2024 17:08:13 +0200 Subject: C++ migration: [[nodiscard]] everywhere. still have to check all std::ignore --- deep_test/deep_test.cpp | 278 ++++++++++++++++++++++++------------------------ src/cancel.cpp | 12 +-- src/cancel.h | 4 +- src/compat.cpp | 6 +- src/deep.cpp | 12 ++- src/deep.h | 10 +- src/keeper.cpp | 58 +++++----- src/keeper.h | 26 ++--- src/lanes.cpp | 55 +++++----- src/lanes.h | 2 +- src/lanes_private.h | 11 +- src/linda.cpp | 12 +-- src/macros_and_utils.h | 15 +-- src/state.cpp | 6 +- src/state.h | 4 +- src/threading.cpp | 4 +- src/threading_osx.h | 10 +- src/tools.cpp | 260 ++++++++++++++++++++++---------------------- src/tools.h | 10 +- src/universe.h | 10 +- 20 files changed, 410 insertions(+), 395 deletions(-) (limited to 'src/tools.h') diff --git a/deep_test/deep_test.cpp b/deep_test/deep_test.cpp index 7c8180f..3467939 100644 --- a/deep_test/deep_test.cpp +++ b/deep_test/deep_test.cpp @@ -10,114 +10,114 @@ // a lanes-deep userdata. needs DeepPrelude and luaG_newdeepuserdata from Lanes code. struct MyDeepUserdata : public DeepPrelude // Deep userdata MUST start with a DeepPrelude { - lua_Integer val{ 0 }; + lua_Integer val{ 0 }; }; // ################################################################################################ -static void* deep_test_id( lua_State* L, DeepOp op_) +[[nodiscard]] static void* deep_test_id(lua_State* L, DeepOp op_) { - switch( op_) - { - case DeepOp::New: - { - MyDeepUserdata* deep_test = new MyDeepUserdata; - return deep_test; - } - - case DeepOp::Delete: - { - MyDeepUserdata* deep_test = static_cast(lua_touserdata( L, 1)); - delete deep_test; - return nullptr; - } - - case DeepOp::Metatable: - { - luaL_getmetatable( L, "deep"); // mt - return nullptr; - } - - case DeepOp::Module: - return (void*)"deep_test"; - - default: - { - return nullptr; - } - } + switch( op_) + { + case DeepOp::New: + { + MyDeepUserdata* deep_test = new MyDeepUserdata; + return deep_test; + } + + case DeepOp::Delete: + { + MyDeepUserdata* deep_test = static_cast(lua_touserdata( L, 1)); + delete deep_test; + return nullptr; + } + + case DeepOp::Metatable: + { + luaL_getmetatable( L, "deep"); // mt + return nullptr; + } + + case DeepOp::Module: + return (void*)"deep_test"; + + default: + { + return nullptr; + } + } } // ################################################################################################ -static int deep_set( lua_State* L) +[[nodiscard]] static int deep_set(lua_State* L) { - MyDeepUserdata* self = static_cast(luaG_todeep(L, deep_test_id, 1)); - lua_Integer i = lua_tointeger( L, 2); - self->val = i; - return 0; + MyDeepUserdata* self = static_cast(luaG_todeep(L, deep_test_id, 1)); + lua_Integer i = lua_tointeger( L, 2); + self->val = i; + return 0; } // ################################################################################################ // won't actually do anything as deep userdata don't have uservalue slots -static int deep_setuv( lua_State* L) +[[nodiscard]] static int deep_setuv(lua_State* L) { - MyDeepUserdata* self = static_cast(luaG_todeep(L, deep_test_id, 1)); - int uv = (int) luaL_optinteger(L, 2, 1); - lua_settop( L, 3); - lua_pushboolean( L, lua_setiuservalue( L, 1, uv) != 0); - return 1; + MyDeepUserdata* self = static_cast(luaG_todeep(L, deep_test_id, 1)); + int uv = (int) luaL_optinteger(L, 2, 1); + lua_settop( L, 3); + lua_pushboolean( L, lua_setiuservalue( L, 1, uv) != 0); + return 1; } // ################################################################################################ // won't actually do anything as deep userdata don't have uservalue slots -static int deep_getuv( lua_State* L) +[[nodiscard]] static int deep_getuv(lua_State* L) { - MyDeepUserdata* self = static_cast(luaG_todeep(L, deep_test_id, 1)); - int uv = (int) luaL_optinteger(L, 2, 1); - lua_getiuservalue( L, 1, uv); - return 1; + MyDeepUserdata* self = static_cast(luaG_todeep(L, deep_test_id, 1)); + int uv = (int) luaL_optinteger(L, 2, 1); + lua_getiuservalue( L, 1, uv); + return 1; } // ################################################################################################ -static int deep_tostring( lua_State* L) +[[nodiscard]] static int deep_tostring(lua_State* L) { - MyDeepUserdata* self = static_cast(luaG_todeep(L, deep_test_id, 1)); - lua_pushfstring(L, "%p:deep(%d)", lua_topointer(L, 1), self->val); - return 1; + MyDeepUserdata* self = static_cast(luaG_todeep(L, deep_test_id, 1)); + lua_pushfstring(L, "%p:deep(%d)", lua_topointer(L, 1), self->val); + return 1; } // ################################################################################################ -static int deep_gc( lua_State* L) +[[nodiscard]] static int deep_gc(lua_State* L) { - MyDeepUserdata* self = static_cast(luaG_todeep(L, deep_test_id, 1)); - return 0; + MyDeepUserdata* self = static_cast(luaG_todeep(L, deep_test_id, 1)); + return 0; } // ################################################################################################ static luaL_Reg const deep_mt[] = { - { "__tostring", deep_tostring}, - { "__gc", deep_gc}, - { "set", deep_set}, - { "setuv", deep_setuv}, - { "getuv", deep_getuv}, - { nullptr, nullptr } + { "__tostring", deep_tostring}, + { "__gc", deep_gc}, + { "set", deep_set}, + { "setuv", deep_setuv}, + { "getuv", deep_getuv}, + { nullptr, nullptr } }; // ################################################################################################ int luaD_new_deep( lua_State* L) { - int nuv = (int) luaL_optinteger( L, 1, 0); - // no additional parameter to luaG_newdeepuserdata! - lua_settop( L, 0); - return luaG_newdeepuserdata( L, deep_test_id, nuv); + int const nuv{ static_cast(luaL_optinteger(L, 1, 0)) }; + // no additional parameter to luaG_newdeepuserdata! + lua_settop(L, 0); + return luaG_newdeepuserdata(Dest{ L }, deep_test_id, nuv); } // ################################################################################################ @@ -125,101 +125,101 @@ int luaD_new_deep( lua_State* L) struct MyClonableUserdata { - lua_Integer val; + lua_Integer val; }; // ################################################################################################ -static int clonable_set( lua_State* L) +[[nodiscard]] static int clonable_set(lua_State* L) { - MyClonableUserdata* self = static_cast(lua_touserdata(L, 1)); - lua_Integer i = lua_tointeger(L, 2); - self->val = i; - return 0; + MyClonableUserdata* self = static_cast(lua_touserdata(L, 1)); + lua_Integer i = lua_tointeger(L, 2); + self->val = i; + return 0; } // ################################################################################################ -static int clonable_setuv( lua_State* L) +[[nodiscard]] static int clonable_setuv(lua_State* L) { - MyClonableUserdata* self = static_cast(lua_touserdata(L, 1)); - int uv = (int) luaL_optinteger(L, 2, 1); - lua_settop( L, 3); - lua_pushboolean( L, lua_setiuservalue( L, 1, uv) != 0); - return 1; + MyClonableUserdata* self = static_cast(lua_touserdata(L, 1)); + int uv = (int) luaL_optinteger(L, 2, 1); + lua_settop( L, 3); + lua_pushboolean( L, lua_setiuservalue( L, 1, uv) != 0); + return 1; } // ################################################################################################ -static int clonable_getuv( lua_State* L) +[[nodiscard]] static int clonable_getuv(lua_State* L) { - MyClonableUserdata* self = static_cast(lua_touserdata(L, 1)); - int uv = (int) luaL_optinteger(L, 2, 1); - lua_getiuservalue( L, 1, uv); - return 1; + MyClonableUserdata* self = static_cast(lua_touserdata(L, 1)); + int uv = (int) luaL_optinteger(L, 2, 1); + lua_getiuservalue( L, 1, uv); + return 1; } // ################################################################################################ -static int clonable_tostring(lua_State* L) +[[nodiscard]] static int clonable_tostring(lua_State* L) { - MyClonableUserdata* self = static_cast(lua_touserdata(L, 1)); - lua_pushfstring(L, "%p:clonable(%d)", lua_topointer(L, 1), self->val); - return 1; + MyClonableUserdata* self = static_cast(lua_touserdata(L, 1)); + lua_pushfstring(L, "%p:clonable(%d)", lua_topointer(L, 1), self->val); + return 1; } // ################################################################################################ -static int clonable_gc( lua_State* L) +[[nodiscard]] static int clonable_gc(lua_State* L) { - MyClonableUserdata* self = static_cast(lua_touserdata(L, 1)); - return 0; + MyClonableUserdata* self = static_cast(lua_touserdata(L, 1)); + return 0; } // ################################################################################################ // this is all we need to make a userdata lanes-clonable. no dependency on Lanes code. -static int clonable_lanesclone( lua_State* L) +[[nodiscard]] static int clonable_lanesclone(lua_State* L) { - switch( lua_gettop( L)) - { - case 3: - { - MyClonableUserdata* self = static_cast(lua_touserdata(L, 1)); - MyClonableUserdata* from = static_cast(lua_touserdata(L, 2)); - size_t len = lua_tointeger( L, 3); - assert( len == sizeof(MyClonableUserdata)); - *self = *from; - } - return 0; - - default: - (void) luaL_error( L, "Lanes called clonable_lanesclone with unexpected parameters"); - } - return 0; + switch( lua_gettop( L)) + { + case 3: + { + MyClonableUserdata* self = static_cast(lua_touserdata(L, 1)); + MyClonableUserdata* from = static_cast(lua_touserdata(L, 2)); + size_t len = lua_tointeger( L, 3); + assert( len == sizeof(MyClonableUserdata)); + *self = *from; + } + return 0; + + default: + (void) luaL_error( L, "Lanes called clonable_lanesclone with unexpected parameters"); + } + return 0; } // ################################################################################################ static luaL_Reg const clonable_mt[] = { - { "__tostring", clonable_tostring}, - { "__gc", clonable_gc}, - { "__lanesclone", clonable_lanesclone}, - { "set", clonable_set}, - { "setuv", clonable_setuv}, - { "getuv", clonable_getuv}, - { nullptr, nullptr } + { "__tostring", clonable_tostring}, + { "__gc", clonable_gc}, + { "__lanesclone", clonable_lanesclone}, + { "set", clonable_set}, + { "setuv", clonable_setuv}, + { "getuv", clonable_getuv}, + { nullptr, nullptr } }; // ################################################################################################ int luaD_new_clonable( lua_State* L) { - int nuv = (int) luaL_optinteger( L, 1, 1); - lua_newuserdatauv( L, sizeof(MyClonableUserdata), nuv); - luaL_setmetatable( L, "clonable"); - return 1; + int const nuv{ static_cast(luaL_optinteger(L, 1, 1)) }; + lua_newuserdatauv( L, sizeof(MyClonableUserdata), nuv); + luaL_setmetatable( L, "clonable"); + return 1; } // ################################################################################################ @@ -227,33 +227,33 @@ int luaD_new_clonable( lua_State* L) static luaL_Reg const deep_module[] = { - { "new_deep", luaD_new_deep}, - { "new_clonable", luaD_new_clonable}, - { nullptr, nullptr } + { "new_deep", luaD_new_deep}, + { "new_clonable", luaD_new_clonable}, + { nullptr, nullptr } }; // ################################################################################################ LANES_API int luaopen_deep_test(lua_State* L) { - luaL_newlib( L, deep_module); // M - - // preregister the metatables for the types we can instantiate so that Lanes can know about them - if( luaL_newmetatable( L, "clonable")) // M mt - { - luaL_setfuncs( L, clonable_mt, 0); - lua_pushvalue(L, -1); // M mt mt - lua_setfield(L, -2, "__index"); // M mt - } - lua_setfield(L, -2, "__clonableMT"); // M - - if( luaL_newmetatable( L, "deep")) // mt - { - luaL_setfuncs( L, deep_mt, 0); - lua_pushvalue(L, -1); // mt mt - lua_setfield(L, -2, "__index"); // mt - } - lua_setfield(L, -2, "__deepMT"); // M - - return 1; + luaL_newlib( L, deep_module); // M + + // preregister the metatables for the types we can instantiate so that Lanes can know about them + if( luaL_newmetatable( L, "clonable")) // M mt + { + luaL_setfuncs( L, clonable_mt, 0); + lua_pushvalue(L, -1); // M mt mt + lua_setfield(L, -2, "__index"); // M mt + } + lua_setfield(L, -2, "__clonableMT"); // M + + if( luaL_newmetatable( L, "deep")) // mt + { + luaL_setfuncs( L, deep_mt, 0); + lua_pushvalue(L, -1); // mt mt + lua_setfield(L, -2, "__index"); // mt + } + lua_setfield(L, -2, "__deepMT"); // M + + return 1; } diff --git a/src/cancel.cpp b/src/cancel.cpp index e08e975..2f3c22e 100644 --- a/src/cancel.cpp +++ b/src/cancel.cpp @@ -51,7 +51,7 @@ THE SOFTWARE. * Returns CANCEL_SOFT/HARD if any locks are to be exited, and 'raise_cancel_error()' called, * to make execution of the lane end. */ -static inline CancelRequest cancel_test(lua_State* L) +[[nodiscard]] static inline CancelRequest cancel_test(lua_State* L) { Lane* const lane{ LANE_POINTER_REGKEY.readLightUserDataValue(L) }; // 'lane' is nullptr for the original main state (and no-one can cancel that) @@ -76,7 +76,7 @@ LUAG_FUNC( cancel_test) // ################################################################################################ // ################################################################################################ -static void cancel_hook(lua_State* L, [[maybe_unused]] lua_Debug* ar) +[[nodiscard]] static void cancel_hook(lua_State* L, [[maybe_unused]] lua_Debug* ar) { DEBUGSPEW_CODE(fprintf(stderr, "cancel_hook\n")); if (cancel_test(L) != CancelRequest::None) @@ -108,7 +108,7 @@ static void cancel_hook(lua_State* L, [[maybe_unused]] lua_Debug* ar) // ################################################################################################ -static CancelResult thread_cancel_soft(Lane* lane_, lua_Duration duration_, bool wake_lane_) +[[nodiscard]] static CancelResult thread_cancel_soft(Lane* lane_, lua_Duration duration_, bool wake_lane_) { lane_->cancel_request = CancelRequest::Soft; // it's now signaled to stop // negative timeout: we don't want to truly abort the lane, we just want it to react to cancel_test() on its own @@ -126,7 +126,7 @@ static CancelResult thread_cancel_soft(Lane* lane_, lua_Duration duration_, bool // ################################################################################################ -static CancelResult thread_cancel_hard(Lane* lane_, lua_Duration duration_, bool wake_lane_) +[[nodiscard]] static CancelResult thread_cancel_hard(Lane* lane_, lua_Duration duration_, bool wake_lane_) { lane_->cancel_request = CancelRequest::Hard; // it's now signaled to stop //lane_->m_thread.get_stop_source().request_stop(); @@ -204,7 +204,7 @@ CancelOp which_cancel_op(char const* op_string_) // ################################################################################################ -static CancelOp which_cancel_op(lua_State* L, int idx_) +[[nodiscard]] static CancelOp which_cancel_op(lua_State* L, int idx_) { if (lua_type(L, idx_) == LUA_TSTRING) { @@ -273,7 +273,7 @@ LUAG_FUNC(thread_cancel) case CancelResult::Cancelled: lua_pushboolean(L, 1); - push_thread_status(L, lane); + std::ignore = push_thread_status(L, lane); break; } // should never happen, only here to prevent the compiler from complaining of "not all control paths returning a value" diff --git a/src/cancel.h b/src/cancel.h index 954b04e..060edb3 100644 --- a/src/cancel.h +++ b/src/cancel.h @@ -49,8 +49,8 @@ enum class CancelOp // crc64/we of string "CANCEL_ERROR" generated at http://www.nitrxgen.net/hashgen/ static constexpr UniqueKey CANCEL_ERROR{ 0xe97d41626cc97577ull }; // 'raise_cancel_error' sentinel -CancelOp which_cancel_op(char const* op_string_); -CancelResult thread_cancel(Lane* lane_, CancelOp op_, int hook_count_, lua_Duration secs_, bool wake_lindas_); +[[nodiscard]] CancelOp which_cancel_op(char const* op_string_); +[[nodiscard]] CancelResult thread_cancel(Lane* lane_, CancelOp op_, int hook_count_, lua_Duration secs_, bool wake_lindas_); [[noreturn]] static inline void raise_cancel_error(lua_State* L) { diff --git a/src/compat.cpp b/src/compat.cpp index 8acab25..9807390 100644 --- a/src/compat.cpp +++ b/src/compat.cpp @@ -15,7 +15,7 @@ // ################################################################################################ // ################################################################################################ -static int luaL_getsubtable (lua_State *L, int idx, const char *fname) +[[nodiscard]] static int luaL_getsubtable(lua_State* L, int idx, const char* fname) { lua_getfield(L, idx, fname); if (lua_istable(L, -1)) @@ -33,12 +33,12 @@ static int luaL_getsubtable (lua_State *L, int idx, const char *fname) // ################################################################################################ -void luaL_requiref (lua_State *L, const char *modname, lua_CFunction openf, int glb) +void luaL_requiref(lua_State *L, const char *modname, lua_CFunction openf, int glb) { lua_pushcfunction(L, openf); lua_pushstring(L, modname); /* argument to open function */ lua_call(L, 1, 1); /* open module */ - luaL_getsubtable(L, LUA_REGISTRYINDEX, LUA_LOADED_TABLE); + std::ignore = luaL_getsubtable(L, LUA_REGISTRYINDEX, LUA_LOADED_TABLE); lua_pushvalue(L, -2); /* make copy of module (call result) */ lua_setfield(L, -2, modname); /* _LOADED[modname] = module */ lua_pop(L, 1); /* remove _LOADED table */ diff --git a/src/deep.cpp b/src/deep.cpp index 55063b3..a3806aa 100644 --- a/src/deep.cpp +++ b/src/deep.cpp @@ -107,7 +107,7 @@ static void get_deep_lookup(lua_State* L) * Return the registered ID function for 'index' (deep userdata proxy), * or nullptr if 'index' is not a deep userdata proxy. */ -static inline luaG_IdFunction get_idfunc(lua_State* L, int index, LookupMode mode_) +[[nodiscard]] static inline luaG_IdFunction get_idfunc(lua_State* L, int index, LookupMode mode_) { // when looking inside a keeper, we are 100% sure the object is a deep userdata if (mode_ == LookupMode::FromKeeper) @@ -160,7 +160,7 @@ void free_deep_prelude(lua_State* L, DeepPrelude* prelude_) * End of life for a proxy object; reduce the deep reference count and clean it up if reaches 0. * */ -static int deep_userdata_gc(lua_State* L) +[[nodiscard]] static int deep_userdata_gc(lua_State* L) { DeepPrelude** const proxy{ lua_tofulluserdata(L, 1) }; DeepPrelude* p = *proxy; @@ -470,10 +470,14 @@ bool copydeep(Universe* U, Dest L2, int L2_cache_i, Source L, int i, LookupMode int const clone_i = lua_gettop( L2); while( nuv) { - inter_copy_one( U, L2, L2_cache_i, L, lua_absindex( L, -1), VT::NORMAL, mode_, upName_); // u uv + std::ignore = inter_copy_one(U + , L2, L2_cache_i + , L, lua_absindex( L, -1) + , VT::NORMAL, mode_, upName_ + ); // u uv lua_pop( L, 1); // ... u [uv]* // this pops the value from the stack - lua_setiuservalue( L2, clone_i, nuv); // u + lua_setiuservalue(L2, clone_i, nuv); // u -- nuv; } } diff --git a/src/deep.h b/src/deep.h index 6cf33ec..7be5c5d 100644 --- a/src/deep.h +++ b/src/deep.h @@ -36,7 +36,7 @@ enum class DeepOp Module, }; -using luaG_IdFunction = void*(*)( lua_State* L, DeepOp op_); +using luaG_IdFunction = void*(*)(lua_State* L, DeepOp op_); // ################################################################################################ @@ -54,8 +54,8 @@ struct DeepPrelude std::atomic m_refcount{ 0 }; }; -char const* push_deep_proxy(Dest L, DeepPrelude* prelude, int nuv_, LookupMode mode_); -void free_deep_prelude( lua_State* L, DeepPrelude* prelude_); +[[nodiscard]] char const* push_deep_proxy(Dest L, DeepPrelude* prelude, int nuv_, LookupMode mode_); +void free_deep_prelude(lua_State* L, DeepPrelude* prelude_); -LANES_API int luaG_newdeepuserdata(Dest L, luaG_IdFunction idfunc, int nuv_); -LANES_API DeepPrelude* luaG_todeep(lua_State* L, luaG_IdFunction idfunc, int index); +LANES_API [[nodiscard]] int luaG_newdeepuserdata(Dest L, luaG_IdFunction idfunc, int nuv_); +LANES_API [[nodiscard]] DeepPrelude* luaG_todeep(lua_State* L, luaG_IdFunction idfunc, int index); diff --git a/src/keeper.cpp b/src/keeper.cpp index 9718bda..19fbd06 100644 --- a/src/keeper.cpp +++ b/src/keeper.cpp @@ -61,12 +61,12 @@ class keeper_fifo int limit{ -1 }; // a fifo full userdata has one uservalue, the table that holds the actual fifo contents - static void* operator new([[maybe_unused]] size_t size_, lua_State* L) noexcept { return lua_newuserdatauv(L, 1); } + [[nodiscard]] static void* operator new([[maybe_unused]] size_t size_, lua_State* L) noexcept { return lua_newuserdatauv(L, 1); } // always embedded somewhere else or "in-place constructed" as a full userdata // can't actually delete the operator because the compiler generates stack unwinding code that could call it in case of exception static void operator delete([[maybe_unused]] void* p_, lua_State* L) { ASSERT_L(!"should never be called") }; - static keeper_fifo* getPtr(lua_State* L, int idx_) + [[nodiscard]] static keeper_fifo* getPtr(lua_State* L, int idx_) { return lua_tofulluserdata(L, idx_); } @@ -77,7 +77,7 @@ static constexpr int CONTENTS_TABLE{ 1 }; // ################################################################################################## // replaces the fifo ud by its uservalue on the stack -static keeper_fifo* prepare_fifo_access(lua_State* L, int idx_) +[[nodiscard]] static keeper_fifo* prepare_fifo_access(lua_State* L, int idx_) { keeper_fifo* const fifo{ keeper_fifo::getPtr(L, idx_) }; if (fifo != nullptr) @@ -95,7 +95,7 @@ static keeper_fifo* prepare_fifo_access(lua_State* L, int idx_) // in: nothing // out: { first = 1, count = 0, limit = -1} -static keeper_fifo* fifo_new(lua_State* L) +[[nodiscard]] static keeper_fifo* fifo_new(lua_State* L) { STACK_GROW(L, 2); STACK_CHECK_START_REL(L, 0); @@ -213,46 +213,46 @@ int keeper_push_linda_storage(Universe* U, Dest L, void* ptr_, uintptr_t magic_) Source const KL{ K ? K->L : nullptr }; if (KL == nullptr) return 0; - STACK_GROW(KL, 4); + STACK_GROW(KL, 4); // KEEPER MAIN STACK_CHECK_START_REL(KL, 0); - FIFOS_KEY.pushValue(KL); // fifos - lua_pushlightuserdata(KL, ptr_); // fifos ud - lua_rawget(KL, -2); // fifos storage - lua_remove(KL, -2); // storage + FIFOS_KEY.pushValue(KL); // fifos + lua_pushlightuserdata(KL, ptr_); // fifos ud + lua_rawget(KL, -2); // fifos storage + lua_remove(KL, -2); // storage if (!lua_istable(KL, -1)) { - lua_pop(KL, 1); // + lua_pop(KL, 1); // STACK_CHECK(KL, 0); return 0; } - // move data from keeper to destination state KEEPER MAIN - lua_pushnil(KL); // storage nil + // move data from keeper to destination state + lua_pushnil(KL); // storage nil STACK_GROW(L, 5); STACK_CHECK_START_REL(L, 0); lua_newtable(L); // out - while (lua_next(KL, -2)) // storage key fifo + while (lua_next(KL, -2)) // storage key fifo { - keeper_fifo* fifo = prepare_fifo_access(KL, -1); // storage key fifotbl - lua_pushvalue(KL, -2); // storage key fifotbl key - luaG_inter_move(U, KL, L, 1, LookupMode::FromKeeper); // storage key fifotbl // out key + keeper_fifo* fifo = prepare_fifo_access(KL, -1); // storage key fifotbl + lua_pushvalue(KL, -2); // storage key fifotbl key + std::ignore = luaG_inter_move(U, KL, L, 1, LookupMode::FromKeeper); // storage key fifotbl // out key STACK_CHECK(L, 2); - lua_newtable(L); // out key keyout - luaG_inter_move(U, KL, L, 1, LookupMode::FromKeeper); // storage key // out key keyout fifotbl - lua_pushinteger(L, fifo->first); // out key keyout fifotbl first + lua_newtable(L); // out key keyout + std::ignore = luaG_inter_move(U, KL, L, 1, LookupMode::FromKeeper); // storage key // out key keyout fifotbl + lua_pushinteger(L, fifo->first); // out key keyout fifotbl first STACK_CHECK(L, 5); - lua_setfield(L, -3, "first"); // out key keyout fifotbl - lua_pushinteger(L, fifo->count); // out key keyout fifobtl count + lua_setfield(L, -3, "first"); // out key keyout fifotbl + lua_pushinteger(L, fifo->count); // out key keyout fifobtl count STACK_CHECK(L, 5); - lua_setfield(L, -3, "count"); // out key keyout fifotbl - lua_pushinteger(L, fifo->limit); // out key keyout fifotbl limit + lua_setfield(L, -3, "count"); // out key keyout fifotbl + lua_pushinteger(L, fifo->limit); // out key keyout fifotbl limit STACK_CHECK(L, 5); - lua_setfield(L, -3, "limit"); // out key keyout fifotbl - lua_setfield(L, -2, "fifo"); // out key keyout - lua_rawset(L, -3); // out + lua_setfield(L, -3, "limit"); // out key keyout fifotbl + lua_setfield(L, -2, "fifo"); // out key keyout + lua_rawset(L, -3); // out STACK_CHECK(L, 1); } STACK_CHECK(L, 1); - lua_pop(KL, 1); // + lua_pop(KL, 1); // STACK_CHECK(KL, 0); return 1; } @@ -287,7 +287,7 @@ int keepercall_send(lua_State* L) if( lua_isnil(L, -1)) { lua_pop(L, 1); // ud key ... fifos - fifo_new(L); // ud key ... fifos fifo + std::ignore = fifo_new(L); // ud key ... fifos fifo lua_pushvalue(L, 2); // ud key ... fifos fifo key lua_pushvalue(L, -2); // ud key ... fifos fifo key fifo lua_rawset(L, -4); // ud key ... fifos fifo @@ -465,7 +465,7 @@ int keepercall_set(lua_State* L) { // fifos key [val [, ...]] nil // no need to wake writers in that case, because a writer can't wait on an inexistent key lua_pop(L, 1); // fifos key [val [, ...]] - fifo_new(L); // fifos key [val [, ...]] fifo + std::ignore = fifo_new(L); // fifos key [val [, ...]] fifo lua_pushvalue(L, 2); // fifos key [val [, ...]] fifo key lua_pushvalue(L, -2); // fifos key [val [, ...]] fifo key fifo lua_rawset(L, 1); // fifos key [val [, ...]] fifo diff --git a/src/keeper.h b/src/keeper.h index 89fa2ab..627c7ea 100644 --- a/src/keeper.h +++ b/src/keeper.h @@ -38,23 +38,23 @@ static constexpr UniqueKey NIL_SENTINEL{ 0x7eaafa003a1d11a1ull }; void init_keepers(Universe* U, lua_State* L); void close_keepers(Universe* U); -Keeper* which_keeper(Keepers* keepers_, uintptr_t magic_); -Keeper* keeper_acquire(Keepers* keepers_, uintptr_t magic_); +[[nodiscard]] Keeper* which_keeper(Keepers* keepers_, uintptr_t magic_); +[[nodiscard]] Keeper* keeper_acquire(Keepers* keepers_, uintptr_t magic_); void keeper_release(Keeper* K_); void keeper_toggle_nil_sentinels(lua_State* L, int val_i_, LookupMode const mode_); -int keeper_push_linda_storage(Universe* U, Dest L, void* ptr_, uintptr_t magic_); +[[nodiscard]] int keeper_push_linda_storage(Universe* U, Dest L, void* ptr_, uintptr_t magic_); using keeper_api_t = lua_CFunction; #define KEEPER_API(_op) keepercall_##_op #define PUSH_KEEPER_FUNC lua_pushcfunction // lua_Cfunctions to run inside a keeper state -int keepercall_clear(lua_State* L); -int keepercall_send(lua_State* L); -int keepercall_receive(lua_State* L); -int keepercall_receive_batched(lua_State* L); -int keepercall_limit(lua_State* L); -int keepercall_get(lua_State* L); -int keepercall_set(lua_State* L); -int keepercall_count(lua_State* L); - -int keeper_call(Universe* U, lua_State* K, keeper_api_t _func, lua_State* L, void* linda, int starting_index); +[[nodiscard]] int keepercall_clear(lua_State* L); +[[nodiscard]] int keepercall_send(lua_State* L); +[[nodiscard]] int keepercall_receive(lua_State* L); +[[nodiscard]] int keepercall_receive_batched(lua_State* L); +[[nodiscard]] int keepercall_limit(lua_State* L); +[[nodiscard]] int keepercall_get(lua_State* L); +[[nodiscard]] int keepercall_set(lua_State* L); +[[nodiscard]] int keepercall_count(lua_State* L); + +[[nodiscard]] int keeper_call(Universe* U, lua_State* K, keeper_api_t _func, lua_State* L, void* linda, int starting_index); diff --git a/src/lanes.cpp b/src/lanes.cpp index 3d0c70d..2a5ebfd 100644 --- a/src/lanes.cpp +++ b/src/lanes.cpp @@ -162,7 +162,7 @@ static void securize_debug_threadname(lua_State* L, Lane* lane_) } #if ERROR_FULL_STACK -static int lane_error(lua_State* L); +[[nodiscard]] static int lane_error(lua_State* L); // crc64/we of string "STACKTRACE_REGKEY" generated at http://www.nitrxgen.net/hashgen/ static constexpr UniqueKey STACKTRACE_REGKEY{ 0x534af7d3226a429full }; #endif // ERROR_FULL_STACK @@ -188,7 +188,7 @@ static constexpr UniqueKey FINALIZER_REGKEY{ 0x188fccb8bf348e09ull }; * Returns: true if a table was pushed * false if no table found, not created, and nothing pushed */ -static bool push_registry_table( lua_State* L, UniqueKey key, bool create) +[[nodiscard]] static bool push_registry_table(lua_State* L, UniqueKey key, bool create) { STACK_GROW(L, 3); STACK_CHECK_START_REL(L, 0); @@ -237,7 +237,7 @@ static void tracking_add(Lane* lane_) /* * A free-running lane has ended; remove it from tracking chain */ -static bool tracking_remove(Lane* lane_) +[[nodiscard]] static bool tracking_remove(Lane* lane_) { bool found{ false }; std::lock_guard guard{ lane_->U->tracking_cs }; @@ -277,7 +277,7 @@ Lane::~Lane() if (U->tracking_first != nullptr) { // Lane was cleaned up, no need to handle at process termination - tracking_remove(this); + std::ignore = tracking_remove(this); } #endif // HAVE_LANE_TRACKING() } @@ -300,10 +300,10 @@ LUAG_FUNC( set_finalizer) { luaL_argcheck(L, lua_isfunction(L, 1), 1, "finalizer should be a function"); luaL_argcheck(L, lua_gettop( L) == 1, 1, "too many arguments"); - // Get the current finalizer table (if any) - push_registry_table(L, FINALIZER_REGKEY, true /*do create if none*/); // finalizer {finalisers} + // Get the current finalizer table (if any), create one if it doesn't exist + std::ignore = push_registry_table(L, FINALIZER_REGKEY, true); // finalizer {finalisers} STACK_GROW(L, 2); - lua_pushinteger(L, lua_rawlen(L, -1) + 1); // finalizer {finalisers} idx + lua_pushinteger(L, lua_rawlen(L, -1) + 1); // finalizer {finalisers} idx lua_pushvalue(L, 1); // finalizer {finalisers} idx finalizer lua_rawset(L, -3); // finalizer {finalisers} lua_pop(L, 2); // @@ -326,7 +326,7 @@ LUAG_FUNC( set_finalizer) // static void push_stack_trace( lua_State* L, int rc_, int stk_base_); -static int run_finalizers( lua_State* L, int lua_rc) +[[nodiscard]] static int run_finalizers(lua_State* L, int lua_rc) { int finalizers_index; int n; @@ -430,7 +430,7 @@ static void selfdestruct_add(Lane* lane_) /* * A free-running lane has ended; remove it from selfdestruct chain */ -static bool selfdestruct_remove(Lane* lane_) +[[nodiscard]] static bool selfdestruct_remove(Lane* lane_) { bool found{ false }; std::lock_guard guard{ lane_->U->selfdestruct_cs }; @@ -465,7 +465,7 @@ static bool selfdestruct_remove(Lane* lane_) /* * Process end; cancel any still free-running threads */ -static int universe_gc( lua_State* L) +[[nodiscard]] static int universe_gc(lua_State* L) { Universe* const U{ lua_tofulluserdata(L, 1) }; lua_Duration const shutdown_timeout{ lua_tonumber(L, lua_upvalueindex(1)) }; @@ -638,7 +638,7 @@ LUAG_FUNC( set_error_reporting) return 0; } -static int lane_error(lua_State* L) +[[nodiscard]] static int lane_error(lua_State* L) { // error message (any type) STACK_CHECK_START_ABS(L, 1); // some_error @@ -1176,7 +1176,10 @@ LUAG_FUNC(lane_new) if (lua_pcall( L2, 1, 1, 0) != LUA_OK) // ret/errcode { // propagate error to main state if any - luaG_inter_move(U, Source{ L2 }, Dest{ L }, 1, LookupMode::LaneBody); // func libs priority globals package required gc_cb [... args ...] n "modname" error + std::ignore = luaG_inter_move(U + , Source{ L2 }, Dest{ L } + , 1, LookupMode::LaneBody + ); // func libs priority globals package required gc_cb [... args ...] n "modname" error raise_lua_error(L); } // after requiring the module, register the functions it exported in our name<->function database @@ -1209,7 +1212,7 @@ LUAG_FUNC(lane_new) lua_pushglobaltable(L2); // _G while( lua_next(L, globals_idx)) // func libs priority globals package required gc_cb [... args ...] k v { - luaG_inter_copy(U, Source{ L }, Dest{ L2 }, 2, LookupMode::LaneBody); // _G k v + std::ignore = luaG_inter_copy(U, Source{ L }, Dest{ L2 }, 2, LookupMode::LaneBody); // _G k v // assign it in L2's globals table lua_rawset(L2, -3); // _G lua_pop(L, 1); // func libs priority globals package required gc_cb [... args ...] k @@ -1290,7 +1293,7 @@ LUAG_FUNC(lane_new) // and the issue of canceling/killing threads at gc is not very nice, either // (would easily cause waits at gc cycle, which we don't want). // -static int lane_gc(lua_State* L) +[[nodiscard]] static int lane_gc(lua_State* L) { bool have_gc_cb{ false }; Lane* const lane{ lua_toLane(L, 1) }; // ud @@ -1356,7 +1359,7 @@ static int lane_gc(lua_State* L) // / "error" finished at an error, error value is there // / "cancelled" execution cancelled by M (state gone) // -static char const * thread_status_string(Lane* lane_) +[[nodiscard]] static char const* thread_status_string(Lane* lane_) { Lane::Status const st{ lane_->m_status }; // read just once (volatile) char const* str = @@ -1624,20 +1627,20 @@ LUAG_FUNC(threads) { Lane* lane{ U->tracking_first }; int index = 0; - lua_newtable(L); // {} + lua_newtable(L); // {} while (lane != TRACKING_END) { // insert a { name, status } tuple, so that several lanes with the same name can't clobber each other - lua_newtable(L); // {} {} - lua_pushstring(L, lane->debug_name); // {} {} "name" - lua_setfield(L, -2, "name"); // {} {} - push_thread_status(L, lane); // {} {} "status" - lua_setfield(L, -2, "status"); // {} {} - lua_rawseti(L, -2, ++index); // {} + lua_newtable(L); // {} {} + lua_pushstring(L, lane->debug_name); // {} {} "name" + lua_setfield(L, -2, "name"); // {} {} + std::ignore = push_thread_status(L, lane); // {} {} "status" + lua_setfield(L, -2, "status"); // {} {} + lua_rawseti(L, -2, ++index); // {} lane = lane->tracking_next; } } - return lua_gettop(L) - top; // 0 or 1 + return lua_gettop(L) - top; // 0 or 1 } #endif // HAVE_LANE_TRACKING() @@ -1723,7 +1726,7 @@ LUAG_FUNC(wakeup_conv) */ extern int LG_linda(lua_State* L); -static const struct luaL_Reg lanes_functions[] = +static struct luaL_Reg const lanes_functions[] = { { "linda", LG_linda }, { "now_secs", LG_now_secs }, @@ -2022,9 +2025,9 @@ LANES_API int luaopen_lanes_core( lua_State* L) return 1; } -static int default_luaopen_lanes( lua_State* L) +[[nodiscard]] static int default_luaopen_lanes(lua_State* L) { - int rc = luaL_loadfile(L, "lanes.lua") || lua_pcall(L, 0, 1, 0); + int const rc{ luaL_loadfile(L, "lanes.lua") || lua_pcall(L, 0, 1, 0) }; if (rc != LUA_OK) { return luaL_error(L, "failed to initialize embedded Lanes"); diff --git a/src/lanes.h b/src/lanes.h index 05a0a5c..bc8de55 100644 --- a/src/lanes.h +++ b/src/lanes.h @@ -20,7 +20,7 @@ extern "C" { #define LANES_VERSION_GREATER_THAN(MAJOR, MINOR, PATCH) ((LANES_VERSION_MAJOR>MAJOR) || (LANES_VERSION_MAJOR==MAJOR && (LANES_VERSION_MINOR>MINOR || (LANES_VERSION_MINOR==MINOR && LANES_VERSION_PATCH>PATCH)))) #define LANES_VERSION_GREATER_OR_EQUAL(MAJOR, MINOR, PATCH) ((LANES_VERSION_MAJOR>MAJOR) || (LANES_VERSION_MAJOR==MAJOR && (LANES_VERSION_MINOR>MINOR || (LANES_VERSION_MINOR==MINOR && LANES_VERSION_PATCH>=PATCH)))) -LANES_API int luaopen_lanes_core(lua_State* L); +LANES_API [[nodiscard]] int luaopen_lanes_core(lua_State* L); // Call this to work with embedded Lanes instead of calling luaopen_lanes_core() LANES_API void luaopen_lanes_embedded(lua_State* L, lua_CFunction _luaopen_lanes); diff --git a/src/lanes_private.h b/src/lanes_private.h index 0fcbbfc..ba40e44 100644 --- a/src/lanes_private.h +++ b/src/lanes_private.h @@ -78,7 +78,7 @@ class Lane // // For tracking only - static void* operator new(size_t size_, Universe* U_) noexcept { return U_->internal_allocator.alloc(size_); } + [[nodiscard]] static void* operator new(size_t size_, Universe* U_) noexcept { return U_->internal_allocator.alloc(size_); } // can't actually delete the operator because the compiler generates stack unwinding code that could call it in case of exception static void operator delete(void* p_, Universe* U_) { U_->internal_allocator.free(p_, sizeof(Lane)); } // this one is for us, to make sure memory is freed by the correct allocator @@ -87,7 +87,7 @@ class Lane Lane(Universe* U_, lua_State* L_); ~Lane(); - bool waitForCompletion(lua_Duration duration_); + [[nodiscard]] bool waitForCompletion(lua_Duration duration_); void startThread(int priority_); }; @@ -98,6 +98,9 @@ static constexpr UniqueKey LANE_POINTER_REGKEY{ 0xB3022205633743BCull }; // used // 'Lane' are malloc/free'd and the handle only carries a pointer. // This is not deep userdata since the handle's not portable among lanes. // -#define lua_toLane(L, i) (*((Lane**) luaL_checkudata( L, i, "Lane"))) +[[nodiscard]] inline Lane* lua_toLane(lua_State* L, int i_) +{ + return *(static_cast(luaL_checkudata(L, i_, "Lane"))); +} -int push_thread_status(lua_State* L, Lane* s); +[[nodiscard]] int push_thread_status(lua_State* L, Lane* s); diff --git a/src/linda.cpp b/src/linda.cpp index 39977bc..50964ad 100644 --- a/src/linda.cpp +++ b/src/linda.cpp @@ -70,7 +70,7 @@ class Linda : public DeepPrelude // Deep userdata MUST start with this header public: // a fifo full userdata has one uservalue, the table that holds the actual fifo contents - static void* operator new(size_t size_, Universe* U_) noexcept { return U_->internal_allocator.alloc(size_); } + [[nodiscard]] static void* operator new(size_t size_, Universe* U_) noexcept { return U_->internal_allocator.alloc(size_); } // always embedded somewhere else or "in-place constructed" as a full userdata // can't actually delete the operator because the compiler generates stack unwinding code that could call it in case of exception static void operator delete(void* p_, Universe* U_) { U_->internal_allocator.free(p_, sizeof(Linda)); } @@ -137,10 +137,10 @@ class Linda : public DeepPrelude // Deep userdata MUST start with this header return nullptr; } }; -static void* linda_id( lua_State*, DeepOp); +[[nodiscard]] static void* linda_id(lua_State*, DeepOp); template -static inline Linda* lua_toLinda(lua_State* L, int idx_) +[[nodiscard]] static inline Linda* lua_toLinda(lua_State* L, int idx_) { Linda* const linda{ static_cast(luaG_todeep(L, linda_id, idx_)) }; if (!OPT) @@ -742,7 +742,7 @@ LUAG_FUNC(linda_deep) */ template -static int linda_tostring(lua_State* L, int idx_) +[[nodiscard]] static int linda_tostring(lua_State* L, int idx_) { Linda* const linda{ lua_toLinda(L, idx_) }; if (linda != nullptr) @@ -851,7 +851,7 @@ LUAG_FUNC(linda_towatch) * For any other strings, the ID function must not react at all. This allows * future extensions of the system. */ -static void* linda_id( lua_State* L, DeepOp op_) +[[nodiscard]] static void* linda_id(lua_State* L, DeepOp op_) { switch( op_) { @@ -907,7 +907,7 @@ static void* linda_id( lua_State* L, DeepOp op_) // Clean associated structures in the keeper state. Keeper* const K{ need_acquire_release ? keeper_acquire(linda->U->keepers, linda->hashSeed()) : myK }; // hopefully this won't ever raise an error as we would jump to the closest pcall site while forgetting to release the keeper mutex... - keeper_call(linda->U, K->L, KEEPER_API(clear), L, linda, 0); + std::ignore = keeper_call(linda->U, K->L, KEEPER_API(clear), L, linda, 0); if (need_acquire_release) { keeper_release(K); diff --git a/src/macros_and_utils.h b/src/macros_and_utils.h index 47ce90c..31ae8bd 100644 --- a/src/macros_and_utils.h +++ b/src/macros_and_utils.h @@ -23,11 +23,13 @@ extern char const* debugspew_indent; #define INDENT_BEGIN "%.*s " #define INDENT_END , (U ? U->debugspew_indent_depth.load(std::memory_order_relaxed) : 0), debugspew_indent #define DEBUGSPEW_CODE(_code) _code -#define DEBUGSPEW_PARAM_COMMA( param_) param_, +#define DEBUGSPEW_OR_NOT(a_, b_) a_ +#define DEBUGSPEW_PARAM_COMMA(param_) param_, #define DEBUGSPEW_COMMA_PARAM( param_) , param_ #else // USE_DEBUG_SPEW() #define DEBUGSPEW_CODE(_code) -#define DEBUGSPEW_PARAM_COMMA( param_) +#define DEBUGSPEW_OR_NOT(a_, b_) b_ +#define DEBUGSPEW_PARAM_COMMA(param_) #define DEBUGSPEW_COMMA_PARAM( param_) #endif // USE_DEBUG_SPEW() @@ -130,20 +132,20 @@ inline void STACK_GROW(lua_State* L, int n_) } } -#define LUAG_FUNC( func_name) int LG_##func_name( lua_State* L) +#define LUAG_FUNC(func_name) [[nodiscard]] int LG_##func_name(lua_State* L) // ################################################################################################# // a small helper to extract a full userdata pointer from the stack in a safe way template -T* lua_tofulluserdata(lua_State* L, int index_) +[[nodiscard]] T* lua_tofulluserdata(lua_State* L, int index_) { ASSERT_L(lua_isnil(L, index_) || lua_type(L, index_) == LUA_TUSERDATA); return static_cast(lua_touserdata(L, index_)); } template -auto lua_tolightuserdata(lua_State* L, int index_) +[[nodiscard]] auto lua_tolightuserdata(lua_State* L, int index_) { ASSERT_L(lua_isnil(L, index_) || lua_islightuserdata(L, index_)); if constexpr (std::is_pointer_v) @@ -157,7 +159,7 @@ auto lua_tolightuserdata(lua_State* L, int index_) } template -T* lua_newuserdatauv(lua_State* L, int nuvalue_) +[[nodiscard]] T* lua_newuserdatauv(lua_State* L, int nuvalue_) { return static_cast(lua_newuserdatauv(L, sizeof(T), nuvalue_)); } @@ -175,6 +177,7 @@ using lua_Duration = std::chrono::template duration; // ################################################################################################# +// A unique type generator template struct Unique { diff --git a/src/state.cpp b/src/state.cpp index 6a9ada7..496e21e 100644 --- a/src/state.cpp +++ b/src/state.cpp @@ -49,7 +49,7 @@ THE SOFTWARE. // // Upvalues: [1]: original 'require' function // -static int luaG_new_require( lua_State* L) +[[nodiscard]] static int luaG_new_require(lua_State* L) { int rc; int const args = lua_gettop( L); // args @@ -110,7 +110,7 @@ void serialize_require(DEBUGSPEW_PARAM_COMMA( Universe* U) lua_State* L) /*---=== luaG_newstate ===---*/ -static int require_lanes_core( lua_State* L) +[[nodiscard]] static int require_lanes_core(lua_State* L) { // leaves a copy of 'lanes.core' module table on the stack luaL_requiref( L, "lanes.core", luaopen_lanes_core, 0); @@ -118,7 +118,7 @@ static int require_lanes_core( lua_State* L) } -static const luaL_Reg libs[] = +static luaL_Reg const libs[] = { { LUA_LOADLIBNAME, luaopen_package}, { LUA_TABLIBNAME, luaopen_table}, diff --git a/src/state.h b/src/state.h index 2601f77..e1c311a 100644 --- a/src/state.h +++ b/src/state.h @@ -10,8 +10,8 @@ void serialize_require(DEBUGSPEW_PARAM_COMMA(Universe* U) lua_State* L); // ################################################################################################ -lua_State* create_state(Universe* U, lua_State* from_); -lua_State* luaG_newstate(Universe* U, Source _from, char const* libs); +[[nodiscard]] lua_State* create_state(Universe* U, lua_State* from_); +[[nodiscard]] lua_State* luaG_newstate(Universe* U, Source _from, char const* libs); // ################################################################################################ diff --git a/src/threading.cpp b/src/threading.cpp index d278bb1..259693a 100644 --- a/src/threading.cpp +++ b/src/threading.cpp @@ -209,7 +209,7 @@ void THREAD_SETNAME(char const* _name) // general its implementation is pretty much trivial, as on Win32 target // just SCHED_OTHER can be supported. #undef pthread_attr_setschedpolicy -static int pthread_attr_setschedpolicy(pthread_attr_t* attr, int policy) +[[nodiscard]] static int pthread_attr_setschedpolicy(pthread_attr_t* attr, int policy) { if (policy != SCHED_OTHER) { @@ -348,7 +348,7 @@ static int const gs_prio_remap[] = #endif // _PRIO_0 }; -static int select_prio(int prio /* -3..+3 */) +[[nodiscard]] static int select_prio(int prio /* -3..+3 */) { if (prio == THREAD_PRIO_DEFAULT) prio = 0; diff --git a/src/threading_osx.h b/src/threading_osx.h index b47d2f6..f4d41e0 100644 --- a/src/threading_osx.h +++ b/src/threading_osx.h @@ -2,8 +2,7 @@ * THREADING_OSX.H * http://yyshen.github.io/2015/01/18/binding_threads_to_cores_osx.html */ -#ifndef __threading_osx_h__ -#define __threading_osx_h__ 1 +#pragma once #include #include @@ -18,9 +17,9 @@ struct cpu_set_t static inline void CPU_ZERO(cpu_set_t *cs) { cs->count = 0; } static inline void CPU_SET(int num, cpu_set_t *cs) { cs->count |= (1 << num); } -static inline int CPU_ISSET(int num, cpu_set_t *cs) { return (cs->count & (1 << num)); } +[[nodiscard]] static inline int CPU_ISSET(int num, cpu_set_t *cs) { return (cs->count & (1 << num)); } -int sched_getaffinity(pid_t pid, size_t cpu_size, cpu_set_t *cpu_set) +[[nodiscard]] int sched_getaffinity(pid_t pid, size_t cpu_size, cpu_set_t *cpu_set) { int32_t core_count = 0; size_t len = sizeof(core_count); @@ -39,7 +38,7 @@ int sched_getaffinity(pid_t pid, size_t cpu_size, cpu_set_t *cpu_set) return 0; } -int pthread_setaffinity_np(pthread_t thread, size_t cpu_size, cpu_set_t *cpu_set) +[[nodiscard]] int pthread_setaffinity_np(pthread_t thread, size_t cpu_size, cpu_set_t *cpu_set) { thread_port_t mach_thread; int core = 0; @@ -57,4 +56,3 @@ int pthread_setaffinity_np(pthread_t thread, size_t cpu_size, cpu_set_t *cpu_set return 0; } -#endif diff --git a/src/tools.cpp b/src/tools.cpp index 07f9ae6..4083a57 100644 --- a/src/tools.cpp +++ b/src/tools.cpp @@ -144,7 +144,7 @@ void luaG_dump( lua_State* L) // ################################################################################################ // same as PUC-Lua l_alloc -extern "C" static void* libc_lua_Alloc([[maybe_unused]] void* ud, [[maybe_unused]] void* ptr_, [[maybe_unused]] size_t osize_, size_t nsize_) +extern "C" [[nodiscard]] static void* libc_lua_Alloc([[maybe_unused]] void* ud, [[maybe_unused]] void* ptr_, [[maybe_unused]] size_t osize_, size_t nsize_) { if (nsize_ == 0) { @@ -159,7 +159,7 @@ extern "C" static void* libc_lua_Alloc([[maybe_unused]] void* ud, [[maybe_unused // ################################################################################################# -static int luaG_provide_protected_allocator(lua_State* L) +[[nodiscard]] static int luaG_provide_protected_allocator(lua_State* L) { Universe* const U{ universe_get(L) }; // push a new full userdata on the stack, giving access to the universe's protected allocator @@ -234,7 +234,7 @@ void initialize_allocator_function(Universe* U, lua_State* L) // ################################################################################################ -static int dummy_writer( lua_State* L, void const* p, size_t sz, void* ud) +[[nodiscard]] static int dummy_writer(lua_State* L, void const* p, size_t sz, void* ud) { (void)L; (void)p; (void)sz; (void) ud; // unused return 666; @@ -291,7 +291,7 @@ FuncSubType luaG_getfuncsubtype( lua_State *L, int _i) // ################################################################################################# -static lua_CFunction luaG_tocfunction(lua_State* L, int _i, FuncSubType* _out) +[[nodiscard]] static lua_CFunction luaG_tocfunction(lua_State* L, int _i, FuncSubType* _out) { lua_CFunction p = lua_tocfunction( L, _i); *_out = luaG_getfuncsubtype( L, _i); @@ -304,7 +304,7 @@ static constexpr UniqueKey LOOKUPCACHE_REGKEY{ 0x837a68dfc6fcb716ull }; // ################################################################################################# // inspired from tconcat() in ltablib.c -static char const* luaG_pushFQN( lua_State* L, int t, int last, size_t* length) +[[nodiscard]] static char const* luaG_pushFQN(lua_State* L, int t, int last, size_t* length) { int i = 1; luaL_Buffer b; @@ -338,7 +338,7 @@ static char const* luaG_pushFQN( lua_State* L, int t, int last, size_t* length) * if we already had an entry of type [o] = ..., replace the name if the new one is shorter * pops the processed object from the stack */ -static void update_lookup_entry( DEBUGSPEW_PARAM_COMMA( Universe* U) lua_State* L, int _ctx_base, int _depth) +static void update_lookup_entry(DEBUGSPEW_PARAM_COMMA( Universe* U) lua_State* L, int _ctx_base, int _depth) { // slot 1 in the stack contains the table that receives everything we found int const dest = _ctx_base; @@ -362,7 +362,7 @@ static void update_lookup_entry( DEBUGSPEW_PARAM_COMMA( Universe* U) lua_State* ++ _depth; lua_rawseti( L, fqn, _depth); // ... {bfc} k o name? // generate name - DEBUGSPEW_CODE( newName =) luaG_pushFQN( L, fqn, _depth, &newNameLength); // ... {bfc} k o name? "f.q.n" + DEBUGSPEW_OR_NOT(newName, std::ignore) = luaG_pushFQN(L, fqn, _depth, &newNameLength);// ... {bfc} k o name? "f.q.n" // Lua 5.2 introduced a hash randomizer seed which causes table iteration to yield a different key order // on different VMs even when the tables are populated the exact same way. // When Lua is built with compatibility options (such as LUA_COMPAT_ALL), @@ -613,7 +613,7 @@ static constexpr UniqueKey REG_MTID{ 0x2e68f9b4751584dcull }; /* * Get a unique ID for metatable at [i]. */ -static lua_Integer get_mt_id( Universe* U, lua_State* L, int i) +[[nodiscard]] static lua_Integer get_mt_id(Universe* U, lua_State* L, int i) { lua_Integer id; @@ -654,25 +654,25 @@ static lua_Integer get_mt_id( Universe* U, lua_State* L, int i) // ################################################################################################# // function sentinel used to transfer native functions from/to keeper states -static int func_lookup_sentinel( lua_State* L) +[[nodiscard]] static int func_lookup_sentinel(lua_State* L) { - return luaL_error( L, "function lookup sentinel for %s, should never be called", lua_tostring( L, lua_upvalueindex( 1))); + return luaL_error(L, "function lookup sentinel for %s, should never be called", lua_tostring(L, lua_upvalueindex(1))); } // ################################################################################################# // function sentinel used to transfer native table from/to keeper states -static int table_lookup_sentinel( lua_State* L) +[[nodiscard]] static int table_lookup_sentinel(lua_State* L) { - return luaL_error( L, "table lookup sentinel for %s, should never be called", lua_tostring( L, lua_upvalueindex( 1))); + return luaL_error(L, "table lookup sentinel for %s, should never be called", lua_tostring(L, lua_upvalueindex(1))); } // ################################################################################################# // function sentinel used to transfer cloned full userdata from/to keeper states -static int userdata_clone_sentinel( lua_State* L) +[[nodiscard]] static int userdata_clone_sentinel(lua_State* L) { - return luaL_error( L, "userdata clone sentinel for %s, should never be called", lua_tostring( L, lua_upvalueindex( 1))); + return luaL_error(L, "userdata clone sentinel for %s, should never be called", lua_tostring(L, lua_upvalueindex(1))); } // ################################################################################################# @@ -680,7 +680,7 @@ static int userdata_clone_sentinel( lua_State* L) /* * retrieve the name of a function/table in the lookup database */ -static char const* find_lookup_name(lua_State* L, int i, LookupMode mode_, char const* upName_, size_t* len_) +[[nodiscard]] static char const* find_lookup_name(lua_State* L, int i, LookupMode mode_, char const* upName_, size_t* len_) { DEBUGSPEW_CODE( Universe* const U = universe_get( L)); char const* fqn; @@ -753,7 +753,7 @@ static char const* find_lookup_name(lua_State* L, int i, LookupMode mode_, char /* * Push a looked-up table, or nothing if we found nothing */ -static bool lookup_table(Dest L2, Source L, int i, LookupMode mode_, char const* upName_) +[[nodiscard]] static bool lookup_table(Dest L2, Source L, int i, LookupMode mode_, char const* upName_) { // get the name of the table we want to send size_t len; @@ -830,13 +830,12 @@ static bool lookup_table(Dest L2, Source L, int i, LookupMode mode_, char const* * Returns true if the table was cached (no need to fill it!); false if * it's a virgin. */ -static bool push_cached_table(lua_State* L2, int L2_cache_i, lua_State* L, int i) +[[nodiscard]] static bool push_cached_table(Dest L2, int L2_cache_i, Source L, int i) { - bool not_found_in_cache; // L2 void const* p{ lua_topointer(L, i) }; ASSERT_L( L2_cache_i != 0); - STACK_GROW( L2, 3); + STACK_GROW( L2, 3); // L2 STACK_CHECK_START_REL(L2, 0); // We don't need to use the from state ('L') in ID since the life span @@ -847,7 +846,7 @@ static bool push_cached_table(lua_State* L2, int L2_cache_i, lua_State* L, int i //fprintf( stderr, "<< ID: %s >>\n", lua_tostring( L2, -1)); lua_rawget( L2, L2_cache_i); // ... {cached|nil} - not_found_in_cache = lua_isnil( L2, -1); + bool const not_found_in_cache{ lua_isnil(L2, -1) }; if( not_found_in_cache) { lua_pop( L2, 1); // ... @@ -866,83 +865,83 @@ static bool push_cached_table(lua_State* L2, int L2_cache_i, lua_State* L, int i /* * Return some name helping to identify an object */ -static int discover_object_name_recur( lua_State* L, int shortest_, int depth_) +[[nodiscard]] static int discover_object_name_recur(lua_State* L, int shortest_, int depth_) { int const what = 1; // o "r" {c} {fqn} ... {?} int const result = 2; int const cache = 3; int const fqn = 4; // no need to scan this table if the name we will discover is longer than one we already know - if( shortest_ <= depth_ + 1) + if (shortest_ <= depth_ + 1) { return shortest_; } - STACK_GROW( L, 3); + STACK_GROW(L, 3); STACK_CHECK_START_REL(L, 0); // stack top contains the table to search in - lua_pushvalue( L, -1); // o "r" {c} {fqn} ... {?} {?} - lua_rawget( L, cache); // o "r" {c} {fqn} ... {?} nil/1 + lua_pushvalue(L, -1); // o "r" {c} {fqn} ... {?} {?} + lua_rawget(L, cache); // o "r" {c} {fqn} ... {?} nil/1 // if table is already visited, we are done - if( !lua_isnil( L, -1)) + if( !lua_isnil(L, -1)) { - lua_pop( L, 1); // o "r" {c} {fqn} ... {?} + lua_pop(L, 1); // o "r" {c} {fqn} ... {?} return shortest_; } // examined table is not in the cache, add it now - lua_pop( L, 1); // o "r" {c} {fqn} ... {?} - lua_pushvalue( L, -1); // o "r" {c} {fqn} ... {?} {?} - lua_pushinteger( L, 1); // o "r" {c} {fqn} ... {?} {?} 1 - lua_rawset( L, cache); // o "r" {c} {fqn} ... {?} + lua_pop(L, 1); // o "r" {c} {fqn} ... {?} + lua_pushvalue(L, -1); // o "r" {c} {fqn} ... {?} {?} + lua_pushinteger(L, 1); // o "r" {c} {fqn} ... {?} {?} 1 + lua_rawset(L, cache); // o "r" {c} {fqn} ... {?} // scan table contents - lua_pushnil( L); // o "r" {c} {fqn} ... {?} nil - while( lua_next( L, -2)) // o "r" {c} {fqn} ... {?} k v + lua_pushnil(L); // o "r" {c} {fqn} ... {?} nil + while (lua_next(L, -2)) // o "r" {c} {fqn} ... {?} k v { - //char const *const strKey = (lua_type( L, -2) == LUA_TSTRING) ? lua_tostring( L, -2) : nullptr; // only for debugging - //lua_Number const numKey = (lua_type( L, -2) == LUA_TNUMBER) ? lua_tonumber( L, -2) : -6666; // only for debugging - STACK_CHECK( L, 2); + //char const *const strKey = (lua_type(L, -2) == LUA_TSTRING) ? lua_tostring(L, -2) : nullptr; // only for debugging + //lua_Number const numKey = (lua_type(L, -2) == LUA_TNUMBER) ? lua_tonumber(L, -2) : -6666; // only for debugging + STACK_CHECK(L, 2); // append key name to fqn stack ++ depth_; - lua_pushvalue( L, -2); // o "r" {c} {fqn} ... {?} k v k - lua_rawseti( L, fqn, depth_); // o "r" {c} {fqn} ... {?} k v - if( lua_rawequal( L, -1, what)) // is it what we are looking for? + lua_pushvalue(L, -2); // o "r" {c} {fqn} ... {?} k v k + lua_rawseti(L, fqn, depth_); // o "r" {c} {fqn} ... {?} k v + if (lua_rawequal(L, -1, what)) // is it what we are looking for? { - STACK_CHECK( L, 2); + STACK_CHECK(L, 2); // update shortest name if( depth_ < shortest_) { shortest_ = depth_; - luaG_pushFQN( L, fqn, depth_, nullptr); // o "r" {c} {fqn} ... {?} k v "fqn" - lua_replace( L, result); // o "r" {c} {fqn} ... {?} k v + std::ignore = luaG_pushFQN(L, fqn, depth_, nullptr); // o "r" {c} {fqn} ... {?} k v "fqn" + lua_replace(L, result); // o "r" {c} {fqn} ... {?} k v } // no need to search further at this level - lua_pop( L, 2); // o "r" {c} {fqn} ... {?} - STACK_CHECK( L, 0); + lua_pop(L, 2); // o "r" {c} {fqn} ... {?} + STACK_CHECK(L, 0); break; } - switch( lua_type( L, -1)) // o "r" {c} {fqn} ... {?} k v + switch (lua_type(L, -1)) // o "r" {c} {fqn} ... {?} k v { default: // nil, boolean, light userdata, number and string aren't identifiable break; case LUA_TTABLE: // o "r" {c} {fqn} ... {?} k {} - STACK_CHECK( L, 2); - shortest_ = discover_object_name_recur( L, shortest_, depth_); + STACK_CHECK(L, 2); + shortest_ = discover_object_name_recur(L, shortest_, depth_); // search in the table's metatable too - if( lua_getmetatable( L, -1)) // o "r" {c} {fqn} ... {?} k {} {mt} + if (lua_getmetatable(L, -1)) // o "r" {c} {fqn} ... {?} k {} {mt} { - if( lua_istable( L, -1)) + if( lua_istable(L, -1)) { ++ depth_; - lua_pushliteral( L, "__metatable"); // o "r" {c} {fqn} ... {?} k {} {mt} "__metatable" - lua_rawseti( L, fqn, depth_); // o "r" {c} {fqn} ... {?} k {} {mt} - shortest_ = discover_object_name_recur( L, shortest_, depth_); - lua_pushnil( L); // o "r" {c} {fqn} ... {?} k {} {mt} nil - lua_rawseti( L, fqn, depth_); // o "r" {c} {fqn} ... {?} k {} {mt} + lua_pushliteral(L, "__metatable"); // o "r" {c} {fqn} ... {?} k {} {mt} "__metatable" + lua_rawseti(L, fqn, depth_); // o "r" {c} {fqn} ... {?} k {} {mt} + shortest_ = discover_object_name_recur(L, shortest_, depth_); + lua_pushnil(L); // o "r" {c} {fqn} ... {?} k {} {mt} nil + lua_rawseti(L, fqn, depth_); // o "r" {c} {fqn} ... {?} k {} {mt} -- depth_; } - lua_pop( L, 1); // o "r" {c} {fqn} ... {?} k {} + lua_pop(L, 1); // o "r" {c} {fqn} ... {?} k {} } - STACK_CHECK( L, 2); + STACK_CHECK(L, 2); break; case LUA_TTHREAD: // o "r" {c} {fqn} ... {?} k T @@ -950,61 +949,61 @@ static int discover_object_name_recur( lua_State* L, int shortest_, int depth_) break; case LUA_TUSERDATA: // o "r" {c} {fqn} ... {?} k U - STACK_CHECK( L, 2); + STACK_CHECK(L, 2); // search in the object's metatable (some modules are built that way) - if( lua_getmetatable( L, -1)) // o "r" {c} {fqn} ... {?} k U {mt} + if (lua_getmetatable(L, -1)) // o "r" {c} {fqn} ... {?} k U {mt} { - if( lua_istable( L, -1)) + if (lua_istable(L, -1)) { ++ depth_; - lua_pushliteral( L, "__metatable"); // o "r" {c} {fqn} ... {?} k U {mt} "__metatable" - lua_rawseti( L, fqn, depth_); // o "r" {c} {fqn} ... {?} k U {mt} - shortest_ = discover_object_name_recur( L, shortest_, depth_); - lua_pushnil( L); // o "r" {c} {fqn} ... {?} k U {mt} nil - lua_rawseti( L, fqn, depth_); // o "r" {c} {fqn} ... {?} k U {mt} + lua_pushliteral(L, "__metatable"); // o "r" {c} {fqn} ... {?} k U {mt} "__metatable" + lua_rawseti(L, fqn, depth_); // o "r" {c} {fqn} ... {?} k U {mt} + shortest_ = discover_object_name_recur(L, shortest_, depth_); + lua_pushnil(L); // o "r" {c} {fqn} ... {?} k U {mt} nil + lua_rawseti(L, fqn, depth_); // o "r" {c} {fqn} ... {?} k U {mt} -- depth_; } - lua_pop( L, 1); // o "r" {c} {fqn} ... {?} k U + lua_pop(L, 1); // o "r" {c} {fqn} ... {?} k U } - STACK_CHECK( L, 2); + STACK_CHECK(L, 2); // search in the object's uservalues { int uvi = 1; - while( lua_getiuservalue( L, -1, uvi) != LUA_TNONE) // o "r" {c} {fqn} ... {?} k U {u} + while (lua_getiuservalue(L, -1, uvi) != LUA_TNONE) // o "r" {c} {fqn} ... {?} k U {u} { - if( lua_istable( L, -1)) // if it is a table, look inside + if( lua_istable(L, -1)) // if it is a table, look inside { ++ depth_; - lua_pushliteral( L, "uservalue"); // o "r" {c} {fqn} ... {?} k v {u} "uservalue" - lua_rawseti( L, fqn, depth_); // o "r" {c} {fqn} ... {?} k v {u} - shortest_ = discover_object_name_recur( L, shortest_, depth_); - lua_pushnil( L); // o "r" {c} {fqn} ... {?} k v {u} nil - lua_rawseti( L, fqn, depth_); // o "r" {c} {fqn} ... {?} k v {u} + lua_pushliteral(L, "uservalue"); // o "r" {c} {fqn} ... {?} k v {u} "uservalue" + lua_rawseti(L, fqn, depth_); // o "r" {c} {fqn} ... {?} k v {u} + shortest_ = discover_object_name_recur(L, shortest_, depth_); + lua_pushnil(L); // o "r" {c} {fqn} ... {?} k v {u} nil + lua_rawseti(L, fqn, depth_); // o "r" {c} {fqn} ... {?} k v {u} -- depth_; } - lua_pop( L, 1); // o "r" {c} {fqn} ... {?} k U + lua_pop(L, 1); // o "r" {c} {fqn} ... {?} k U ++ uvi; } // when lua_getiuservalue() returned LUA_TNONE, it pushed a nil. pop it now - lua_pop( L, 1); // o "r" {c} {fqn} ... {?} k U + lua_pop(L, 1); // o "r" {c} {fqn} ... {?} k U } - STACK_CHECK( L, 2); + STACK_CHECK(L, 2); break; } // make ready for next iteration - lua_pop( L, 1); // o "r" {c} {fqn} ... {?} k + lua_pop(L, 1); // o "r" {c} {fqn} ... {?} k // remove name from fqn stack - lua_pushnil( L); // o "r" {c} {fqn} ... {?} k nil - lua_rawseti( L, fqn, depth_); // o "r" {c} {fqn} ... {?} k - STACK_CHECK( L, 1); + lua_pushnil(L); // o "r" {c} {fqn} ... {?} k nil + lua_rawseti(L, fqn, depth_); // o "r" {c} {fqn} ... {?} k + STACK_CHECK(L, 1); -- depth_; } // o "r" {c} {fqn} ... {?} - STACK_CHECK( L, 0); + STACK_CHECK(L, 0); // remove the visited table from the cache, in case a shorter path to the searched object exists - lua_pushvalue( L, -1); // o "r" {c} {fqn} ... {?} {?} - lua_pushnil( L); // o "r" {c} {fqn} ... {?} {?} nil - lua_rawset( L, cache); // o "r" {c} {fqn} ... {?} - STACK_CHECK( L, 0); + lua_pushvalue(L, -1); // o "r" {c} {fqn} ... {?} {?} + lua_pushnil(L); // o "r" {c} {fqn} ... {?} {?} nil + lua_rawset(L, cache); // o "r" {c} {fqn} ... {?} + STACK_CHECK(L, 0); return shortest_; } @@ -1169,7 +1168,7 @@ static char const* vt_names[] = // we have to do it that way because we can't unbalance the stack between buffer operations // namely, this means we can't push a function on top of the stack *after* we initialize the buffer! // luckily, this also works with earlier Lua versions -static int buf_writer( lua_State* L, void const* b, size_t size, void* ud) +[[nodiscard]] static int buf_writer(lua_State* L, void const* b, size_t size, void* ud) { luaL_Buffer* B = (luaL_Buffer*) ud; if( !B->L) @@ -1339,7 +1338,7 @@ static void copy_func(Universe* U, Dest L2, int L2_cache_i, Source L, int i, Loo static void copy_cached_func(Universe* U, Dest L2, int L2_cache_i, Source L, int i, LookupMode mode_, char const* upName_) { FuncSubType funcSubType; - /*lua_CFunction cfunc =*/ luaG_tocfunction( L, i, &funcSubType); // nullptr for LuaJIT-fast && bytecode functions + std::ignore = luaG_tocfunction(L, i, &funcSubType); // nullptr for LuaJIT-fast && bytecode functions if( funcSubType == FST_Bytecode) { void* const aspointer = (void*)lua_topointer( L, i); @@ -1391,7 +1390,7 @@ static void copy_cached_func(Universe* U, Dest L2, int L2_cache_i, Source L, int // ################################################################################################# -static bool push_cached_metatable(Universe* U, Dest L2, int L2_cache_i, Source L, int i, LookupMode mode_, char const* upName_) +[[nodiscard]] static bool push_cached_metatable(Universe* U, Dest L2, int L2_cache_i, Source L, int i, LookupMode mode_, char const* upName_) { STACK_CHECK_START_REL(L, 0); if( lua_getmetatable( L, i)) // ... mt @@ -1442,7 +1441,7 @@ static bool push_cached_metatable(Universe* U, Dest L2, int L2_cache_i, Source L // ################################################################################################# -static void inter_copy_keyvaluepair(Universe* U, Dest L2, int L2_cache_i, Source L, VT vt_, LookupMode mode_, char const* upName_) +[[nodiscard]] static void inter_copy_keyvaluepair(Universe* U, Dest L2, int L2_cache_i, Source L, VT vt_, LookupMode mode_, char const* upName_) { int val_i = lua_gettop(L); int key_i = val_i - 1; @@ -1514,7 +1513,7 @@ static void inter_copy_keyvaluepair(Universe* U, Dest L2, int L2_cache_i, Source */ static constexpr UniqueKey CLONABLES_CACHE_KEY{ 0xD04EE018B3DEE8F5ull }; -static bool copyclone(Universe* U, Dest L2, int L2_cache_i, Source L, int source_i_, LookupMode mode_, char const* upName_) +[[nodiscard]] static bool copyclone(Universe* U, Dest L2, int L2_cache_i, Source L, int source_i_, LookupMode mode_, char const* upName_) { void* const source = lua_touserdata( L, source_i_); source_i_ = lua_absindex( L, source_i_); @@ -1600,7 +1599,11 @@ static bool copyclone(Universe* U, Dest L2, int L2_cache_i, Source L, int source // assign uservalues while( uvi > 0) { - inter_copy_one(U, L2, L2_cache_i, L, lua_absindex( L, -1), VT::NORMAL, mode_, upName_); // ... u uv + std::ignore = inter_copy_one(U + , L2, L2_cache_i + , L, lua_absindex(L, -1) + , VT::NORMAL, mode_, upName_ + ); // ... u uv lua_pop( L, 1); // ... mt __lanesclone [uv]* // this pops the value from the stack lua_setiuservalue( L2, -2, uvi); // ... u @@ -1629,7 +1632,7 @@ static bool copyclone(Universe* U, Dest L2, int L2_cache_i, Source L, int source // ################################################################################################# -static bool inter_copy_userdata(Universe* U, Dest L2, int L2_cache_i, Source L, int i, VT vt_, LookupMode mode_, char const* upName_) +[[nodiscard]] static bool inter_copy_userdata(Universe* U, Dest L2, int L2_cache_i, Source L, int i, VT vt_, LookupMode mode_, char const* upName_) { STACK_CHECK_START_REL(L, 0); STACK_CHECK_START_REL(L2, 0); @@ -1679,7 +1682,7 @@ static bool inter_copy_userdata(Universe* U, Dest L2, int L2_cache_i, Source L, // ################################################################################################# -static bool inter_copy_function(Universe* U, Dest L2, int L2_cache_i, Source L, int source_i_, VT vt_, LookupMode mode_, char const* upName_) +[[nodiscard]] static bool inter_copy_function(Universe* U, Dest L2, int L2_cache_i, Source L, int source_i_, VT vt_, LookupMode mode_, char const* upName_) { if (vt_ == VT::KEY) { @@ -1693,13 +1696,10 @@ static bool inter_copy_function(Universe* U, Dest L2, int L2_cache_i, Source L, if( lua_tocfunction( L, source_i_) == userdata_clone_sentinel) // we are actually copying a clonable full userdata from a keeper { // clone the full userdata again - size_t userdata_size = 0; - void* source; - void* clone; // let's see if we already restored this userdata lua_getupvalue( L, source_i_, 2); // ... u - source = lua_touserdata( L, -1); + void* source = lua_touserdata( L, -1); lua_pushlightuserdata( L2, source); // ... source lua_rawget( L2, L2_cache_i); // ... u? if( !lua_isnil( L2, -1)) @@ -1712,12 +1712,13 @@ static bool inter_copy_function(Universe* U, Dest L2, int L2_cache_i, Source L, lua_pop( L2, 1); // ... // this function has 2 upvalues: the fqn of its metatable, and the userdata itself - lookup_table( L2, L, source_i_, mode_, upName_); // ... mt + std::ignore = lookup_table( L2, L, source_i_, mode_, upName_); // ... mt // originally 'source_i_' slot was the proxy closure, but from now on it indexes the actual userdata we extracted from it source_i_ = lua_gettop( L); source = lua_touserdata( L, -1); + void* clone{ nullptr }; // get the number of bytes to allocate for the clone - userdata_size = (size_t) lua_rawlen( L, -1); + size_t const userdata_size { lua_rawlen(L, -1) }; { // extract uservalues (don't transfer them yet) int uvi = 0; @@ -1738,7 +1739,11 @@ static bool inter_copy_function(Universe* U, Dest L2, int L2_cache_i, Source L, // transfer and assign uservalues while( uvi > 0) { - inter_copy_one(U, L2, L2_cache_i, L, lua_absindex(L, -1), vt_, mode_, upName_); // ... mt u uv + std::ignore = inter_copy_one(U + , L2, L2_cache_i + , L, lua_absindex(L, -1) + , vt_, mode_, upName_ + ); // ... mt u uv lua_pop( L, 1); // ... u [uv]* // this pops the value from the stack lua_setiuservalue( L2, -2, uvi); // ... mt u @@ -1774,7 +1779,7 @@ static bool inter_copy_function(Universe* U, Dest L2, int L2_cache_i, Source L, // ################################################################################################# -static bool inter_copy_table(Universe* U, Dest L2, int L2_cache_i, Source L, int i, VT vt_, LookupMode mode_, char const* upName_) +[[nodiscard]] static bool inter_copy_table(Universe* U, Dest L2, int L2_cache_i, Source L, int i, VT vt_, LookupMode mode_, char const* upName_) { if (vt_ == VT::KEY) { @@ -1846,7 +1851,7 @@ static bool inter_copy_table(Universe* U, Dest L2, int L2_cache_i, Source L, int * * Returns true if value was pushed, false if its type is non-supported. */ -bool inter_copy_one(Universe* U, Dest L2, int L2_cache_i, Source L, int i, VT vt_, LookupMode mode_, char const* upName_) +[[nodiscard]] bool inter_copy_one(Universe* U, Dest L2, int L2_cache_i, Source L, int i, VT vt_, LookupMode mode_, char const* upName_) { bool ret{ true }; int val_type = lua_type( L, i); @@ -1971,78 +1976,77 @@ bool inter_copy_one(Universe* U, Dest L2, int L2_cache_i, Source L, int i, VT vt * * Note: Parameters are in this order ('L' = from first) to be same as 'lua_xmove'. */ -int luaG_inter_copy(Universe* U, Source L, Dest L2, int n, LookupMode mode_) +[[nodiscard]] int luaG_inter_copy(Universe* U, Source L, Dest L2, int n, LookupMode mode_) { - int top_L = lua_gettop(L); // ... {}n - int top_L2 = lua_gettop(L2); // ... - int i, j; + int const top_L{ lua_gettop(L) }; // ... {}n + int const top_L2{ lua_gettop(L2) }; // ... char tmpBuf[16]; - char const* pBuf = U->verboseErrors ? tmpBuf : "?"; - bool copyok{ true }; + char const* pBuf{ U->verboseErrors ? tmpBuf : "?" }; - DEBUGSPEW_CODE( fprintf( stderr, INDENT_BEGIN "luaG_inter_copy()\n" INDENT_END)); + DEBUGSPEW_CODE(fprintf(stderr, INDENT_BEGIN "luaG_inter_copy()\n" INDENT_END)); DEBUGSPEW_CODE(U->debugspew_indent_depth.fetch_add(1, std::memory_order_relaxed)); - if( n > top_L) + if (n > top_L) { // requesting to copy more than is available? - DEBUGSPEW_CODE( fprintf( stderr, INDENT_BEGIN "nothing to copy()\n" INDENT_END)); + DEBUGSPEW_CODE(fprintf(stderr, INDENT_BEGIN "nothing to copy()\n" INDENT_END)); DEBUGSPEW_CODE(U->debugspew_indent_depth.fetch_sub(1, std::memory_order_relaxed)); return -1; } STACK_CHECK_START_REL(L2, 0); - STACK_GROW( L2, n + 1); + STACK_GROW(L2, n + 1); /* * Make a cache table for the duration of this copy. Collects tables and * function entries, avoiding the same entries to be passed on as multiple * copies. ESSENTIAL i.e. for handling upvalue tables in the right manner! */ - lua_newtable( L2); // ... cache + lua_newtable(L2); // ... cache STACK_CHECK_START_REL(L, 0); - for( i = top_L - n + 1, j = 1; i <= top_L; ++ i, ++ j) + bool copyok{ true }; + for (int i = top_L - n + 1, j = 1; i <= top_L; ++i, ++j) { - if( U->verboseErrors) + if (U->verboseErrors) { - sprintf( tmpBuf, "arg_%d", j); + sprintf(tmpBuf, "arg_%d", j); } copyok = inter_copy_one(U, L2, top_L2 + 1, L, i, VT::NORMAL, mode_, pBuf); // ... cache {}n - if( !copyok) + if (!copyok) { break; } } - STACK_CHECK( L, 0); + STACK_CHECK(L, 0); DEBUGSPEW_CODE(U->debugspew_indent_depth.fetch_sub(1, std::memory_order_relaxed)); - if( copyok) + if (copyok) { - STACK_CHECK( L2, n + 1); + STACK_CHECK(L2, n + 1); // Remove the cache table. Persistent caching would cause i.e. multiple // messages passed in the same table to use the same table also in receiving end. - lua_remove( L2, top_L2 + 1); + lua_remove(L2, top_L2 + 1); return 0; } // error -> pop everything from the target state stack - lua_settop( L2, top_L2); - STACK_CHECK( L2, 0); + lua_settop(L2, top_L2); + STACK_CHECK(L2, 0); return -2; } // ################################################################################################# -int luaG_inter_move(Universe* U, Source L, Dest L2, int n, LookupMode mode_) +[[nodiscard]] int luaG_inter_move(Universe* U, Source L, Dest L2, int n, LookupMode mode_) { - int ret = luaG_inter_copy( U, L, L2, n, mode_); - lua_pop( L, (int) n); + int const ret{ luaG_inter_copy(U, L, L2, n, mode_) }; + lua_pop( L, n); return ret; } -int luaG_inter_copy_package(Universe* U, Source L, Dest L2, int package_idx_, LookupMode mode_) +[[nodiscard]] int luaG_inter_copy_package(Universe* U, Source L, Dest L2, int package_idx_, LookupMode mode_) { DEBUGSPEW_CODE(fprintf(stderr, INDENT_BEGIN "luaG_inter_copy_package()\n" INDENT_END)); DEBUGSPEW_CODE(U->debugspew_indent_depth.fetch_add(1, std::memory_order_relaxed)); @@ -2080,7 +2084,7 @@ int luaG_inter_copy_package(Universe* U, Source L, Dest L2, int package_idx_, Lo else { DEBUGSPEW_CODE(U->debugspew_indent_depth.fetch_add(1, std::memory_order_relaxed)); - luaG_inter_move(U, L, L2, 1, mode_); // moves the entry to L2 + std::ignore = luaG_inter_move(U, L, L2, 1, mode_); // moves the entry to L2 DEBUGSPEW_CODE(U->debugspew_indent_depth.fetch_sub(1, std::memory_order_relaxed)); lua_setfield(L2, -2, entry); // set package[entry] } diff --git a/src/tools.h b/src/tools.h index 8e95a4f..f0de7ec 100644 --- a/src/tools.h +++ b/src/tools.h @@ -25,16 +25,16 @@ enum class VT KEY, METATABLE }; -bool inter_copy_one(Universe* U, Dest L2, int L2_cache_i, Source L, int i, VT vt_, LookupMode mode_, char const* upName_); +[[nodiscard]] bool inter_copy_one(Universe* U, Dest L2, int L2_cache_i, Source L, int i, VT vt_, LookupMode mode_, char const* upName_); // ################################################################################################ -int luaG_inter_copy_package(Universe* U, Source L, Dest L2, int package_idx_, LookupMode mode_); +[[nodiscard]] int luaG_inter_copy_package(Universe* U, Source L, Dest L2, int package_idx_, LookupMode mode_); -int luaG_inter_copy(Universe* U, Source L, Dest L2, int n, LookupMode mode_); -int luaG_inter_move(Universe* U, Source L, Dest L2, int n, LookupMode mode_); +[[nodiscard]] int luaG_inter_copy(Universe* U, Source L, Dest L2, int n, LookupMode mode_); +[[nodiscard]] int luaG_inter_move(Universe* U, Source L, Dest L2, int n, LookupMode mode_); -int luaG_nameof(lua_State* L); +[[nodiscard]] int luaG_nameof(lua_State* L); void populate_func_lookup_table(lua_State* L, int _i, char const* _name); void initialize_allocator_function(Universe* U, lua_State* L); diff --git a/src/universe.h b/src/universe.h index f4211af..113ed21 100644 --- a/src/universe.h +++ b/src/universe.h @@ -35,7 +35,7 @@ class AllocatorDefinition lua_Alloc m_allocF{ nullptr }; void* m_allocUD{ nullptr }; - static void* operator new(size_t size_, lua_State* L) noexcept { return lua_newuserdatauv(L, size_, 0); } + [[nodiscard]] static void* operator new(size_t size_, lua_State* L) noexcept { return lua_newuserdatauv(L, size_, 0); } // always embedded somewhere else or "in-place constructed" as a full userdata // can't actually delete the operator because the compiler generates stack unwinding code that could call it in case of exception static void operator delete([[maybe_unused]] void* p_, lua_State* L) { ASSERT_L(!"should never be called") }; @@ -81,7 +81,7 @@ class ProtectedAllocator : public AllocatorDefinition std::mutex m_lock; - static void* protected_lua_Alloc(void* ud_, void* ptr_, size_t osize_, size_t nsize_) + [[nodiscard]] static void* protected_lua_Alloc(void* ud_, void* ptr_, size_t osize_, size_t nsize_) { ProtectedAllocator* const allocator{ static_cast(ud_) }; std::lock_guard guard{ allocator->m_lock }; @@ -91,7 +91,7 @@ class ProtectedAllocator : public AllocatorDefinition public: // we are not like our base class: we can't be created inside a full userdata (or we would have to install a metatable and __gc handler to destroy ourselves properly) - static void* operator new(size_t size_, lua_State* L) noexcept = delete; + [[nodiscard]] static void* operator new(size_t size_, lua_State* L) noexcept = delete; static void operator delete(void* p_, lua_State* L) = delete; AllocatorDefinition makeDefinition() @@ -185,6 +185,6 @@ class Universe // ################################################################################################ -Universe* universe_get(lua_State* L); -Universe* universe_create(lua_State* L); +[[nodiscard]] Universe* universe_get(lua_State* L); +[[nodiscard]] Universe* universe_create(lua_State* L); void universe_store(lua_State* L, Universe* U); -- cgit v1.2.3-55-g6feb From 408f8a5bf7934e7a5aa113fd3a55899db70dd73a Mon Sep 17 00:00:00 2001 From: Benoit Germain Date: Wed, 10 Apr 2024 09:48:59 +0200 Subject: C++ migration: luaG_inter_copy_* now return an enum class instead of an anonymous int --- src/cancel.cpp | 2 +- src/deep.cpp | 2 +- src/keeper.cpp | 12 ++++++------ src/lanes.cpp | 36 ++++++++++++++++++------------------ src/linda.cpp | 2 +- src/macros_and_utils.h | 8 ++++---- src/state.cpp | 34 ++++++++++++++++++++++++---------- src/tools.cpp | 38 ++++++++++++++++++++++++-------------- src/tools.h | 23 ++++++++++++++--------- 9 files changed, 93 insertions(+), 64 deletions(-) (limited to 'src/tools.h') diff --git a/src/cancel.cpp b/src/cancel.cpp index 3a01ac2..b3e52b6 100644 --- a/src/cancel.cpp +++ b/src/cancel.cpp @@ -213,7 +213,7 @@ CancelOp which_cancel_op(char const* op_string_) lua_remove(L, idx_); // argument is processed, remove it if (op == CancelOp::Invalid) { - std::ignore = luaL_error(L, "invalid hook option %s", str); // doesn't return + luaL_error(L, "invalid hook option %s", str); // doesn't return } return op; } diff --git a/src/deep.cpp b/src/deep.cpp index 418724e..d0b8123 100644 --- a/src/deep.cpp +++ b/src/deep.cpp @@ -488,7 +488,7 @@ bool copydeep(Universe* U, Dest L2, int L2_cache_i, Source L, int i, LookupMode { // raise the error in the proper state (not the keeper) lua_State* const errL{ (mode_ == LookupMode::FromKeeper) ? L2 : L }; - std::ignore = luaL_error(errL, errmsg); // doesn't return + luaL_error(errL, errmsg); // doesn't return } return true; } \ No newline at end of file diff --git a/src/keeper.cpp b/src/keeper.cpp index 37fcba5..f56c50c 100644 --- a/src/keeper.cpp +++ b/src/keeper.cpp @@ -656,7 +656,7 @@ void init_keepers(Universe* U, lua_State* L) lua_pop(L, 1); // if (nb_keepers < 1) { - std::ignore = luaL_error(L, "Bad number of keepers (%d)", nb_keepers); // doesn't return + luaL_error(L, "Bad number of keepers (%d)", nb_keepers); // doesn't return } STACK_CHECK(L, 0); @@ -671,7 +671,7 @@ void init_keepers(Universe* U, lua_State* L) U->keepers = static_cast(U->internal_allocator.alloc(bytes)); if (U->keepers == nullptr) { - std::ignore = luaL_error(L, "init_keepers() failed while creating keeper array; out of memory"); // doesn't return + luaL_error(L, "init_keepers() failed while creating keeper array; out of memory"); // doesn't return } U->keepers->Keepers::Keepers(); U->keepers->gc_threshold = keepers_gc_threshold; @@ -688,7 +688,7 @@ void init_keepers(Universe* U, lua_State* L) lua_State* const K{ create_state(U, L) }; if (K == nullptr) { - std::ignore = luaL_error(L, "init_keepers() failed while creating keeper states; out of memory"); // doesn't return + luaL_error(L, "init_keepers() failed while creating keeper states; out of memory"); // doesn't return } U->keepers->keeper_array[i].L = K; @@ -717,7 +717,7 @@ void init_keepers(Universe* U, lua_State* L) if (!lua_isnil(L, -1)) { // when copying with mode LookupMode::ToKeeper, error message is pushed at the top of the stack, not raised immediately - if (luaG_inter_copy_package(U, Source{ L }, Dest{ K }, -1, LookupMode::ToKeeper)) + if (luaG_inter_copy_package(U, Source{ L }, Dest{ K }, -1, LookupMode::ToKeeper) != InterCopyResult::Success) { // if something went wrong, the error message is at the top of the stack lua_remove(L, -2); // error_msg @@ -840,7 +840,7 @@ int keeper_call(Universe* U, lua_State* K, keeper_api_t func_, lua_State* L, voi lua_pushlightuserdata(K, linda); - if ((args == 0) || luaG_inter_copy(U, Source{ L }, Dest{ K }, args, LookupMode::ToKeeper) == 0) // L->K + if ((args == 0) || luaG_inter_copy(U, Source{ L }, Dest{ K }, args, LookupMode::ToKeeper) == InterCopyResult::Success) // L->K { lua_call(K, 1 + args, LUA_MULTRET); retvals = lua_gettop(K) - Ktos; @@ -848,7 +848,7 @@ int keeper_call(Universe* U, lua_State* K, keeper_api_t func_, lua_State* L, voi // this may interrupt a lane, causing the destruction of the underlying OS thread // after this, another lane making use of this keeper can get an error code from the mutex-locking function // when attempting to grab the mutex again (WINVER <= 0x400 does this, but locks just fine, I don't know about pthread) - if ((retvals > 0) && luaG_inter_move(U, Source{ K }, Dest{ L }, retvals, LookupMode::FromKeeper) != 0) // K->L + if ((retvals > 0) && luaG_inter_move(U, Source{ K }, Dest{ L }, retvals, LookupMode::FromKeeper) != InterCopyResult::Success) // K->L { retvals = -1; } diff --git a/src/lanes.cpp b/src/lanes.cpp index f549359..462de0f 100644 --- a/src/lanes.cpp +++ b/src/lanes.cpp @@ -540,7 +540,7 @@ static void selfdestruct_add(Lane* lane_) if (lane != SELFDESTRUCT_END) { // this causes a leak because we don't call U's destructor (which could be bad if the still running lanes are accessing it) - std::ignore = luaL_error(L, "Zombie thread %s refuses to die!", lane->debug_name); // doesn't return + luaL_error(L, "Zombie thread %s refuses to die!", lane->debug_name); // doesn't return } } @@ -1134,7 +1134,8 @@ LUAG_FUNC(lane_new) { DEBUGSPEW_CODE(fprintf(stderr, INDENT_BEGIN "lane_new: update 'package'\n" INDENT_END)); // when copying with mode LookupMode::LaneBody, should raise an error in case of problem, not leave it one the stack - std::ignore = luaG_inter_copy_package(U, Source{ L }, Dest{ L2 }, package_idx, LookupMode::LaneBody); + [[maybe_unused]] InterCopyResult const ret{ luaG_inter_copy_package(U, Source{ L }, Dest{ L2 }, package_idx, LookupMode::LaneBody) }; + ASSERT_L(ret == InterCopyResult::Success); // either all went well, or we should not even get here } // modules to require in the target lane *before* the function is transfered! @@ -1146,7 +1147,7 @@ LUAG_FUNC(lane_new) // should not happen, was checked in lanes.lua before calling lane_new() if (lua_type(L, required_idx) != LUA_TTABLE) { - return luaL_error(L, "expected required module list as a table, got %s", luaL_typename(L, required_idx)); + luaL_error(L, "expected required module list as a table, got %s", luaL_typename(L, required_idx)); // doesn't return } lua_pushnil(L); // func libs priority globals package required gc_cb [... args ...] nil @@ -1154,7 +1155,7 @@ LUAG_FUNC(lane_new) { if (lua_type(L, -1) != LUA_TSTRING || lua_type(L, -2) != LUA_TNUMBER || lua_tonumber(L, -2) != nbRequired) { - return luaL_error(L, "required module list should be a list of strings"); + luaL_error(L, "required module list should be a list of strings"); // doesn't return } else { @@ -1168,7 +1169,7 @@ LUAG_FUNC(lane_new) if (lua_isnil( L2, -1)) { lua_pop( L2, 1); // - return luaL_error(L, "cannot pre-require modules without loading 'package' library first"); + luaL_error(L, "cannot pre-require modules without loading 'package' library first"); // doesn't return } else { @@ -1203,7 +1204,7 @@ LUAG_FUNC(lane_new) DEBUGSPEW_CODE( fprintf( stderr, INDENT_BEGIN "lane_new: transfer globals\n" INDENT_END)); if (!lua_istable(L, globals_idx)) { - return luaL_error(L, "Expected table, got %s", luaL_typename(L, globals_idx)); + luaL_error(L, "Expected table, got %s", luaL_typename(L, globals_idx)); // doesn't return } DEBUGSPEW_CODE(U->debugspew_indent_depth.fetch_add(1, std::memory_order_relaxed)); @@ -1230,11 +1231,11 @@ LUAG_FUNC(lane_new) DEBUGSPEW_CODE(fprintf( stderr, INDENT_BEGIN "lane_new: transfer lane body\n" INDENT_END)); DEBUGSPEW_CODE(U->debugspew_indent_depth.fetch_add(1, std::memory_order_relaxed)); lua_pushvalue(L, 1); // func libs priority globals package required gc_cb [... args ...] func - int const res{ luaG_inter_move(U, Source{ L }, Dest{ L2 }, 1, LookupMode::LaneBody) }; // func libs priority globals package required gc_cb [... args ...] // func + InterCopyResult const res{ luaG_inter_move(U, Source{ L }, Dest{ L2 }, 1, LookupMode::LaneBody) }; // func libs priority globals package required gc_cb [... args ...] // func DEBUGSPEW_CODE(U->debugspew_indent_depth.fetch_sub(1, std::memory_order_relaxed)); - if (res != 0) + if (res != InterCopyResult::Success) { - return luaL_error(L, "tried to copy unsupported types"); + luaL_error(L, "tried to copy unsupported types"); // doesn't return } } else if (lua_type(L, 1) == LUA_TSTRING) @@ -1243,7 +1244,7 @@ LUAG_FUNC(lane_new) // compile the string if (luaL_loadstring(L2, lua_tostring(L, 1)) != 0) // func { - return luaL_error(L, "error when parsing lane function code"); + luaL_error(L, "error when parsing lane function code"); // doesn't return } } STACK_CHECK(L, 0); @@ -1253,14 +1254,13 @@ LUAG_FUNC(lane_new) // revive arguments if (nargs > 0) { - int res; DEBUGSPEW_CODE(fprintf( stderr, INDENT_BEGIN "lane_new: transfer lane arguments\n" INDENT_END)); DEBUGSPEW_CODE(U->debugspew_indent_depth.fetch_add(1, std::memory_order_relaxed)); - res = luaG_inter_move(U, Source{ L }, Dest{ L2 }, nargs, LookupMode::LaneBody); // func libs priority globals package required gc_cb // func [... args ...] + InterCopyResult const res{ luaG_inter_move(U, Source{ L }, Dest{ L2 }, nargs, LookupMode::LaneBody) }; // func libs priority globals package required gc_cb // func [... args ...] DEBUGSPEW_CODE(U->debugspew_indent_depth.fetch_sub(1, std::memory_order_relaxed)); - if (res != 0) + if (res != InterCopyResult::Success) { - return luaL_error(L, "tried to copy unsupported types"); + luaL_error(L, "tried to copy unsupported types"); // doesn't return } } STACK_CHECK(L, -nargs); @@ -1420,9 +1420,9 @@ LUAG_FUNC(thread_join) case Lane::Done: { int const n{ lua_gettop(L2) }; // whole L2 stack - if ((n > 0) && (luaG_inter_move(U, Source{ L2 }, Dest{ L }, n, LookupMode::LaneBody) != 0)) + if ((n > 0) && (luaG_inter_move(U, Source{ L2 }, Dest{ L }, n, LookupMode::LaneBody) != InterCopyResult::Success)) { - return luaL_error(L, "tried to copy unsupported types"); + luaL_error(L, "tried to copy unsupported types"); // doesn't return } ret = n; } @@ -1434,9 +1434,9 @@ LUAG_FUNC(thread_join) STACK_GROW(L, 3); lua_pushnil(L); // even when ERROR_FULL_STACK, if the error is not LUA_ERRRUN, the handler wasn't called, and we only have 1 error message on the stack ... - if (luaG_inter_move(U, Source{ L2 }, Dest{ L }, n, LookupMode::LaneBody) != 0) // nil "err" [trace] + if (luaG_inter_move(U, Source{ L2 }, Dest{ L }, n, LookupMode::LaneBody) != InterCopyResult::Success) // nil "err" [trace] { - return luaL_error(L, "tried to copy unsupported types: %s", lua_tostring(L, -n)); + luaL_error(L, "tried to copy unsupported types: %s", lua_tostring(L, -n)); // doesn't return } ret = 1 + n; } diff --git a/src/linda.cpp b/src/linda.cpp index 50964ad..e749f52 100644 --- a/src/linda.cpp +++ b/src/linda.cpp @@ -162,7 +162,7 @@ static void check_key_types(lua_State* L, int start_, int end_) { continue; } - std::ignore = luaL_error(L, "argument #%d: invalid key type (not a boolean, string, number or light userdata)", i); + luaL_error(L, "argument #%d: invalid key type (not a boolean, string, number or light userdata)", i); // doesn't return } } diff --git a/src/macros_and_utils.h b/src/macros_and_utils.h index 31ae8bd..99e49f9 100644 --- a/src/macros_and_utils.h +++ b/src/macros_and_utils.h @@ -77,7 +77,7 @@ class StackChecker if ((offset_ < 0) || (m_oldtop < 0)) { assert(false); - std::ignore = luaL_error(m_L, "STACK INIT ASSERT failed (%d not %d): %s:%d", lua_gettop(m_L), offset_, file_, line_); + luaL_error(m_L, "STACK INIT ASSERT failed (%d not %d): %s:%d", lua_gettop(m_L), offset_, file_, line_); // doesn't return } } @@ -88,7 +88,7 @@ class StackChecker if (lua_gettop(m_L) != pos_) { assert(false); - std::ignore = luaL_error(m_L, "STACK INIT ASSERT failed (%d not %d): %s:%d", lua_gettop(m_L), pos_, file_, line_); + luaL_error(m_L, "STACK INIT ASSERT failed (%d not %d): %s:%d", lua_gettop(m_L), pos_, file_, line_); // doesn't return } } @@ -108,7 +108,7 @@ class StackChecker if (actual != expected_) { assert(false); - std::ignore = luaL_error(m_L, "STACK ASSERT failed (%d not %d): %s:%d", actual, expected_, file_, line_); + luaL_error(m_L, "STACK ASSERT failed (%d not %d): %s:%d", actual, expected_, file_, line_); // doesn't return } } } @@ -128,7 +128,7 @@ inline void STACK_GROW(lua_State* L, int n_) { if (!lua_checkstack(L, n_)) { - std::ignore = luaL_error(L, "Cannot grow stack!"); + luaL_error(L, "Cannot grow stack!"); // doesn't return } } diff --git a/src/state.cpp b/src/state.cpp index 496e21e..4a5f995 100644 --- a/src/state.cpp +++ b/src/state.cpp @@ -8,7 +8,7 @@ =============================================================================== Copyright (C) 2002-10 Asko Kauppi -2011-21 benoit Germain +2011-24 benoit Germain Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal @@ -79,6 +79,8 @@ THE SOFTWARE. return lua_gettop(L); // result(s) } +// ################################################################################################# + /* * Serialize calls to 'require', if it exists */ @@ -117,6 +119,7 @@ void serialize_require(DEBUGSPEW_PARAM_COMMA( Universe* U) lua_State* L) return 1; } +// ################################################################################################# static luaL_Reg const libs[] = { @@ -154,7 +157,9 @@ static luaL_Reg const libs[] = { nullptr, nullptr } }; -static void open1lib( DEBUGSPEW_PARAM_COMMA( Universe* U) lua_State* L, char const* name_, size_t len_) +// ################################################################################################# + +static void open1lib(DEBUGSPEW_PARAM_COMMA(Universe* U) lua_State* L, char const* name_, size_t len_) { int i; for( i = 0; libs[i].name; ++ i) @@ -183,6 +188,7 @@ static void open1lib( DEBUGSPEW_PARAM_COMMA( Universe* U) lua_State* L, char con } } +// ################################################################################################# // just like lua_xmove, args are (from, to) static void copy_one_time_settings(Universe* U, Source L, Dest L2) @@ -194,19 +200,21 @@ static void copy_one_time_settings(Universe* U, Source L, Dest L2) DEBUGSPEW_CODE(fprintf( stderr, INDENT_BEGIN "copy_one_time_settings()\n" INDENT_END)); DEBUGSPEW_CODE(U->debugspew_indent_depth.fetch_add(1, std::memory_order_relaxed)); - CONFIG_REGKEY.pushValue(L); // config + CONFIG_REGKEY.pushValue(L); // config // copy settings from from source to destination registry - if( luaG_inter_move( U, L, L2, 1, LookupMode::LaneBody) < 0) // // config + if (luaG_inter_move(U, L, L2, 1, LookupMode::LaneBody) != InterCopyResult::Success) // // config { - (void) luaL_error( L, "failed to copy settings when loading lanes.core"); + luaL_error( L, "failed to copy settings when loading lanes.core"); // doesn't return } // set L2:_R[CONFIG_REGKEY] = settings - CONFIG_REGKEY.setValue(L2, [](lua_State* L) { lua_insert(L, -2); }); // config - STACK_CHECK( L2, 0); - STACK_CHECK( L, 0); + CONFIG_REGKEY.setValue(L2, [](lua_State* L) { lua_insert(L, -2); }); // config + STACK_CHECK(L2, 0); + STACK_CHECK(L, 0); DEBUGSPEW_CODE(U->debugspew_indent_depth.fetch_sub(1, std::memory_order_relaxed)); } +// ################################################################################################# + void initialize_on_state_create( Universe* U, lua_State* L) { STACK_CHECK_START_REL(L, 1); // settings @@ -238,7 +246,9 @@ void initialize_on_state_create( Universe* U, lua_State* L) STACK_CHECK(L, 1); } -lua_State* create_state( Universe* U, lua_State* from_) +// ################################################################################################# + +lua_State* create_state(Universe* U, lua_State* from_) { lua_State* L; #if LUAJIT_FLAVOR() == 64 @@ -264,11 +274,13 @@ lua_State* create_state( Universe* U, lua_State* from_) if (L == nullptr) { - std::ignore = luaL_error( from_, "luaG_newstate() failed while creating state; out of memory"); + luaL_error(from_, "luaG_newstate() failed while creating state; out of memory"); // doesn't return } return L; } +// ################################################################################################# + void call_on_state_create(Universe* U, lua_State* L, lua_State* from_, LookupMode mode_) { if (U->on_state_create_func != nullptr) @@ -304,6 +316,8 @@ void call_on_state_create(Universe* U, lua_State* L, lua_State* from_, LookupMod } } +// ################################################################################################# + /* * Like 'luaL_openlibs()' but allows the set of libraries be selected * diff --git a/src/tools.cpp b/src/tools.cpp index 9207df6..aa95f04 100644 --- a/src/tools.cpp +++ b/src/tools.cpp @@ -1415,7 +1415,7 @@ static void copy_cached_func(Universe* U, Dest L2, int L2_cache_i, Source L, int lua_pop(L2, 1); // _R[REG_MTID] if (!inter_copy_one(U, L2, L2_cache_i, L, lua_gettop(L), VT::METATABLE, mode_, upName_)) // _R[REG_MTID] mt? { - std::ignore = luaL_error(L, "Error copying a metatable"); // doesn't return + luaL_error(L, "Error copying a metatable"); // doesn't return } STACK_CHECK(L2, 2); // _R[REG_MTID] mt @@ -1600,7 +1600,7 @@ static constexpr UniqueKey CLONABLES_CACHE_KEY{ 0xD04EE018B3DEE8F5ull }; { if (!inter_copy_one(U, L2, L2_cache_i, L, lua_absindex(L, -1), VT::NORMAL, mode_, upName_)) // ... u uv { - std::ignore = luaL_error(L, "Cannot copy upvalue type '%s'", luaL_typename(L, -1)); // doesn't return + luaL_error(L, "Cannot copy upvalue type '%s'", luaL_typename(L, -1)); // doesn't return } lua_pop( L, 1); // ... mt __lanesclone [uv]* // this pops the value from the stack @@ -1670,7 +1670,7 @@ static constexpr UniqueKey CLONABLES_CACHE_KEY{ 0xD04EE018B3DEE8F5ull }; } else // raise an error { - std::ignore = luaL_error(L, "can't copy non-deep full userdata across lanes"); // doesn't return + luaL_error(L, "can't copy non-deep full userdata across lanes"); // doesn't return } STACK_CHECK(L2, 1); @@ -1739,7 +1739,7 @@ static constexpr UniqueKey CLONABLES_CACHE_KEY{ 0xD04EE018B3DEE8F5ull }; { if (!inter_copy_one(U, L2, L2_cache_i, L, lua_absindex(L, -1), vt_, mode_, upName_)) // ... mt u uv { - std::ignore = luaL_error(L, "Cannot copy upvalue type '%s'", luaL_typename(L, -1)); // doesn't return + luaL_error(L, "Cannot copy upvalue type '%s'", luaL_typename(L, -1)); // doesn't return } lua_pop(L, 1); // ... u [uv]* // this pops the value from the stack @@ -1973,7 +1973,7 @@ static constexpr UniqueKey CLONABLES_CACHE_KEY{ 0xD04EE018B3DEE8F5ull }; * * Note: Parameters are in this order ('L' = from first) to be same as 'lua_xmove'. */ -[[nodiscard]] int luaG_inter_copy(Universe* U, Source L, Dest L2, int n, LookupMode mode_) +[[nodiscard]] InterCopyResult luaG_inter_copy(Universe* U, Source L, Dest L2, int n, LookupMode mode_) { int const top_L{ lua_gettop(L) }; // ... {}n int const top_L2{ lua_gettop(L2) }; // ... @@ -1988,7 +1988,7 @@ static constexpr UniqueKey CLONABLES_CACHE_KEY{ 0xD04EE018B3DEE8F5ull }; // requesting to copy more than is available? DEBUGSPEW_CODE(fprintf(stderr, INDENT_BEGIN "nothing to copy()\n" INDENT_END)); DEBUGSPEW_CODE(U->debugspew_indent_depth.fetch_sub(1, std::memory_order_relaxed)); - return -1; + return InterCopyResult::NotEnoughValues; } STACK_CHECK_START_REL(L2, 0); @@ -2025,25 +2025,31 @@ static constexpr UniqueKey CLONABLES_CACHE_KEY{ 0xD04EE018B3DEE8F5ull }; // Remove the cache table. Persistent caching would cause i.e. multiple // messages passed in the same table to use the same table also in receiving end. lua_remove(L2, top_L2 + 1); - return 0; + return InterCopyResult::Success; } // error -> pop everything from the target state stack lua_settop(L2, top_L2); STACK_CHECK(L2, 0); - return -2; + return InterCopyResult::Error; } // ################################################################################################# -[[nodiscard]] int luaG_inter_move(Universe* U, Source L, Dest L2, int n, LookupMode mode_) +[[nodiscard]] InterCopyResult luaG_inter_move(Universe* U, Source L, Dest L2, int n_, LookupMode mode_) { - int const ret{ luaG_inter_copy(U, L, L2, n, mode_) }; - lua_pop( L, n); + InterCopyResult const ret{ luaG_inter_copy(U, L, L2, n_, mode_) }; + lua_pop( L, n_); return ret; } -[[nodiscard]] int luaG_inter_copy_package(Universe* U, Source L, Dest L2, int package_idx_, LookupMode mode_) +// ################################################################################################# + +// transfers stuff from L->_G["package"] to L2->_G["package"] +// returns InterCopyResult::Success if everything is fine +// returns InterCopyResult::Error if pushed an error message in L +// else raise an error in L +[[nodiscard]] InterCopyResult luaG_inter_copy_package(Universe* U, Source L, Dest L2, int package_idx_, LookupMode mode_) { DEBUGSPEW_CODE(fprintf(stderr, INDENT_BEGIN "luaG_inter_copy_package()\n" INDENT_END)); DEBUGSPEW_CODE(U->debugspew_indent_depth.fetch_add(1, std::memory_order_relaxed)); @@ -2056,7 +2062,11 @@ static constexpr UniqueKey CLONABLES_CACHE_KEY{ 0xD04EE018B3DEE8F5ull }; lua_pushfstring(L, "expected package as table, got %s", luaL_typename(L, package_idx_)); STACK_CHECK(L, 1); // raise the error when copying from lane to lane, else just leave it on the stack to be raised later - return (mode_ == LookupMode::LaneBody) ? lua_error(L) : 1; + if (mode_ == LookupMode::LaneBody) + { + lua_error(L); // doesn't return + } + return InterCopyResult::Error; } lua_getglobal(L2, "package"); if (!lua_isnil(L2, -1)) // package library not loaded: do nothing @@ -2095,5 +2105,5 @@ static constexpr UniqueKey CLONABLES_CACHE_KEY{ 0xD04EE018B3DEE8F5ull }; STACK_CHECK(L2, 0); STACK_CHECK(L, 0); DEBUGSPEW_CODE(U->debugspew_indent_depth.fetch_sub(1, std::memory_order_relaxed)); - return 0; + return InterCopyResult::Success; } diff --git a/src/tools.h b/src/tools.h index f0de7ec..dce7378 100644 --- a/src/tools.h +++ b/src/tools.h @@ -1,8 +1,6 @@ #pragma once -#include "threading.h" #include "deep.h" - #include "macros_and_utils.h" // forwards @@ -11,13 +9,13 @@ class Universe; // ################################################################################################ #ifdef _DEBUG -void luaG_dump( lua_State* L); +void luaG_dump(lua_State* L); #endif // _DEBUG // ################################################################################################ -void push_registry_subtable_mode( lua_State* L, UniqueKey key_, const char* mode_); -void push_registry_subtable( lua_State* L, UniqueKey key_); +void push_registry_subtable_mode(lua_State* L, UniqueKey key_, const char* mode_); +void push_registry_subtable(lua_State* L, UniqueKey key_); enum class VT { @@ -25,14 +23,21 @@ enum class VT KEY, METATABLE }; + +enum class InterCopyResult +{ + Success, + NotEnoughValues, + Error +}; + [[nodiscard]] bool inter_copy_one(Universe* U, Dest L2, int L2_cache_i, Source L, int i, VT vt_, LookupMode mode_, char const* upName_); // ################################################################################################ -[[nodiscard]] int luaG_inter_copy_package(Universe* U, Source L, Dest L2, int package_idx_, LookupMode mode_); - -[[nodiscard]] int luaG_inter_copy(Universe* U, Source L, Dest L2, int n, LookupMode mode_); -[[nodiscard]] int luaG_inter_move(Universe* U, Source L, Dest L2, int n, LookupMode mode_); +[[nodiscard]] InterCopyResult luaG_inter_copy_package(Universe* U, Source L, Dest L2, int package_idx_, LookupMode mode_); +[[nodiscard]] InterCopyResult luaG_inter_copy(Universe* U, Source L, Dest L2, int n, LookupMode mode_); +[[nodiscard]] InterCopyResult luaG_inter_move(Universe* U, Source L, Dest L2, int n, LookupMode mode_); [[nodiscard]] int luaG_nameof(lua_State* L); -- cgit v1.2.3-55-g6feb