diff options
-rw-r--r-- | src/cancel.cpp | 4 | ||||
-rw-r--r-- | src/cancel.h | 2 | ||||
-rw-r--r-- | src/lanes.cpp | 88 | ||||
-rw-r--r-- | src/lanes_private.h | 45 | ||||
-rw-r--r-- | src/universe.h | 2 |
5 files changed, 76 insertions, 65 deletions
diff --git a/src/cancel.cpp b/src/cancel.cpp index 0f7a6af..07c316a 100644 --- a/src/cancel.cpp +++ b/src/cancel.cpp | |||
@@ -162,7 +162,7 @@ static CancelResult thread_cancel_hard(lua_State* L, Lane* lane_, double secs_, | |||
162 | (void) waitkill_timeout_; // unused | 162 | (void) waitkill_timeout_; // unused |
163 | (void) L; // unused | 163 | (void) L; // unused |
164 | #endif // THREADAPI == THREADAPI_PTHREAD | 164 | #endif // THREADAPI == THREADAPI_PTHREAD |
165 | lane_->mstatus = ThreadStatus::Killed; // mark 'gc' to wait for it | 165 | lane_->mstatus = Lane::Killed; // mark 'gc' to wait for it |
166 | // note that lane_->status value must remain to whatever it was at the time of the kill | 166 | // note that lane_->status value must remain to whatever it was at the time of the kill |
167 | // because we need to know if we can lua_close() the Lua State or not. | 167 | // because we need to know if we can lua_close() the Lua State or not. |
168 | result = CancelResult::Killed; | 168 | result = CancelResult::Killed; |
@@ -176,7 +176,7 @@ CancelResult thread_cancel(lua_State* L, Lane* lane_, CancelOp op_, double secs_ | |||
176 | { | 176 | { |
177 | // remember that lanes are not transferable: only one thread can cancel a lane, so no multithreading issue here | 177 | // remember that lanes are not transferable: only one thread can cancel a lane, so no multithreading issue here |
178 | // We can read 'lane_->status' without locks, but not wait for it (if Posix no PTHREAD_TIMEDJOIN) | 178 | // We can read 'lane_->status' without locks, but not wait for it (if Posix no PTHREAD_TIMEDJOIN) |
179 | if (lane_->mstatus == ThreadStatus::Killed) | 179 | if (lane_->mstatus == Lane::Killed) |
180 | { | 180 | { |
181 | return CancelResult::Killed; | 181 | return CancelResult::Killed; |
182 | } | 182 | } |
diff --git a/src/cancel.h b/src/cancel.h index 481cca4..0c1de14 100644 --- a/src/cancel.h +++ b/src/cancel.h | |||
@@ -15,7 +15,7 @@ extern "C" { | |||
15 | 15 | ||
16 | // ################################################################################################ | 16 | // ################################################################################################ |
17 | 17 | ||
18 | struct Lane; // forward | 18 | class Lane; // forward |
19 | 19 | ||
20 | /* | 20 | /* |
21 | * Lane cancellation request modes | 21 | * Lane cancellation request modes |
diff --git a/src/lanes.cpp b/src/lanes.cpp index 327c2df..079b880 100644 --- a/src/lanes.cpp +++ b/src/lanes.cpp | |||
@@ -99,6 +99,27 @@ THE SOFTWARE. | |||
99 | # include <sys/types.h> | 99 | # include <sys/types.h> |
100 | #endif | 100 | #endif |
101 | 101 | ||
102 | // forwarding (will do things better later) | ||
103 | static void tracking_add(Lane* lane_); | ||
104 | |||
105 | Lane::Lane(Universe* U_, lua_State* L_) | ||
106 | : U{ U_ } | ||
107 | , L{ L_ } | ||
108 | { | ||
109 | #if THREADWAIT_METHOD == THREADWAIT_CONDVAR | ||
110 | MUTEX_INIT(&done_lock); | ||
111 | SIGNAL_INIT(&done_signal); | ||
112 | #endif // THREADWAIT_METHOD == THREADWAIT_CONDVAR | ||
113 | |||
114 | #if HAVE_LANE_TRACKING() | ||
115 | if (U->tracking_first) | ||
116 | { | ||
117 | tracking_add(this); | ||
118 | } | ||
119 | #endif // HAVE_LANE_TRACKING() | ||
120 | } | ||
121 | |||
122 | |||
102 | /* Do you want full call stacks, or just the line where the error happened? | 123 | /* Do you want full call stacks, or just the line where the error happened? |
103 | * | 124 | * |
104 | * TBD: The full stack feature does not seem to work (try 'make error'). | 125 | * TBD: The full stack feature does not seem to work (try 'make error'). |
@@ -228,27 +249,22 @@ static bool tracking_remove(Lane* lane_) | |||
228 | 249 | ||
229 | // ################################################################################################# | 250 | // ################################################################################################# |
230 | 251 | ||
231 | //--- | 252 | Lane::~Lane() |
232 | // low-level cleanup | ||
233 | |||
234 | static void lane_cleanup(Lane* lane_) | ||
235 | { | 253 | { |
236 | // Clean up after a (finished) thread | 254 | // Clean up after a (finished) thread |
237 | // | 255 | // |
238 | #if THREADWAIT_METHOD == THREADWAIT_CONDVAR | 256 | #if THREADWAIT_METHOD == THREADWAIT_CONDVAR |
239 | SIGNAL_FREE(&lane_->done_signal); | 257 | SIGNAL_FREE(&done_signal); |
240 | MUTEX_FREE(&lane_->done_lock); | 258 | MUTEX_FREE(&done_lock); |
241 | #endif // THREADWAIT_METHOD == THREADWAIT_CONDVAR | 259 | #endif // THREADWAIT_METHOD == THREADWAIT_CONDVAR |
242 | 260 | ||
243 | #if HAVE_LANE_TRACKING() | 261 | #if HAVE_LANE_TRACKING() |
244 | if (lane_->U->tracking_first != nullptr) | 262 | if (U->tracking_first != nullptr) |
245 | { | 263 | { |
246 | // Lane was cleaned up, no need to handle at process termination | 264 | // Lane was cleaned up, no need to handle at process termination |
247 | tracking_remove(lane_); | 265 | tracking_remove(this); |
248 | } | 266 | } |
249 | #endif // HAVE_LANE_TRACKING() | 267 | #endif // HAVE_LANE_TRACKING() |
250 | |||
251 | lane_->U->internal_allocator.free(lane_, sizeof(Lane)); | ||
252 | } | 268 | } |
253 | 269 | ||
254 | /* | 270 | /* |
@@ -536,7 +552,7 @@ static int selfdestruct_gc( lua_State* L) | |||
536 | #endif // THREADAPI == THREADAPI_PTHREAD | 552 | #endif // THREADAPI == THREADAPI_PTHREAD |
537 | } | 553 | } |
538 | // NO lua_close() in this case because we don't know where execution of the state was interrupted | 554 | // NO lua_close() in this case because we don't know where execution of the state was interrupted |
539 | lane_cleanup(lane); | 555 | delete lane; |
540 | lane = next_s; | 556 | lane = next_s; |
541 | ++n; | 557 | ++n; |
542 | } | 558 | } |
@@ -565,7 +581,7 @@ static int selfdestruct_gc( lua_State* L) | |||
565 | U->timer_deep = nullptr; | 581 | U->timer_deep = nullptr; |
566 | } | 582 | } |
567 | 583 | ||
568 | close_keepers( U); | 584 | close_keepers(U); |
569 | 585 | ||
570 | // remove the protected allocator, if any | 586 | // remove the protected allocator, if any |
571 | U->protected_allocator.removeFrom(L); | 587 | U->protected_allocator.removeFrom(L); |
@@ -919,7 +935,7 @@ static THREAD_RETURN_T THREAD_CALLCONV lane_main(void* vs) | |||
919 | // STACK_DUMP(L); | 935 | // STACK_DUMP(L); |
920 | // Call finalizers, if the script has set them up. | 936 | // Call finalizers, if the script has set them up. |
921 | // | 937 | // |
922 | int rc2 = run_finalizers(L, rc); | 938 | int rc2{ run_finalizers(L, rc) }; |
923 | DEBUGSPEW_CODE(fprintf(stderr, INDENT_BEGIN "Lane %p finalizer: %s\n" INDENT_END, L, get_errcode_name(rc2))); | 939 | DEBUGSPEW_CODE(fprintf(stderr, INDENT_BEGIN "Lane %p finalizer: %s\n" INDENT_END, L, get_errcode_name(rc2))); |
924 | if (rc2 != LUA_OK) // Error within a finalizer! | 940 | if (rc2 != LUA_OK) // Error within a finalizer! |
925 | { | 941 | { |
@@ -938,7 +954,7 @@ static THREAD_RETURN_T THREAD_CALLCONV lane_main(void* vs) | |||
938 | --lane->U->selfdestructing_count; | 954 | --lane->U->selfdestructing_count; |
939 | lane->U->selfdestruct_cs.unlock(); | 955 | lane->U->selfdestruct_cs.unlock(); |
940 | 956 | ||
941 | lane_cleanup(lane); // s is freed at this point | 957 | delete lane; |
942 | } | 958 | } |
943 | else | 959 | else |
944 | { | 960 | { |
@@ -1207,35 +1223,14 @@ LUAG_FUNC( lane_new) | |||
1207 | STACK_CHECK( L2, 1 + nargs); | 1223 | STACK_CHECK( L2, 1 + nargs); |
1208 | 1224 | ||
1209 | // 'lane' is allocated from heap, not Lua, since its life span may surpass the handle's (if free running thread) | 1225 | // 'lane' is allocated from heap, not Lua, since its life span may surpass the handle's (if free running thread) |
1210 | // | 1226 | Lane* const lane{ new (U) Lane{ U, L2 } }; |
1211 | // a Lane full userdata needs a single uservalue | ||
1212 | Lane** const ud{ lua_newuserdatauv<Lane*>(L, 1) }; // func libs priority globals package required gc_cb lane | ||
1213 | Lane* const lane{ *ud = static_cast<Lane*>(U->internal_allocator.alloc(sizeof(Lane))) }; // don't forget to store the pointer in the userdata! | ||
1214 | if (lane == nullptr) | 1227 | if (lane == nullptr) |
1215 | { | 1228 | { |
1216 | return luaL_error(L, "could not create lane: out of memory"); | 1229 | return luaL_error(L, "could not create lane: out of memory"); |
1217 | } | 1230 | } |
1218 | 1231 | // a Lane full userdata needs a single uservalue | |
1219 | lane->L = L2; | 1232 | Lane** const ud{ lua_newuserdatauv<Lane*>(L, 1) }; // func libs priority globals package required gc_cb lane |
1220 | lane->U = U; | 1233 | *ud = lane; // don't forget to store the pointer in the userdata! |
1221 | lane->status = PENDING; | ||
1222 | lane->waiting_on = nullptr; | ||
1223 | lane->debug_name = "<unnamed>"; | ||
1224 | lane->cancel_request = CancelRequest::None; | ||
1225 | |||
1226 | #if THREADWAIT_METHOD == THREADWAIT_CONDVAR | ||
1227 | MUTEX_INIT(&lane->done_lock); | ||
1228 | SIGNAL_INIT(&lane->done_signal); | ||
1229 | #endif // THREADWAIT_METHOD == THREADWAIT_CONDVAR | ||
1230 | lane->mstatus = ThreadStatus::Normal; | ||
1231 | lane->selfdestruct_next = nullptr; | ||
1232 | #if HAVE_LANE_TRACKING() | ||
1233 | lane->tracking_next = nullptr; | ||
1234 | if (lane->U->tracking_first) | ||
1235 | { | ||
1236 | tracking_add(lane); | ||
1237 | } | ||
1238 | #endif // HAVE_LANE_TRACKING() | ||
1239 | 1234 | ||
1240 | // Set metatable for the userdata | 1235 | // Set metatable for the userdata |
1241 | // | 1236 | // |
@@ -1245,7 +1240,7 @@ LUAG_FUNC( lane_new) | |||
1245 | 1240 | ||
1246 | // Create uservalue for the userdata | 1241 | // Create uservalue for the userdata |
1247 | // (this is where lane body return values will be stored when the handle is indexed by a numeric key) | 1242 | // (this is where lane body return values will be stored when the handle is indexed by a numeric key) |
1248 | lua_newtable( L); // func libs cancelstep priority globals package required gc_cb lane uv | 1243 | lua_newtable(L); // func libs cancelstep priority globals package required gc_cb lane uv |
1249 | 1244 | ||
1250 | // Store the gc_cb callback in the uservalue | 1245 | // Store the gc_cb callback in the uservalue |
1251 | if (gc_cb_idx > 0) | 1246 | if (gc_cb_idx > 0) |
@@ -1263,6 +1258,7 @@ LUAG_FUNC( lane_new) | |||
1263 | STACK_CHECK(L, 1); | 1258 | STACK_CHECK(L, 1); |
1264 | STACK_CHECK( L2, 1 + nargs); | 1259 | STACK_CHECK( L2, 1 + nargs); |
1265 | 1260 | ||
1261 | // TODO: launch thread earlier, sync with a std::latch to parallelize OS thread warmup and L2 preparation | ||
1266 | DEBUGSPEW_CODE( fprintf( stderr, INDENT_BEGIN "lane_new: launching thread\n" INDENT_END)); | 1262 | DEBUGSPEW_CODE( fprintf( stderr, INDENT_BEGIN "lane_new: launching thread\n" INDENT_END)); |
1267 | THREAD_CREATE(&lane->thread, lane_main, lane, priority); | 1263 | THREAD_CREATE(&lane->thread, lane_main, lane, priority); |
1268 | 1264 | ||
@@ -1306,7 +1302,7 @@ LUAG_FUNC(thread_gc) | |||
1306 | 1302 | ||
1307 | // We can read 'lane->status' without locks, but not wait for it | 1303 | // We can read 'lane->status' without locks, but not wait for it |
1308 | // test Killed state first, as it doesn't need to enter the selfdestruct chain | 1304 | // test Killed state first, as it doesn't need to enter the selfdestruct chain |
1309 | if (lane->mstatus == ThreadStatus::Killed) | 1305 | if (lane->mstatus == Lane::Killed) |
1310 | { | 1306 | { |
1311 | // Make sure a kill has proceeded, before cleaning up the data structure. | 1307 | // Make sure a kill has proceeded, before cleaning up the data structure. |
1312 | // | 1308 | // |
@@ -1350,7 +1346,7 @@ LUAG_FUNC(thread_gc) | |||
1350 | } | 1346 | } |
1351 | 1347 | ||
1352 | // Clean up after a (finished) thread | 1348 | // Clean up after a (finished) thread |
1353 | lane_cleanup(lane); | 1349 | delete lane; |
1354 | 1350 | ||
1355 | // do this after lane cleanup in case the callback triggers an error | 1351 | // do this after lane cleanup in case the callback triggers an error |
1356 | if (have_gc_cb) | 1352 | if (have_gc_cb) |
@@ -1377,7 +1373,7 @@ static char const * thread_status_string(Lane* lane_) | |||
1377 | { | 1373 | { |
1378 | enum e_status const st{ lane_->status }; // read just once (volatile) | 1374 | enum e_status const st{ lane_->status }; // read just once (volatile) |
1379 | char const* str = | 1375 | char const* str = |
1380 | (lane_->mstatus == ThreadStatus::Killed) ? "killed" : // new to v3.3.0! | 1376 | (lane_->mstatus == Lane::Killed) ? "killed" : // new to v3.3.0! |
1381 | (st == PENDING) ? "pending" : | 1377 | (st == PENDING) ? "pending" : |
1382 | (st == RUNNING) ? "running" : // like in 'co.status()' | 1378 | (st == RUNNING) ? "running" : // like in 'co.status()' |
1383 | (st == WAITING) ? "waiting" : | 1379 | (st == WAITING) ? "waiting" : |
@@ -1413,7 +1409,6 @@ LUAG_FUNC(thread_join) | |||
1413 | Lane* const lane{ lua_toLane(L, 1) }; | 1409 | Lane* const lane{ lua_toLane(L, 1) }; |
1414 | lua_Number const wait_secs{ luaL_optnumber(L, 2, -1.0) }; | 1410 | lua_Number const wait_secs{ luaL_optnumber(L, 2, -1.0) }; |
1415 | lua_State* const L2{ lane->L }; | 1411 | lua_State* const L2{ lane->L }; |
1416 | int ret; | ||
1417 | bool const done{ THREAD_ISNULL(lane->thread) || THREAD_WAIT(&lane->thread, wait_secs, &lane->done_signal, &lane->done_lock, &lane->status) }; | 1412 | bool const done{ THREAD_ISNULL(lane->thread) || THREAD_WAIT(&lane->thread, wait_secs, &lane->done_signal, &lane->done_lock, &lane->status) }; |
1418 | if (!done || !L2) | 1413 | if (!done || !L2) |
1419 | { | 1414 | { |
@@ -1426,7 +1421,8 @@ LUAG_FUNC(thread_join) | |||
1426 | STACK_CHECK_START_REL(L, 0); | 1421 | STACK_CHECK_START_REL(L, 0); |
1427 | // Thread is DONE/ERROR_ST/CANCELLED; all ours now | 1422 | // Thread is DONE/ERROR_ST/CANCELLED; all ours now |
1428 | 1423 | ||
1429 | if (lane->mstatus == ThreadStatus::Killed) // OS thread was killed if thread_cancel was forced | 1424 | int ret{ 0 }; |
1425 | if (lane->mstatus == Lane::Killed) // OS thread was killed if thread_cancel was forced | ||
1430 | { | 1426 | { |
1431 | // in that case, even if the thread was killed while DONE/ERROR_ST/CANCELLED, ignore regular return values | 1427 | // in that case, even if the thread was killed while DONE/ERROR_ST/CANCELLED, ignore regular return values |
1432 | STACK_GROW(L, 2); | 1428 | STACK_GROW(L, 2); |
@@ -1536,7 +1532,7 @@ LUAG_FUNC(thread_index) | |||
1536 | switch (lane->status) | 1532 | switch (lane->status) |
1537 | { | 1533 | { |
1538 | default: | 1534 | default: |
1539 | if (lane->mstatus != ThreadStatus::Killed) | 1535 | if (lane->mstatus != Lane::Killed) |
1540 | { | 1536 | { |
1541 | // this is an internal error, we probably never get here | 1537 | // this is an internal error, we probably never get here |
1542 | lua_settop(L, 0); | 1538 | lua_settop(L, 0); |
diff --git a/src/lanes_private.h b/src/lanes_private.h index 1530b9e..7c52876 100644 --- a/src/lanes_private.h +++ b/src/lanes_private.h | |||
@@ -4,40 +4,46 @@ | |||
4 | #include "cancel.h" | 4 | #include "cancel.h" |
5 | #include "universe.h" | 5 | #include "universe.h" |
6 | 6 | ||
7 | enum class ThreadStatus | ||
8 | { | ||
9 | Normal, // normal master side state | ||
10 | Killed // issued an OS kill | ||
11 | }; | ||
12 | |||
13 | // NOTE: values to be changed by either thread, during execution, without | 7 | // NOTE: values to be changed by either thread, during execution, without |
14 | // locking, are marked "volatile" | 8 | // locking, are marked "volatile" |
15 | // | 9 | // |
16 | struct Lane | 10 | class Lane |
17 | { | 11 | { |
12 | private: | ||
13 | |||
14 | enum class ThreadStatus | ||
15 | { | ||
16 | Normal, // normal master side state | ||
17 | Killed // issued an OS kill | ||
18 | }; | ||
19 | |||
20 | public: | ||
21 | |||
22 | using enum ThreadStatus; | ||
23 | |||
18 | THREAD_T thread; | 24 | THREAD_T thread; |
19 | // | 25 | // |
20 | // M: sub-thread OS thread | 26 | // M: sub-thread OS thread |
21 | // S: not used | 27 | // S: not used |
22 | 28 | ||
23 | char const* debug_name; | 29 | char const* debug_name{ "<unnamed>" }; |
24 | 30 | ||
31 | Universe* const U; | ||
25 | lua_State* L; | 32 | lua_State* L; |
26 | Universe* U; | ||
27 | // | 33 | // |
28 | // M: prepares the state, and reads results | 34 | // M: prepares the state, and reads results |
29 | // S: while S is running, M must keep out of modifying the state | 35 | // S: while S is running, M must keep out of modifying the state |
30 | 36 | ||
31 | volatile enum e_status status; | 37 | volatile enum e_status status{ PENDING }; |
32 | // | 38 | // |
33 | // M: sets to PENDING (before launching) | 39 | // M: sets to PENDING (before launching) |
34 | // S: updates -> RUNNING/WAITING -> DONE/ERROR_ST/CANCELLED | 40 | // S: updates -> RUNNING/WAITING -> DONE/ERROR_ST/CANCELLED |
35 | 41 | ||
36 | SIGNAL_T* volatile waiting_on; | 42 | SIGNAL_T* volatile waiting_on{ nullptr }; |
37 | // | 43 | // |
38 | // When status is WAITING, points on the linda's signal the thread waits on, else nullptr | 44 | // When status is WAITING, points on the linda's signal the thread waits on, else nullptr |
39 | 45 | ||
40 | volatile CancelRequest cancel_request; | 46 | volatile CancelRequest cancel_request{ CancelRequest::None }; |
41 | // | 47 | // |
42 | // M: sets to false, flags true for cancel request | 48 | // M: sets to false, flags true for cancel request |
43 | // S: reads to see if cancel is requested | 49 | // S: reads to see if cancel is requested |
@@ -54,22 +60,31 @@ struct Lane | |||
54 | // lane status changes to DONE/ERROR_ST/CANCELLED. | 60 | // lane status changes to DONE/ERROR_ST/CANCELLED. |
55 | #endif // THREADWAIT_METHOD == THREADWAIT_CONDVAR | 61 | #endif // THREADWAIT_METHOD == THREADWAIT_CONDVAR |
56 | 62 | ||
57 | volatile ThreadStatus mstatus; | 63 | volatile ThreadStatus mstatus{ Normal }; |
58 | // | 64 | // |
59 | // M: sets to Normal, if issued a kill changes to Killed | 65 | // M: sets to Normal, if issued a kill changes to Killed |
60 | // S: not used | 66 | // S: not used |
61 | 67 | ||
62 | Lane* volatile selfdestruct_next; | 68 | Lane* volatile selfdestruct_next{ nullptr }; |
63 | // | 69 | // |
64 | // M: sets to non-nullptr if facing lane handle '__gc' cycle but the lane | 70 | // M: sets to non-nullptr if facing lane handle '__gc' cycle but the lane |
65 | // is still running | 71 | // is still running |
66 | // S: cleans up after itself if non-nullptr at lane exit | 72 | // S: cleans up after itself if non-nullptr at lane exit |
67 | 73 | ||
68 | #if HAVE_LANE_TRACKING() | 74 | #if HAVE_LANE_TRACKING() |
69 | Lane* volatile tracking_next; | 75 | Lane* volatile tracking_next{ nullptr }; |
70 | #endif // HAVE_LANE_TRACKING() | 76 | #endif // HAVE_LANE_TRACKING() |
71 | // | 77 | // |
72 | // For tracking only | 78 | // For tracking only |
79 | |||
80 | static void* operator new(size_t size_, Universe* U_) noexcept { return U_->internal_allocator.alloc(size_); } | ||
81 | // can't actually delete the operator because the compiler generates stack unwinding code that could call it in case of exception | ||
82 | static void operator delete(void* p_, Universe* U_) { U_->internal_allocator.free(p_, sizeof(Lane)); } | ||
83 | // this one is for us, to make sure memory is freed by the correct allocator | ||
84 | static void operator delete(void* p_) { static_cast<Lane*>(p_)->U->internal_allocator.free(p_, sizeof(Lane)); } | ||
85 | |||
86 | Lane(Universe* U_, lua_State* L_); | ||
87 | ~Lane(); | ||
73 | }; | 88 | }; |
74 | 89 | ||
75 | // xxh64 of string "LANE_POINTER_REGKEY" generated at https://www.pelock.com/products/hash-calculator | 90 | // xxh64 of string "LANE_POINTER_REGKEY" generated at https://www.pelock.com/products/hash-calculator |
diff --git a/src/universe.h b/src/universe.h index 1079915..a2ad5f5 100644 --- a/src/universe.h +++ b/src/universe.h | |||
@@ -16,7 +16,7 @@ extern "C" { | |||
16 | // forwards | 16 | // forwards |
17 | struct DeepPrelude; | 17 | struct DeepPrelude; |
18 | struct Keepers; | 18 | struct Keepers; |
19 | struct Lane; | 19 | class Lane; |
20 | 20 | ||
21 | // ################################################################################################ | 21 | // ################################################################################################ |
22 | 22 | ||