UDPConnection/cpp_impl/src/TSQueue.hpp

132 lines
2.8 KiB
C++

#ifndef UDPC_THREADSAFE_QUEUE_HPP
#define UDPC_THREADSAFE_QUEUE_HPP
#define UDPC_TSQUEUE_DEFAULT_CAPACITY 32
#include <cstdlib>
#include <memory>
#include <mutex>
#include <RB/RingBuffer.hpp>
template <typename T>
class TSQueue {
public:
TSQueue(unsigned int capacity = UDPC_TSQUEUE_DEFAULT_CAPACITY);
~TSQueue();
// disable copy
TSQueue(const TSQueue &other) = delete;
TSQueue &operator=(const TSQueue &other) = delete;
// enable move
TSQueue(TSQueue &&other);
TSQueue &operator=(TSQueue &&other);
bool push(const T &data);
T top();
bool pop();
void clear();
void changeCapacity(unsigned int newCapacity);
unsigned int size();
unsigned int capacity();
bool empty();
private:
std::mutex mutex;
RB::RingBuffer<T> rb;
};
template <typename T>
TSQueue<T>::TSQueue(unsigned int capacity) :
mutex(),
rb(capacity)
{
rb.setResizePolicy(false);
}
template <typename T>
TSQueue<T>::TSQueue(TSQueue &&other) :
TSQueue<T>::TSQueue(other.rb.getCapacity())
{
std::lock_guard<std::mutex> lock(other.mutex);
for(unsigned int i = 0; i < other.rb.getSize(); ++i) {
rb.push(other.rb[i]);
}
}
template <typename T>
TSQueue<T>& TSQueue<T>::operator =(TSQueue &&other)
{
std::scoped_lock lock(other.mutex, mutex);
rb.resize(0);
rb.changeCapacity(other.rb.getCapacity());
for(unsigned int i = 0; i < other.rb.getSize(); ++i) {
rb.push(other.rb[i]);
}
}
template <typename T>
TSQueue<T>::~TSQueue()
{}
template <typename T>
bool TSQueue<T>::push(const T &data) {
std::lock_guard<std::mutex> lock(mutex);
if(rb.getSize() == rb.getCapacity()) {
return false;
}
rb.push(data);
return true;
}
template <typename T>
T TSQueue<T>::top() {
std::lock_guard<std::mutex> lock(mutex);
T value = rb.top();
return value;
}
template <typename T>
bool TSQueue<T>::pop() {
std::lock_guard<std::mutex> lock(mutex);
if(rb.empty()) {
return false;
}
rb.pop();
return true;
}
template <typename T>
void TSQueue<T>::clear() {
std::lock_guard<std::mutex> lock(mutex);
rb.resize(0);
}
template <typename T>
void TSQueue<T>::changeCapacity(unsigned int newCapacity) {
std::lock_guard<std::mutex> lock(mutex);
rb.changeCapacity(newCapacity);
}
template <typename T>
unsigned int TSQueue<T>::size() {
std::lock_guard<std::mutex> lock(mutex);
unsigned int size = rb.getSize();
return size;
}
template <typename T>
unsigned int TSQueue<T>::capacity() {
std::lock_guard<std::mutex> lock(mutex);
unsigned int capacity = rb.getCapacity();
return capacity;
}
template <typename T>
bool TSQueue<T>::empty() {
// No lock required, since this is calling size() that uses a lock
unsigned int size = this->size();
return size == 0;
}
#endif