158 lines
5.7 KiB
C++
158 lines
5.7 KiB
C++
|
|
/*
|
||
|
|
* Distributed under the Boost Software License, Version 1.0.
|
||
|
|
* (See accompanying file LICENSE_1_0.txt or copy at
|
||
|
|
* http://www.boost.org/LICENSE_1_0.txt)
|
||
|
|
*
|
||
|
|
* Copyright (c) 2011 Helge Bahmann
|
||
|
|
* Copyright (c) 2013 Tim Blechmann
|
||
|
|
* Copyright (c) 2014, 2020-2025 Andrey Semashev
|
||
|
|
*/
|
||
|
|
/*!
|
||
|
|
* \file atomic/detail/atomic_flag_impl.hpp
|
||
|
|
*
|
||
|
|
* This header contains implementation of \c atomic_flag.
|
||
|
|
*/
|
||
|
|
|
||
|
|
#ifndef BOOST_ATOMIC_DETAIL_ATOMIC_FLAG_IMPL_HPP_INCLUDED_
|
||
|
|
#define BOOST_ATOMIC_DETAIL_ATOMIC_FLAG_IMPL_HPP_INCLUDED_
|
||
|
|
|
||
|
|
#include <chrono>
|
||
|
|
#include <utility>
|
||
|
|
#include <type_traits>
|
||
|
|
#include <boost/assert.hpp>
|
||
|
|
#include <boost/memory_order.hpp>
|
||
|
|
#include <boost/atomic/wait_result.hpp>
|
||
|
|
#include <boost/atomic/detail/config.hpp>
|
||
|
|
#include <boost/atomic/detail/core_operations.hpp>
|
||
|
|
#include <boost/atomic/detail/wait_operations.hpp>
|
||
|
|
#include <boost/atomic/detail/aligned_variable.hpp>
|
||
|
|
#include <boost/atomic/detail/header.hpp>
|
||
|
|
|
||
|
|
#ifdef BOOST_HAS_PRAGMA_ONCE
|
||
|
|
#pragma once
|
||
|
|
#endif
|
||
|
|
|
||
|
|
/*
|
||
|
|
* IMPLEMENTATION NOTE: All interface functions MUST be declared with BOOST_FORCEINLINE,
|
||
|
|
* see comment for convert_memory_order_to_gcc in gcc_atomic_memory_order_utils.hpp.
|
||
|
|
*/
|
||
|
|
|
||
|
|
namespace boost {
|
||
|
|
namespace atomics {
|
||
|
|
namespace detail {
|
||
|
|
|
||
|
|
#define BOOST_ATOMIC_FLAG_INIT {}
|
||
|
|
|
||
|
|
//! Atomic flag implementation
|
||
|
|
template< bool IsInterprocess >
|
||
|
|
struct atomic_flag_impl
|
||
|
|
{
|
||
|
|
// Prefer 4-byte storage as most platforms support waiting/notifying operations without a lock pool for 32-bit integers
|
||
|
|
using core_operations = atomics::detail::core_operations< 4u, false, IsInterprocess >;
|
||
|
|
using wait_operations = atomics::detail::wait_operations< core_operations >;
|
||
|
|
using storage_type = typename core_operations::storage_type;
|
||
|
|
|
||
|
|
static constexpr bool is_always_lock_free = core_operations::is_always_lock_free;
|
||
|
|
static constexpr bool always_has_native_wait_notify = wait_operations::always_has_native_wait_notify;
|
||
|
|
|
||
|
|
BOOST_ATOMIC_DETAIL_ALIGNED_VAR_TPL(core_operations::storage_alignment, storage_type, m_storage);
|
||
|
|
|
||
|
|
BOOST_FORCEINLINE constexpr atomic_flag_impl() noexcept : m_storage(0u)
|
||
|
|
{
|
||
|
|
}
|
||
|
|
|
||
|
|
atomic_flag_impl(atomic_flag_impl const&) = delete;
|
||
|
|
atomic_flag_impl& operator= (atomic_flag_impl const&) = delete;
|
||
|
|
|
||
|
|
BOOST_FORCEINLINE bool is_lock_free() const volatile noexcept
|
||
|
|
{
|
||
|
|
return is_always_lock_free;
|
||
|
|
}
|
||
|
|
|
||
|
|
BOOST_FORCEINLINE bool has_native_wait_notify() const volatile noexcept
|
||
|
|
{
|
||
|
|
return wait_operations::has_native_wait_notify(m_storage);
|
||
|
|
}
|
||
|
|
|
||
|
|
BOOST_FORCEINLINE bool test(memory_order order = memory_order_seq_cst) const volatile noexcept
|
||
|
|
{
|
||
|
|
BOOST_ASSERT(order != memory_order_release);
|
||
|
|
BOOST_ASSERT(order != memory_order_acq_rel);
|
||
|
|
return !!core_operations::load(m_storage, order);
|
||
|
|
}
|
||
|
|
|
||
|
|
BOOST_FORCEINLINE bool test_and_set(memory_order order = memory_order_seq_cst) volatile noexcept
|
||
|
|
{
|
||
|
|
return core_operations::test_and_set(m_storage, order);
|
||
|
|
}
|
||
|
|
|
||
|
|
BOOST_FORCEINLINE void clear(memory_order order = memory_order_seq_cst) volatile noexcept
|
||
|
|
{
|
||
|
|
BOOST_ASSERT(order != memory_order_consume);
|
||
|
|
BOOST_ASSERT(order != memory_order_acquire);
|
||
|
|
BOOST_ASSERT(order != memory_order_acq_rel);
|
||
|
|
core_operations::clear(m_storage, order);
|
||
|
|
}
|
||
|
|
|
||
|
|
BOOST_FORCEINLINE bool wait(bool old_val, memory_order order = memory_order_seq_cst) const volatile noexcept
|
||
|
|
{
|
||
|
|
BOOST_ASSERT(order != memory_order_release);
|
||
|
|
BOOST_ASSERT(order != memory_order_acq_rel);
|
||
|
|
|
||
|
|
return !!wait_operations::wait(m_storage, static_cast< storage_type >(old_val), order);
|
||
|
|
}
|
||
|
|
|
||
|
|
template< typename Clock, typename Duration >
|
||
|
|
BOOST_FORCEINLINE wait_result< bool >
|
||
|
|
wait_until(bool old_val, std::chrono::time_point< Clock, Duration > timeout, memory_order order = memory_order_seq_cst) const volatile
|
||
|
|
noexcept(noexcept(wait_operations::wait_until(
|
||
|
|
std::declval< storage_type const volatile& >(), std::declval< storage_type >(), timeout, order, std::declval< bool& >())))
|
||
|
|
{
|
||
|
|
BOOST_ASSERT(order != memory_order_release);
|
||
|
|
BOOST_ASSERT(order != memory_order_acq_rel);
|
||
|
|
|
||
|
|
bool timed_out = false;
|
||
|
|
storage_type new_value = wait_operations::wait_until(m_storage, static_cast< storage_type >(old_val), timeout, order, timed_out);
|
||
|
|
return wait_result< bool >(!!new_value, timed_out);
|
||
|
|
}
|
||
|
|
|
||
|
|
template< typename Rep, typename Period >
|
||
|
|
BOOST_FORCEINLINE wait_result< bool >
|
||
|
|
wait_for(bool old_val, std::chrono::duration< Rep, Period > timeout, memory_order order = memory_order_seq_cst) const volatile
|
||
|
|
noexcept(noexcept(wait_operations::wait_for(
|
||
|
|
std::declval< storage_type const volatile& >(), std::declval< storage_type >(), timeout, order, std::declval< bool& >())))
|
||
|
|
{
|
||
|
|
BOOST_ASSERT(order != memory_order_release);
|
||
|
|
BOOST_ASSERT(order != memory_order_acq_rel);
|
||
|
|
|
||
|
|
bool timed_out = false;
|
||
|
|
storage_type new_value = wait_operations::wait_for(m_storage, static_cast< storage_type >(old_val), timeout, order, timed_out);
|
||
|
|
return wait_result< bool >(!!new_value, timed_out);
|
||
|
|
}
|
||
|
|
|
||
|
|
BOOST_FORCEINLINE void notify_one() volatile noexcept
|
||
|
|
{
|
||
|
|
wait_operations::notify_one(m_storage);
|
||
|
|
}
|
||
|
|
|
||
|
|
BOOST_FORCEINLINE void notify_all() volatile noexcept
|
||
|
|
{
|
||
|
|
wait_operations::notify_all(m_storage);
|
||
|
|
}
|
||
|
|
};
|
||
|
|
|
||
|
|
#if defined(BOOST_NO_CXX17_INLINE_VARIABLES)
|
||
|
|
template< bool IsInterprocess >
|
||
|
|
constexpr bool atomic_flag_impl< IsInterprocess >::is_always_lock_free;
|
||
|
|
template< bool IsInterprocess >
|
||
|
|
constexpr bool atomic_flag_impl< IsInterprocess >::always_has_native_wait_notify;
|
||
|
|
#endif
|
||
|
|
|
||
|
|
} // namespace detail
|
||
|
|
} // namespace atomics
|
||
|
|
} // namespace boost
|
||
|
|
|
||
|
|
#include <boost/atomic/detail/footer.hpp>
|
||
|
|
|
||
|
|
#endif // BOOST_ATOMIC_DETAIL_ATOMIC_FLAG_IMPL_HPP_INCLUDED_
|