Files
firmware/src/mesh/TypedQueue.h

114 lines
3.0 KiB
C
Raw Normal View History

#pragma once
2020-02-02 09:59:00 -08:00
#include <cassert>
#include <type_traits>
2020-10-10 09:57:57 +08:00
#include "concurrency/OSThread.h"
#include "freertosinc.h"
2020-02-02 09:59:00 -08:00
2020-06-16 17:01:50 -07:00
#ifdef HAS_FREE_RTOS
2020-02-02 09:59:00 -08:00
/**
* A wrapper for freertos queues. Note: each element object should be small
* and POD (Plain Old Data type) as elements are memcpied by value.
2020-02-02 09:59:00 -08:00
*/
template <class T> class TypedQueue
{
static_assert(std::is_pod<T>::value, "T must be pod");
2020-02-02 09:59:00 -08:00
QueueHandle_t h;
2020-10-10 09:57:57 +08:00
concurrency::OSThread *reader = NULL;
public:
TypedQueue(int maxElements)
{
2020-02-02 09:59:00 -08:00
h = xQueueCreate(maxElements, sizeof(T));
assert(h);
}
~TypedQueue() { vQueueDelete(h); }
2020-02-02 09:59:00 -08:00
int numFree() { return uxQueueSpacesAvailable(h); }
2020-02-06 21:26:40 -08:00
bool isEmpty() { return uxQueueMessagesWaiting(h) == 0; }
2020-10-10 09:57:57 +08:00
bool enqueue(T x, TickType_t maxWait = portMAX_DELAY)
{
if (reader) {
reader->setInterval(0);
concurrency::mainDelay.interrupt();
}
return xQueueSendToBack(h, &x, maxWait) == pdTRUE;
}
2020-02-02 09:59:00 -08:00
2020-10-10 09:57:57 +08:00
bool enqueueFromISR(T x, BaseType_t *higherPriWoken)
{
if (reader) {
reader->setInterval(0);
concurrency::mainDelay.interruptFromISR(higherPriWoken);
}
return xQueueSendToBackFromISR(h, &x, higherPriWoken) == pdTRUE;
}
bool dequeue(T *p, TickType_t maxWait = portMAX_DELAY) { return xQueueReceive(h, p, maxWait) == pdTRUE; }
bool dequeueFromISR(T *p, BaseType_t *higherPriWoken) { return xQueueReceiveFromISR(h, p, higherPriWoken); }
2020-10-10 09:57:57 +08:00
/**
* Set a thread that is reading from this queue
* If a message is pushed to this queue that thread will be scheduled to run ASAP.
*
* Note: thread will not be automatically enabled, just have its interval set to 0
*/
void setReader(concurrency::OSThread *t) { reader = t; }
2020-02-02 09:59:00 -08:00
};
2020-06-16 17:01:50 -07:00
#else
#include <queue>
/**
* A wrapper for freertos queues. Note: each element object should be small
* and POD (Plain Old Data type) as elements are memcpied by value.
*/
template <class T> class TypedQueue
{
std::queue<T> q;
2020-10-11 09:18:47 +08:00
concurrency::OSThread *reader = NULL;
2020-06-16 17:01:50 -07:00
public:
TypedQueue(int maxElements) {}
int numFree() { return 1; } // Always claim 1 free, because we can grow to any size
2020-06-16 17:01:50 -07:00
bool isEmpty() { return q.empty(); }
bool enqueue(T x, TickType_t maxWait = portMAX_DELAY)
{
2020-10-11 09:18:47 +08:00
if (reader) {
reader->setInterval(0);
concurrency::mainDelay.interrupt();
}
2020-06-16 17:01:50 -07:00
q.push(x);
return true;
}
// bool enqueueFromISR(T x, BaseType_t *higherPriWoken) { return xQueueSendToBackFromISR(h, &x, higherPriWoken) == pdTRUE; }
bool dequeue(T *p, TickType_t maxWait = portMAX_DELAY)
{
if (isEmpty())
return false;
else {
*p = q.front();
q.pop();
return true;
}
}
// bool dequeueFromISR(T *p, BaseType_t *higherPriWoken) { return xQueueReceiveFromISR(h, p, higherPriWoken); }
2020-10-11 09:18:47 +08:00
void setReader(concurrency::OSThread *t) { reader = t; }
2020-06-16 17:01:50 -07:00
};
#endif