mbed-os/platform/CircularBuffer.h

217 lines
5.3 KiB
C
Raw Normal View History

/* mbed Microcontroller Library
* Copyright (c) 2015 ARM Limited
2018-11-09 11:31:20 +00:00
* SPDX-License-Identifier: Apache-2.0
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#ifndef MBED_CIRCULARBUFFER_H
#define MBED_CIRCULARBUFFER_H
#include <stdint.h>
#include "platform/mbed_critical.h"
2017-12-21 10:22:11 +00:00
#include "platform/mbed_assert.h"
namespace mbed {
namespace internal {
/* Detect if CounterType of the Circular buffer is of unsigned type. */
template<typename T>
2018-06-27 14:09:15 +00:00
struct is_unsigned {
static const bool value = false;
};
template<>
2018-06-27 14:09:15 +00:00
struct is_unsigned<unsigned char> {
static const bool value = true;
};
template<>
2018-06-27 14:09:15 +00:00
struct is_unsigned<unsigned short> {
static const bool value = true;
};
template<>
2018-06-27 14:09:15 +00:00
struct is_unsigned<unsigned int> {
static const bool value = true;
};
template<>
2018-06-27 14:09:15 +00:00
struct is_unsigned<unsigned long> {
static const bool value = true;
};
template<>
2018-06-27 14:09:15 +00:00
struct is_unsigned<unsigned long long> {
static const bool value = true;
};
};
2016-10-04 20:02:44 +00:00
/** \addtogroup platform */
2017-10-24 15:05:45 +00:00
/** @{*/
/**
* \defgroup platform_CircularBuffer CircularBuffer functions
* @{
*/
/** Templated Circular buffer class
*
* @note Synchronization level: Interrupt safe
* @note CounterType must be unsigned and consistent with BufferSize
*/
template<typename T, uint32_t BufferSize, typename CounterType = uint32_t>
class CircularBuffer {
public:
2018-06-27 14:09:15 +00:00
CircularBuffer() : _head(0), _tail(0), _full(false)
{
MBED_STATIC_ASSERT(
internal::is_unsigned<CounterType>::value,
"CounterType must be unsigned"
);
MBED_STATIC_ASSERT(
(sizeof(CounterType) >= sizeof(uint32_t)) ||
(BufferSize < (((uint64_t) 1) << (sizeof(CounterType) * 8))),
"Invalid BufferSize for the CounterType"
);
}
2018-06-27 14:09:15 +00:00
~CircularBuffer()
{
}
/** Push the transaction to the buffer. This overwrites the buffer if it's
* full
*
* @param data Data to be pushed to the buffer
*/
2018-06-27 14:09:15 +00:00
void push(const T &data)
{
core_util_critical_section_enter();
if (full()) {
_tail++;
if (_tail == BufferSize) {
_tail = 0;
}
}
_pool[_head++] = data;
if (_head == BufferSize) {
2018-10-10 16:29:21 +00:00
_head = 0;
}
if (_head == _tail) {
_full = true;
}
core_util_critical_section_exit();
}
/** Pop the transaction from the buffer
*
* @param data Data to be popped from the buffer
* @return True if the buffer is not empty and data contains a transaction, false otherwise
*/
2018-06-27 14:09:15 +00:00
bool pop(T &data)
{
bool data_popped = false;
core_util_critical_section_enter();
if (!empty()) {
data = _pool[_tail++];
if (_tail == BufferSize) {
_tail = 0;
}
_full = false;
data_popped = true;
}
core_util_critical_section_exit();
return data_popped;
}
/** Check if the buffer is empty
*
* @return True if the buffer is empty, false if not
*/
2018-06-27 14:09:15 +00:00
bool empty() const
{
core_util_critical_section_enter();
bool is_empty = (_head == _tail) && !_full;
core_util_critical_section_exit();
return is_empty;
}
/** Check if the buffer is full
*
* @return True if the buffer is full, false if not
*/
2018-06-27 14:09:15 +00:00
bool full() const
{
core_util_critical_section_enter();
bool full = _full;
core_util_critical_section_exit();
return full;
}
/** Reset the buffer
*
*/
2018-06-27 14:09:15 +00:00
void reset()
{
core_util_critical_section_enter();
_head = 0;
_tail = 0;
_full = false;
core_util_critical_section_exit();
}
/** Get the number of elements currently stored in the circular_buffer */
2018-06-27 14:09:15 +00:00
CounterType size() const
{
core_util_critical_section_enter();
CounterType elements;
2017-10-20 20:06:01 +00:00
if (!_full) {
if (_head < _tail) {
elements = BufferSize + _head - _tail;
2017-10-31 15:07:33 +00:00
} else {
elements = _head - _tail;
2017-10-20 20:06:01 +00:00
}
2017-10-31 15:07:33 +00:00
} else {
elements = BufferSize;
2017-10-20 20:06:01 +00:00
}
core_util_critical_section_exit();
return elements;
}
/** Peek into circular buffer without popping
*
* @param data Data to be peeked from the buffer
* @return True if the buffer is not empty and data contains a transaction, false otherwise
*/
2018-06-27 14:09:15 +00:00
bool peek(T &data) const
{
bool data_updated = false;
core_util_critical_section_enter();
if (!empty()) {
data = _pool[_tail];
data_updated = true;
}
core_util_critical_section_exit();
return data_updated;
}
2018-06-27 14:09:15 +00:00
private:
T _pool[BufferSize];
CounterType _head;
CounterType _tail;
bool _full;
};
2017-10-24 15:05:45 +00:00
/**@}*/
/**@}*/
}
#endif