cppcoro/lib/cancellation_state.hpp
2017-06-11 20:19:31 +09:30

109 lines
3.8 KiB
C++

///////////////////////////////////////////////////////////////////////////////
// Copyright (c) Lewis Baker
// Licenced under MIT license. See LICENSE.txt for details.
///////////////////////////////////////////////////////////////////////////////
#ifndef CPPCORO_CANCELLATION_STATE_HPP_INCLUDED
#define CPPCORO_CANCELLATION_STATE_HPP_INCLUDED
#include <cppcoro/cancellation_token.hpp>
#include <thread>
#include <atomic>
#include <cstdint>
namespace cppcoro
{
namespace detail
{
struct cancellation_registration_state;
class cancellation_state
{
public:
/// Allocates a new cancellation_state object.
///
/// \throw std::bad_alloc
/// If there was insufficient memory to allocate one.
static cancellation_state* create();
~cancellation_state();
/// Increment the reference count of cancellation_token and
/// cancellation_registration objects referencing this state.
void add_token_ref() noexcept;
/// Decrement the reference count of cancellation_token and
/// cancellation_registration objects referencing this state.
void release_token_ref() noexcept;
/// Increment the reference count of cancellation_source objects.
void add_source_ref() noexcept;
/// Decrement the reference count of cancellation_souce objects.
///
/// The cancellation_state will no longer be cancellable once the
/// cancellation_source ref count reaches zero.
void release_source_ref() noexcept;
/// Query if the cancellation_state can have cancellation requested.
///
/// \return
/// Returns true if there are no more references to a cancellation_source
/// object.
bool can_be_cancelled() const noexcept;
/// Query if some thread has called request_cancellation().
bool is_cancellation_requested() const noexcept;
/// Flag state has having cancellation_requested and execute any
/// registered callbacks.
void request_cancellation();
/// Try to register the cancellation_registration as a callback to be executed
/// when cancellation is requested.
///
/// \return
/// true if the callback was successfully registered, false if the callback was
/// not registered because cancellation had already been requested.
///
/// \throw std::bad_alloc
/// If callback was unable to be registered due to insufficient memory.
bool try_register_callback(cancellation_registration* registration);
/// Deregister a callback previously registered successfully in a call to try_register_callback().
///
/// If the callback is currently being executed on another
/// thread that is concurrently calling request_cancellation()
/// then this call will block until the callback has finished executing.
void deregister_callback(cancellation_registration* registration) noexcept;
private:
cancellation_state() noexcept;
bool is_cancellation_notification_complete() const noexcept;
static constexpr std::uint64_t cancellation_requested_flag = 1;
static constexpr std::uint64_t cancellation_notification_complete_flag = 2;
static constexpr std::uint64_t cancellation_source_ref_increment = 4;
static constexpr std::uint64_t cancellation_token_ref_increment = UINT64_C(1) << 33;
static constexpr std::uint64_t can_be_cancelled_mask = cancellation_token_ref_increment - 1;
static constexpr std::uint64_t cancellation_ref_count_mask =
~(cancellation_requested_flag | cancellation_notification_complete_flag);
// A value that has:
// - bit 0 - indicates whether cancellation has been requested.
// - bit 1 - indicates whether cancellation notification is complete.
// - bits 2-32 - ref-count for cancellation_source instances.
// - bits 33-63 - ref-count for cancellation_token/cancellation_registration instances.
std::atomic<std::uint64_t> m_state;
std::atomic<cancellation_registration_state*> m_registrationState;
};
}
}
#endif