#ifndef INCLUDED_SRC_BUILDTOOL_MULTITHREADING_ATOMIC_VALUE_HPP #define INCLUDED_SRC_BUILDTOOL_MULTITHREADING_ATOMIC_VALUE_HPP #include #include #include #include "src/utils/cpp/atomic.hpp" // Value that can be set and get atomically. Reset is not thread-safe. template class AtomicValue { public: AtomicValue() noexcept = default; AtomicValue(AtomicValue const& other) noexcept = delete; AtomicValue(AtomicValue&& other) noexcept : data_{other.data_.load()} {} ~AtomicValue() noexcept = default; auto operator=(AtomicValue const& other) noexcept = delete; auto operator=(AtomicValue&& other) noexcept = delete; // Atomically set value once and return its reference. If this method is // called multiple times concurrently, the setter is called only once. In // any case, this method blocks until the value is ready. [[nodiscard]] auto SetOnceAndGet( std::function const& setter) const& noexcept -> T const& { if (data_.load() == nullptr) { if (not load_.exchange(true)) { data_.store(std::make_shared(setter())); data_.notify_all(); } else { data_.wait(nullptr); } } return *data_.load(); } [[nodiscard]] auto SetOnceAndGet(std::function const& setter) && = delete; // Reset, not thread-safe! void Reset() noexcept { load_ = false; data_ = nullptr; } private: mutable std::atomic load_{false}; mutable atomic_shared_ptr data_{nullptr}; }; #endif // INCLUDED_SRC_BUILDTOOL_MULTITHREADING_ATOMIC_VALUE_HPP