LBPM/threadpool/atomic_list.h
2023-10-22 09:53:45 -04:00

200 lines
5.7 KiB
C++
Raw Permalink Blame History

This file contains invisible Unicode characters

This file contains invisible Unicode characters that are indistinguishable to humans but may be processed differently by a computer. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

#ifndef included_AtomicModelAtomicList
#define included_AtomicModelAtomicList
#include <atomic>
#include <csignal>
#include <functional>
#include "threadpool/atomic_helpers.h"
/** \class AtomicList
*
* \brief Maintain a sorted list of entries
* \details This class implements a basic sorted list that is thread-safe and lock-free.
* Entries are stored smallest to largest according to the compare operator
*/
template<class TYPE, class COMPARE = std::less<TYPE>>
class AtomicList final
{
public:
//! Default constructor
AtomicList( size_t capacity = 1024, const TYPE &default_value = TYPE(),
const COMPARE &comp = COMPARE() );
//! Destructor
~AtomicList();
/*!
* \brief Remove an item from the list
* \details Find and remove first entry that meets the given criteria
* @return Return the item that matches the criteria,
* or the default item if no item matches
* @param compare Comparison function object (i.e. an object that satisfies
* the requirements of Compare) which returns true if the
* given value meets the selection criteria.
* The signature of the comparison function should be equivalent to:
* bool cmp( const TYPE& value, ... );
* @param args Additional arguments for the comparison
*/
template<typename Compare, class... Args>
inline TYPE remove( Compare compare, const Args &... args );
//! Remove the first from the list
inline TYPE remove_first();
/*!
* \brief Insert an item
* \details Insert an item into the list
* @param x Item to insert
*/
inline void insert( const TYPE &x );
/*!
* \brief Return the size of the list
* \details Return the number of items in the list
*/
inline size_t size() const { return AtomicOperations::atomic_get( &d_N ); }
/*!
* \brief Check if the list is empty
* \details Return true if the list is empty
*/
inline bool empty() const { return AtomicOperations::atomic_get( &d_N ) == 0; }
/*!
* \brief Clear the list
* \details Removes all entries from the list
*/
inline void clear()
{
while ( !empty() ) {
remove_first();
}
}
/*!
* \brief Return the capacity of the list
* \details Return the maximum number of items the list can hold
*/
inline constexpr size_t capacity() const { return d_capacity; }
/*!
* \brief Check the list
* \details Perform a series of checks to verify the list is in a stable state.
* Note: This function is only partially thread-safe: it will block all other
* operations on the list, but check may fail if we caught a thread modifing the list.
* It is intended for debugging purposes only!
* @return This function returns true if the list is in a good working state
*/
inline bool check();
//! Return the total number of inserts since object creation
inline size_t N_insert() const { return AtomicOperations::atomic_get( &d_N_insert ); }
//! Return the total number of removals since object creation
inline size_t N_remove() const { return AtomicOperations::atomic_get( &d_N_remove ); }
private:
// Data members
COMPARE d_compare;
const size_t d_capacity;
volatile TYPE d_default;
volatile TYPE *d_objects;
volatile AtomicOperations::int32_atomic d_N;
volatile AtomicOperations::int32_atomic *d_next;
volatile AtomicOperations::int32_atomic d_unused;
volatile AtomicOperations::int64_atomic d_N_insert;
volatile AtomicOperations::int64_atomic d_N_remove;
private:
inline int lock( int i )
{
if ( i == -1 )
return -1;
int tmp = 0;
do {
tmp = AtomicOperations::atomic_fetch_and_and( &d_next[i], 0 );
} while ( tmp == 0 );
return tmp;
}
inline void unlock( int i, int value )
{
if ( i != -1 )
AtomicOperations::atomic_fetch_and_or( &d_next[i], value );
}
inline int get_unused()
{
int i = 0;
do {
i = AtomicOperations::atomic_fetch_and_and( &d_unused, 0 );
} while ( i == 0 );
AtomicOperations::atomic_fetch_and_or( &d_unused, -( d_next[i] + 4 ) + 1 );
d_next[i] = -3;
return i;
}
inline void put_unused( int i )
{
int j = 0;
do {
AtomicOperations::atomic_swap( &d_unused, &j );
} while ( j == 0 );
d_next[i] = -3 - j;
AtomicOperations::atomic_fetch_and_or( &d_unused, i );
}
public:
AtomicList( const AtomicList & ) = delete;
AtomicList &operator=( const AtomicList & ) = delete;
};
/** \class MemoryPool
*
* \brief Pool allocator
* \details This class implements a basic fast pool allocator that is thread-safe.
*/
template<class TYPE, class INT_TYPE = int>
class MemoryPool final
{
public:
//! Default constructor
explicit MemoryPool( size_t size );
//! destructor
~MemoryPool();
/*!
* \brief Allocate an object
* \details Allocates a new object from the pool
* @return Return the new pointer, or nullptr if there is no more room in the pool
*/
inline TYPE *allocate();
/*!
* \brief Insert an item
* \details Insert an item into the list
* @param ptr The pointer to free
*/
inline void free( TYPE *ptr );
private:
// Data members
volatile TYPE *d_objects;
volatile AtomicOperations::int32_atomic d_next;
private:
MemoryPool( const MemoryPool & );
MemoryPool &operator=( const MemoryPool & );
};
#include "threadpool/atomic_list.hpp"
#endif