From c41575c0d8955dca381d6f9d4c85a213b728fb1f Mon Sep 17 00:00:00 2001 From: Jacek Sieka Date: Tue, 25 Apr 2023 16:34:55 +0200 Subject: [PATCH] disallow reentrancy Reentrancy causes event reordering and stack explosions, in addition to leading to hard-to-debug scenarios. Since none of chronos is actually tested under reentrant conditions (ie that all features such as exceptions, cancellations, buffer operations, timers etc work reliably when the loop is reentered), this is hard to support over time and prevents useful optimizations - this PR simply detects and disallows the behaviour to remove the uncertainty, simplifying reasoning about the event loop in general. --- chronos/internal/asyncengine.nim | 63 +++++++++++++------------------- tests/testbugs.nim | 37 ------------------- tests/testmacro.nim | 11 ++++++ 3 files changed, 36 insertions(+), 75 deletions(-) diff --git a/chronos/internal/asyncengine.nim b/chronos/internal/asyncengine.nim index d794f72b6..b252349f9 100644 --- a/chronos/internal/asyncengine.nim +++ b/chronos/internal/asyncengine.nim @@ -64,20 +64,23 @@ type ticks*: Deque[AsyncCallback] trackers*: Table[string, TrackerBase] counters*: Table[string, TrackerCounter] - -proc sentinelCallbackImpl(arg: pointer) {.gcsafe, noreturn.} = - raiseAssert "Sentinel callback MUST not be scheduled" - -const - SentinelCallback = AsyncCallback(function: sentinelCallbackImpl, - udata: nil) - -proc isSentinel(acb: AsyncCallback): bool = - acb == SentinelCallback + polling*: bool + ## The event loop is currently running proc `<`(a, b: TimerCallback): bool = result = a.finishAt < b.finishAt +template preparePoll(loop: PDispatcherBase) = + # If you hit this assert, you've called `poll`, `runForever` or `waitFor` + # from within an async function which is not supported due to the difficulty + # to control stack depth and event ordering + # If you're using `waitFor`, switch to `await` and / or propagate the + # up the call stack. + doAssert not loop.polling, "The event loop and chronos functions in general are not reentrant" + + loop.polling = true + defer: loop.polling = false + func getAsyncTimestamp*(a: Duration): auto {.inline.} = ## Return rounded up value of duration with milliseconds resolution. ## @@ -142,10 +145,10 @@ template processTicks(loop: untyped) = loop.callbacks.addLast(loop.ticks.popFirst()) template processCallbacks(loop: untyped) = - while true: - let callable = loop.callbacks.popFirst() # len must be > 0 due to sentinel - if isSentinel(callable): - break + # Process existing callbacks but not those that follow, to allow the network + # to regain control regularly + for _ in 0..