QuickCppLib 0.10
Eliminate all the tedious hassle when making state-of-the-art C++ 14 - 23 libraries!
Loading...
Searching...
No Matches
quickcpplib::_xxx::configurable_spinlock::spinlock< T, spinpolicy2, spinpolicy3, spinpolicy4 > Class Template Reference

A non-FIFO policy configurable spin lock meeting the Mutex concept providing the fastest possible spin lock. More...

#include "spinlock.hpp"

Inheritance diagram for quickcpplib::_xxx::configurable_spinlock::spinlock< T, spinpolicy2, spinpolicy3, spinpolicy4 >:

Public Member Functions

constexpr spinlock ()
 
 spinlock (const spinlock &)=delete
 
constexpr spinlock (spinlock &&o) noexcept
 
void lock () noexcept
 
bool lock (T only_if_not_this) noexcept
 Locks if the atomic is not the supplied value, else returning false.
 

Detailed Description

template<typename T, template< class > class spinpolicy2 = spins_to_loop<125>::policy, template< class > class spinpolicy3 = spins_to_yield<250>::policy, template< class > class spinpolicy4 = spins_to_sleep::policy>
class quickcpplib::_xxx::configurable_spinlock::spinlock< T, spinpolicy2, spinpolicy3, spinpolicy4 >

A non-FIFO policy configurable spin lock meeting the Mutex concept providing the fastest possible spin lock.

Template Parameters
TAn integral type capable of atomic usage

sizeof(spinlock<T>) == sizeof(T). Suitable for usage in shared memory.

Meets the requirements of BasicLockable and Lockable. Provides a get() and set() for the type used for the spin lock. Suitable for limited usage in constexpr.

Warning
spinlock<bool> which might seem obvious is usually slower than spinlock<uintptr_t> on most architectures.

So why reinvent the wheel?

  1. Policy configurable spin.
  2. Implemented in pure C++ 11 atomics so the thread sanitiser works as expected.
  3. Multistate locks are possible instead of just 0|1.
  4. I don't much care for doing writes during the spin which a lot of other spinlocks do. It generates an unnecessary amount of cache line invalidation traffic. Better to spin-read and only write when the read suggests you might have a chance.
  5. This spin lock can use a pointer to memory as the spin lock at the cost of some performance. It uses the bottom bit as the locked flag. See locked_ptr<T>.

Constructor & Destructor Documentation

◆ spinlock() [1/3]

template<typename T , template< class > class spinpolicy2 = spins_to_loop<125>::policy, template< class > class spinpolicy3 = spins_to_yield<250>::policy, template< class > class spinpolicy4 = spins_to_sleep::policy>
constexpr quickcpplib::_xxx::configurable_spinlock::spinlock< T, spinpolicy2, spinpolicy3, spinpolicy4 >::spinlock ( )
inlineconstexpr
698{}

◆ spinlock() [2/3]

template<typename T , template< class > class spinpolicy2 = spins_to_loop<125>::policy, template< class > class spinpolicy3 = spins_to_yield<250>::policy, template< class > class spinpolicy4 = spins_to_sleep::policy>
quickcpplib::_xxx::configurable_spinlock::spinlock< T, spinpolicy2, spinpolicy3, spinpolicy4 >::spinlock ( const spinlock< T, spinpolicy2, spinpolicy3, spinpolicy4 > &  )
delete

◆ spinlock() [3/3]

template<typename T , template< class > class spinpolicy2 = spins_to_loop<125>::policy, template< class > class spinpolicy3 = spins_to_yield<250>::policy, template< class > class spinpolicy4 = spins_to_sleep::policy>
constexpr quickcpplib::_xxx::configurable_spinlock::spinlock< T, spinpolicy2, spinpolicy3, spinpolicy4 >::spinlock ( spinlock< T, spinpolicy2, spinpolicy3, spinpolicy4 > &&  o)
inlineconstexprnoexcept
701 : parenttype(std::move(o))
702 {
703 }

Member Function Documentation

◆ lock() [1/2]

template<typename T , template< class > class spinpolicy2 = spins_to_loop<125>::policy, template< class > class spinpolicy3 = spins_to_yield<250>::policy, template< class > class spinpolicy4 = spins_to_sleep::policy>
void quickcpplib::_xxx::configurable_spinlock::spinlock< T, spinpolicy2, spinpolicy3, spinpolicy4 >::lock ( )
inlinenoexcept
705 {
706 for(size_t n = 0;; n++)
707 {
708 if(parenttype::try_lock())
709 return;
710 parenttype::int_yield(n);
711 }
712 }

◆ lock() [2/2]

template<typename T , template< class > class spinpolicy2 = spins_to_loop<125>::policy, template< class > class spinpolicy3 = spins_to_yield<250>::policy, template< class > class spinpolicy4 = spins_to_sleep::policy>
bool quickcpplib::_xxx::configurable_spinlock::spinlock< T, spinpolicy2, spinpolicy3, spinpolicy4 >::lock ( only_if_not_this)
inlinenoexcept

Locks if the atomic is not the supplied value, else returning false.

715 {
716 for(size_t n = 0;; n++)
717 {
718 T expected = 0;
719 if(parenttype::try_lock(expected))
720 return true;
721 if(expected == only_if_not_this)
722 return false;
723 parenttype::int_yield(n);
724 }
725 }

The documentation for this class was generated from the following file: