diff options
author | Benoit Germain <benoit.germain@ubisoft.com> | 2024-04-09 15:49:18 +0200 |
---|---|---|
committer | Benoit Germain <benoit.germain@ubisoft.com> | 2024-04-09 15:49:18 +0200 |
commit | 95932749a53f46ae5901798a17e1f0c4c33ac6b2 (patch) | |
tree | 327d85f6f634edfacc01dc587492d96f9200344d | |
parent | 98ff4191c2cd215c7d6a429e9ddd66f7a6a30316 (diff) | |
download | lanes-95932749a53f46ae5901798a17e1f0c4c33ac6b2.tar.gz lanes-95932749a53f46ae5901798a17e1f0c4c33ac6b2.tar.bz2 lanes-95932749a53f46ae5901798a17e1f0c4c33ac6b2.zip |
C++ migration: use strong type safety for source and destination states in transfer functions
Diffstat (limited to '')
-rw-r--r-- | CHANGES | 744 | ||||
-rw-r--r-- | lanes-4.0.0-0.rockspec (renamed from lanes-3.16.3-0.rockspec) | 4 | ||||
-rw-r--r-- | src/cancel.cpp | 7 | ||||
-rw-r--r-- | src/compat.cpp | 35 | ||||
-rw-r--r-- | src/deep.cpp | 40 | ||||
-rw-r--r-- | src/deep.h | 4 | ||||
-rw-r--r-- | src/keeper.cpp | 16 | ||||
-rw-r--r-- | src/keeper.h | 2 | ||||
-rw-r--r-- | src/lanes.cpp | 20 | ||||
-rw-r--r-- | src/lanes_private.h | 4 | ||||
-rw-r--r-- | src/linda.cpp | 10 | ||||
-rw-r--r-- | src/macros_and_utils.h | 18 | ||||
-rw-r--r-- | src/state.cpp | 29 | ||||
-rw-r--r-- | src/state.h | 3 | ||||
-rw-r--r-- | src/threading.cpp | 12 | ||||
-rw-r--r-- | src/threading.h | 1 | ||||
-rw-r--r-- | src/tools.cpp | 85 | ||||
-rw-r--r-- | src/tools.h | 14 | ||||
-rw-r--r-- | src/universe.cpp | 8 |
19 files changed, 161 insertions, 895 deletions
@@ -1,746 +1,6 @@ | |||
1 | CHANGES: | 1 | CHANGES: |
2 | 2 | ||
3 | CHANGE 159: BGe 19-Mar-24 | 3 | CHANGE 1: BGe 9-Apr-24 |
4 | * fix small internal issue with when hitting timeout on thread kill during thread_cancel() in pthread implementation | 4 | * reset changelog, next entry will list API changes since last C-implementation. |
5 | |||
6 | CHANGE 158: BGe 22-Feb-24 | ||
7 | * naive luajit detection in PUC-Lua-based builds, and vice-versa to detect mismatches | ||
8 | * internal version bumped to 3.16.3 | ||
9 | |||
10 | CHANGE 157: Mitalie 17-Aug-23 | ||
11 | * Prevent crash on linux as non-root | ||
12 | * internal version bumped to 3.16.2 | ||
13 | |||
14 | CHANGE 156: BGe 9-Aug-23 | ||
15 | * new configuration option .internal_allocator to help LuaJIT users. | ||
16 | * internal version bumped to 3.16.1 | ||
17 | |||
18 | CHANGE 155: BGe 28-Jul-23 | ||
19 | * 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). | ||
20 | |||
21 | CHANGE 154: eligovision 1-Mar-22 | ||
22 | * Fix 3-parametrized __lanesclone | ||
23 | |||
24 | CHANGE 153: BGe 17-Feb-22 | ||
25 | * NEVER use allocator obtained from lua_getallocf to allocate stuff manually when compiling for LuaJIT | ||
26 | |||
27 | CHANGE 152: BGe 7-Feb-22 | ||
28 | * bumped version to 3.16.0 | ||
29 | * __lanesclone is now called only once with 3 parameters dest, source, size -> BREAKS CUSTOM DEEP USERDATA API | ||
30 | |||
31 | CHANGE 151: BGe 7-Feb-22 | ||
32 | * bumped version to 3.15.2 | ||
33 | * Lanes no longer relies on malloc/free for internal allocations, but uses the primary alloc function from the master Lua state | ||
34 | |||
35 | CHANGE 150: BGe 22-Sep-21 | ||
36 | * fix require() wrapper to return all values returned by original require() | ||
37 | |||
38 | CHANGE 149: BGe 8-Jul-21 | ||
39 | * bumped version to 3.15.1 | ||
40 | * fix function transfer with lua_dump for Lua 5.4 failing for functions big enough to necessitate a buffer reallocation | ||
41 | |||
42 | CHANGE 148: BGe 23-Jun-21 | ||
43 | * __lanesclone now receives the original as light userdata the first time it is called -> BREAKS CUSTOM DEEP USERDATA API | ||
44 | |||
45 | CHANGE 147: BGe 16-Jun-21 | ||
46 | * changed lanes.threads() output so that several lanes with the same name don't clobber each other in the result table -> BREAKS API | ||
47 | * bumped version to 3.15 because of the API change | ||
48 | |||
49 | CHANGE 146: BGe 26-Apr-19 | ||
50 | * lane:cancel() rework (see doc). | ||
51 | * opt.cancelstep is gone, hook is installed by lane:cancel() if requested | ||
52 | |||
53 | CHANGE 145: BGe 28-Nov-18 | ||
54 | * more code refacto | ||
55 | * don't test __lanesignore for POD types (-> slightly faster when trasnfering lots of data) | ||
56 | |||
57 | CHANGE 144: BGe 28-Nov-18 | ||
58 | * some code refacto | ||
59 | |||
60 | CHANGE 143: BGe 27-Nov-18 | ||
61 | * Lua 5.4 support | ||
62 | * __lanesclone and lanes.nameof support userdata uservalue(s) | ||
63 | |||
64 | CHANGE 142: BGe 26-Nov-18 | ||
65 | * Version is available in public header | ||
66 | |||
67 | CHANGE 141: BGe 25-Nov-18 | ||
68 | * protect_allocator configure option is gone, long live allocator (more embedders-friendly) | ||
69 | |||
70 | CHANGE 140: BGe 22-Nov-18 | ||
71 | * Raise an error instead of crashing when attempting to transfer a non-deep full userdata | ||
72 | |||
73 | CHANGE 139: BGe 21-Nov-18 | ||
74 | * more DEBUGSPEW | ||
75 | |||
76 | CHANGE 138: BGe 19-Nov-18 | ||
77 | * Registry access code utility macros | ||
78 | * CONFIG_REGKEY and LOOKUP_REGKEY are now lightuserdata instead of strings | ||
79 | * Stack checking debug macros improvements | ||
80 | |||
81 | CHANGE 137: BGe 15-Nov-18 | ||
82 | * Deep userdata must embed DeepPrelude to save an allocation (also changes Deep protocol) | ||
83 | |||
84 | CHANGE 136: BGe 15-Nov-18 | ||
85 | * split linda code in a separate file | ||
86 | * rockspec for version v3.13.0 | ||
87 | |||
88 | CHANGE 135: BGe 11-Nov-18 | ||
89 | * fix a bunch of compilation warnings | ||
90 | |||
91 | CHANGE 134: BGe 3-Dec-13 | ||
92 | * new API lanes.set_thread_affinity() | ||
93 | * set_debug_threadname implemented with win32 pthread | ||
94 | |||
95 | CHANGE 133: BGe 8-Nov-18 | ||
96 | * Make sure any linda operation that can raise an error won't ever leave a mutex unreleased | ||
97 | * lane:join() now returns nil, "timeout" in case of timeout | ||
98 | |||
99 | CHANGE 132: BGe 7-Nov-18 | ||
100 | * __lanesclone mechanism should actually work now | ||
101 | |||
102 | CHANGE 131: BGe 7-Nov-18 | ||
103 | * Fix potential crash at application shutdown when deep userdata were created before Lanes is required | ||
104 | |||
105 | CHANGE 130: BGe 2-Nov-18 | ||
106 | * always duplicate the config structure in new lanes even when no libraries are initialized by the generator | ||
107 | (fixes an internal error trying to call on_state_create in a lane without any libs loaded) | ||
108 | |||
109 | CHANGE 129: BGe 2-Nov-18 | ||
110 | * Bumped version to 3.13 | ||
111 | * fix error when autodetecting protect_allocator when running under LuaJIT | ||
112 | |||
113 | CHANGE 128: BGe 31-Oct-18 | ||
114 | * Better default value autodetection for protect_allocator setting | ||
115 | |||
116 | CHANGE 127: BGe 30-Oct-18 | ||
117 | * restrict internal light userdata constants to 47 significant bits when building against LuaJIT-x64 | ||
118 | |||
119 | CHANGE 126: Bge 29-Oct-18 | ||
120 | * Add deep user data cloning support | ||
121 | |||
122 | CHANGE 125: BGe 25-Oct-18 | ||
123 | * Fix Lanes build by reorganizing types around a bit | ||
124 | |||
125 | CHANGE 124: BGe 9-Jul-18 | ||
126 | * Fix a stack overflow when copying large tables with verbose_errors option enabled | ||
127 | * Support for integer formatting in verbose errors | ||
128 | |||
129 | CHANGE 123: BGe 2-Aug-17 | ||
130 | * added support for user-provided __gc in deep userdata | ||
131 | * more complete deep userdata sample | ||
132 | |||
133 | CHANGE 122: BGe 1-Aug-17 | ||
134 | * fix crash trying to use a deep-aware module while not requiring Lanes | ||
135 | * bumped version to 3.12 | ||
136 | |||
137 | CHANGE 121: BGe 13-Jun-17 | ||
138 | * no longer internally assert when an error message is not a string | ||
139 | |||
140 | CHANGE 120: BGe 5-Jun-17 | ||
141 | * new API function lanes.register( "name", module) to manually register a module table after it was required | ||
142 | * Transfering registered module tables will link the equivalent in the destination state instead of cloning it | ||
143 | * bumped version to 3.11 | ||
144 | |||
145 | CHANGE 119: BGe 10-May-17 | ||
146 | * Fixed some compilation warnings | ||
147 | * Improved LuaJIT support | ||
148 | |||
149 | CHANGE 118: trukanduk 21-Nov-16 | ||
150 | * bumped version to 3.10.1 | ||
151 | * objects with a metatable that contains __lanesignore are skipped during data transfers | ||
152 | |||
153 | CHANGE 117: mpeterv 21-Nov-16 | ||
154 | * Fix an implicit number-to-string conversion | ||
155 | |||
156 | CHANGE 116: BGe, mpeterv 27-Apr-15 | ||
157 | * bumped version to 3.10.0 | ||
158 | * segfault fixed in LG_lane_new | ||
159 | * Lua 5.3 support | ||
160 | |||
161 | CHANGE 115: BGe 18-Sep-14 | ||
162 | * bumped version to 3.9.7 | ||
163 | * new function lanes.sleep() | ||
164 | |||
165 | CHANGE 114: BGe 8-Jul-14 | ||
166 | * Postponed _G scan for function lookup database to after on_state_create invocation | ||
167 | * Fixed a crash when USE_DEBUG_SPEW == 1 | ||
168 | |||
169 | CHANGE 113: BGe 17-Jun-14 | ||
170 | * bumped version to 3.9.6 | ||
171 | * 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 | ||
172 | 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() | ||
173 | |||
174 | CHANGE 112 BGe 16-May-14 | ||
175 | * bumped version to 3.9.5 | ||
176 | * fix linda.__towatch to return non-nil when the linda is empty | ||
177 | * lanes.gen() error reporting improvements | ||
178 | |||
179 | CHANGE 111 BGe 24-Apr-14 | ||
180 | * fixed linda:send() possibly returning an undefined value | ||
181 | |||
182 | CHANGE 110 Stepets 20-Apr-14 | ||
183 | * fix LuaJIT detection issues | ||
184 | |||
185 | CHANGE 109 BGe 03-Apr-14 | ||
186 | * moved some Lua-version compatibility code in separate source files | ||
187 | |||
188 | CHANGE 108: BGe 20-Mar-14 | ||
189 | * bumped version to 3.9.4 | ||
190 | * set_finalizer throws an error if provided finalizer isn't a function | ||
191 | * fix error handling when the error doesn't generate an error handler call (IOW, all errors but LUA_ERRRUN) | ||
192 | * provide callstack if LUA_ERRRUN occurs inside a finalizer | ||
193 | |||
194 | CHANGE 107: BGe 19-Mar-14 | ||
195 | * Make sure we don't mutex-wrap require() more than once, just in case | ||
196 | |||
197 | CHANGE 106: BGe 17-Mar-14 | ||
198 | * Fixed crash when using protect_allocator option | ||
199 | |||
200 | CHANGE 105: BGe 27-Feb-14 | ||
201 | * Bumped version to 3.9.3 | ||
202 | * new exposed variable linda.null that exposes the internal NIL_SENTINEL marker | ||
203 | * linda:send() interprets send key linda.null as authorization to silently send a single nil when not provided with anything to send | ||
204 | (useful when sending results of a function that can return nothing) | ||
205 | |||
206 | CHANGE 104: BGe 25-Feb-14 | ||
207 | * Bumped version to 3.9.2 | ||
208 | * Internal rework: the whole Lanes engine now works "per universe" to allow concurrent Lanes execution in more than one embedded master state | ||
209 | * this universe is a full userdata created in the master state, selfdestruct_gc is the __gc for this userdata | ||
210 | * most of what was initialized only once is now per-universe | ||
211 | * Fixed potential crashes at desinit if problems occur during keeper states initialisation | ||
212 | * Fixed require() not always serialized properly | ||
213 | * Raise an error instead of crashing on deep userdata prelude memory allocation failure | ||
214 | * Added forgotten mutex desinitialisation at universe shutdown | ||
215 | |||
216 | CHANGE 103: BGe 24-Feb-14 | ||
217 | * Fix lookup database table not being created when it should if Lanes is required in more than one Lua master state | ||
218 | |||
219 | CHANGE 102: BGe 18-Feb-14 | ||
220 | * raise an error instead of dereferencing a NULL pointer on deep userdata creation and lane struct creation | ||
221 | |||
222 | CHANGE 101: BGe 18-Feb-14 | ||
223 | * version 3.9.1 | ||
224 | * removed some keeper desinit legacy dead code | ||
225 | * keeper array is allocated with master state's alloc function instead of malloc()/free() | ||
226 | * prevent application crash when specifying a very large number of keepers in the configuration options | ||
227 | * any error occuring during one-time inits is raised outside the one-time mutex protected code region | ||
228 | |||
229 | CHANGE 100: BGe 17-Feb-14 | ||
230 | * lanes.linda() accepts an optional integer group to give control on keeper state repartition | ||
231 | |||
232 | CHANGE 99: BGe 17-Feb-14 | ||
233 | * version 3.9.0 | ||
234 | * keepers now require "package", receive package.path & package.cpath, and call on_state_create() if it is a C function | ||
235 | * changed the deep public API (improved deep idfunc signature, renamed luaG_deep_userdata to luaG_newdeepuserdata) | ||
236 | * if an error occurs while copying a deep userdata, don't raise inside the keeper state | ||
237 | * fixed situations where raised errors could lead to memory leaks (deep gc) | ||
238 | |||
239 | CHANGE 98: BGe 13-Feb-14 | ||
240 | * version 3.8.5 | ||
241 | * linda:limit() returns lanes.cancel_error on a limited linda | ||
242 | * lanes.genlock() and lanes.genatomic() support cancelled lindas by returning lanes.cancel_error whenever appropriate | ||
243 | * fixed a possible Lua stack overflow when calling linda:dump() | ||
244 | * fixed cases where linda:send() and linda:receive() would not return lanes.cancel_error when they should | ||
245 | |||
246 | CHANGE 97: BGe 10-Feb-14 | ||
247 | * version 3.8.4 | ||
248 | * new API linda:cancel("read"|"write"|"both"|"none") | ||
249 | * all linda operations return lanes.cancel_error on a cancelled linda | ||
250 | * raised an internal string length so that longer linda names are fully output before truncation applies when doing tostring( linda) | ||
251 | |||
252 | CHANGE 96: BGe 24-Jan-14 | ||
253 | * another Lua stack overflow fix when sending complex function through lindas or as lane body | ||
254 | |||
255 | CHANGE 95: BGe 22-Jan-14 | ||
256 | * version 3.8.3 | ||
257 | * fixed a possible Lua stack overflow when sending complex function through lindas or as lane body | ||
258 | * experimental: lanes.nameof() scans the registry if a regular search didn't yield anything interesting | ||
259 | * fixed lanes.nameof() misbehaving when encountering a LUA_TTHREAD object | ||
260 | |||
261 | CHANGE 94: BGe 22-Jan-14 | ||
262 | * version 3.8.2 | ||
263 | * new lane launcher option gc_cb to set a callback that is invoked when a lane is garbage collected | ||
264 | * Fix more invalid memory accesses when fetching the name of a joined lane with lanes:threads() (because its lua_State is closed) | ||
265 | |||
266 | CHANGE 93: BGe 20-Jan-14 | ||
267 | * slightly improve linda performance when the producer/consumer scenario leaves leave the key empty | ||
268 | |||
269 | CHANGE 92: BGe 20-Jan-14 | ||
270 | * version 3.8.1 | ||
271 | * new function lane:get_debug_threadname() | ||
272 | * Fix invalid memory accesses when fetching the name of a joined lane with lanes:threads() (because its lua_State is closed) | ||
273 | * use luaL_newmetatable() to create the metatable for lane objects | ||
274 | * prevent malicious code from crashing by calling lane methods without passing the lane as first argument (raise an error instead) | ||
275 | * 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 | ||
276 | |||
277 | CHANGE 91: BGe 20-Jan-14 | ||
278 | * version 3.8.0 | ||
279 | * linda:set() accepts multiple values to set in the specified slot | ||
280 | * linda:get() accepts an optional count to peek several values at once | ||
281 | |||
282 | CHANGE 90: BGe 16-Jan-14 | ||
283 | * version 3.7.8 | ||
284 | * lane:cancel() now accepts a boolean second argument when soft cancelling (negative timeout) to wake the thread if necessary | ||
285 | * if a blocked linda send() or receive() call is interrupted by a cancellation request, | ||
286 | it returns CANCEL_ERROR so that this case can be differentiated from a simple timeout | ||
287 | * fixed WIN32 THREAD_CREATE() wrong _beginthreadex() error detection | ||
288 | * fatal WIN32 threading errors retrieve and output the error description string with FormatMessage() | ||
289 | * fixed missing lanes.set_singlethreaded | ||
290 | * fixed perftest.lua | ||
291 | * added test/cancel.lua | ||
292 | |||
293 | CHANGE 89: BGe 09-Jan-14 | ||
294 | * version 3.7.7 | ||
295 | * fix crash when calling linda:count() on unknown keys | ||
296 | * purge key storage with linda:set( key, nil) on an unlimited key to reduce memory usage with lots of keys | ||
297 | * linda:limit() wakes write-blocked threads if necessary when the new limit enables writes to occur again | ||
298 | * linda:set() wakes write-blocked threads if necessary if the operation created some room to write into | ||
299 | |||
300 | CHANGE 88: BGe 06-Jan-14 | ||
301 | * version 3.7.6 | ||
302 | * if config.on_state_create() is a C function, call it by direct C closure reconstruction in newly created states | ||
303 | |||
304 | CHANGE 87: BGe 20-Dec-13 | ||
305 | * version 3.7.5 | ||
306 | * 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 | ||
307 | * new setting demote_full_userdata to select between light userdata demotion or raising an error when attempting to transfer a non-deep full userdata | ||
308 | |||
309 | CHANGE 86: BGe 3-Dec-13 | ||
310 | * version 3.7.4 | ||
311 | * internal refactoring of pthread priority management code | ||
312 | * new API lanes.set_thread_priority() | ||
313 | |||
314 | CHANGE 85: BGe 28-Nov-13 | ||
315 | * version 3.7.3 | ||
316 | * set pthread thread cancel type to PTHREAD_CANCEL_ASYNCHRONOUS | ||
317 | * lane_h:cancel() accepts a 3rd timeout argument used when waiting for actual thread termination (hitting the timeout raises an error) | ||
318 | * added PROPAGATE_ALLOCF macro to select state creation mode (lua_newstate or luaL_newstate) | ||
319 | |||
320 | CHANGE 84: BGe 18-Nov-13 | ||
321 | * Fix a deadlock when GCing during a linda operation. | ||
322 | * Fix a compilation warning about an unused variable | ||
323 | * Get rid of uintptr_t to remove dependency on stdint.h | ||
324 | * Fix internal error at lane creation when the generator doesn't open any base library | ||
325 | |||
326 | CHANGE 83: BGe 16-Nov-13 | ||
327 | * version 3.7.2 | ||
328 | * Fixed function returned by lanes.genlock() not handling numeric keys properly when release lock | ||
329 | * Enable lanes.genlock() to attempt lock with an optional "try" mode | ||
330 | * make EnableCrashingOnCrashes a one-time operation | ||
331 | |||
332 | CHANGE 82: BGe 13-Nov-13 | ||
333 | * Fix a case where an error could be raised inside a keeper state | ||
334 | |||
335 | CHANGE 81: BGe 07-Nov-13 | ||
336 | * Make set_finalizer(), set_debug_threadname(), cancel_test() and set_error_reporting() transferable from lane to lane | ||
337 | * Improved some DEBUGSPEW output | ||
338 | |||
339 | CHANGE 80: BGe 06-Nov-13 | ||
340 | * Fix a few compilation warnings about uninitialized variables | ||
341 | * Fixed a bad extern variable declaration resulting in multiple instances (crashes the Pelles-C build) | ||
342 | |||
343 | CHANGE 79: BGe 04-Nov-13 | ||
344 | * Fix lanes.nameof() crashing when encountering a light userdata | ||
345 | |||
346 | CHANGE 78: BGe 25-Oct-13 | ||
347 | * Fix windows build not exporting public 'deep' API | ||
348 | * Don't call on_state_create in keeper states, as it is no longer necessary | ||
349 | * Remove inclusion of stdint.h | ||
350 | * Fix windows build for WINVER > 0x400 | ||
351 | |||
352 | CHANGE 77: BGe 22-Oct-13 | ||
353 | * version 3.7.1 | ||
354 | * errors inside finalizers generate a full stack just like any other error | ||
355 | |||
356 | CHANGE 76: BGe 10-Oct-13 | ||
357 | * version 3.7.0 | ||
358 | * fix lanes.threads() not being available in a lane where lanes.configure() settings didn't contain track_lanes although the initial configure() call did. | ||
359 | |||
360 | CHANGE 75: BGe 7-Oct-13 | ||
361 | * require "lanes".configure() sequence is only necessary at the first require "lanes". | ||
362 | |||
363 | CHANGE 74: BGe 7-Oct-13 | ||
364 | * fix a crash at application shutdown where in some situations we could deinitialize the protected allocator mutex while a lane was still using it. | ||
365 | |||
366 | CHANGE 73: BGe 4-Oct-13 | ||
367 | * fix timers broken by change 69 | ||
368 | |||
369 | CHANGE 72: BGe 3-Oct-13 | ||
370 | * bugfix: no longer create a global named "lanes.core" inside lanes having "*" as library list | ||
371 | |||
372 | CHANGE 71: BGe 30-Sept-13 | ||
373 | * version 3.6.6 | ||
374 | * properly handle cases when a Lua C module is a C function | ||
375 | |||
376 | CHANGE 70: BGe 27-Step-13 | ||
377 | * no longer call core.configure with dummy params when requiring lanes more than once (fixes potential multithreading issues with LuaJIT allocator) | ||
378 | * activated EnableCrashingOnCrashes() is active on Win32 debug builds | ||
379 | * fixed some comments in code | ||
380 | |||
381 | CHANGE 69: BGe 26-Sept-13 | ||
382 | * version 3.6.5 | ||
383 | * Reduce memory footprint, simplify module order setup in conjuction with Lanes, and send over native functions a bit faster as well | ||
384 | * Lanes no longer has to internally require modules inside the keeper states because they no longer need a lookup database | ||
385 | the lookup name is stored as-is and actually converted in the destination state | ||
386 | * optimisation: bypass cache when sending native functions over | ||
387 | * removed all the KEEPER_MODEL_LUA code, as it can no longer work anyway | ||
388 | |||
389 | CHANGE 68: BGe 24-Sept-13 | ||
390 | * version 3.6.4 | ||
391 | * Fix possible application hang at shutdown if a keeper state referenced a linda. | ||
392 | |||
393 | CHANGE 67: BGe 2-Aug-13 | ||
394 | * version 3.6.3 | ||
395 | * lane:cancel(<negative-timeout>) only causes cancel_test() to return true but won't interrupt execution of the lane during linda operations | ||
396 | |||
397 | CHANGE 66: BGe 31-Jul-13 | ||
398 | * more explicit errors when trying to transfer unknown source functions (with new configure option verbose_errors) | ||
399 | |||
400 | CHANGE 65: BGe 23-Jul-13 | ||
401 | * default options wrap allocator around a mutex when run by LuaJIT | ||
402 | |||
403 | CHANGE 64: BGe 20-Jul-13 | ||
404 | * WIN32 builds against pre-Vista versions no longer use PulseEvent to fix occasional hangs when a wake event is missed | ||
405 | |||
406 | CHANGE 63: BGe 20-May-13 | ||
407 | * version 3.6.2 | ||
408 | * WIN32 builds use condition variables instead of PulseEvent() when available. | ||
409 | * first steps toward fixing make-vc.cmd | ||
410 | |||
411 | CHANGE 62: BGe 05-Apr-13 | ||
412 | * version 3.6.1 | ||
413 | * 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 | ||
414 | |||
415 | CHANGE 61: BGe 14-Mar-13 | ||
416 | * version 3.6.0 | ||
417 | * protect_allocator is an API change -> version bump | ||
418 | * bugfix: allocator protection should be done once per primary Lua state, not once only the first time ever Lanes is required | ||
419 | |||
420 | CHANGE 60: BGe 13-Mar-13 | ||
421 | * version 3.5.2 | ||
422 | * stricter validation of with_timers config option: validator was accepting any non-boolean value | ||
423 | * new configuration option protect_allocator for VMs with thread unsafe allocators (such as LuaJIT) | ||
424 | * removed some obsolete bits of dead code | ||
425 | |||
426 | CHANGE 59: BGe 12-Feb-13 | ||
427 | * version 3.5.1 | ||
428 | * new lanes.h header and API call luaopen_lanes_embedded() for embedders | ||
429 | * "lanes.core" is an acceptable library in the generator libs argument | ||
430 | * library "*" wildcard also opens lanes.core | ||
431 | * tweaked code for Xbox 360 build | ||
432 | |||
433 | CHANGE 58: BGe 30-Jan-13 | ||
434 | * version 3.5.0 | ||
435 | * new: API lanes.require(), use it instead of regular require() for modules that export C functions you need to send over. | ||
436 | * new: lanes no longer require 'lanes.core' by default in every created state. Use {required={"lanes.core"}} if you need to transfer lanes functions. | ||
437 | * internal: because of the above, reworked the timer implementation to remove upvalue-dependency on lanes.core | ||
438 | * new: API lanes.timer_lane, to be able to operate on timer lane if need be | ||
439 | * improved: if a module is a full userdata, scan its metatable for function database population | ||
440 | * improved: on_state_create can be a Lua function | ||
441 | * changed: on_state_create is called after the base libraries are loaded | ||
442 | * package[loaders|searchers] is no longer transfered as function naming depends on slot order | ||
443 | * internal: changed separator from '.' to '/' in lookup databases to be able to distinguish search levels and dot coming from module names | ||
444 | * added some mode debug spew | ||
445 | * updated tests to reflect the above changes | ||
446 | |||
447 | CHANGE 57: BGe 28-Jan-13 | ||
448 | * More detailed DEBUG_SPEW logs | ||
449 | * A bit of code cosmetics | ||
450 | |||
451 | CHANGE 56: BGe 25-Jan-13 | ||
452 | * version 3.4.4 | ||
453 | * bugfix: take into account the fact that "coroutine" is no longer part of base library in Lua 5.2 | ||
454 | * bugfix: if "bit32" was listed in the libraries, it wouldn't open (library list parsing failing on digits) | ||
455 | * bugfix: Use luaL_requiref() to open standard libraries in Lua 5.2 as we should | ||
456 | * bugfix: any Lua state created by Lanes reuses the allocator function of the originating state | ||
457 | * bugfix: don't call on_state_create() while GC is suspended during lua state initialization | ||
458 | |||
459 | CHANGE 55: BGe 24-Jan-13 | ||
460 | * version 3.4.3 | ||
461 | * raise an error if lane generator libs specification contains a lib more than once | ||
462 | * bit32 is a valid lib name in the libs specification (silently ignored by the Lua 5.1 build) | ||
463 | * improved lanes.nameof to search inside table- and userdata- metatables for an object's name | ||
464 | * bugfix: fixed an unwarranted error when trying to discover a function name upon a failed transfer | ||
465 | * contents of package.[path,cpath,preload,loaders|searchers] are pulled *only once* inside keeper states at initialisation | ||
466 | * Lua function upvalues equal to the global environment aren't copied by value, but bound to the destination's global environment | ||
467 | especially useful for Lua 5.2 _ENV | ||
468 | * bugfix: fixed loading of base libraries that didn't create the global tables when built for Lua 5.2 | ||
469 | |||
470 | CHANGE 54: BGe 10-Jan-13 | ||
471 | * version 3.4.2 | ||
472 | * Don't pull "package" settings in the timer lane | ||
473 | * removed a limitation preventing Lua functions with indirect recursive upvalue references from being transferable | ||
474 | |||
475 | CHANGE 53: BGe 11-Dec-2012 | ||
476 | * version 3.4.1 | ||
477 | * new function lanes.timers(), returns a list of all active timers. | ||
478 | |||
479 | CHANGE 52: BGe 03-Dec-2012 | ||
480 | * linda:send() and linda:receive() no longer triggers string->number autocoercion when checking for the optional timeout argument: | ||
481 | a string is always a linda slot, even if coercible. | ||
482 | |||
483 | CHANGE 51: BGe 27-Nov-2012 | ||
484 | * linux flavors with older glibc use prctl instead of pthread_setname_np | ||
485 | * selfdestruct chain handling is now the same on all platforms | ||
486 | |||
487 | CHANGE 50: BGe 22-Nov-2012 | ||
488 | * bugfix: linda:set() no longer clears the storage limit | ||
489 | |||
490 | CHANGE 49: BGe 21-Nov-2012 | ||
491 | * fix application shutdown crash by not registering anything with atexit() | ||
492 | * rockspec for version v3.4.0 | ||
493 | |||
494 | CHANGE 48: BGe 25-Sep-2012 | ||
495 | * version 3.4.0 | ||
496 | * new method linda:dump() that outputs the full contents of a linda as a table, also linked to __towatch for Decoda support | ||
497 | * linda:receive() API change! | ||
498 | * instead of [val, key], linda:receive( timeout, key) returns [key, val] | ||
499 | * instead of [val, [...]], linda:receive( timeout, linda.batched key) returns [key, val[, ...]] | ||
500 | 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 | ||
501 | * fixed Lua 5.2 build to take into account the "loaders"->"searchers" name change in 'package' module. | ||
502 | * a bit of html cleanup and added some infos in the documentation regarding the Lanes internals | ||
503 | |||
504 | CHANGE 47: BGe 13-Sep-2012 | ||
505 | * implemented set_debug_threadname() for pthread builds where possible | ||
506 | * refactored linda __tostring and __concat | ||
507 | |||
508 | CHANGE 46: BGe 10-Sep-2012 | ||
509 | * version 3.3.0 | ||
510 | * lane.status can return "killed" if lane was forcefully killed with lanes:cancel() | ||
511 | * lane:join(): return nil, "killed" if called on a killed lane. | ||
512 | * lane[<n>]: produces [1] = nil, [2] = "killed" if the lane was killed | ||
513 | * lane:join(): fixed an assertion in debug builds when joining a lane forcefully cancelled with lane:cancel( <x>, true). | ||
514 | * indexing a lane with a string other than "join", "cancel" or "status" raises an error. | ||
515 | * fixed configure() to correctly apply defaults when they are missing from the provided settings | ||
516 | * added a shutdown_timeout to control the duration Lanes will wait for graceful termination of running lanes at application shutdown. Default is 0.25. | ||
517 | |||
518 | CHANGE 45: BGe 21-Aug-2012 | ||
519 | * keeper internals implemented in C instead of Lua for better performances | ||
520 | * fixed arguments checks in linda:limit() and linda:set() | ||
521 | |||
522 | CHANGE 44: BGe 13-Aug-2012 | ||
523 | * lanes code updated to build against Lua 5.1 and Lua 5.2 | ||
524 | * removed the search for MSVCR80.DLL when building for MinGW32 since it no longer seems to be necessary | ||
525 | |||
526 | CHANGE 43: BGe 09-Aug-2012 | ||
527 | * fix possible crash at application shutdown when a race condition causes linda objects to be collected after the keeper states are cleaned up. | ||
528 | |||
529 | CHANGE 42: BGe 06-Aug-2012 | ||
530 | * lanes.linda() accepts an optional name for debug purposes | ||
531 | |||
532 | CHANGE 41: BGe 07-Jul-2012 | ||
533 | * lua51-lanes renamed lanes/core | ||
534 | * keeper state microcode is no longer embedded inside lanes.core, but located and loaded with package.loaders[2] | ||
535 | * changed rockspec build type from "make" to "builtin" | ||
536 | |||
537 | CHANGE 40: BGe 26-Jun-2012 | ||
538 | * when a transfered function is not found in source, guess its name to help the user find out what's wrong | ||
539 | * new function lanes.nameof() | ||
540 | |||
541 | CHANGE 39: BGe 23-Jun-2012 | ||
542 | * lanes.timer() accepts a first_secs=nil to stop a timer | ||
543 | * timer lane catches errors and prints them | ||
544 | * fixed some typos in manual | ||
545 | |||
546 | CHANGE 38: BGe 11-Jun-2012 | ||
547 | * linda:receive() batched mode now accepts a max_count optional argument | ||
548 | |||
549 | CHANGE 37: BGe 4-Jun-2012 (fix and idea courtesy of sonoro1234) | ||
550 | * fixed thread_cancel() not working when called without argument | ||
551 | * new lane-global function set_error_reporting() to enable more data detailed data provided by lane_error() | ||
552 | |||
553 | CHANGE 36 BGe 26-Apr-2012 | ||
554 | * improved LuaJIT2 compatibility by handling "*" library set through luaL_openlibs() | ||
555 | |||
556 | CHANGE 35 BGe 17-Feb-2012 | ||
557 | * changed lanes.configure signature to receive a table instead of individual parameters | ||
558 | * added support for an on_state_create callback called to load custom functions in a state in addition to the base libraries | ||
559 | |||
560 | CHANGE 34 BGe 14-Nov-2011 | ||
561 | * removed packagepath and packagecpath options, replaced by a package table, whose fields path, cpath, loaders, preload are transfered | ||
562 | * code cleanup to facilitate transition between WIN32 and PTHREAD implementations | ||
563 | * tentative fix for desinit crashes when free running lanes are killed at process shutdown | ||
564 | |||
565 | CHANGE 33 BGe 5-Nov-2011: Lanes version 3.0-beta | ||
566 | * process exit change: close everything at GC when main state closes, not when atexit() handlers are processed | ||
567 | * Lua 5.2-style module: | ||
568 | * module() is no longer used to implement lanes.lua | ||
569 | * a global "lanes" variable is no longer created when the module is required | ||
570 | * the Lanes module table is returned instead | ||
571 | * Lanes must be initialized before used: | ||
572 | * the first occurence of 'require "lanes"' produces a minimal interface that only contains a configure() function | ||
573 | * the remainder of the interface is made available once this function is called | ||
574 | * subsequent calls to configure() do nothing | ||
575 | * configure() controls the number of keeper states and the startup of timers | ||
576 | * LuaJIT 2 compatibility | ||
577 | * non-Lua functions are no longer copied by creating a C closure from a C pointer, but through 2-way lookup tables | ||
578 | * 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 | ||
579 | * introduces a change in configuration .globals management: contents are copied *after* std libs are loaded | ||
580 | * new .required configuration entry to list modules that must be require()'ed before lane body is transferred | ||
581 | * lane:cancel() wakes up waiting lindas like what is done at lane shutdown | ||
582 | |||
583 | CHANGE 32 BGe 14-May-2011 | ||
584 | * raise an error when linda:send() has nothing to send | ||
585 | |||
586 | CHANGE 31 BGe 17-Apr-2011 | ||
587 | * linda uses a fast FIFO implementation to speed up data exchanges | ||
588 | * new linda:count() method | ||
589 | * new linda batched data read mode | ||
590 | * proper key type check in all linda methods | ||
591 | * fix setup-vc.cmd to support Visual Studio 2010 and Windows 7 64 bits | ||
592 | * bugfix: release keeper state mutex at desinit | ||
593 | |||
594 | CHANGE 30 BGe 30-Mar-2011 | ||
595 | * linda honors __tostring and __concat | ||
596 | * new accessor linda:keys(), to retrieve the list of keys with pending data inside a linda | ||
597 | * new lanes options packagepath and packagecpath, in case one needs to set them differently than the default | ||
598 | |||
599 | CHANGE 29 BGe 1-Mar-2011 | ||
600 | fixed potential crash at application shutdown when calling lua_close() on a killed thread's VM. | ||
601 | exposed cancel_test() in the lanes to enable manual testing for cancellation requests. | ||
602 | removed kludgy {globals={threadName}} support, replaced with a new function set_debug_threadname(). | ||
603 | |||
604 | CHANGE 28 BGe 18-Feb-2011 | ||
605 | - moved keeper-related code in a separate source file | ||
606 | - keeper.lua is now embedded in text form instead of bytecode to improve LuaJIT2-compatibility | ||
607 | |||
608 | CHANGE 27 BGe 17-Feb-2011 | ||
609 | - we know Lanes is loaded in the master state, so we don't force it | ||
610 | to be required in every lane too when a linda deep userdata is copied | ||
611 | - Refactor lane proxy implementation: it is now a full userdata instead | ||
612 | of a table, and its methods are implemented in C instead of Lua | ||
613 | * its metatable is no longer accessible | ||
614 | * writing to the proxy raises an error | ||
615 | * it is no longer possible to overwrite its join() and cancel() methods | ||
616 | - when a deep userdata idfunc requests a module to be required, manually | ||
617 | check that it is not loaded before requiring it instead of relying on | ||
618 | the require function's loop detection feature | ||
619 | - when a module must be required, raise an error if the 'require' function | ||
620 | is not found in the target state | ||
621 | |||
622 | CHANGE 26 BGe 14-Feb-2011: | ||
623 | Fixed application hang-up because keeper state was not released in case of errors thrown by | ||
624 | inter-state data copy for unsupported types | ||
625 | |||
626 | CHANGE 25 BGe 12-Feb-2011: | ||
627 | Changed idfunc signature and contract to clarify that fact it is not lua-callable | ||
628 | and to be able to require the module it was exported from in the target lanes | ||
629 | |||
630 | CHANGE 24 DPtr 25-Jan-2011: | ||
631 | Changed lanes.c to export functions as a module rather than writing them directly to the globals table. | ||
632 | |||
633 | CHANGE 23 DPtr 23-Jan-2011: | ||
634 | Fixed bug where reference to Linda object was dropped for a short time ( crashing if GC was run during that time ). | ||
635 | Changed the atexit code to trip the timer thread's write signal. | ||
636 | |||
637 | CHANGE 22 DPtr 19-Jan-2011: | ||
638 | Changed luaG_push_proxy to cache deep userdata proxies. | ||
639 | |||
640 | CHANGE 21 (bugfixes) BGe 3-Jan-2011: | ||
641 | Several fixes by Martin Krpan: | ||
642 | - linda_send was waiting on the wrong signal | ||
643 | - buildfix when using i586-mingw32msvc-gcc cross compiler | ||
644 | - lanes_h:cancel() returns a boolean as it should | ||
645 | - timers could get blocked sometimes because they were initialized with negative values | ||
646 | - prepare_timeout could generate an illegal setting | ||
647 | |||
648 | CHANGE 20 BGe 3-Dec-2010: | ||
649 | Enable to specify a string as lane code instead of a function so that we dont use lua_dump, which | ||
650 | isn't supported by LuaJIT. | ||
651 | |||
652 | CHANGE 19 BGe 2-Dec-2010: | ||
653 | 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. | ||
654 | |||
655 | CHANGE 18 BGe 6-Oct-2010: | ||
656 | Fixed 'memory leak' in some situations where a free running lane is collected before application shutdown | ||
657 | A bit of code cleanup | ||
658 | |||
659 | CHANGE 17 BGe 21-Sept-2010: | ||
660 | Fixed stupid compilation errors. | ||
661 | |||
662 | CHANGE 16 PLM 24-Aug-2010: | ||
663 | Releasing memory at gc / atexit. | ||
664 | Finalizers actually get error strings. | ||
665 | Fixed missing argument propagation in lane:cancel | ||
666 | Added variable threadName sent trough globals-table. Set in s_lane, and in debuggers on windows. | ||
667 | Added argument checking for linda-objects, where missing them caused crashes. | ||
668 | |||
669 | CHANGE 15 (minor) BGe 27-Jul-2010: | ||
670 | Version bump for a true upgrade release (2.0.4 package was only a renamed 2.0.3) | ||
671 | |||
672 | CHANGE 14 (bug fix) BGe 09-Jul-2010: | ||
673 | Fixed lane status to be correctly returned as "waiting" when it should. | ||
674 | |||
675 | CHANGE 13 (fix for multithreaded host apps) AKa 24-Jun-2009: | ||
676 | <borisusun-at-gmail> mentioned Lanes expects the host application to be singlethreaded, | ||
677 | and there are troubles if Lanes is used from multiple threads, opened by the host | ||
678 | (before requiring Lanes). This is true, and fix should now be in place. | ||
679 | |||
680 | CHANGE 12 (bug fix on Windows, 2.0.3) AKa 25-Jan-2009: | ||
681 | Did CHANGE 9 the way it should be done. | ||
682 | |||
683 | CHANGE 11 (new feature, 2.0.3) AKa 23-Jan-2009: | ||
684 | Finalizers ('set_finalizer()') for being able to do cleanup of a lane's | ||
685 | resources, whether it returned succesfully or via an error. | ||
686 | |||
687 | CHANGE 10 (new feature, 2.0.3) AKa 23-Jan-2009: | ||
688 | Call stack showing where an error occurred is not merged with the error | ||
689 | message, but delivered as a separate stack table ({ "filename:line" [, ...] }). | ||
690 | Getting call stacks of errorred lanes is now possible. | ||
691 | |||
692 | CHANGE 9 (bug fix on Windows) AKa 10-Dec-2008 (> 2.0.2): | ||
693 | Applied patch from Kriss Daniels to avoid issues on 'now_time()' in Win32 | ||
694 | (http://luaforge.net/forum/forum.php?thread_id=22704&forum_id=1781). | ||
695 | |||
696 | CHANGE 8 (bug fix) AKa 26-Oct-2008: | ||
697 | Avoids occasional segfault at process exit (on multicore CPUs). Does this | ||
698 | by keeping track of "free running" threads (s.a. the time thread) and | ||
699 | cancelling them at process exit. | ||
700 | |||
701 | Tested (2.0.2) on Linux 64,x86, OS X, WinXP. | ||
702 | |||
703 | CHANGE 7 (bug fix) AKa 15-Oct-2008: | ||
704 | Recursive functions that use themselves as direct upvalue can now be | ||
705 | passed to other lanes, and used as a lane function. | ||
706 | |||
707 | CHANGE 6 (bug fix) AKa 15-Oct-2008: | ||
708 | Added local caches of the following to src/lanes.lua (was otherwise getting | ||
709 | errors at least in 'tests/irayo_recursive.lua'). | ||
710 | |||
711 | local assert= assert | ||
712 | local string_gmatch= assert( string.gmatch ) | ||
713 | local select= assert( select ) | ||
714 | local type= assert( type ) | ||
715 | local pairs= assert( pairs ) | ||
716 | local tostring= assert( tostring ) | ||
717 | local error= assert( error ) | ||
718 | local setmetatable= assert( setmetatable ) | ||
719 | local rawget= assert( rawget ) | ||
720 | |||
721 | Thanks to Irayo for detecting and reporting this. | ||
722 | |||
723 | CHANGE 5 (new feature): | ||
724 | Modifying Makefile so it's better suited to LuaRocks. | ||
725 | |||
726 | CHANGE 4 (new feature): | ||
727 | Metatable copying, allowing Lua objects to be copied across lanes. | ||
728 | |||
729 | CHANGE 3 (bug fix) AKa 5-Aug-2008: | ||
730 | The '__gc' method was not tied to thread userdata, at all. Caused memory | ||
731 | lifespan problems at least on OS X when threads were cancelled (EINVAL). | ||
732 | |||
733 | CHANGE 2 (bug fix) AKa 5-Aug-2008: | ||
734 | Better calculation of timeouts, always making them absolute (even in Win32) | ||
735 | to allow for events that wake the lane up but don't read/write the Linda | ||
736 | key that it was observing. | ||
737 | |||
738 | CHANGE 1 (bug fix) AKa 4-Aug-2008: | ||
739 | Signalling woke up only one waiting thread, not all. This caused i.e. | ||
740 | receive to not wake up if there was another thread waiting on the same | ||
741 | Linda object. | ||
742 | |||
743 | PThread fix: using 'pthread_cond_broadcast()' instead of 'pthread_cond_signal()' | ||
744 | Win32 fix: using manual events and 'PulseEvent()' | ||
745 | 5 | ||
746 | (end) | 6 | (end) |
diff --git a/lanes-3.16.3-0.rockspec b/lanes-4.0.0-0.rockspec index 106a2d9..4e1b370 100644 --- a/lanes-3.16.3-0.rockspec +++ b/lanes-4.0.0-0.rockspec | |||
@@ -7,11 +7,11 @@ | |||
7 | 7 | ||
8 | package = "Lanes" | 8 | package = "Lanes" |
9 | 9 | ||
10 | version = "3.16.3-0" | 10 | version = "4.0.0-0" |
11 | 11 | ||
12 | source= { | 12 | source= { |
13 | url= "git+https://github.com/LuaLanes/lanes.git", | 13 | url= "git+https://github.com/LuaLanes/lanes.git", |
14 | branch= "v3.16.3" | 14 | branch= "v4.0.0" |
15 | } | 15 | } |
16 | 16 | ||
17 | description = { | 17 | description = { |
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 @@ | |||
1 | /* | 1 | /* |
2 | -- | 2 | -- |
3 | -- CANCEL.C | 3 | -- CANCEL.CPP |
4 | -- | 4 | -- |
5 | -- Lane cancellation support | 5 | -- Lane cancellation support |
6 | -- | 6 | -- |
@@ -9,7 +9,7 @@ | |||
9 | --[[ | 9 | --[[ |
10 | =============================================================================== | 10 | =============================================================================== |
11 | 11 | ||
12 | Copyright (C) 2011-2019 Benoit Germain <bnt.germain@gmail.com> | 12 | Copyright (C) 2011-2024 Benoit Germain <bnt.germain@gmail.com> |
13 | 13 | ||
14 | Permission is hereby granted, free of charge, to any person obtaining a copy | 14 | Permission is hereby granted, free of charge, to any person obtaining a copy |
15 | of this software and associated documentation files (the "Software"), to deal | 15 | of this software and associated documentation files (the "Software"), to deal |
@@ -35,9 +35,6 @@ THE SOFTWARE. | |||
35 | 35 | ||
36 | #include "cancel.h" | 36 | #include "cancel.h" |
37 | 37 | ||
38 | // #include <assert.h> | ||
39 | //#include <string.h> | ||
40 | |||
41 | #include "lanes_private.h" | 38 | #include "lanes_private.h" |
42 | #include "threading.h" | 39 | #include "threading.h" |
43 | #include "tools.h" | 40 | #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 @@ | |||
1 | /* | 1 | /* |
2 | * ############################################################################################### | 2 | * ############################################################################################### |
3 | * ######################################### Lua 5.1/5.2 ######################################### | 3 | * ####################################### Lua 5.1/5.2/5.3 ####################################### |
4 | * ############################################################################################### | 4 | * ############################################################################################### |
5 | */ | 5 | */ |
6 | #include "compat.h" | 6 | #include "compat.h" |
@@ -9,7 +9,12 @@ | |||
9 | /* | 9 | /* |
10 | ** Copied from Lua 5.2 loadlib.c | 10 | ** Copied from Lua 5.2 loadlib.c |
11 | */ | 11 | */ |
12 | // ################################################################################################ | ||
13 | // ################################################################################################ | ||
12 | #if LUA_VERSION_NUM == 501 | 14 | #if LUA_VERSION_NUM == 501 |
15 | // ################################################################################################ | ||
16 | // ################################################################################################ | ||
17 | |||
13 | static int luaL_getsubtable (lua_State *L, int idx, const char *fname) | 18 | static int luaL_getsubtable (lua_State *L, int idx, const char *fname) |
14 | { | 19 | { |
15 | lua_getfield(L, idx, fname); | 20 | lua_getfield(L, idx, fname); |
@@ -26,6 +31,8 @@ static int luaL_getsubtable (lua_State *L, int idx, const char *fname) | |||
26 | } | 31 | } |
27 | } | 32 | } |
28 | 33 | ||
34 | // ################################################################################################ | ||
35 | |||
29 | void luaL_requiref (lua_State *L, const char *modname, lua_CFunction openf, int glb) | 36 | void luaL_requiref (lua_State *L, const char *modname, lua_CFunction openf, int glb) |
30 | { | 37 | { |
31 | lua_pushcfunction(L, openf); | 38 | lua_pushcfunction(L, openf); |
@@ -43,24 +50,30 @@ void luaL_requiref (lua_State *L, const char *modname, lua_CFunction openf, int | |||
43 | } | 50 | } |
44 | #endif // LUA_VERSION_NUM | 51 | #endif // LUA_VERSION_NUM |
45 | 52 | ||
53 | // ################################################################################################ | ||
54 | // ################################################################################################ | ||
46 | #if LUA_VERSION_NUM < 504 | 55 | #if LUA_VERSION_NUM < 504 |
56 | // ################################################################################################ | ||
57 | // ################################################################################################ | ||
47 | 58 | ||
48 | void* lua_newuserdatauv( lua_State* L, size_t sz, int nuvalue) | 59 | void* lua_newuserdatauv( lua_State* L, size_t sz, int nuvalue) |
49 | { | 60 | { |
50 | ASSERT_L( nuvalue <= 1); | 61 | ASSERT_L( nuvalue <= 1); |
51 | return lua_newuserdata( L, sz); | 62 | return lua_newuserdata(L, sz); |
52 | } | 63 | } |
53 | 64 | ||
65 | // ################################################################################################ | ||
66 | |||
54 | // push on stack uservalue #n of full userdata at idx | 67 | // push on stack uservalue #n of full userdata at idx |
55 | int lua_getiuservalue(lua_State* L, int idx, int n) | 68 | int lua_getiuservalue(lua_State* L, int idx, int n) |
56 | { | 69 | { |
57 | // full userdata can have only 1 uservalue before 5.4 | 70 | // full userdata can have only 1 uservalue before 5.4 |
58 | if( n > 1) | 71 | if( n > 1) |
59 | { | 72 | { |
60 | lua_pushnil( L); | 73 | lua_pushnil(L); |
61 | return LUA_TNONE; | 74 | return LUA_TNONE; |
62 | } | 75 | } |
63 | lua_getuservalue( L, idx); | 76 | lua_getuservalue(L, idx); |
64 | 77 | ||
65 | #if LUA_VERSION_NUM == 501 | 78 | #if LUA_VERSION_NUM == 501 |
66 | /* default environment is not a nil (see lua_getfenv) */ | 79 | /* default environment is not a nil (see lua_getfenv) */ |
@@ -68,30 +81,32 @@ int lua_getiuservalue(lua_State* L, int idx, int n) | |||
68 | if (lua_rawequal(L, -2, -1) || lua_rawequal(L, -2, LUA_GLOBALSINDEX)) | 81 | if (lua_rawequal(L, -2, -1) || lua_rawequal(L, -2, LUA_GLOBALSINDEX)) |
69 | { | 82 | { |
70 | lua_pop(L, 2); | 83 | lua_pop(L, 2); |
71 | lua_pushnil( L); | 84 | lua_pushnil(L); |
72 | 85 | ||
73 | return LUA_TNONE; | 86 | return LUA_TNONE; |
74 | } | 87 | } |
75 | lua_pop(L, 1); /* remove package */ | 88 | lua_pop(L, 1); /* remove package */ |
76 | #endif | 89 | #endif |
77 | 90 | ||
78 | return lua_type( L, -1); | 91 | return lua_type(L, -1); |
79 | } | 92 | } |
80 | 93 | ||
94 | // ################################################################################################ | ||
95 | |||
81 | // pop stack top, sets it a uservalue #n of full userdata at idx | 96 | // pop stack top, sets it a uservalue #n of full userdata at idx |
82 | int lua_setiuservalue( lua_State* L, int idx, int n) | 97 | int lua_setiuservalue(lua_State* L, int idx, int n) |
83 | { | 98 | { |
84 | if( n > 1 | 99 | if( n > 1 |
85 | #if LUA_VERSION_NUM == 501 | 100 | #if LUA_VERSION_NUM == 501 |
86 | || lua_type( L, -1) != LUA_TTABLE | 101 | || lua_type(L, -1) != LUA_TTABLE |
87 | #endif | 102 | #endif |
88 | ) | 103 | ) |
89 | { | 104 | { |
90 | lua_pop( L, 1); | 105 | lua_pop(L, 1); |
91 | return 0; | 106 | return 0; |
92 | } | 107 | } |
93 | 108 | ||
94 | (void) lua_setuservalue( L, idx); | 109 | std::ignore = lua_setuservalue(L, idx); |
95 | return 1; // I guess anything non-0 is ok | 110 | return 1; // I guess anything non-0 is ok |
96 | } | 111 | } |
97 | 112 | ||
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 @@ | |||
1 | /* | 1 | /* |
2 | * DEEP.C Copyright (c) 2017, Benoit Germain | 2 | * DEEP.CPP Copyright (c) 2024, Benoit Germain |
3 | * | 3 | * |
4 | * Deep userdata support, separate in its own source file to help integration | 4 | * Deep userdata support, separate in its own source file to help integration |
5 | * without enforcing a Lanes dependency | 5 | * without enforcing a Lanes dependency |
@@ -9,7 +9,7 @@ | |||
9 | =============================================================================== | 9 | =============================================================================== |
10 | 10 | ||
11 | Copyright (C) 2002-10 Asko Kauppi <akauppi@gmail.com> | 11 | Copyright (C) 2002-10 Asko Kauppi <akauppi@gmail.com> |
12 | 2011-17 Benoit Germain <bnt.germain@gmail.com> | 12 | 2011-24 Benoit Germain <bnt.germain@gmail.com> |
13 | 13 | ||
14 | Permission is hereby granted, free of charge, to any person obtaining a copy | 14 | Permission is hereby granted, free of charge, to any person obtaining a copy |
15 | of this software and associated documentation files (the "Software"), to deal | 15 | of this software and associated documentation files (the "Software"), to deal |
@@ -41,13 +41,6 @@ THE SOFTWARE. | |||
41 | 41 | ||
42 | #include <bit> | 42 | #include <bit> |
43 | #include <cassert> | 43 | #include <cassert> |
44 | #include <ctype.h> | ||
45 | #include <stdio.h> | ||
46 | #include <string.h> | ||
47 | #include <stdlib.h> | ||
48 | #if !defined(__APPLE__) | ||
49 | #include <malloc.h> | ||
50 | #endif | ||
51 | 44 | ||
52 | /*-- Metatable copying --*/ | 45 | /*-- Metatable copying --*/ |
53 | 46 | ||
@@ -73,7 +66,7 @@ static constexpr UniqueKey DEEP_PROXY_CACHE_KEY{ 0x05773d6fc26be106ull }; | |||
73 | * Sets up [-1]<->[-2] two-way lookups, and ensures the lookup table exists. | 66 | * Sets up [-1]<->[-2] two-way lookups, and ensures the lookup table exists. |
74 | * Pops the both values off the stack. | 67 | * Pops the both values off the stack. |
75 | */ | 68 | */ |
76 | static void set_deep_lookup( lua_State* L) | 69 | static void set_deep_lookup(lua_State* L) |
77 | { | 70 | { |
78 | STACK_GROW( L, 3); | 71 | STACK_GROW( L, 3); |
79 | STACK_CHECK_START_REL(L, 2); // a b | 72 | STACK_CHECK_START_REL(L, 2); // a b |
@@ -88,11 +81,13 @@ static void set_deep_lookup( lua_State* L) | |||
88 | STACK_CHECK( L, 0); | 81 | STACK_CHECK( L, 0); |
89 | } | 82 | } |
90 | 83 | ||
84 | // ################################################################################################ | ||
85 | |||
91 | /* | 86 | /* |
92 | * Pops the key (metatable or idfunc) off the stack, and replaces with the | 87 | * Pops the key (metatable or idfunc) off the stack, and replaces with the |
93 | * deep lookup value (idfunc/metatable/nil). | 88 | * deep lookup value (idfunc/metatable/nil). |
94 | */ | 89 | */ |
95 | static void get_deep_lookup( lua_State* L) | 90 | static void get_deep_lookup(lua_State* L) |
96 | { | 91 | { |
97 | STACK_GROW( L, 1); | 92 | STACK_GROW( L, 1); |
98 | STACK_CHECK_START_REL(L, 1); // a | 93 | STACK_CHECK_START_REL(L, 1); // a |
@@ -106,11 +101,13 @@ static void get_deep_lookup( lua_State* L) | |||
106 | STACK_CHECK( L, 1); | 101 | STACK_CHECK( L, 1); |
107 | } | 102 | } |
108 | 103 | ||
104 | // ################################################################################################ | ||
105 | |||
109 | /* | 106 | /* |
110 | * Return the registered ID function for 'index' (deep userdata proxy), | 107 | * Return the registered ID function for 'index' (deep userdata proxy), |
111 | * or nullptr if 'index' is not a deep userdata proxy. | 108 | * or nullptr if 'index' is not a deep userdata proxy. |
112 | */ | 109 | */ |
113 | static inline luaG_IdFunction get_idfunc( lua_State* L, int index, LookupMode mode_) | 110 | static inline luaG_IdFunction get_idfunc(lua_State* L, int index, LookupMode mode_) |
114 | { | 111 | { |
115 | // when looking inside a keeper, we are 100% sure the object is a deep userdata | 112 | // when looking inside a keeper, we are 100% sure the object is a deep userdata |
116 | if (mode_ == LookupMode::FromKeeper) | 113 | if (mode_ == LookupMode::FromKeeper) |
@@ -142,8 +139,9 @@ static inline luaG_IdFunction get_idfunc( lua_State* L, int index, LookupMode mo | |||
142 | } | 139 | } |
143 | } | 140 | } |
144 | 141 | ||
142 | // ################################################################################################ | ||
145 | 143 | ||
146 | void free_deep_prelude( lua_State* L, DeepPrelude* prelude_) | 144 | void free_deep_prelude(lua_State* L, DeepPrelude* prelude_) |
147 | { | 145 | { |
148 | ASSERT_L(prelude_->idfunc); | 146 | ASSERT_L(prelude_->idfunc); |
149 | STACK_CHECK_START_REL(L, 0); | 147 | STACK_CHECK_START_REL(L, 0); |
@@ -154,6 +152,7 @@ void free_deep_prelude( lua_State* L, DeepPrelude* prelude_) | |||
154 | STACK_CHECK(L, 0); | 152 | STACK_CHECK(L, 0); |
155 | } | 153 | } |
156 | 154 | ||
155 | // ################################################################################################ | ||
157 | 156 | ||
158 | /* | 157 | /* |
159 | * void= mt.__gc( proxy_ud ) | 158 | * void= mt.__gc( proxy_ud ) |
@@ -161,7 +160,7 @@ void free_deep_prelude( lua_State* L, DeepPrelude* prelude_) | |||
161 | * End of life for a proxy object; reduce the deep reference count and clean it up if reaches 0. | 160 | * End of life for a proxy object; reduce the deep reference count and clean it up if reaches 0. |
162 | * | 161 | * |
163 | */ | 162 | */ |
164 | static int deep_userdata_gc( lua_State* L) | 163 | static int deep_userdata_gc(lua_State* L) |
165 | { | 164 | { |
166 | DeepPrelude** const proxy{ lua_tofulluserdata<DeepPrelude*>(L, 1) }; | 165 | DeepPrelude** const proxy{ lua_tofulluserdata<DeepPrelude*>(L, 1) }; |
167 | DeepPrelude* p = *proxy; | 166 | DeepPrelude* p = *proxy; |
@@ -193,6 +192,7 @@ static int deep_userdata_gc( lua_State* L) | |||
193 | return 0; | 192 | return 0; |
194 | } | 193 | } |
195 | 194 | ||
195 | // ################################################################################################ | ||
196 | 196 | ||
197 | /* | 197 | /* |
198 | * Push a proxy userdata on the stack. | 198 | * Push a proxy userdata on the stack. |
@@ -203,7 +203,7 @@ static int deep_userdata_gc( lua_State* L) | |||
203 | * used in this Lua state (metatable, registring it). Otherwise, increments the | 203 | * used in this Lua state (metatable, registring it). Otherwise, increments the |
204 | * reference count. | 204 | * reference count. |
205 | */ | 205 | */ |
206 | char const* push_deep_proxy(lua_State* L, DeepPrelude* prelude, int nuv_, LookupMode mode_) | 206 | char const* push_deep_proxy(Dest L, DeepPrelude* prelude, int nuv_, LookupMode mode_) |
207 | { | 207 | { |
208 | // Check if a proxy already exists | 208 | // Check if a proxy already exists |
209 | push_registry_subtable_mode( L, DEEP_PROXY_CACHE_KEY, "v"); // DPC | 209 | 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 | |||
278 | // this is needed because we must make sure the shared library is still loaded as long as we hold a pointer on the idfunc | 278 | // this is needed because we must make sure the shared library is still loaded as long as we hold a pointer on the idfunc |
279 | { | 279 | { |
280 | int oldtop_module = lua_gettop( L); | 280 | int oldtop_module = lua_gettop( L); |
281 | modname = (char const*) prelude->idfunc( L, DeepOp::Module); // DPC proxy metatable | 281 | modname = (char const*) prelude->idfunc( L, DeepOp::Module); // DPC proxy metatable |
282 | // make sure the function pushed nothing on the stack! | 282 | // make sure the function pushed nothing on the stack! |
283 | if( lua_gettop( L) - oldtop_module != 0) | 283 | if( lua_gettop( L) - oldtop_module != 0) |
284 | { | 284 | { |
@@ -348,6 +348,8 @@ char const* push_deep_proxy(lua_State* L, DeepPrelude* prelude, int nuv_, Lookup | |||
348 | return nullptr; | 348 | return nullptr; |
349 | } | 349 | } |
350 | 350 | ||
351 | // ################################################################################################ | ||
352 | |||
351 | /* | 353 | /* |
352 | * Create a deep userdata | 354 | * Create a deep userdata |
353 | * | 355 | * |
@@ -370,7 +372,7 @@ char const* push_deep_proxy(lua_State* L, DeepPrelude* prelude, int nuv_, Lookup | |||
370 | * | 372 | * |
371 | * Returns: 'proxy' userdata for accessing the deep data via 'luaG_todeep()' | 373 | * Returns: 'proxy' userdata for accessing the deep data via 'luaG_todeep()' |
372 | */ | 374 | */ |
373 | int luaG_newdeepuserdata( lua_State* L, luaG_IdFunction idfunc, int nuv_) | 375 | int luaG_newdeepuserdata(Dest L, luaG_IdFunction idfunc, int nuv_) |
374 | { | 376 | { |
375 | STACK_GROW( L, 1); | 377 | STACK_GROW( L, 1); |
376 | STACK_CHECK_START_REL(L, 0); | 378 | STACK_CHECK_START_REL(L, 0); |
@@ -409,6 +411,7 @@ int luaG_newdeepuserdata( lua_State* L, luaG_IdFunction idfunc, int nuv_) | |||
409 | return 1; | 411 | return 1; |
410 | } | 412 | } |
411 | 413 | ||
414 | // ################################################################################################ | ||
412 | 415 | ||
413 | /* | 416 | /* |
414 | * Access deep userdata through a proxy. | 417 | * Access deep userdata through a proxy. |
@@ -430,6 +433,7 @@ DeepPrelude* luaG_todeep(lua_State* L, luaG_IdFunction idfunc, int index) | |||
430 | return *proxy; | 433 | return *proxy; |
431 | } | 434 | } |
432 | 435 | ||
436 | // ################################################################################################ | ||
433 | 437 | ||
434 | /* | 438 | /* |
435 | * Copy deep userdata between two separate Lua states (from L to L2) | 439 | * 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) | |||
438 | * the id function of the copied value, or nullptr for non-deep userdata | 442 | * the id function of the copied value, or nullptr for non-deep userdata |
439 | * (not copied) | 443 | * (not copied) |
440 | */ | 444 | */ |
441 | bool copydeep(Universe* U, lua_State* L2, int L2_cache_i, lua_State* L, int i, LookupMode mode_, char const* upName_) | 445 | bool copydeep(Universe* U, Dest L2, int L2_cache_i, Source L, int i, LookupMode mode_, char const* upName_) |
442 | { | 446 | { |
443 | luaG_IdFunction const idfunc { get_idfunc(L, i, mode_) }; | 447 | luaG_IdFunction const idfunc { get_idfunc(L, i, mode_) }; |
444 | if (idfunc == nullptr) | 448 | if (idfunc == nullptr) |
@@ -54,8 +54,8 @@ struct DeepPrelude | |||
54 | std::atomic<int> m_refcount{ 0 }; | 54 | std::atomic<int> m_refcount{ 0 }; |
55 | }; | 55 | }; |
56 | 56 | ||
57 | char const* push_deep_proxy(lua_State* L, DeepPrelude* prelude, int nuv_, LookupMode mode_); | 57 | char const* push_deep_proxy(Dest L, DeepPrelude* prelude, int nuv_, LookupMode mode_); |
58 | void free_deep_prelude( lua_State* L, DeepPrelude* prelude_); | 58 | void free_deep_prelude( lua_State* L, DeepPrelude* prelude_); |
59 | 59 | ||
60 | LANES_API int luaG_newdeepuserdata( lua_State* L, luaG_IdFunction idfunc, int nuv_); | 60 | LANES_API int luaG_newdeepuserdata(Dest L, luaG_IdFunction idfunc, int nuv_); |
61 | LANES_API DeepPrelude* luaG_todeep(lua_State* L, luaG_IdFunction idfunc, int index); | 61 | 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 @@ | |||
14 | --[[ | 14 | --[[ |
15 | =============================================================================== | 15 | =============================================================================== |
16 | 16 | ||
17 | Copyright (C) 2011-2023 Benoit Germain <bnt.germain@gmail.com> | 17 | Copyright (C) 2011-2024 Benoit Germain <bnt.germain@gmail.com> |
18 | 18 | ||
19 | Permission is hereby granted, free of charge, to any person obtaining a copy | 19 | Permission is hereby granted, free of charge, to any person obtaining a copy |
20 | of this software and associated documentation files (the "Software"), to deal | 20 | of this software and associated documentation files (the "Software"), to deal |
@@ -207,10 +207,10 @@ static void push_table(lua_State* L, int idx_) | |||
207 | 207 | ||
208 | // ################################################################################################## | 208 | // ################################################################################################## |
209 | 209 | ||
210 | int keeper_push_linda_storage(Universe* U, lua_State* L, void* ptr_, uintptr_t magic_) | 210 | int keeper_push_linda_storage(Universe* U, Dest L, void* ptr_, uintptr_t magic_) |
211 | { | 211 | { |
212 | Keeper* const K{ which_keeper(U->keepers, magic_) }; | 212 | Keeper* const K{ which_keeper(U->keepers, magic_) }; |
213 | lua_State* const KL{ K ? K->L : nullptr }; | 213 | Source const KL{ K ? K->L : nullptr }; |
214 | if (KL == nullptr) | 214 | if (KL == nullptr) |
215 | return 0; | 215 | return 0; |
216 | STACK_GROW(KL, 4); | 216 | STACK_GROW(KL, 4); |
@@ -412,7 +412,7 @@ int keepercall_limit(lua_State* L) | |||
412 | // set the new limit | 412 | // set the new limit |
413 | fifo->limit = limit; | 413 | fifo->limit = limit; |
414 | // return 0 or 1 value | 414 | // return 0 or 1 value |
415 | return lua_gettop( L); | 415 | return lua_gettop(L); |
416 | } | 416 | } |
417 | 417 | ||
418 | // ################################################################################################## | 418 | // ################################################################################################## |
@@ -485,7 +485,7 @@ int keepercall_set(lua_State* L) | |||
485 | lua_insert(L, 3); // fifos key fifotbl [val [, ...]] | 485 | lua_insert(L, 3); // fifos key fifotbl [val [, ...]] |
486 | fifo_push(L, fifo, count); // fifos key fifotbl | 486 | fifo_push(L, fifo, count); // fifos key fifotbl |
487 | } | 487 | } |
488 | return should_wake_writers ? (lua_pushboolean( L, 1), 1) : 0; | 488 | return should_wake_writers ? (lua_pushboolean(L, 1), 1) : 0; |
489 | } | 489 | } |
490 | 490 | ||
491 | // ################################################################################################## | 491 | // ################################################################################################## |
@@ -717,7 +717,7 @@ void init_keepers(Universe* U, lua_State* L) | |||
717 | if (!lua_isnil(L, -1)) | 717 | if (!lua_isnil(L, -1)) |
718 | { | 718 | { |
719 | // when copying with mode LookupMode::ToKeeper, error message is pushed at the top of the stack, not raised immediately | 719 | // when copying with mode LookupMode::ToKeeper, error message is pushed at the top of the stack, not raised immediately |
720 | if (luaG_inter_copy_package(U, L, K, -1, LookupMode::ToKeeper)) | 720 | if (luaG_inter_copy_package(U, Source{ L }, Dest{ K }, -1, LookupMode::ToKeeper)) |
721 | { | 721 | { |
722 | // if something went wrong, the error message is at the top of the stack | 722 | // if something went wrong, the error message is at the top of the stack |
723 | lua_remove(L, -2); // error_msg | 723 | 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 | |||
840 | 840 | ||
841 | lua_pushlightuserdata(K, linda); | 841 | lua_pushlightuserdata(K, linda); |
842 | 842 | ||
843 | if ((args == 0) || luaG_inter_copy(U, L, K, args, LookupMode::ToKeeper) == 0) // L->K | 843 | if ((args == 0) || luaG_inter_copy(U, Source{ L }, Dest{ K }, args, LookupMode::ToKeeper) == 0) // L->K |
844 | { | 844 | { |
845 | lua_call(K, 1 + args, LUA_MULTRET); | 845 | lua_call(K, 1 + args, LUA_MULTRET); |
846 | retvals = lua_gettop(K) - Ktos; | 846 | 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 | |||
848 | // this may interrupt a lane, causing the destruction of the underlying OS thread | 848 | // this may interrupt a lane, causing the destruction of the underlying OS thread |
849 | // after this, another lane making use of this keeper can get an error code from the mutex-locking function | 849 | // after this, another lane making use of this keeper can get an error code from the mutex-locking function |
850 | // when attempting to grab the mutex again (WINVER <= 0x400 does this, but locks just fine, I don't know about pthread) | 850 | // when attempting to grab the mutex again (WINVER <= 0x400 does this, but locks just fine, I don't know about pthread) |
851 | if ((retvals > 0) && luaG_inter_move(U, K, L, retvals, LookupMode::FromKeeper) != 0) // K->L | 851 | if ((retvals > 0) && luaG_inter_move(U, Source{ K }, Dest{ L }, retvals, LookupMode::FromKeeper) != 0) // K->L |
852 | { | 852 | { |
853 | retvals = -1; | 853 | retvals = -1; |
854 | } | 854 | } |
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_); | |||
42 | Keeper* keeper_acquire(Keepers* keepers_, uintptr_t magic_); | 42 | Keeper* keeper_acquire(Keepers* keepers_, uintptr_t magic_); |
43 | void keeper_release(Keeper* K_); | 43 | void keeper_release(Keeper* K_); |
44 | void keeper_toggle_nil_sentinels(lua_State* L, int val_i_, LookupMode const mode_); | 44 | void keeper_toggle_nil_sentinels(lua_State* L, int val_i_, LookupMode const mode_); |
45 | int keeper_push_linda_storage(Universe* U, lua_State* L, void* ptr_, uintptr_t magic_); | 45 | int keeper_push_linda_storage(Universe* U, Dest L, void* ptr_, uintptr_t magic_); |
46 | 46 | ||
47 | using keeper_api_t = lua_CFunction; | 47 | using keeper_api_t = lua_CFunction; |
48 | #define KEEPER_API(_op) keepercall_##_op | 48 | #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) | |||
1027 | DEBUGSPEW_CODE(U->debugspew_indent_depth.fetch_add(1, std::memory_order_relaxed)); | 1027 | DEBUGSPEW_CODE(U->debugspew_indent_depth.fetch_add(1, std::memory_order_relaxed)); |
1028 | 1028 | ||
1029 | // populate with selected libraries at the same time | 1029 | // populate with selected libraries at the same time |
1030 | lua_State* const L2{ luaG_newstate(U, L, libs_str) }; // L // L2 | 1030 | lua_State* const L2{ luaG_newstate(U, Source{ L }, libs_str) }; // L // L2 |
1031 | 1031 | ||
1032 | // 'lane' is allocated from heap, not Lua, since its life span may surpass the handle's (if free running thread) | 1032 | // 'lane' is allocated from heap, not Lua, since its life span may surpass the handle's (if free running thread) |
1033 | Lane* const lane{ new (U) Lane{ U, L2 } }; | 1033 | Lane* const lane{ new (U) Lane{ U, L2 } }; |
@@ -1134,7 +1134,7 @@ LUAG_FUNC(lane_new) | |||
1134 | { | 1134 | { |
1135 | DEBUGSPEW_CODE(fprintf(stderr, INDENT_BEGIN "lane_new: update 'package'\n" INDENT_END)); | 1135 | DEBUGSPEW_CODE(fprintf(stderr, INDENT_BEGIN "lane_new: update 'package'\n" INDENT_END)); |
1136 | // when copying with mode LookupMode::LaneBody, should raise an error in case of problem, not leave it one the stack | 1136 | // when copying with mode LookupMode::LaneBody, should raise an error in case of problem, not leave it one the stack |
1137 | (void) luaG_inter_copy_package(U, L, L2, package_idx, LookupMode::LaneBody); | 1137 | std::ignore = luaG_inter_copy_package(U, Source{ L }, Dest{ L2 }, package_idx, LookupMode::LaneBody); |
1138 | } | 1138 | } |
1139 | 1139 | ||
1140 | // modules to require in the target lane *before* the function is transfered! | 1140 | // modules to require in the target lane *before* the function is transfered! |
@@ -1150,7 +1150,7 @@ LUAG_FUNC(lane_new) | |||
1150 | } | 1150 | } |
1151 | 1151 | ||
1152 | lua_pushnil(L); // func libs priority globals package required gc_cb [... args ...] nil | 1152 | lua_pushnil(L); // func libs priority globals package required gc_cb [... args ...] nil |
1153 | while( lua_next(L, required_idx) != 0) // func libs priority globals package required gc_cb [... args ...] n "modname" | 1153 | while (lua_next(L, required_idx) != 0) // func libs priority globals package required gc_cb [... args ...] n "modname" |
1154 | { | 1154 | { |
1155 | if (lua_type(L, -1) != LUA_TSTRING || lua_type(L, -2) != LUA_TNUMBER || lua_tonumber(L, -2) != nbRequired) | 1155 | if (lua_type(L, -1) != LUA_TSTRING || lua_type(L, -2) != LUA_TNUMBER || lua_tonumber(L, -2) != nbRequired) |
1156 | { | 1156 | { |
@@ -1176,7 +1176,7 @@ LUAG_FUNC(lane_new) | |||
1176 | if (lua_pcall( L2, 1, 1, 0) != LUA_OK) // ret/errcode | 1176 | if (lua_pcall( L2, 1, 1, 0) != LUA_OK) // ret/errcode |
1177 | { | 1177 | { |
1178 | // propagate error to main state if any | 1178 | // propagate error to main state if any |
1179 | luaG_inter_move(U, L2, L, 1, LookupMode::LaneBody); // func libs priority globals package required gc_cb [... args ...] n "modname" error | 1179 | luaG_inter_move(U, Source{ L2 }, Dest{ L }, 1, LookupMode::LaneBody); // func libs priority globals package required gc_cb [... args ...] n "modname" error |
1180 | raise_lua_error(L); | 1180 | raise_lua_error(L); |
1181 | } | 1181 | } |
1182 | // after requiring the module, register the functions it exported in our name<->function database | 1182 | // after requiring the module, register the functions it exported in our name<->function database |
@@ -1209,7 +1209,7 @@ LUAG_FUNC(lane_new) | |||
1209 | lua_pushglobaltable(L2); // _G | 1209 | lua_pushglobaltable(L2); // _G |
1210 | while( lua_next(L, globals_idx)) // func libs priority globals package required gc_cb [... args ...] k v | 1210 | while( lua_next(L, globals_idx)) // func libs priority globals package required gc_cb [... args ...] k v |
1211 | { | 1211 | { |
1212 | luaG_inter_copy(U, L, L2, 2, LookupMode::LaneBody); // _G k v | 1212 | luaG_inter_copy(U, Source{ L }, Dest{ L2 }, 2, LookupMode::LaneBody); // _G k v |
1213 | // assign it in L2's globals table | 1213 | // assign it in L2's globals table |
1214 | lua_rawset(L2, -3); // _G | 1214 | lua_rawset(L2, -3); // _G |
1215 | lua_pop(L, 1); // func libs priority globals package required gc_cb [... args ...] k | 1215 | lua_pop(L, 1); // func libs priority globals package required gc_cb [... args ...] k |
@@ -1227,7 +1227,7 @@ LUAG_FUNC(lane_new) | |||
1227 | DEBUGSPEW_CODE(fprintf( stderr, INDENT_BEGIN "lane_new: transfer lane body\n" INDENT_END)); | 1227 | DEBUGSPEW_CODE(fprintf( stderr, INDENT_BEGIN "lane_new: transfer lane body\n" INDENT_END)); |
1228 | DEBUGSPEW_CODE(U->debugspew_indent_depth.fetch_add(1, std::memory_order_relaxed)); | 1228 | DEBUGSPEW_CODE(U->debugspew_indent_depth.fetch_add(1, std::memory_order_relaxed)); |
1229 | lua_pushvalue(L, 1); // func libs priority globals package required gc_cb [... args ...] func | 1229 | lua_pushvalue(L, 1); // func libs priority globals package required gc_cb [... args ...] func |
1230 | int const res{ luaG_inter_move(U, L, L2, 1, LookupMode::LaneBody) };// func libs priority globals package required gc_cb [... args ...] // func | 1230 | int const res{ luaG_inter_move(U, Source{ L }, Dest{ L2 }, 1, LookupMode::LaneBody) }; // func libs priority globals package required gc_cb [... args ...] // func |
1231 | DEBUGSPEW_CODE(U->debugspew_indent_depth.fetch_sub(1, std::memory_order_relaxed)); | 1231 | DEBUGSPEW_CODE(U->debugspew_indent_depth.fetch_sub(1, std::memory_order_relaxed)); |
1232 | if (res != 0) | 1232 | if (res != 0) |
1233 | { | 1233 | { |
@@ -1253,7 +1253,7 @@ LUAG_FUNC(lane_new) | |||
1253 | int res; | 1253 | int res; |
1254 | DEBUGSPEW_CODE(fprintf( stderr, INDENT_BEGIN "lane_new: transfer lane arguments\n" INDENT_END)); | 1254 | DEBUGSPEW_CODE(fprintf( stderr, INDENT_BEGIN "lane_new: transfer lane arguments\n" INDENT_END)); |
1255 | DEBUGSPEW_CODE(U->debugspew_indent_depth.fetch_add(1, std::memory_order_relaxed)); | 1255 | DEBUGSPEW_CODE(U->debugspew_indent_depth.fetch_add(1, std::memory_order_relaxed)); |
1256 | res = luaG_inter_move(U, L, L2, nargs, LookupMode::LaneBody); // func libs priority globals package required gc_cb // func [... args ...] | 1256 | res = luaG_inter_move(U, Source{ L }, Dest{ L2 }, nargs, LookupMode::LaneBody); // func libs priority globals package required gc_cb // func [... args ...] |
1257 | DEBUGSPEW_CODE(U->debugspew_indent_depth.fetch_sub(1, std::memory_order_relaxed)); | 1257 | DEBUGSPEW_CODE(U->debugspew_indent_depth.fetch_sub(1, std::memory_order_relaxed)); |
1258 | if (res != 0) | 1258 | if (res != 0) |
1259 | { | 1259 | { |
@@ -1418,7 +1418,7 @@ LUAG_FUNC(thread_join) | |||
1418 | case Lane::Done: | 1418 | case Lane::Done: |
1419 | { | 1419 | { |
1420 | int const n{ lua_gettop(L2) }; // whole L2 stack | 1420 | int const n{ lua_gettop(L2) }; // whole L2 stack |
1421 | if ((n > 0) && (luaG_inter_move(U, L2, L, n, LookupMode::LaneBody) != 0)) | 1421 | if ((n > 0) && (luaG_inter_move(U, Source{ L2 }, Dest{ L }, n, LookupMode::LaneBody) != 0)) |
1422 | { | 1422 | { |
1423 | return luaL_error(L, "tried to copy unsupported types"); | 1423 | return luaL_error(L, "tried to copy unsupported types"); |
1424 | } | 1424 | } |
@@ -1432,7 +1432,7 @@ LUAG_FUNC(thread_join) | |||
1432 | STACK_GROW(L, 3); | 1432 | STACK_GROW(L, 3); |
1433 | lua_pushnil(L); | 1433 | lua_pushnil(L); |
1434 | // 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 ... | 1434 | // 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 ... |
1435 | if (luaG_inter_move(U, L2, L, n, LookupMode::LaneBody) != 0) // nil "err" [trace] | 1435 | if (luaG_inter_move(U, Source{ L2 }, Dest{ L }, n, LookupMode::LaneBody) != 0) // nil "err" [trace] |
1436 | { | 1436 | { |
1437 | return luaL_error(L, "tried to copy unsupported types: %s", lua_tostring(L, -n)); | 1437 | return luaL_error(L, "tried to copy unsupported types: %s", lua_tostring(L, -n)); |
1438 | } | 1438 | } |
@@ -1832,7 +1832,7 @@ LUAG_FUNC(configure) | |||
1832 | STACK_CHECK(L, 2); | 1832 | STACK_CHECK(L, 2); |
1833 | 1833 | ||
1834 | { | 1834 | { |
1835 | char const* errmsg{ push_deep_proxy(L, U->timer_deep, 0, LookupMode::LaneBody) }; // settings M timer_deep | 1835 | char const* errmsg{ push_deep_proxy(Dest{ L }, U->timer_deep, 0, LookupMode::LaneBody) }; // settings M timer_deep |
1836 | if (errmsg != nullptr) | 1836 | if (errmsg != nullptr) |
1837 | { | 1837 | { |
1838 | return luaL_error(L, errmsg); | 1838 | 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 | |||
98 | // 'Lane' are malloc/free'd and the handle only carries a pointer. | 98 | // 'Lane' are malloc/free'd and the handle only carries a pointer. |
99 | // This is not deep userdata since the handle's not portable among lanes. | 99 | // This is not deep userdata since the handle's not portable among lanes. |
100 | // | 100 | // |
101 | #define lua_toLane( L, i) (*((Lane**) luaL_checkudata( L, i, "Lane"))) | 101 | #define lua_toLane(L, i) (*((Lane**) luaL_checkudata( L, i, "Lane"))) |
102 | 102 | ||
103 | int push_thread_status( lua_State* L, Lane* s); | 103 | 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) | |||
804 | LUAG_FUNC(linda_dump) | 804 | LUAG_FUNC(linda_dump) |
805 | { | 805 | { |
806 | Linda* const linda{ lua_toLinda<false>(L, 1) }; | 806 | Linda* const linda{ lua_toLinda<false>(L, 1) }; |
807 | return keeper_push_linda_storage(linda->U, L, linda, linda->hashSeed()); | 807 | return keeper_push_linda_storage(linda->U, Dest{ L }, linda, linda->hashSeed()); |
808 | } | 808 | } |
809 | 809 | ||
810 | // ################################################################################################# | 810 | // ################################################################################################# |
@@ -816,7 +816,7 @@ LUAG_FUNC(linda_dump) | |||
816 | LUAG_FUNC(linda_towatch) | 816 | LUAG_FUNC(linda_towatch) |
817 | { | 817 | { |
818 | Linda* const linda{ lua_toLinda<false>(L, 1) }; | 818 | Linda* const linda{ lua_toLinda<false>(L, 1) }; |
819 | int pushed{ keeper_push_linda_storage(linda->U, L, linda, linda->hashSeed()) }; | 819 | int pushed{ keeper_push_linda_storage(linda->U, Dest{ L }, linda, linda->hashSeed()) }; |
820 | if (pushed == 0) | 820 | if (pushed == 0) |
821 | { | 821 | { |
822 | // if the linda is empty, don't return nil | 822 | // if the linda is empty, don't return nil |
@@ -1009,11 +1009,11 @@ static void* linda_id( lua_State* L, DeepOp op_) | |||
1009 | */ | 1009 | */ |
1010 | LUAG_FUNC(linda) | 1010 | LUAG_FUNC(linda) |
1011 | { | 1011 | { |
1012 | int const top = lua_gettop(L); | 1012 | int const top{ lua_gettop(L) }; |
1013 | luaL_argcheck(L, top <= 2, top, "too many arguments"); | 1013 | luaL_argcheck(L, top <= 2, top, "too many arguments"); |
1014 | if (top == 1) | 1014 | if (top == 1) |
1015 | { | 1015 | { |
1016 | int const t = lua_type(L, 1); | 1016 | int const t{ lua_type(L, 1) }; |
1017 | luaL_argcheck(L, t == LUA_TSTRING || t == LUA_TNUMBER, 1, "wrong parameter (should be a string or a number)"); | 1017 | luaL_argcheck(L, t == LUA_TSTRING || t == LUA_TNUMBER, 1, "wrong parameter (should be a string or a number)"); |
1018 | } | 1018 | } |
1019 | else if (top == 2) | 1019 | else if (top == 2) |
@@ -1021,5 +1021,5 @@ LUAG_FUNC(linda) | |||
1021 | luaL_checktype(L, 1, LUA_TSTRING); | 1021 | luaL_checktype(L, 1, LUA_TSTRING); |
1022 | luaL_checktype(L, 2, LUA_TNUMBER); | 1022 | luaL_checktype(L, 2, LUA_TNUMBER); |
1023 | } | 1023 | } |
1024 | return luaG_newdeepuserdata(L, linda_id, 0); | 1024 | return luaG_newdeepuserdata(Dest{ L }, linda_id, 0); |
1025 | } | 1025 | } |
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; | |||
44 | 44 | ||
45 | #else // NDEBUG | 45 | #else // NDEBUG |
46 | 46 | ||
47 | #define _ASSERT_L( L, cond_) if( (cond_) == 0) { (void) luaL_error( L, "ASSERT failed: %s:%d '%s'", __FILE__, __LINE__, #cond_);} | 47 | #define _ASSERT_L(L, cond_) if( (cond_) == 0) { (void) luaL_error(L, "ASSERT failed: %s:%d '%s'", __FILE__, __LINE__, #cond_);} |
48 | #define STACK_DUMP( L) luaG_dump( L) | 48 | #define STACK_DUMP(L) luaG_dump(L) |
49 | 49 | ||
50 | class StackChecker | 50 | class StackChecker |
51 | { | 51 | { |
@@ -172,3 +172,17 @@ T* lua_newuserdatauv(lua_State* L, int nuvalue_) | |||
172 | } | 172 | } |
173 | 173 | ||
174 | using lua_Duration = std::chrono::template duration<lua_Number>; | 174 | using lua_Duration = std::chrono::template duration<lua_Number>; |
175 | |||
176 | // ################################################################################################# | ||
177 | |||
178 | template <typename T, auto = []{}> | ||
179 | struct Unique | ||
180 | { | ||
181 | T m_val; | ||
182 | Unique() = default; | ||
183 | operator T() const { return m_val; } | ||
184 | explicit Unique(T b_) : m_val{ b_ } {} | ||
185 | }; | ||
186 | |||
187 | using Source = Unique<lua_State*>; | ||
188 | using Dest = Unique<lua_State*>; | ||
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 @@ | |||
1 | /* | 1 | /* |
2 | * STATE.C | 2 | * STATE.CPP |
3 | * | 3 | * |
4 | * Lua tools to support Lanes. | 4 | * Lua tools to support Lanes. |
5 | */ | 5 | */ |
@@ -31,20 +31,11 @@ THE SOFTWARE. | |||
31 | =============================================================================== | 31 | =============================================================================== |
32 | */ | 32 | */ |
33 | 33 | ||
34 | #include <stdio.h> | 34 | #include "state.h" |
35 | #include <assert.h> | 35 | |
36 | #include <string.h> | ||
37 | #include <ctype.h> | ||
38 | #include <stdlib.h> | ||
39 | #if !defined(__APPLE__) | ||
40 | #include <malloc.h> | ||
41 | #endif // __APPLE__ | ||
42 | |||
43 | #include "compat.h" | ||
44 | #include "macros_and_utils.h" | ||
45 | #include "universe.h" | ||
46 | #include "tools.h" | ||
47 | #include "lanes.h" | 36 | #include "lanes.h" |
37 | #include "tools.h" | ||
38 | #include "universe.h" | ||
48 | 39 | ||
49 | // ################################################################################################ | 40 | // ################################################################################################ |
50 | 41 | ||
@@ -194,9 +185,9 @@ static void open1lib( DEBUGSPEW_PARAM_COMMA( Universe* U) lua_State* L, char con | |||
194 | 185 | ||
195 | 186 | ||
196 | // just like lua_xmove, args are (from, to) | 187 | // just like lua_xmove, args are (from, to) |
197 | static void copy_one_time_settings( Universe* U, lua_State* L, lua_State* L2) | 188 | static void copy_one_time_settings(Universe* U, Source L, Dest L2) |
198 | { | 189 | { |
199 | STACK_GROW( L, 2); | 190 | STACK_GROW(L, 2); |
200 | STACK_CHECK_START_REL(L, 0); | 191 | STACK_CHECK_START_REL(L, 0); |
201 | STACK_CHECK_START_REL(L2, 0); | 192 | STACK_CHECK_START_REL(L2, 0); |
202 | 193 | ||
@@ -326,11 +317,11 @@ void call_on_state_create(Universe* U, lua_State* L, lua_State* from_, LookupMod | |||
326 | * *NOT* called for keeper states! | 317 | * *NOT* called for keeper states! |
327 | * | 318 | * |
328 | */ | 319 | */ |
329 | lua_State* luaG_newstate( Universe* U, lua_State* from_, char const* libs_) | 320 | lua_State* luaG_newstate(Universe* U, Source from_, char const* libs_) |
330 | { | 321 | { |
331 | lua_State* L = create_state( U, from_); | 322 | Dest const L{ create_state(U, from_) }; |
332 | 323 | ||
333 | STACK_GROW( L, 2); | 324 | STACK_GROW(L, 2); |
334 | STACK_CHECK_START_ABS(L, 0); | 325 | STACK_CHECK_START_ABS(L, 0); |
335 | 326 | ||
336 | // copy the universe as a light userdata (only the master state holds the full userdata) | 327 | // 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 @@ | |||
3 | #include "macros_and_utils.h" | 3 | #include "macros_and_utils.h" |
4 | 4 | ||
5 | // forwards | 5 | // forwards |
6 | enum class LookupMode; | ||
6 | class Universe; | 7 | class Universe; |
7 | 8 | ||
8 | void serialize_require(DEBUGSPEW_PARAM_COMMA(Universe* U) lua_State* L); | 9 | 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); | |||
10 | // ################################################################################################ | 11 | // ################################################################################################ |
11 | 12 | ||
12 | lua_State* create_state(Universe* U, lua_State* from_); | 13 | lua_State* create_state(Universe* U, lua_State* from_); |
13 | lua_State* luaG_newstate(Universe* U, lua_State* _from, char const* libs); | 14 | lua_State* luaG_newstate(Universe* U, Source _from, char const* libs); |
14 | 15 | ||
15 | // ################################################################################################ | 16 | // ################################################################################################ |
16 | 17 | ||
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 @@ | |||
1 | /* | 1 | /* |
2 | * THREADING.C Copyright (c) 2007-08, Asko Kauppi | 2 | * THREADING.CPP Copyright (c) 2007-08, Asko Kauppi |
3 | * Copyright (C) 2009-19, Benoit Germain | 3 | * Copyright (C) 2009-24, Benoit Germain |
4 | * | 4 | * |
5 | * Lua Lanes OS threading specific code. | 5 | * Lua Lanes OS threading specific code. |
6 | * | 6 | * |
@@ -12,7 +12,7 @@ | |||
12 | =============================================================================== | 12 | =============================================================================== |
13 | 13 | ||
14 | Copyright (C) 2007-10 Asko Kauppi <akauppi@gmail.com> | 14 | Copyright (C) 2007-10 Asko Kauppi <akauppi@gmail.com> |
15 | Copyright (C) 2009-14, Benoit Germain <bnt.germain@gmail.com> | 15 | Copyright (C) 2009-24, Benoit Germain <bnt.germain@gmail.com> |
16 | 16 | ||
17 | Permission is hereby granted, free of charge, to any person obtaining a copy | 17 | Permission is hereby granted, free of charge, to any person obtaining a copy |
18 | of this software and associated documentation files (the "Software"), to deal | 18 | of this software and associated documentation files (the "Software"), to deal |
@@ -47,12 +47,6 @@ THE SOFTWARE. | |||
47 | 47 | ||
48 | #endif // __linux__ | 48 | #endif // __linux__ |
49 | 49 | ||
50 | #include <stdio.h> | ||
51 | #include <stdlib.h> | ||
52 | #include <assert.h> | ||
53 | #include <errno.h> | ||
54 | #include <math.h> | ||
55 | |||
56 | #include "threading.h" | 50 | #include "threading.h" |
57 | 51 | ||
58 | #if !defined( PLATFORM_XBOX) && !defined( PLATFORM_WIN32) && !defined( PLATFORM_POCKETPC) | 52 | #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 @@ | |||
2 | 2 | ||
3 | #include "platform.h" | 3 | #include "platform.h" |
4 | 4 | ||
5 | #include <time.h> | ||
6 | #include <thread> | 5 | #include <thread> |
7 | 6 | ||
8 | #define THREADAPI_WINDOWS 1 | 7 | #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. | |||
31 | =============================================================================== | 31 | =============================================================================== |
32 | */ | 32 | */ |
33 | 33 | ||
34 | #include <stdio.h> | ||
35 | #include <assert.h> | ||
36 | #include <string.h> | ||
37 | #include <ctype.h> | ||
38 | #include <stdlib.h> | ||
39 | #if !defined(__APPLE__) | ||
40 | #include <malloc.h> | ||
41 | #endif // __APPLE__ | ||
42 | |||
43 | #include "tools.h" | 34 | #include "tools.h" |
44 | #include "compat.h" | 35 | |
45 | #include "universe.h" | 36 | #include "universe.h" |
46 | #include "keeper.h" | ||
47 | #include "lanes.h" | ||
48 | #include "uniquekey.h" | ||
49 | 37 | ||
50 | // functions implemented in deep.c | 38 | // functions implemented in deep.c |
51 | extern bool copydeep(Universe* U, lua_State* L2, int L2_cache_i, lua_State* L, int i, LookupMode mode_, char const* upName_); | 39 | extern bool copydeep(Universe* U, Dest L2, int L2_cache_i, Source L, int i, LookupMode mode_, char const* upName_); |
52 | extern void push_registry_subtable( lua_State* L, UniqueKey key_); | 40 | extern void push_registry_subtable( lua_State* L, UniqueKey key_); |
53 | 41 | ||
54 | DEBUGSPEW_CODE( char const* debugspew_indent = "----+----!----+----!----+----!----+----!----+----!----+----!----+----!----+"); | 42 | DEBUGSPEW_CODE( char const* debugspew_indent = "----+----!----+----!----+----!----+----!----+----!----+----!----+----!----+"); |
@@ -554,10 +542,10 @@ static void populate_func_lookup_table_recur(DEBUGSPEW_PARAM_COMMA(Universe* U) | |||
554 | /* | 542 | /* |
555 | * create a "fully.qualified.name" <-> function equivalence database | 543 | * create a "fully.qualified.name" <-> function equivalence database |
556 | */ | 544 | */ |
557 | void populate_func_lookup_table( lua_State* L, int _i, char const* name_) | 545 | void populate_func_lookup_table(lua_State* L, int i_, char const* name_) |
558 | { | 546 | { |
559 | int const ctx_base = lua_gettop( L) + 1; | 547 | int const ctx_base = lua_gettop(L) + 1; |
560 | int const in_base = lua_absindex( L, _i); | 548 | int const in_base = lua_absindex(L, i_); |
561 | int start_depth = 0; | 549 | int start_depth = 0; |
562 | DEBUGSPEW_CODE( Universe* U = universe_get( L)); | 550 | DEBUGSPEW_CODE( Universe* U = universe_get( L)); |
563 | DEBUGSPEW_CODE( fprintf( stderr, INDENT_BEGIN "%p: populate_func_lookup_table('%s')\n" INDENT_END, L, name_ ? name_ : "nullptr")); | 551 | 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 | |||
765 | /* | 753 | /* |
766 | * Push a looked-up table, or nothing if we found nothing | 754 | * Push a looked-up table, or nothing if we found nothing |
767 | */ | 755 | */ |
768 | static bool lookup_table(lua_State* L2, lua_State* L, int i, LookupMode mode_, char const* upName_) | 756 | static bool lookup_table(Dest L2, Source L, int i, LookupMode mode_, char const* upName_) |
769 | { | 757 | { |
770 | // get the name of the table we want to send | 758 | // get the name of the table we want to send |
771 | size_t len; | 759 | size_t len; |
@@ -775,7 +763,7 @@ static bool lookup_table(lua_State* L2, lua_State* L, int i, LookupMode mode_, c | |||
775 | return false; | 763 | return false; |
776 | } | 764 | } |
777 | // push the equivalent table in the destination's stack, retrieved from the lookup table | 765 | // push the equivalent table in the destination's stack, retrieved from the lookup table |
778 | STACK_CHECK_START_REL(L2, 0); // L // L2 | 766 | STACK_CHECK_START_REL(L2, 0); // L // L2 |
779 | STACK_GROW( L2, 3); // up to 3 slots are necessary on error | 767 | STACK_GROW( L2, 3); // up to 3 slots are necessary on error |
780 | switch( mode_) | 768 | switch( mode_) |
781 | { | 769 | { |
@@ -785,34 +773,34 @@ static bool lookup_table(lua_State* L2, lua_State* L, int i, LookupMode mode_, c | |||
785 | 773 | ||
786 | case LookupMode::ToKeeper: | 774 | case LookupMode::ToKeeper: |
787 | // push a sentinel closure that holds the lookup name as upvalue | 775 | // push a sentinel closure that holds the lookup name as upvalue |
788 | lua_pushlstring( L2, fqn, len); // "f.q.n" | 776 | lua_pushlstring(L2, fqn, len); // "f.q.n" |
789 | lua_pushcclosure( L2, table_lookup_sentinel, 1); // f | 777 | lua_pushcclosure(L2, table_lookup_sentinel, 1); // f |
790 | break; | 778 | break; |
791 | 779 | ||
792 | case LookupMode::LaneBody: | 780 | case LookupMode::LaneBody: |
793 | case LookupMode::FromKeeper: | 781 | case LookupMode::FromKeeper: |
794 | LOOKUP_REGKEY.pushValue(L2); // {} | 782 | LOOKUP_REGKEY.pushValue(L2); // {} |
795 | STACK_CHECK( L2, 1); | 783 | STACK_CHECK(L2, 1); |
796 | ASSERT_L( lua_istable( L2, -1)); | 784 | ASSERT_L(lua_istable(L2, -1)); |
797 | lua_pushlstring( L2, fqn, len); // {} "f.q.n" | 785 | lua_pushlstring(L2, fqn, len); // {} "f.q.n" |
798 | lua_rawget( L2, -2); // {} t | 786 | lua_rawget(L2, -2); // {} t |
799 | // we accept destination lookup failures in the case of transfering the Lanes body function (this will result in the source table being cloned instead) | 787 | // we accept destination lookup failures in the case of transfering the Lanes body function (this will result in the source table being cloned instead) |
800 | // but not when we extract something out of a keeper, as there is nothing to clone! | 788 | // but not when we extract something out of a keeper, as there is nothing to clone! |
801 | if (lua_isnil(L2, -1) && mode_ == LookupMode::LaneBody) | 789 | if (lua_isnil(L2, -1) && mode_ == LookupMode::LaneBody) |
802 | { | 790 | { |
803 | lua_pop( L2, 2); // | 791 | lua_pop(L2, 2); // |
804 | STACK_CHECK( L2, 0); | 792 | STACK_CHECK(L2, 0); |
805 | return false; | 793 | return false; |
806 | } | 794 | } |
807 | else if( !lua_istable( L2, -1)) | 795 | else if( !lua_istable(L2, -1)) |
808 | { | 796 | { |
809 | char const* from, *to; | 797 | char const* from, *to; |
810 | lua_getglobal( L, "decoda_name"); // ... t ... decoda_name | 798 | lua_getglobal(L, "decoda_name"); // ... t ... decoda_name |
811 | from = lua_tostring( L, -1); | 799 | from = lua_tostring(L, -1); |
812 | lua_pop( L, 1); // ... t ... | 800 | lua_pop(L, 1); // ... t ... |
813 | lua_getglobal( L2, "decoda_name"); // {} t decoda_name | 801 | lua_getglobal(L2, "decoda_name"); // {} t decoda_name |
814 | to = lua_tostring( L2, -1); | 802 | to = lua_tostring( L2, -1); |
815 | lua_pop( L2, 1); // {} t | 803 | lua_pop(L2, 1); // {} t |
816 | // 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 | 804 | // 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 |
817 | (void) luaL_error( | 805 | (void) luaL_error( |
818 | (mode_ == LookupMode::FromKeeper) ? L2 : L | 806 | (mode_ == LookupMode::FromKeeper) ? L2 : L |
@@ -823,7 +811,7 @@ static bool lookup_table(lua_State* L2, lua_State* L, int i, LookupMode mode_, c | |||
823 | ); | 811 | ); |
824 | return false; | 812 | return false; |
825 | } | 813 | } |
826 | lua_remove( L2, -2); // t | 814 | lua_remove(L2, -2); // t |
827 | break; | 815 | break; |
828 | } | 816 | } |
829 | STACK_CHECK( L2, 1); | 817 | STACK_CHECK( L2, 1); |
@@ -1194,7 +1182,7 @@ static int buf_writer( lua_State* L, void const* b, size_t size, void* ud) | |||
1194 | 1182 | ||
1195 | // ################################################################################################# | 1183 | // ################################################################################################# |
1196 | 1184 | ||
1197 | static void copy_func(Universe* U, lua_State* L2, int L2_cache_i, lua_State* L, int i, LookupMode mode_, char const* upName_) | 1185 | static void copy_func(Universe* U, Dest L2, int L2_cache_i, Source L, int i, LookupMode mode_, char const* upName_) |
1198 | { | 1186 | { |
1199 | int n, needToPush; | 1187 | int n, needToPush; |
1200 | luaL_Buffer B; | 1188 | luaL_Buffer B; |
@@ -1348,7 +1336,7 @@ static void copy_func(Universe* U, lua_State* L2, int L2_cache_i, lua_State* L, | |||
1348 | * | 1336 | * |
1349 | * Always pushes a function to 'L2'. | 1337 | * Always pushes a function to 'L2'. |
1350 | */ | 1338 | */ |
1351 | static void copy_cached_func(Universe* U, lua_State* L2, int L2_cache_i, lua_State* L, int i, LookupMode mode_, char const* upName_) | 1339 | static void copy_cached_func(Universe* U, Dest L2, int L2_cache_i, Source L, int i, LookupMode mode_, char const* upName_) |
1352 | { | 1340 | { |
1353 | FuncSubType funcSubType; | 1341 | FuncSubType funcSubType; |
1354 | /*lua_CFunction cfunc =*/ luaG_tocfunction( L, i, &funcSubType); // nullptr for LuaJIT-fast && bytecode functions | 1342 | /*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 | |||
1403 | 1391 | ||
1404 | // ################################################################################################# | 1392 | // ################################################################################################# |
1405 | 1393 | ||
1406 | static bool push_cached_metatable(Universe* U, lua_State* L2, int L2_cache_i, lua_State* L, int i, LookupMode mode_, char const* upName_) | 1394 | static bool push_cached_metatable(Universe* U, Dest L2, int L2_cache_i, Source L, int i, LookupMode mode_, char const* upName_) |
1407 | { | 1395 | { |
1408 | STACK_CHECK_START_REL(L, 0); | 1396 | STACK_CHECK_START_REL(L, 0); |
1409 | if( lua_getmetatable( L, i)) // ... mt | 1397 | 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 | |||
1454 | 1442 | ||
1455 | // ################################################################################################# | 1443 | // ################################################################################################# |
1456 | 1444 | ||
1457 | static void inter_copy_keyvaluepair(Universe* U, lua_State* L2, int L2_cache_i, lua_State* L, VT vt_, LookupMode mode_, char const* upName_) | 1445 | static void inter_copy_keyvaluepair(Universe* U, Dest L2, int L2_cache_i, Source L, VT vt_, LookupMode mode_, char const* upName_) |
1458 | { | 1446 | { |
1459 | int val_i = lua_gettop(L); | 1447 | int val_i = lua_gettop(L); |
1460 | int key_i = val_i - 1; | 1448 | int key_i = val_i - 1; |
@@ -1526,7 +1514,7 @@ static void inter_copy_keyvaluepair(Universe* U, lua_State* L2, int L2_cache_i, | |||
1526 | */ | 1514 | */ |
1527 | static constexpr UniqueKey CLONABLES_CACHE_KEY{ 0xD04EE018B3DEE8F5ull }; | 1515 | static constexpr UniqueKey CLONABLES_CACHE_KEY{ 0xD04EE018B3DEE8F5ull }; |
1528 | 1516 | ||
1529 | static bool copyclone(Universe* U, lua_State* L2, int L2_cache_i, lua_State* L, int source_i_, LookupMode mode_, char const* upName_) | 1517 | static bool copyclone(Universe* U, Dest L2, int L2_cache_i, Source L, int source_i_, LookupMode mode_, char const* upName_) |
1530 | { | 1518 | { |
1531 | void* const source = lua_touserdata( L, source_i_); | 1519 | void* const source = lua_touserdata( L, source_i_); |
1532 | source_i_ = lua_absindex( L, source_i_); | 1520 | 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, | |||
1641 | 1629 | ||
1642 | // ################################################################################################# | 1630 | // ################################################################################################# |
1643 | 1631 | ||
1644 | 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_) | 1632 | static bool inter_copy_userdata(Universe* U, Dest L2, int L2_cache_i, Source L, int i, VT vt_, LookupMode mode_, char const* upName_) |
1645 | { | 1633 | { |
1646 | STACK_CHECK_START_REL(L, 0); | 1634 | STACK_CHECK_START_REL(L, 0); |
1647 | STACK_CHECK_START_REL(L2, 0); | 1635 | 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_ | |||
1691 | 1679 | ||
1692 | // ################################################################################################# | 1680 | // ################################################################################################# |
1693 | 1681 | ||
1694 | 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_) | 1682 | 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_) |
1695 | { | 1683 | { |
1696 | if (vt_ == VT::KEY) | 1684 | if (vt_ == VT::KEY) |
1697 | { | 1685 | { |
@@ -1786,7 +1774,7 @@ static bool inter_copy_function(Universe* U, lua_State* L2, int L2_cache_i, lua_ | |||
1786 | 1774 | ||
1787 | // ################################################################################################# | 1775 | // ################################################################################################# |
1788 | 1776 | ||
1789 | 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_) | 1777 | static bool inter_copy_table(Universe* U, Dest L2, int L2_cache_i, Source L, int i, VT vt_, LookupMode mode_, char const* upName_) |
1790 | { | 1778 | { |
1791 | if (vt_ == VT::KEY) | 1779 | if (vt_ == VT::KEY) |
1792 | { | 1780 | { |
@@ -1858,7 +1846,7 @@ static bool inter_copy_table(Universe* U, lua_State* L2, int L2_cache_i, lua_Sta | |||
1858 | * | 1846 | * |
1859 | * Returns true if value was pushed, false if its type is non-supported. | 1847 | * Returns true if value was pushed, false if its type is non-supported. |
1860 | */ | 1848 | */ |
1861 | 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_) | 1849 | bool inter_copy_one(Universe* U, Dest L2, int L2_cache_i, Source L, int i, VT vt_, LookupMode mode_, char const* upName_) |
1862 | { | 1850 | { |
1863 | bool ret{ true }; | 1851 | bool ret{ true }; |
1864 | int val_type = lua_type( L, i); | 1852 | 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 | |||
1974 | return ret; | 1962 | return ret; |
1975 | } | 1963 | } |
1976 | 1964 | ||
1965 | // ################################################################################################# | ||
1966 | |||
1977 | /* | 1967 | /* |
1978 | * Akin to 'lua_xmove' but copies values between _any_ Lua states. | 1968 | * Akin to 'lua_xmove' but copies values between _any_ Lua states. |
1979 | * | 1969 | * |
@@ -1981,7 +1971,7 @@ bool inter_copy_one(Universe* U, lua_State* L2, int L2_cache_i, lua_State* L, in | |||
1981 | * | 1971 | * |
1982 | * Note: Parameters are in this order ('L' = from first) to be same as 'lua_xmove'. | 1972 | * Note: Parameters are in this order ('L' = from first) to be same as 'lua_xmove'. |
1983 | */ | 1973 | */ |
1984 | int luaG_inter_copy(Universe* U, lua_State* L, lua_State* L2, int n, LookupMode mode_) | 1974 | int luaG_inter_copy(Universe* U, Source L, Dest L2, int n, LookupMode mode_) |
1985 | { | 1975 | { |
1986 | int top_L = lua_gettop(L); // ... {}n | 1976 | int top_L = lua_gettop(L); // ... {}n |
1987 | int top_L2 = lua_gettop(L2); // ... | 1977 | 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 | |||
2043 | return -2; | 2033 | return -2; |
2044 | } | 2034 | } |
2045 | 2035 | ||
2036 | // ################################################################################################# | ||
2046 | 2037 | ||
2047 | int luaG_inter_move(Universe* U, lua_State* L, lua_State* L2, int n, LookupMode mode_) | 2038 | int luaG_inter_move(Universe* U, Source L, Dest L2, int n, LookupMode mode_) |
2048 | { | 2039 | { |
2049 | int ret = luaG_inter_copy( U, L, L2, n, mode_); | 2040 | int ret = luaG_inter_copy( U, L, L2, n, mode_); |
2050 | lua_pop( L, (int) n); | 2041 | lua_pop( L, (int) n); |
2051 | return ret; | 2042 | return ret; |
2052 | } | 2043 | } |
2053 | 2044 | ||
2054 | int luaG_inter_copy_package(Universe* U, lua_State* L, lua_State* L2, int package_idx_, LookupMode mode_) | 2045 | int luaG_inter_copy_package(Universe* U, Source L, Dest L2, int package_idx_, LookupMode mode_) |
2055 | { | 2046 | { |
2056 | DEBUGSPEW_CODE(fprintf(stderr, INDENT_BEGIN "luaG_inter_copy_package()\n" INDENT_END)); | 2047 | DEBUGSPEW_CODE(fprintf(stderr, INDENT_BEGIN "luaG_inter_copy_package()\n" INDENT_END)); |
2057 | DEBUGSPEW_CODE(U->debugspew_indent_depth.fetch_add(1, std::memory_order_relaxed)); | 2048 | 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 | |||
25 | KEY, | 25 | KEY, |
26 | METATABLE | 26 | METATABLE |
27 | }; | 27 | }; |
28 | 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_); | 28 | bool inter_copy_one(Universe* U, Dest L2, int L2_cache_i, Source L, int i, VT vt_, LookupMode mode_, char const* upName_); |
29 | 29 | ||
30 | // ################################################################################################ | 30 | // ################################################################################################ |
31 | 31 | ||
32 | int luaG_inter_copy_package( Universe* U, lua_State* L, lua_State* L2, int package_idx_, LookupMode mode_); | 32 | int luaG_inter_copy_package(Universe* U, Source L, Dest L2, int package_idx_, LookupMode mode_); |
33 | 33 | ||
34 | int luaG_inter_copy(Universe* U, lua_State* L, lua_State* L2, int n, LookupMode mode_); | 34 | int luaG_inter_copy(Universe* U, Source L, Dest L2, int n, LookupMode mode_); |
35 | int luaG_inter_move(Universe* U, lua_State* L, lua_State* L2, int n, LookupMode mode_); | 35 | int luaG_inter_move(Universe* U, Source L, Dest L2, int n, LookupMode mode_); |
36 | 36 | ||
37 | int luaG_nameof( lua_State* L); | 37 | int luaG_nameof(lua_State* L); |
38 | 38 | ||
39 | void populate_func_lookup_table( lua_State* L, int _i, char const* _name); | 39 | void populate_func_lookup_table(lua_State* L, int _i, char const* _name); |
40 | void initialize_allocator_function( Universe* U, lua_State* L); | 40 | void initialize_allocator_function(Universe* U, lua_State* L); |
41 | 41 | ||
42 | // ################################################################################################ | 42 | // ################################################################################################ |
43 | 43 | ||
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) | |||
80 | U->Universe::Universe(); | 80 | U->Universe::Universe(); |
81 | STACK_CHECK_START_REL(L, 1); | 81 | STACK_CHECK_START_REL(L, 1); |
82 | UNIVERSE_FULL_REGKEY.setValue(L, [](lua_State* L) { lua_pushvalue(L, -2); }); | 82 | UNIVERSE_FULL_REGKEY.setValue(L, [](lua_State* L) { lua_pushvalue(L, -2); }); |
83 | UNIVERSE_LIGHT_REGKEY.setValue(L, [U](lua_State* L) { lua_pushlightuserdata( L, U); }); | 83 | UNIVERSE_LIGHT_REGKEY.setValue(L, [U](lua_State* L) { lua_pushlightuserdata(L, U); }); |
84 | STACK_CHECK(L, 1); | 84 | STACK_CHECK(L, 1); |
85 | return U; | 85 | return U; |
86 | } | 86 | } |
@@ -91,8 +91,8 @@ void universe_store(lua_State* L, Universe* U) | |||
91 | { | 91 | { |
92 | ASSERT_L(!U || universe_get(L) == nullptr); | 92 | ASSERT_L(!U || universe_get(L) == nullptr); |
93 | STACK_CHECK_START_REL(L, 0); | 93 | STACK_CHECK_START_REL(L, 0); |
94 | UNIVERSE_LIGHT_REGKEY.setValue(L, [U](lua_State* L) { U ? lua_pushlightuserdata( L, U) : lua_pushnil( L); }); | 94 | UNIVERSE_LIGHT_REGKEY.setValue(L, [U](lua_State* L) { U ? lua_pushlightuserdata(L, U) : lua_pushnil(L); }); |
95 | STACK_CHECK( L, 0); | 95 | STACK_CHECK(L, 0); |
96 | } | 96 | } |
97 | 97 | ||
98 | // ################################################################################################ | 98 | // ################################################################################################ |
@@ -101,6 +101,6 @@ Universe* universe_get(lua_State* L) | |||
101 | { | 101 | { |
102 | STACK_CHECK_START_REL(L, 0); | 102 | STACK_CHECK_START_REL(L, 0); |
103 | Universe* const universe{ UNIVERSE_LIGHT_REGKEY.readLightUserDataValue<Universe>(L) }; | 103 | Universe* const universe{ UNIVERSE_LIGHT_REGKEY.readLightUserDataValue<Universe>(L) }; |
104 | STACK_CHECK( L, 0); | 104 | STACK_CHECK(L, 0); |
105 | return universe; | 105 | return universe; |
106 | } | 106 | } |