Commit 2e3b3413 authored by Kenton Varda's avatar Kenton Varda

Simplify event queuing -- always preempt from same thread, yield cross-thread.

parent 860af726
...@@ -46,7 +46,7 @@ public: ...@@ -46,7 +46,7 @@ public:
while (size > 0) { while (size > 0) {
size_t n = rand() % size + 1; size_t n = rand() % size + 1;
inner.write(buffer, n); inner.write(buffer, n);
usleep(5); usleep(10000);
buffer = reinterpret_cast<const byte*>(buffer) + n; buffer = reinterpret_cast<const byte*>(buffer) + n;
size -= n; size -= n;
} }
......
...@@ -337,6 +337,38 @@ TEST(Async, Ordering) { ...@@ -337,6 +337,38 @@ TEST(Async, Ordering) {
EXPECT_EQ(7, counter); EXPECT_EQ(7, counter);
} }
TEST(Async, Spark) {
// Tests that EventLoop::there() only runs eagerly when queued cross-thread.
SimpleEventLoop loop;
auto notification = newPromiseAndFulfiller<void>();;
Promise<void> unsparked = nullptr;
Promise<void> then = nullptr;
Promise<void> later = nullptr;
// `sparked` will evaluate eagerly, even though we never wait on it, because there() is being
// called from outside the event loop.
Promise<void> sparked = loop.there(Promise<void>(READY_NOW), [&]() {
// `unsparked` will never execute because it's attached to the current loop and we never wait
// on it.
unsparked = loop.there(Promise<void>(READY_NOW), [&]() {
ADD_FAILURE() << "This continuation shouldn't happen because no one waits on it.";
});
// `then` will similarly never execute.
then = Promise<void>(READY_NOW).then([&]() {
ADD_FAILURE() << "This continuation shouldn't happen because no one waits on it.";
});
// `evalLater` *does* eagerly execute even when queued to the same loop.
later = loop.evalLater([&]() {
notification.fulfiller->fulfill();
});
});
loop.wait(kj::mv(notification.promise));
}
TEST(Async, Fork) { TEST(Async, Fork) {
SimpleEventLoop loop; SimpleEventLoop loop;
......
...@@ -53,6 +53,20 @@ public: ...@@ -53,6 +53,20 @@ public:
} }
}; };
class YieldPromiseNode final: public _::PromiseNode {
public:
bool onReady(EventLoop::Event& event) noexcept override {
event.arm(false);
return false;
}
void get(_::ExceptionOrValue& output) noexcept override {
output.as<_::Void>().value = _::Void();
}
Maybe<const EventLoop&> getSafeEventLoop() noexcept override {
return nullptr;
}
};
} // namespace } // namespace
EventLoop& EventLoop::current() { EventLoop& EventLoop::current() {
...@@ -61,6 +75,10 @@ EventLoop& EventLoop::current() { ...@@ -61,6 +75,10 @@ EventLoop& EventLoop::current() {
return *result; return *result;
} }
bool EventLoop::isCurrent() const {
return threadLocalEventLoop == this;
}
void EventLoop::EventListHead::fire() { void EventLoop::EventListHead::fire() {
KJ_FAIL_ASSERT("Fired event list head."); KJ_FAIL_ASSERT("Fired event list head.");
} }
...@@ -114,6 +132,10 @@ void EventLoop::waitImpl(Own<_::PromiseNode> node, _::ExceptionOrValue& result) ...@@ -114,6 +132,10 @@ void EventLoop::waitImpl(Own<_::PromiseNode> node, _::ExceptionOrValue& result)
node->get(result); node->get(result);
} }
Promise<void> EventLoop::yieldIfSameThread() const {
return Promise<void>(false, kj::heap<YieldPromiseNode>());
}
EventLoop::Event::~Event() noexcept(false) { EventLoop::Event::~Event() noexcept(false) {
if (this != &loop.queue) { if (this != &loop.queue) {
KJ_ASSERT(next == nullptr || std::uncaught_exception(), KJ_ASSERT(next == nullptr || std::uncaught_exception(),
...@@ -122,36 +144,32 @@ EventLoop::Event::~Event() noexcept(false) { ...@@ -122,36 +144,32 @@ EventLoop::Event::~Event() noexcept(false) {
} }
} }
void EventLoop::Event::arm(Schedule schedule) { void EventLoop::Event::arm(bool preemptIfSameThread) {
loop.queue.mutex.lock(_::Mutex::EXCLUSIVE); loop.queue.mutex.lock(_::Mutex::EXCLUSIVE);
KJ_DEFER(loop.queue.mutex.unlock(_::Mutex::EXCLUSIVE)); KJ_DEFER(loop.queue.mutex.unlock(_::Mutex::EXCLUSIVE));
if (next == nullptr) { if (next == nullptr) {
bool queueIsEmpty = loop.queue.next == &loop.queue; bool queueIsEmpty = loop.queue.next == &loop.queue;
switch (schedule) { if (preemptIfSameThread && threadLocalEventLoop == &loop) {
case PREEMPT: // Insert the event into the queue. We put it at the front rather than the back so that
// Insert the event into the queue. We put it at the front rather than the back so that // related events are executed together and so that increasing the granularity of events
// related events are executed together and so that increasing the granularity of events // does not cause your code to "lose priority" compared to simultaneously-running code
// does not cause your code to "lose priority" compared to simultaneously-running code // with less granularity.
// with less granularity. next = loop.insertPoint;
next = loop.insertPoint; prev = next->prev;
prev = next->prev; next->prev = this;
next->prev = this; prev->next = this;
prev->next = this; } else {
break; // Insert the node at the *end* of the queue.
prev = loop.queue.prev;
case YIELD: next = prev->next;
// Insert the node at the *end* of the queue. prev->next = this;
prev = loop.queue.prev; next->prev = this;
next = prev->next;
prev->next = this; if (loop.insertPoint == &loop.queue) {
next->prev = this; loop.insertPoint = this;
}
if (loop.insertPoint == &loop.queue) {
loop.insertPoint = this;
}
break;
} }
if (queueIsEmpty) { if (queueIsEmpty) {
...@@ -284,8 +302,7 @@ public: ...@@ -284,8 +302,7 @@ public:
Task(const Impl& taskSet, Own<_::PromiseNode>&& nodeParam) Task(const Impl& taskSet, Own<_::PromiseNode>&& nodeParam)
: EventLoop::Event(taskSet.loop), taskSet(taskSet), node(kj::mv(nodeParam)) { : EventLoop::Event(taskSet.loop), taskSet(taskSet), node(kj::mv(nodeParam)) {
if (node->onReady(*this)) { if (node->onReady(*this)) {
// TODO(soon): Only yield cross-thread. arm();
arm(EventLoop::Event::YIELD);
} }
} }
...@@ -361,8 +378,7 @@ bool PromiseNode::atomicOnReady(EventLoop::Event*& onReadyEvent, EventLoop::Even ...@@ -361,8 +378,7 @@ bool PromiseNode::atomicOnReady(EventLoop::Event*& onReadyEvent, EventLoop::Even
} }
} }
void PromiseNode::atomicReady(EventLoop::Event*& onReadyEvent, void PromiseNode::atomicReady(EventLoop::Event*& onReadyEvent) {
EventLoop::Event::Schedule schedule) {
// If onReadyEvent is null, atomically set it to _kJ_ALREADY_READY. // If onReadyEvent is null, atomically set it to _kJ_ALREADY_READY.
// Otherwise, arm whatever it points at. // Otherwise, arm whatever it points at.
// Useful for firing events in conjuction with atomicOnReady(). // Useful for firing events in conjuction with atomicOnReady().
...@@ -370,7 +386,7 @@ void PromiseNode::atomicReady(EventLoop::Event*& onReadyEvent, ...@@ -370,7 +386,7 @@ void PromiseNode::atomicReady(EventLoop::Event*& onReadyEvent,
EventLoop::Event* oldEvent = nullptr; EventLoop::Event* oldEvent = nullptr;
if (!__atomic_compare_exchange_n(&onReadyEvent, &oldEvent, _kJ_ALREADY_READY, false, if (!__atomic_compare_exchange_n(&onReadyEvent, &oldEvent, _kJ_ALREADY_READY, false,
__ATOMIC_ACQ_REL, __ATOMIC_ACQUIRE)) { __ATOMIC_ACQ_REL, __ATOMIC_ACQUIRE)) {
oldEvent->arm(schedule); oldEvent->arm();
} }
} }
...@@ -439,8 +455,7 @@ ForkBranchBase::~ForkBranchBase() noexcept(false) { ...@@ -439,8 +455,7 @@ ForkBranchBase::~ForkBranchBase() noexcept(false) {
} }
void ForkBranchBase::hubReady() noexcept { void ForkBranchBase::hubReady() noexcept {
// TODO(soon): This should only yield if queuing cross-thread. atomicReady(onReadyEvent);
atomicReady(onReadyEvent, EventLoop::Event::YIELD);
} }
void ForkBranchBase::releaseHub(ExceptionOrValue& output) { void ForkBranchBase::releaseHub(ExceptionOrValue& output) {
...@@ -467,9 +482,7 @@ ForkHubBase::ForkHubBase(const EventLoop& loop, Own<PromiseNode>&& inner, ...@@ -467,9 +482,7 @@ ForkHubBase::ForkHubBase(const EventLoop& loop, Own<PromiseNode>&& inner,
ExceptionOrValue& resultRef) ExceptionOrValue& resultRef)
: EventLoop::Event(loop), inner(kj::mv(inner)), resultRef(resultRef) { : EventLoop::Event(loop), inner(kj::mv(inner)), resultRef(resultRef) {
KJ_DREQUIRE(this->inner->isSafeEventLoop(loop)); KJ_DREQUIRE(this->inner->isSafeEventLoop(loop));
arm();
// TODO(soon): This should only yield if queuing cross-thread.
arm(YIELD);
} }
ForkHubBase::~ForkHubBase() noexcept(false) { ForkHubBase::~ForkHubBase() noexcept(false) {
...@@ -503,10 +516,10 @@ void ForkHubBase::fire() { ...@@ -503,10 +516,10 @@ void ForkHubBase::fire() {
// ------------------------------------------------------------------- // -------------------------------------------------------------------
ChainPromiseNode::ChainPromiseNode(const EventLoop& loop, Own<PromiseNode> inner, Schedule schedule) ChainPromiseNode::ChainPromiseNode(const EventLoop& loop, Own<PromiseNode> inner)
: Event(loop), state(PRE_STEP1), inner(kj::mv(inner)) { : Event(loop), state(PRE_STEP1), inner(kj::mv(inner)) {
KJ_DREQUIRE(this->inner->isSafeEventLoop(loop)); KJ_DREQUIRE(this->inner->isSafeEventLoop(loop));
arm(schedule); arm();
} }
ChainPromiseNode::~ChainPromiseNode() noexcept(false) { ChainPromiseNode::~ChainPromiseNode() noexcept(false) {
...@@ -573,7 +586,7 @@ void ChainPromiseNode::fire() { ...@@ -573,7 +586,7 @@ void ChainPromiseNode::fire() {
if (onReadyEvent != nullptr) { if (onReadyEvent != nullptr) {
if (inner->onReady(*onReadyEvent)) { if (inner->onReady(*onReadyEvent)) {
onReadyEvent->arm(PREEMPT); onReadyEvent->arm();
} }
} }
} }
...@@ -584,12 +597,7 @@ CrossThreadPromiseNodeBase::CrossThreadPromiseNodeBase( ...@@ -584,12 +597,7 @@ CrossThreadPromiseNodeBase::CrossThreadPromiseNodeBase(
const EventLoop& loop, Own<PromiseNode>&& dependency, ExceptionOrValue& resultRef) const EventLoop& loop, Own<PromiseNode>&& dependency, ExceptionOrValue& resultRef)
: Event(loop), dependency(kj::mv(dependency)), resultRef(resultRef) { : Event(loop), dependency(kj::mv(dependency)), resultRef(resultRef) {
KJ_DREQUIRE(this->dependency->isSafeEventLoop(loop)); KJ_DREQUIRE(this->dependency->isSafeEventLoop(loop));
arm();
// The constructor may be called from any thread, so before we can even call onReady() we need
// to switch threads. We yield here so that the event is added to the end of the queue, which
// ensures that multiple events added in sequence are added in order. If we used PREEMPT, events
// we queue cross-thread would end up executing in a non-deterministic order.
arm(YIELD);
} }
CrossThreadPromiseNodeBase::~CrossThreadPromiseNodeBase() noexcept(false) { CrossThreadPromiseNodeBase::~CrossThreadPromiseNodeBase() noexcept(false) {
...@@ -616,7 +624,7 @@ void CrossThreadPromiseNodeBase::fire() { ...@@ -616,7 +624,7 @@ void CrossThreadPromiseNodeBase::fire() {
} }
// If onReadyEvent is null, set it to _kJ_ALREADY_READY. Otherwise, arm it. // If onReadyEvent is null, set it to _kJ_ALREADY_READY. Otherwise, arm it.
PromiseNode::atomicReady(onReadyEvent, YIELD); PromiseNode::atomicReady(onReadyEvent);
} }
} }
......
This diff is collapsed.
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment