Rework cbuf to use FreeRTOS Ringbuffer (#7860)
* Rework cbuf to use FreeRTOS Ringbuffer * Update cbuf.cpp * Fix typo * Initialize with NULL * Implement peek method * Add initializer --------- Co-authored-by: Lucas Saavedra Vaz <lucas.vaz@espressif.com> Co-authored-by: Lucas Saavedra Vaz <32426024+lucasssvaz@users.noreply.github.com>
This commit is contained in:
parent
e1a3525b5a
commit
d59efdcecd
2 changed files with 199 additions and 117 deletions
|
|
@ -19,178 +19,272 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include "cbuf.h"
|
#include "cbuf.h"
|
||||||
|
#include "esp32-hal-log.h"
|
||||||
|
|
||||||
|
#if CONFIG_DISABLE_HAL_LOCKS
|
||||||
|
#define CBUF_MUTEX_CREATE()
|
||||||
|
#define CBUF_MUTEX_LOCK()
|
||||||
|
#define CBUF_MUTEX_UNLOCK()
|
||||||
|
#define CBUF_MUTEX_DELETE()
|
||||||
|
#else
|
||||||
|
#define CBUF_MUTEX_CREATE() if(_lock == NULL){_lock = xSemaphoreCreateMutex(); if(_lock == NULL){log_e("failed to create mutex");}}
|
||||||
|
#define CBUF_MUTEX_LOCK() if(_lock != NULL){xSemaphoreTakeRecursive(_lock, portMAX_DELAY);}
|
||||||
|
#define CBUF_MUTEX_UNLOCK() if(_lock != NULL){xSemaphoreGiveRecursive(_lock);}
|
||||||
|
#define CBUF_MUTEX_DELETE() if(_lock != NULL){SemaphoreHandle_t l = _lock; _lock = NULL; vSemaphoreDelete(l);}
|
||||||
|
#endif
|
||||||
|
|
||||||
cbuf::cbuf(size_t size) :
|
cbuf::cbuf(size_t size) :
|
||||||
next(NULL), _size(size+1), _buf(new char[size+1]), _bufend(_buf + size + 1), _begin(_buf), _end(_begin)
|
next(NULL),
|
||||||
|
has_peek(false),
|
||||||
|
peek_byte(0),
|
||||||
|
_buf(xRingbufferCreate(size, RINGBUF_TYPE_BYTEBUF))
|
||||||
{
|
{
|
||||||
|
if(_buf == NULL) {
|
||||||
|
log_e("failed to allocate ring buffer");
|
||||||
|
}
|
||||||
|
CBUF_MUTEX_CREATE();
|
||||||
}
|
}
|
||||||
|
|
||||||
cbuf::~cbuf()
|
cbuf::~cbuf()
|
||||||
{
|
{
|
||||||
delete[] _buf;
|
CBUF_MUTEX_LOCK();
|
||||||
|
if(_buf != NULL){
|
||||||
|
RingbufHandle_t b = _buf;
|
||||||
|
_buf = NULL;
|
||||||
|
vRingbufferDelete(b);
|
||||||
|
}
|
||||||
|
CBUF_MUTEX_UNLOCK();
|
||||||
|
CBUF_MUTEX_DELETE();
|
||||||
}
|
}
|
||||||
|
|
||||||
size_t cbuf::resizeAdd(size_t addSize)
|
size_t cbuf::resizeAdd(size_t addSize)
|
||||||
{
|
{
|
||||||
return resize(_size + addSize);
|
return resize(size() + addSize);
|
||||||
}
|
}
|
||||||
|
|
||||||
size_t cbuf::resize(size_t newSize)
|
size_t cbuf::resize(size_t newSize)
|
||||||
{
|
{
|
||||||
|
CBUF_MUTEX_LOCK();
|
||||||
|
size_t _size = size();
|
||||||
|
if(newSize == _size) {
|
||||||
|
return _size;
|
||||||
|
}
|
||||||
|
|
||||||
size_t bytes_available = available();
|
|
||||||
newSize += 1;
|
|
||||||
// not lose any data
|
// not lose any data
|
||||||
// if data can be lost use remove or flush before resize
|
// if data can be lost use remove or flush before resize
|
||||||
if((newSize < bytes_available) || (newSize == _size)) {
|
size_t bytes_available = available();
|
||||||
|
if(newSize < bytes_available) {
|
||||||
|
CBUF_MUTEX_UNLOCK();
|
||||||
|
log_e("new size is less than the currently available data size");
|
||||||
return _size;
|
return _size;
|
||||||
}
|
}
|
||||||
|
|
||||||
char *newbuf = new char[newSize];
|
RingbufHandle_t newbuf = xRingbufferCreate(newSize, RINGBUF_TYPE_BYTEBUF);
|
||||||
char *oldbuf = _buf;
|
if(newbuf == NULL) {
|
||||||
|
CBUF_MUTEX_UNLOCK();
|
||||||
if(!newbuf) {
|
log_e("failed to allocate new ring buffer");
|
||||||
return _size;
|
return _size;
|
||||||
}
|
}
|
||||||
|
|
||||||
if(_buf) {
|
if(_buf != NULL) {
|
||||||
read(newbuf, bytes_available);
|
if(bytes_available){
|
||||||
memset((newbuf + bytes_available), 0x00, (newSize - bytes_available));
|
char * old_data = (char *)malloc(bytes_available);
|
||||||
|
if(old_data == NULL){
|
||||||
|
vRingbufferDelete(newbuf);
|
||||||
|
CBUF_MUTEX_UNLOCK();
|
||||||
|
log_e("failed to allocate temporary buffer");
|
||||||
|
return _size;
|
||||||
|
}
|
||||||
|
bytes_available = read(old_data, bytes_available);
|
||||||
|
if(!bytes_available){
|
||||||
|
free(old_data);
|
||||||
|
vRingbufferDelete(newbuf);
|
||||||
|
CBUF_MUTEX_UNLOCK();
|
||||||
|
log_e("failed to read previous data");
|
||||||
|
return _size;
|
||||||
|
}
|
||||||
|
if(xRingbufferSend(newbuf, (void*)old_data, bytes_available, 0) != pdTRUE){
|
||||||
|
write(old_data, bytes_available);
|
||||||
|
free(old_data);
|
||||||
|
vRingbufferDelete(newbuf);
|
||||||
|
CBUF_MUTEX_UNLOCK();
|
||||||
|
log_e("failed to restore previous data");
|
||||||
|
return _size;
|
||||||
|
}
|
||||||
|
free(old_data);
|
||||||
}
|
}
|
||||||
|
|
||||||
_begin = newbuf;
|
RingbufHandle_t b = _buf;
|
||||||
_end = newbuf + bytes_available;
|
|
||||||
_bufend = newbuf + newSize;
|
|
||||||
_size = newSize;
|
|
||||||
|
|
||||||
_buf = newbuf;
|
_buf = newbuf;
|
||||||
delete[] oldbuf;
|
vRingbufferDelete(b);
|
||||||
|
} else {
|
||||||
return _size;
|
_buf = newbuf;
|
||||||
|
}
|
||||||
|
CBUF_MUTEX_UNLOCK();
|
||||||
|
return newSize;
|
||||||
}
|
}
|
||||||
|
|
||||||
size_t cbuf::available() const
|
size_t cbuf::available() const
|
||||||
{
|
{
|
||||||
if(_end >= _begin) {
|
size_t available = 0;
|
||||||
return _end - _begin;
|
if(_buf != NULL){
|
||||||
|
vRingbufferGetInfo(_buf, NULL, NULL, NULL, NULL, (UBaseType_t *)&available);
|
||||||
}
|
}
|
||||||
return _size - (_begin - _end);
|
if (has_peek) available++;
|
||||||
|
return available;
|
||||||
}
|
}
|
||||||
|
|
||||||
size_t cbuf::size()
|
size_t cbuf::size()
|
||||||
{
|
{
|
||||||
|
size_t _size = 0;
|
||||||
|
if(_buf != NULL){
|
||||||
|
_size = xRingbufferGetMaxItemSize(_buf);
|
||||||
|
}
|
||||||
return _size;
|
return _size;
|
||||||
}
|
}
|
||||||
|
|
||||||
size_t cbuf::room() const
|
size_t cbuf::room() const
|
||||||
{
|
{
|
||||||
if(_end >= _begin) {
|
size_t _room = 0;
|
||||||
return _size - (_end - _begin) - 1;
|
if(_buf != NULL){
|
||||||
|
_room = xRingbufferGetCurFreeSize(_buf);
|
||||||
}
|
}
|
||||||
return _begin - _end - 1;
|
return _room;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool cbuf::empty() const
|
||||||
|
{
|
||||||
|
return available() == 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool cbuf::full() const
|
||||||
|
{
|
||||||
|
return room() == 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
int cbuf::peek()
|
int cbuf::peek()
|
||||||
{
|
{
|
||||||
if(empty()) {
|
if (!available()) {
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
return static_cast<int>(*_begin);
|
int c;
|
||||||
}
|
|
||||||
|
|
||||||
size_t cbuf::peek(char *dst, size_t size)
|
CBUF_MUTEX_LOCK();
|
||||||
{
|
if (has_peek) {
|
||||||
size_t bytes_available = available();
|
c = peek_byte;
|
||||||
size_t size_to_read = (size < bytes_available) ? size : bytes_available;
|
} else {
|
||||||
size_t size_read = size_to_read;
|
c = read();
|
||||||
char * begin = _begin;
|
if (c >= 0) {
|
||||||
if(_end < _begin && size_to_read > (size_t) (_bufend - _begin)) {
|
has_peek = true;
|
||||||
size_t top_size = _bufend - _begin;
|
peek_byte = c;
|
||||||
memcpy(dst, _begin, top_size);
|
|
||||||
begin = _buf;
|
|
||||||
size_to_read -= top_size;
|
|
||||||
dst += top_size;
|
|
||||||
}
|
}
|
||||||
memcpy(dst, begin, size_to_read);
|
}
|
||||||
return size_read;
|
CBUF_MUTEX_UNLOCK();
|
||||||
|
return c;
|
||||||
}
|
}
|
||||||
|
|
||||||
int cbuf::read()
|
int cbuf::read()
|
||||||
{
|
{
|
||||||
if(empty()) {
|
char result = 0;
|
||||||
|
if(!read(&result, 1)){
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
char result = *_begin;
|
|
||||||
_begin = wrap_if_bufend(_begin + 1);
|
|
||||||
return static_cast<int>(result);
|
return static_cast<int>(result);
|
||||||
}
|
}
|
||||||
|
|
||||||
size_t cbuf::read(char* dst, size_t size)
|
size_t cbuf::read(char* dst, size_t size)
|
||||||
{
|
{
|
||||||
|
CBUF_MUTEX_LOCK();
|
||||||
size_t bytes_available = available();
|
size_t bytes_available = available();
|
||||||
size_t size_to_read = (size < bytes_available) ? size : bytes_available;
|
if(!bytes_available || !size){
|
||||||
size_t size_read = size_to_read;
|
CBUF_MUTEX_UNLOCK();
|
||||||
if(_end < _begin && size_to_read > (size_t) (_bufend - _begin)) {
|
return 0;
|
||||||
size_t top_size = _bufend - _begin;
|
|
||||||
memcpy(dst, _begin, top_size);
|
|
||||||
_begin = _buf;
|
|
||||||
size_to_read -= top_size;
|
|
||||||
dst += top_size;
|
|
||||||
}
|
}
|
||||||
memcpy(dst, _begin, size_to_read);
|
|
||||||
_begin = wrap_if_bufend(_begin + size_to_read);
|
if (has_peek) {
|
||||||
|
if (dst != NULL) {
|
||||||
|
*dst++ = peek_byte;
|
||||||
|
}
|
||||||
|
size--;
|
||||||
|
}
|
||||||
|
|
||||||
|
size_t size_read = 0;
|
||||||
|
if (size) {
|
||||||
|
size_t received_size = 0;
|
||||||
|
size_t size_to_read = (size < bytes_available) ? size : bytes_available;
|
||||||
|
uint8_t *received_buff = (uint8_t *)xRingbufferReceiveUpTo(_buf, &received_size, 0, size_to_read);
|
||||||
|
if (received_buff != NULL) {
|
||||||
|
if(dst != NULL){
|
||||||
|
memcpy(dst, received_buff, received_size);
|
||||||
|
}
|
||||||
|
vRingbufferReturnItem(_buf, received_buff);
|
||||||
|
size_read = received_size;
|
||||||
|
size_to_read -= received_size;
|
||||||
|
// wrap around data
|
||||||
|
if(size_to_read){
|
||||||
|
received_size = 0;
|
||||||
|
received_buff = (uint8_t *)xRingbufferReceiveUpTo(_buf, &received_size, 0, size_to_read);
|
||||||
|
if (received_buff != NULL) {
|
||||||
|
if(dst != NULL){
|
||||||
|
memcpy(dst+size_read, received_buff, received_size);
|
||||||
|
}
|
||||||
|
vRingbufferReturnItem(_buf, received_buff);
|
||||||
|
size_read += received_size;
|
||||||
|
} else {
|
||||||
|
log_e("failed to read wrap around data from ring buffer");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
log_e("failed to read from ring buffer");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (has_peek) {
|
||||||
|
has_peek = false;
|
||||||
|
size_read++;
|
||||||
|
}
|
||||||
|
|
||||||
|
CBUF_MUTEX_UNLOCK();
|
||||||
return size_read;
|
return size_read;
|
||||||
}
|
}
|
||||||
|
|
||||||
size_t cbuf::write(char c)
|
size_t cbuf::write(char c)
|
||||||
{
|
{
|
||||||
if(full()) {
|
return write(&c, 1);
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
*_end = c;
|
|
||||||
_end = wrap_if_bufend(_end + 1);
|
|
||||||
return 1;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
size_t cbuf::write(const char* src, size_t size)
|
size_t cbuf::write(const char* src, size_t size)
|
||||||
{
|
{
|
||||||
|
CBUF_MUTEX_LOCK();
|
||||||
size_t bytes_available = room();
|
size_t bytes_available = room();
|
||||||
size_t size_to_write = (size < bytes_available) ? size : bytes_available;
|
if(!bytes_available || !size){
|
||||||
size_t size_written = size_to_write;
|
CBUF_MUTEX_UNLOCK();
|
||||||
if(_end >= _begin && size_to_write > (size_t) (_bufend - _end)) {
|
return 0;
|
||||||
size_t top_size = _bufend - _end;
|
|
||||||
memcpy(_end, src, top_size);
|
|
||||||
_end = _buf;
|
|
||||||
size_to_write -= top_size;
|
|
||||||
src += top_size;
|
|
||||||
}
|
}
|
||||||
memcpy(_end, src, size_to_write);
|
size_t size_to_write = (size < bytes_available) ? size : bytes_available;
|
||||||
_end = wrap_if_bufend(_end + size_to_write);
|
if(xRingbufferSend(_buf, (void*)src, size_to_write, 0) != pdTRUE){
|
||||||
return size_written;
|
CBUF_MUTEX_UNLOCK();
|
||||||
|
log_e("failed to write to ring buffer");
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
CBUF_MUTEX_UNLOCK();
|
||||||
|
return size_to_write;
|
||||||
}
|
}
|
||||||
|
|
||||||
void cbuf::flush()
|
void cbuf::flush()
|
||||||
{
|
{
|
||||||
_begin = _buf;
|
read(NULL, available());
|
||||||
_end = _buf;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
size_t cbuf::remove(size_t size)
|
size_t cbuf::remove(size_t size)
|
||||||
{
|
{
|
||||||
|
CBUF_MUTEX_LOCK();
|
||||||
size_t bytes_available = available();
|
size_t bytes_available = available();
|
||||||
if(size >= bytes_available) {
|
if(bytes_available && size){
|
||||||
flush();
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
size_t size_to_remove = (size < bytes_available) ? size : bytes_available;
|
size_t size_to_remove = (size < bytes_available) ? size : bytes_available;
|
||||||
if(_end < _begin && size_to_remove > (size_t) (_bufend - _begin)) {
|
bytes_available -= read(NULL, size_to_remove);
|
||||||
size_t top_size = _bufend - _begin;
|
|
||||||
_begin = _buf;
|
|
||||||
size_to_remove -= top_size;
|
|
||||||
}
|
}
|
||||||
_begin = wrap_if_bufend(_begin + size_to_remove);
|
CBUF_MUTEX_UNLOCK();
|
||||||
return available();
|
return bytes_available;
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -18,12 +18,15 @@
|
||||||
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#ifndef __cbuf_h
|
#pragma once
|
||||||
#define __cbuf_h
|
|
||||||
|
|
||||||
#include <stddef.h>
|
#include <stddef.h>
|
||||||
#include <stdint.h>
|
#include <stdint.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
#include "sdkconfig.h"
|
||||||
|
#include "freertos/FreeRTOS.h"
|
||||||
|
#include "freertos/ringbuf.h"
|
||||||
|
#include "freertos/semphr.h"
|
||||||
|
|
||||||
class cbuf
|
class cbuf
|
||||||
{
|
{
|
||||||
|
|
@ -33,23 +36,14 @@ public:
|
||||||
|
|
||||||
size_t resizeAdd(size_t addSize);
|
size_t resizeAdd(size_t addSize);
|
||||||
size_t resize(size_t newSize);
|
size_t resize(size_t newSize);
|
||||||
|
|
||||||
size_t available() const;
|
size_t available() const;
|
||||||
size_t size();
|
size_t size();
|
||||||
|
|
||||||
size_t room() const;
|
size_t room() const;
|
||||||
|
bool empty() const;
|
||||||
inline bool empty() const
|
bool full() const;
|
||||||
{
|
|
||||||
return _begin == _end;
|
|
||||||
}
|
|
||||||
|
|
||||||
inline bool full() const
|
|
||||||
{
|
|
||||||
return wrap_if_bufend(_end + 1) == _begin;
|
|
||||||
}
|
|
||||||
|
|
||||||
int peek();
|
int peek();
|
||||||
size_t peek(char *dst, size_t size);
|
|
||||||
|
|
||||||
int read();
|
int read();
|
||||||
size_t read(char* dst, size_t size);
|
size_t read(char* dst, size_t size);
|
||||||
|
|
@ -61,19 +55,13 @@ public:
|
||||||
size_t remove(size_t size);
|
size_t remove(size_t size);
|
||||||
|
|
||||||
cbuf *next;
|
cbuf *next;
|
||||||
|
bool has_peek;
|
||||||
|
uint8_t peek_byte;
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
inline char* wrap_if_bufend(char* ptr) const
|
RingbufHandle_t _buf = NULL;
|
||||||
{
|
#if !CONFIG_DISABLE_HAL_LOCKS
|
||||||
return (ptr == _bufend) ? _buf : ptr;
|
SemaphoreHandle_t _lock = NULL;
|
||||||
}
|
#endif
|
||||||
|
|
||||||
size_t _size;
|
|
||||||
char* _buf;
|
|
||||||
const char* _bufend;
|
|
||||||
char* _begin;
|
|
||||||
char* _end;
|
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif//__cbuf_h
|
|
||||||
|
|
|
||||||
Loading…
Reference in a new issue