1#ifndef CPP_UTILITIES_MEMORYSAFETY_RWSPINLOCK_HPP
2#define CPP_UTILITIES_MEMORYSAFETY_RWSPINLOCK_HPP
151#include "../Common.h"
165 enum : int32_t { READER = 4, UPGRADED = 2, WRITER = 1 };
175 uint_fast32_t count = 0;
177 if (++count > 1000) {
178 std::this_thread::yield();
185 static_assert(READER > WRITER + UPGRADED,
"wrong bits!");
186 bits_.fetch_and(~(WRITER | UPGRADED), std::memory_order_release);
191 uint_fast32_t count = 0;
193 if (++count > 1000) {
194 std::this_thread::yield();
200 bits_.fetch_add(-READER, std::memory_order_release);
205 bits_.fetch_add(READER, std::memory_order_acquire);
211 uint_fast32_t count = 0;
213 if (++count > 1000) {
214 std::this_thread::yield();
220 bits_.fetch_add(-UPGRADED, std::memory_order_acq_rel);
227 if (++count > 1000) {
228 std::this_thread::yield();
235 bits_.fetch_add(READER - UPGRADED, std::memory_order_acq_rel);
242 bits_.fetch_or(UPGRADED, std::memory_order_acquire);
243 bits_.fetch_add(-WRITER, std::memory_order_release);
249 return bits_.compare_exchange_strong(
250 expect, WRITER, std::memory_order_acq_rel);
265 int32_t value = bits_.fetch_add(READER, std::memory_order_acquire);
266 if (value & (WRITER | UPGRADED)) {
267 bits_.fetch_add(-READER, std::memory_order_release);
275 int32_t expect = UPGRADED;
276 return bits_.compare_exchange_strong(
277 expect, WRITER, std::memory_order_acq_rel);
289 int32_t value = bits_.fetch_or(UPGRADED, std::memory_order_acquire);
291 return ((value & (UPGRADED | WRITER)) == 0);
296 return bits_.load(std::memory_order_acquire);
300 class UpgradedHolder;
320 other.lock_ =
nullptr;
325 upgraded.lock_ =
nullptr;
332 writer.lock_ =
nullptr;
340 swap(lock_, other.lock_);
367 std::swap(lock_, other.lock_);
393 lock_ = writer.lock_;
394 writer.lock_ =
nullptr;
401 other.lock_ =
nullptr;
406 swap(lock_, other.lock_);
434 swap(lock_, other.lock_);
461 lock_ = upgraded.lock_;
462 upgraded.lock_ =
nullptr;
469 other.lock_ =
nullptr;
474 swap(lock_, other.lock_);
502 swap(lock_, other->lock_);
512 std::atomic<int32_t> bits_;
#define UTILITIES_NAMESPACE_END
Define for end namespace declaration, nothing will be generated if UTILITIES_NAMESPACE isn't defined.
Definition: Common.h:92
#define UTILITIES_NAMESPACE_BEGIN
Define for begin namespace declaration, nothing will be generated if UTILITIES_NAMESPACE isn't define...
Definition: Common.h:91
RAII guard for read lock with RWSpinLock::lock_shared() on construction and RWSpinLock::unlock_shared...
Definition: RWSpinLock.hpp:307
~ReadHolder()
Definition: RWSpinLock.hpp:347
void reset(RWSpinLock *lock=nullptr)
Definition: RWSpinLock.hpp:353
ReadHolder(ReadHolder &&other) noexcept
Definition: RWSpinLock.hpp:319
void swap(ReadHolder &other)
Definition: RWSpinLock.hpp:366
ReadHolder(WriteHolder &&writer)
Definition: RWSpinLock.hpp:331
ReadHolder & operator=(ReadHolder &&other)
Definition: RWSpinLock.hpp:338
ReadHolder(RWSpinLock *lock)
Definition: RWSpinLock.hpp:309
ReadHolder(const ReadHolder &other)=delete
ReadHolder(RWSpinLock &lock)
Definition: RWSpinLock.hpp:315
ReadHolder(UpgradedHolder &&upgraded)
down-grade
Definition: RWSpinLock.hpp:324
ReadHolder & operator=(const ReadHolder &other)=delete
RAII guard for upgrade lock with RWSpinLock::lock_upgrade() on construction and RWSpinLock::unlock_up...
Definition: RWSpinLock.hpp:380
void swap(UpgradedHolder &other)
Definition: RWSpinLock.hpp:432
UpgradedHolder & operator=(const UpgradedHolder &other)=delete
~UpgradedHolder()
Definition: RWSpinLock.hpp:413
UpgradedHolder(RWSpinLock &lock)
Definition: RWSpinLock.hpp:388
void reset(RWSpinLock *lock=nullptr)
Definition: RWSpinLock.hpp:419
UpgradedHolder(WriteHolder &&writer)
Definition: RWSpinLock.hpp:392
UpgradedHolder(UpgradedHolder &&other) noexcept
Definition: RWSpinLock.hpp:400
UpgradedHolder(const UpgradedHolder &other)=delete
UpgradedHolder(RWSpinLock *lock)
Definition: RWSpinLock.hpp:382
UpgradedHolder & operator=(UpgradedHolder &&other)
Definition: RWSpinLock.hpp:404
RAII guard for write lock with RWSpinLock::lock() on construction and RWSpinLock::unlock() on destruc...
Definition: RWSpinLock.hpp:447
WriteHolder(const WriteHolder &other)=delete
WriteHolder & operator=(WriteHolder &&other)
Definition: RWSpinLock.hpp:472
void reset(RWSpinLock *lock=nullptr)
Definition: RWSpinLock.hpp:487
void swap(WriteHolder *other)
Definition: RWSpinLock.hpp:500
WriteHolder(RWSpinLock *lock)
Definition: RWSpinLock.hpp:449
WriteHolder & operator=(const WriteHolder &other)=delete
WriteHolder(WriteHolder &&other) noexcept
Definition: RWSpinLock.hpp:468
WriteHolder(RWSpinLock &lock)
Definition: RWSpinLock.hpp:455
WriteHolder(UpgradedHolder &&upgraded)
promoted from an upgrade lock holder
Definition: RWSpinLock.hpp:460
~WriteHolder()
Definition: RWSpinLock.hpp:481
High-performance read-write-spinlock, see RWSpinLock.hpp for details.
Definition: RWSpinLock.hpp:164
void unlock_upgrade()
Definition: RWSpinLock.hpp:219
RWSpinLock & operator=(RWSpinLock const &)=delete
RWSpinLock(RWSpinLock const &)=delete
void lock_upgrade()
UpgradeLockable Concept.
Definition: RWSpinLock.hpp:210
bool try_lock_shared()
Try to get reader permission on the lock. This can fail if we find out someone is a writer or upgrade...
Definition: RWSpinLock.hpp:262
bool try_lock()
Definition: RWSpinLock.hpp:247
constexpr RWSpinLock()
Definition: RWSpinLock.hpp:168
int32_t bits() const
mainly for debugging purposes.
Definition: RWSpinLock.hpp:295
void lock()
Lockable Concept.
Definition: RWSpinLock.hpp:174
void unlock_shared()
Definition: RWSpinLock.hpp:199
void unlock_upgrade_and_lock()
unlock upgrade and try to acquire write lock
Definition: RWSpinLock.hpp:224
void unlock_and_lock_upgrade()
write unlock and upgrade lock atomically
Definition: RWSpinLock.hpp:239
void unlock_and_lock_shared()
Downgrade the lock from writer status to reader status.
Definition: RWSpinLock.hpp:204
bool try_unlock_upgrade_and_lock()
try to unlock upgrade and write lock atomically
Definition: RWSpinLock.hpp:274
bool try_lock_upgrade()
try to acquire an upgradable lock.
Definition: RWSpinLock.hpp:288
void unlock_upgrade_and_lock_shared()
unlock upgrade and read lock atomically
Definition: RWSpinLock.hpp:234
void unlock()
Writer is responsible for clearing up both the UPGRADED and WRITER bits.
Definition: RWSpinLock.hpp:184
void lock_shared()
SharedLockable Concept.
Definition: RWSpinLock.hpp:190
Namespace for all classes, typedefs and functions of memory safety. See Memory Safety for more instru...
Definition: RWSpinLock.hpp:159