mirror of
https://github.com/iriselia/xgmsv.git
synced 2025-04-06 01:48:27 +08:00
154 lines
4.0 KiB
C++
154 lines
4.0 KiB
C++
// Example code for blog post 'Understanding Awaitables'
|
|
//
|
|
// Copyright (c) Lewis Baker
|
|
|
|
#include <experimental/coroutine>
|
|
#include <atomic>
|
|
|
|
class async_manual_reset_event
|
|
{
|
|
public:
|
|
|
|
async_manual_reset_event(bool initiallySet = false) noexcept;
|
|
|
|
// No copying/moving
|
|
async_manual_reset_event(const async_manual_reset_event&) = delete;
|
|
async_manual_reset_event(async_manual_reset_event&&) = delete;
|
|
async_manual_reset_event& operator=(const async_manual_reset_event&) = delete;
|
|
async_manual_reset_event& operator=(async_manual_reset_event&&) = delete;
|
|
|
|
bool is_set() const noexcept;
|
|
|
|
struct awaiter;
|
|
awaiter operator co_await() const noexcept;
|
|
|
|
void set() noexcept;
|
|
void reset() noexcept;
|
|
|
|
private:
|
|
|
|
friend struct awaiter;
|
|
|
|
// - 'this' => set state
|
|
// - otherwise => not set, head of linked list of awaiter*.
|
|
mutable std::atomic<void*> m_state;
|
|
|
|
};
|
|
|
|
struct async_manual_reset_event::awaiter
|
|
{
|
|
awaiter(const async_manual_reset_event& event) noexcept
|
|
: m_event(event)
|
|
{}
|
|
|
|
bool await_ready() const noexcept;
|
|
bool await_suspend(std::experimental::coroutine_handle<> awaitingCoroutine) noexcept;
|
|
void await_resume() noexcept {}
|
|
|
|
private:
|
|
friend struct async_manual_reset_event;
|
|
|
|
const async_manual_reset_event& m_event;
|
|
std::experimental::coroutine_handle<> m_awaitingCoroutine;
|
|
awaiter* m_next;
|
|
};
|
|
|
|
bool async_manual_reset_event::awaiter::await_ready() const noexcept
|
|
{
|
|
return m_event.is_set();
|
|
}
|
|
|
|
bool async_manual_reset_event::awaiter::await_suspend(
|
|
std::experimental::coroutine_handle<> awaitingCoroutine) noexcept
|
|
{
|
|
// Special m_state value that indicates the event is in the 'set' state.
|
|
const void* const setState = &m_event;
|
|
|
|
// Stash the handle of the awaiting coroutine.
|
|
m_awaitingCoroutine = awaitingCoroutine;
|
|
|
|
// Try to atomically push this awaiter onto the front of the list.
|
|
void* oldValue = m_event.m_state.load(std::memory_order_acquire);
|
|
do
|
|
{
|
|
// Resume immediately if already in 'set' state.
|
|
if (oldValue == setState) return false;
|
|
|
|
// Update linked list to point at current head.
|
|
m_next = static_cast<awaiter*>(oldValue);
|
|
|
|
// Finally, try to swap the old list head, inserting this awaiter
|
|
// as the new list head.
|
|
} while (!m_event.m_state.compare_exchange_weak(
|
|
oldValue,
|
|
this,
|
|
std::memory_order_release,
|
|
std::memory_order_acquire));
|
|
|
|
// Successfully enqueued. Remain suspended.
|
|
return true;
|
|
}
|
|
|
|
async_manual_reset_event::async_manual_reset_event(
|
|
bool initiallySet) noexcept
|
|
: m_state(initiallySet ? this : nullptr)
|
|
{}
|
|
|
|
bool async_manual_reset_event::is_set() const noexcept
|
|
{
|
|
return m_state.load(std::memory_order_acquire) == this;
|
|
}
|
|
|
|
void async_manual_reset_event::reset() noexcept
|
|
{
|
|
void* oldValue = this;
|
|
m_state.compare_exchange_strong(oldValue, nullptr, std::memory_order_acquire);
|
|
}
|
|
|
|
void async_manual_reset_event::set() noexcept
|
|
{
|
|
// Needs to be 'release' so that subsequent 'co_await' has
|
|
// visibility of our prior writes.
|
|
// Needs to be 'acquire' so that we have visibility of prior
|
|
// writes by awaiting coroutines.
|
|
void* oldValue = m_state.exchange(this, std::memory_order_acq_rel);
|
|
if (oldValue != this)
|
|
{
|
|
// Wasn't already in 'set' state.
|
|
// Treat old value as head of a linked-list of waiters
|
|
// which we have now acquired and need to resume.
|
|
auto* waiters = static_cast<awaiter*>(oldValue);
|
|
while (waiters != nullptr)
|
|
{
|
|
// Read m_next before resuming the coroutine as resuming
|
|
// the coroutine will likely destroy the awaiter object.
|
|
auto* next = waiters->m_next;
|
|
waiters->m_awaitingCoroutine.resume();
|
|
waiters = next;
|
|
}
|
|
}
|
|
}
|
|
|
|
async_manual_reset_event::awaiter
|
|
async_manual_reset_event::operator co_await() const noexcept
|
|
{
|
|
return awaiter{ *this };
|
|
}
|
|
|
|
// A simple task-class for void-returning coroutines.
|
|
struct task
|
|
{
|
|
struct promise_type
|
|
{
|
|
task get_return_object() { return {}; }
|
|
std::experimental::suspend_never initial_suspend() { return {}; }
|
|
std::experimental::suspend_never final_suspend() { return {}; }
|
|
void return_void() {}
|
|
void unhandled_exception() {}
|
|
};
|
|
};
|
|
|
|
task example(async_manual_reset_event& event)
|
|
{
|
|
co_await event;
|
|
} |