Files
espcpputils/lockingqueue.h

72 lines
1.5 KiB
C
Raw Normal View History

2021-01-04 20:11:38 +01:00
#pragma once
// system includes
#include <optional>
// espressif includes
#include <freertos/FreeRTOS.h>
#include <freertos/semphr.h>
// local includes
#include "esputils.h"
#include "delayedconstruction.h"
#include "recursive_mutex_semaphore.h"
#include "recursivelockhelper.h"
namespace espcpputils {
template<typename T>
class LockingQueue
{
public:
void push(const T &val);
void push(T &&val);
std::optional<T> tryPop();
void clear();
std::size_t size() const { return m_size; }
private:
espcpputils::recursive_mutex_semaphore m_lock;
std::vector<T> m_queue;
std::size_t m_size{}; // double-buffered to allow for reading without taking a lock
};
template<typename T>
void LockingQueue<T>::push(const T &val)
{
RecursiveLockHelper helper{m_lock.handle};
m_queue.push_back(val);
m_size = m_queue.size();
}
template<typename T>
void LockingQueue<T>::push(T &&val)
{
RecursiveLockHelper helper{m_lock.handle};
m_queue.emplace_back(std::move(val));
m_size = m_queue.size();
}
template<typename T>
std::optional<T> LockingQueue<T>::tryPop()
{
RecursiveLockHelper helper{m_lock.handle};
if (m_queue.empty())
return std::nullopt;
std::optional<T> temp = std::move(m_queue.front());
m_queue.erase(std::begin(m_queue));
m_size = m_queue.size();
return temp;
}
template<typename T>
void LockingQueue<T>::clear()
{
RecursiveLockHelper helper{m_lock.handle};
m_queue.clear();
}
} // namespace espcpputils