cbuffer and queue working
parent
3aa9683f54
commit
e4bc7f9946
Binary file not shown.
Binary file not shown.
Binary file not shown.
@ -0,0 +1,178 @@
|
|||||||
|
---
|
||||||
|
Language: Cpp
|
||||||
|
# BasedOnStyle: Microsoft
|
||||||
|
AccessModifierOffset: -2
|
||||||
|
AlignAfterOpenBracket: Align
|
||||||
|
AlignArrayOfStructures: None
|
||||||
|
AlignConsecutiveMacros: None
|
||||||
|
AlignConsecutiveAssignments: None
|
||||||
|
AlignConsecutiveBitFields: None
|
||||||
|
AlignConsecutiveDeclarations: None
|
||||||
|
AlignEscapedNewlines: Right
|
||||||
|
AlignOperands: Align
|
||||||
|
AlignTrailingComments: true
|
||||||
|
AllowAllArgumentsOnNextLine: true
|
||||||
|
AllowAllConstructorInitializersOnNextLine: true
|
||||||
|
AllowAllParametersOfDeclarationOnNextLine: true
|
||||||
|
AllowShortEnumsOnASingleLine: false
|
||||||
|
AllowShortBlocksOnASingleLine: Never
|
||||||
|
AllowShortCaseLabelsOnASingleLine: false
|
||||||
|
AllowShortFunctionsOnASingleLine: None
|
||||||
|
AllowShortLambdasOnASingleLine: All
|
||||||
|
AllowShortIfStatementsOnASingleLine: Never
|
||||||
|
AllowShortLoopsOnASingleLine: true
|
||||||
|
AlwaysBreakAfterDefinitionReturnType: None
|
||||||
|
AlwaysBreakAfterReturnType: None
|
||||||
|
AlwaysBreakBeforeMultilineStrings: false
|
||||||
|
AlwaysBreakTemplateDeclarations: MultiLine
|
||||||
|
AttributeMacros:
|
||||||
|
- __capability
|
||||||
|
BinPackArguments: true
|
||||||
|
BinPackParameters: true
|
||||||
|
BraceWrapping:
|
||||||
|
AfterCaseLabel: true
|
||||||
|
AfterClass: true
|
||||||
|
AfterControlStatement: Always
|
||||||
|
AfterEnum: true
|
||||||
|
AfterFunction: true
|
||||||
|
AfterNamespace: true
|
||||||
|
AfterObjCDeclaration: true
|
||||||
|
AfterStruct: true
|
||||||
|
AfterUnion: true
|
||||||
|
AfterExternBlock: true
|
||||||
|
BeforeCatch: true
|
||||||
|
BeforeElse: true
|
||||||
|
BeforeLambdaBody: false
|
||||||
|
BeforeWhile: false
|
||||||
|
IndentBraces: false
|
||||||
|
SplitEmptyFunction: false
|
||||||
|
SplitEmptyRecord: true
|
||||||
|
SplitEmptyNamespace: true
|
||||||
|
BreakBeforeBinaryOperators: None
|
||||||
|
BreakBeforeConceptDeclarations: true
|
||||||
|
BreakBeforeBraces: Custom
|
||||||
|
BreakBeforeInheritanceComma: false
|
||||||
|
BreakInheritanceList: BeforeColon
|
||||||
|
BreakBeforeTernaryOperators: true
|
||||||
|
BreakConstructorInitializersBeforeComma: false
|
||||||
|
BreakConstructorInitializers: BeforeColon
|
||||||
|
BreakAfterJavaFieldAnnotations: false
|
||||||
|
BreakStringLiterals: true
|
||||||
|
ColumnLimit: 80
|
||||||
|
CommentPragmas: '^ IWYU pragma:'
|
||||||
|
CompactNamespaces: false
|
||||||
|
ConstructorInitializerAllOnOneLineOrOnePerLine: false
|
||||||
|
ConstructorInitializerIndentWidth: 4
|
||||||
|
ContinuationIndentWidth: 4
|
||||||
|
Cpp11BracedListStyle: true
|
||||||
|
#DeriveLineEnding: true
|
||||||
|
DerivePointerAlignment: false
|
||||||
|
DisableFormat: false
|
||||||
|
EmptyLineAfterAccessModifier: Never
|
||||||
|
EmptyLineBeforeAccessModifier: LogicalBlock
|
||||||
|
ExperimentalAutoDetectBinPacking: false
|
||||||
|
FixNamespaceComments: true
|
||||||
|
ForEachMacros:
|
||||||
|
- foreach
|
||||||
|
- Q_FOREACH
|
||||||
|
- BOOST_FOREACH
|
||||||
|
IfMacros:
|
||||||
|
- KJ_IF_MAYBE
|
||||||
|
IncludeBlocks: Preserve
|
||||||
|
IncludeCategories:
|
||||||
|
- Regex: '^"(llvm|llvm-c|clang|clang-c)/'
|
||||||
|
Priority: 2
|
||||||
|
SortPriority: 0
|
||||||
|
CaseSensitive: false
|
||||||
|
- Regex: '^(<|"(gtest|gmock|isl|json)/)'
|
||||||
|
Priority: 3
|
||||||
|
SortPriority: 0
|
||||||
|
CaseSensitive: false
|
||||||
|
- Regex: '.*'
|
||||||
|
Priority: 1
|
||||||
|
SortPriority: 0
|
||||||
|
CaseSensitive: false
|
||||||
|
IncludeIsMainRegex: '(Test)?$'
|
||||||
|
IncludeIsMainSourceRegex: ''
|
||||||
|
IndentAccessModifiers: false
|
||||||
|
IndentCaseLabels: true
|
||||||
|
IndentCaseBlocks: false
|
||||||
|
IndentGotoLabels: true
|
||||||
|
IndentPPDirectives: None
|
||||||
|
IndentExternBlock: AfterExternBlock
|
||||||
|
IndentRequires: false
|
||||||
|
IndentWidth: 4
|
||||||
|
IndentWrappedFunctionNames: false
|
||||||
|
InsertTrailingCommas: None
|
||||||
|
JavaScriptQuotes: Leave
|
||||||
|
JavaScriptWrapImports: true
|
||||||
|
KeepEmptyLinesAtTheStartOfBlocks: true
|
||||||
|
LambdaBodyIndentation: Signature
|
||||||
|
MacroBlockBegin: ''
|
||||||
|
MacroBlockEnd: ''
|
||||||
|
MaxEmptyLinesToKeep: 1
|
||||||
|
NamespaceIndentation: None
|
||||||
|
ObjCBinPackProtocolList: Auto
|
||||||
|
ObjCBlockIndentWidth: 2
|
||||||
|
ObjCBreakBeforeNestedBlockParam: true
|
||||||
|
ObjCSpaceAfterProperty: false
|
||||||
|
ObjCSpaceBeforeProtocolList: true
|
||||||
|
PenaltyBreakAssignment: 2
|
||||||
|
PenaltyBreakBeforeFirstCallParameter: 19
|
||||||
|
PenaltyBreakComment: 300
|
||||||
|
PenaltyBreakFirstLessLess: 120
|
||||||
|
PenaltyBreakString: 1000
|
||||||
|
PenaltyBreakTemplateDeclaration: 10
|
||||||
|
PenaltyExcessCharacter: 1000000
|
||||||
|
PenaltyReturnTypeOnItsOwnLine: 1000
|
||||||
|
PenaltyIndentedWhitespace: 0
|
||||||
|
PointerAlignment: Left
|
||||||
|
PPIndentWidth: -1
|
||||||
|
ReferenceAlignment: Pointer
|
||||||
|
ReflowComments: true
|
||||||
|
ShortNamespaceLines: 1
|
||||||
|
SortIncludes: CaseSensitive
|
||||||
|
SortJavaStaticImport: Before
|
||||||
|
SortUsingDeclarations: true
|
||||||
|
SpaceAfterCStyleCast: false
|
||||||
|
SpaceAfterLogicalNot: false
|
||||||
|
SpaceAfterTemplateKeyword: true
|
||||||
|
SpaceBeforeAssignmentOperators: true
|
||||||
|
SpaceBeforeCaseColon: false
|
||||||
|
SpaceBeforeCpp11BracedList: false
|
||||||
|
SpaceBeforeCtorInitializerColon: true
|
||||||
|
SpaceBeforeInheritanceColon: true
|
||||||
|
SpaceBeforeParens: ControlStatements
|
||||||
|
SpaceAroundPointerQualifiers: Default
|
||||||
|
SpaceBeforeRangeBasedForLoopColon: true
|
||||||
|
SpaceInEmptyBlock: false
|
||||||
|
SpaceInEmptyParentheses: false
|
||||||
|
SpacesBeforeTrailingComments: 1
|
||||||
|
SpacesInAngles: Never
|
||||||
|
SpacesInConditionalStatement: false
|
||||||
|
SpacesInContainerLiterals: true
|
||||||
|
SpacesInCStyleCastParentheses: false
|
||||||
|
SpacesInLineCommentPrefix:
|
||||||
|
Minimum: 1
|
||||||
|
Maximum: -1
|
||||||
|
SpacesInParentheses: false
|
||||||
|
SpacesInSquareBrackets: false
|
||||||
|
SpaceBeforeSquareBrackets: false
|
||||||
|
BitFieldColonSpacing: Both
|
||||||
|
Standard: Latest
|
||||||
|
StatementAttributeLikeMacros:
|
||||||
|
- Q_EMIT
|
||||||
|
StatementMacros:
|
||||||
|
- Q_UNUSED
|
||||||
|
- QT_REQUIRE_VERSION
|
||||||
|
TabWidth: 4
|
||||||
|
UseCRLF: false
|
||||||
|
UseTab: Always
|
||||||
|
WhitespaceSensitiveMacros:
|
||||||
|
- STRINGIZE
|
||||||
|
- PP_STRINGIZE
|
||||||
|
- BOOST_PP_STRINGIZE
|
||||||
|
- NS_SWIFT_NAME
|
||||||
|
- CF_SWIFT_NAME
|
||||||
|
...
|
||||||
|
|
@ -1,6 +0,0 @@
|
|||||||
((c-mode . ((lsp-clients-clangd-args . ("--compile-commands-dir=build/"
|
|
||||||
"--pch-storage=memory"
|
|
||||||
"--background-index"
|
|
||||||
"-j=4"
|
|
||||||
))
|
|
||||||
)))
|
|
@ -1,77 +0,0 @@
|
|||||||
#include <PCircularBuffer.h>
|
|
||||||
|
|
||||||
// Error handler used for debugging only
|
|
||||||
|
|
||||||
// Circular Buffer Prototypes -- Double
|
|
||||||
static PB_CB_STATUS p_cb_double_push(p_cb_double *cbuffer, double value);
|
|
||||||
static PB_CB_STATUS p_cb_double_empty(p_cb_double *cbuffer);
|
|
||||||
|
|
||||||
// Circular Buffer Definitions -- double
|
|
||||||
PB_CB_STATUS p_cb_double_init(p_cb_double *circ_buffer, double *buff, uint32_t max_length)
|
|
||||||
{
|
|
||||||
PB_CB_STATUS ret = PB_CB_GOOD;
|
|
||||||
do
|
|
||||||
{
|
|
||||||
// Make sure the buffer isn't bad (null)
|
|
||||||
if (!buff)
|
|
||||||
{
|
|
||||||
ret = PB_CB_NULL_BUFFER;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Make sure the max buffer is a useable size
|
|
||||||
if (max_length > PB_CB_MAX_BUFFER_SIZE || max_length <= 0)
|
|
||||||
{
|
|
||||||
ret = PB_CB_BAD_BUFFER_SIZE;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
circ_buffer->buffer = buff;
|
|
||||||
circ_buffer->max_len = (uint16_t)max_length;
|
|
||||||
circ_buffer->head = 0;
|
|
||||||
circ_buffer->push = p_cb_double_push;
|
|
||||||
circ_buffer->empty = p_cb_double_empty;
|
|
||||||
circ_buffer->empty(circ_buffer);
|
|
||||||
} while (0);
|
|
||||||
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
PB_CB_STATUS p_cb_double_push(p_cb_double *cbuffer, double value)
|
|
||||||
{
|
|
||||||
PB_CB_STATUS ret = PB_CB_GOOD;
|
|
||||||
|
|
||||||
if (!cbuffer)
|
|
||||||
{
|
|
||||||
ret = PB_CB_NULL_CBUFFER;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
cbuffer->buffer[cbuffer->head] = value;
|
|
||||||
cbuffer->head = (cbuffer->head + 1) % cbuffer->max_len;
|
|
||||||
}
|
|
||||||
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
PB_CB_STATUS p_cb_double_empty(p_cb_double *cbuffer)
|
|
||||||
{
|
|
||||||
PB_CB_STATUS ret = PB_CB_GOOD;
|
|
||||||
do
|
|
||||||
{
|
|
||||||
if (!cbuffer)
|
|
||||||
{
|
|
||||||
ret = PB_CB_NULL_CBUFFER;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!cbuffer->buffer)
|
|
||||||
{
|
|
||||||
ret = PB_CB_NULL_BUFFER;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
memset(cbuffer->buffer, 0, sizeof(double) * cbuffer->max_len);
|
|
||||||
cbuffer->head = 0;
|
|
||||||
cbuffer->b_empty = true;
|
|
||||||
cbuffer->b_filled = false;
|
|
||||||
} while (0);
|
|
||||||
return ret;
|
|
||||||
}
|
|
@ -1,64 +0,0 @@
|
|||||||
// Resource/Inspiration: https://embedjournal.com/implementing-circular-buffer-embedded-c/
|
|
||||||
|
|
||||||
/*
|
|
||||||
Penguin's Circular Buffer -- a simple floating queue designed for low memory usage (mainly for embedded)
|
|
||||||
|
|
||||||
This is a ring buffer with limited capabilities. It is meant as a container for moving data.
|
|
||||||
Normally included features such as checks to see if the buffer is full or empty have been omitted
|
|
||||||
because this type of buffer is being implemented mainly for sensor data usage. Data is almost never read
|
|
||||||
individually, and even if it is, it isn't meant to be cleared on read. It is a simple moving buffer that
|
|
||||||
automatically writes over old data for the purpose of keeping track of the most up to date data.
|
|
||||||
|
|
||||||
Notes:
|
|
||||||
- Initialization is mandatory using p_cb_<type>_init.
|
|
||||||
- If a circular buffer is not initialized, you will run into a LOT of hard-to-debug problems. Just initialize it.
|
|
||||||
- By default, all cb types should be defined as disabled. This is to save on size. Enable the ones you want to use.
|
|
||||||
|
|
||||||
Behavior:
|
|
||||||
- Oldest data is overwritten
|
|
||||||
- popping of data isn't implemented. If this feature is required, use a ring buffer.
|
|
||||||
|
|
||||||
Acronyms:
|
|
||||||
- p: penguin
|
|
||||||
- pb: penguin buffer
|
|
||||||
- cb: circular buffer
|
|
||||||
- uX, where X is bit size: unsigned X bit integer
|
|
||||||
- iX, where X is bit size: signed X bit integer
|
|
||||||
*/
|
|
||||||
|
|
||||||
#ifndef _PCIRCULARBUFFER_H_
|
|
||||||
#define _PCIRCULARBUFFER_H_
|
|
||||||
|
|
||||||
#include <stdbool.h>
|
|
||||||
#include <stdint.h>
|
|
||||||
|
|
||||||
typedef enum PB_CB_STATUS
|
|
||||||
{
|
|
||||||
PB_CB_GOOD = 0,
|
|
||||||
PB_CB_BAD = 1,
|
|
||||||
PB_CB_BAD_BUFFER_SIZE = 2,
|
|
||||||
PB_CB_NULL_BUFFER = 3,
|
|
||||||
PB_CB_NULL_CBUFFER = 4
|
|
||||||
} PB_CB_STATUS;
|
|
||||||
|
|
||||||
#if PB_CB_DOUBLE
|
|
||||||
typedef struct p_cb_double
|
|
||||||
{
|
|
||||||
double *buffer;
|
|
||||||
uint16_t head;
|
|
||||||
uint16_t max_len;
|
|
||||||
// Signifies the buffer being filled at least once.
|
|
||||||
// Useful for initializing sensor data
|
|
||||||
bool b_filled;
|
|
||||||
// Signifies the buffer being empty
|
|
||||||
// Useful for knowing if data is being received
|
|
||||||
bool b_empty;
|
|
||||||
|
|
||||||
PB_CB_STATUS (*push)(struct p_cb_double *cbuffer, double value);
|
|
||||||
PB_CB_STATUS (*empty)(struct p_cb_double *cbuffer);
|
|
||||||
} p_cb_double;
|
|
||||||
|
|
||||||
PB_CB_STATUS p_cb_double_init(p_cb_double *circ_buffer, double *buff, uint32_t max_length);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#endif
|
|
@ -0,0 +1,32 @@
|
|||||||
|
CC=gcc
|
||||||
|
|
||||||
|
CFLAGS=\
|
||||||
|
-Wall \
|
||||||
|
-std=gnu11 \
|
||||||
|
|
||||||
|
ifdef $(DEBUG)
|
||||||
|
CFLAGS+=\
|
||||||
|
-DDEBUG\
|
||||||
|
-ffunctions-sections\
|
||||||
|
-O3\
|
||||||
|
-g3
|
||||||
|
endif
|
||||||
|
|
||||||
|
|
||||||
|
MK_DIR=mkdir -p
|
||||||
|
|
||||||
|
.PHONY: clean examples tests
|
||||||
|
|
||||||
|
all: examples
|
||||||
|
|
||||||
|
clean:
|
||||||
|
rm -rf bin
|
||||||
|
|
||||||
|
|
||||||
|
examples: clean
|
||||||
|
$(MK_DIR) bin/
|
||||||
|
$(CC) $(CFLAGS) -I. ./examples/p_queue_example.c -o ./bin/p_queue_example
|
||||||
|
$(CC) $(CFLAGS) -I. ./examples/p_cbuffer_example.c -o ./bin/p_cbuffer_example
|
||||||
|
|
||||||
|
|
||||||
|
|
Binary file not shown.
Binary file not shown.
@ -0,0 +1,16 @@
|
|||||||
|
[
|
||||||
|
{
|
||||||
|
"arguments": [
|
||||||
|
"gcc",
|
||||||
|
"-c",
|
||||||
|
"-Wall",
|
||||||
|
"-std=gnu11",
|
||||||
|
"-I.",
|
||||||
|
"-o",
|
||||||
|
"./bin/p_queue_example",
|
||||||
|
"examples/p_queue_example.c"
|
||||||
|
],
|
||||||
|
"directory": "/storage/Shared/Projects/PenguinBuffers",
|
||||||
|
"file": "examples/p_queue_example.c"
|
||||||
|
}
|
||||||
|
]
|
@ -0,0 +1,90 @@
|
|||||||
|
/*
|
||||||
|
(WORK IN PROGRESS)
|
||||||
|
Penguin's Circular Buffer Example -- Turning a noisy bell curve into a less
|
||||||
|
noisy bell curve
|
||||||
|
|
||||||
|
Here's an example of using the circular buffer library for sensor data:
|
||||||
|
|
||||||
|
Let's say I have a sensor that gives me temperature at 1khz (using ideals so
|
||||||
|
it is exactly 1khz) If this sensor was only giving me raw analog data, I
|
||||||
|
might want to do some processing on these values so that I can make sure the
|
||||||
|
values are as clean as possible. The more samples we have, the closer we are
|
||||||
|
to the actual value. The faster we gather samples, the closer we get to
|
||||||
|
representing our data in realtime.
|
||||||
|
|
||||||
|
Using the sample rate, I can decide on a sampling window (in seconds) and an
|
||||||
|
OSR (Oversampling Rate). The larger the buffer size, the more memory I need,
|
||||||
|
so it is useful to find a happy medium between a large buffer and clean
|
||||||
|
values.
|
||||||
|
|
||||||
|
A sampling window is the window of time in which we can accept values for an
|
||||||
|
average value. Depending on your application, a sampling window may need to
|
||||||
|
be extremely small or maybe not so small. A rocket going extremely fast using
|
||||||
|
some sensor for real time controls will want an extremely small sampling
|
||||||
|
window as well as a lot of measurements for both clean, near noiseless data
|
||||||
|
and as close to realtime data as possible.
|
||||||
|
|
||||||
|
The sampling windows is usually an engineering requirement given that can
|
||||||
|
match the sampling rate of the sensor with the following: sampling window (in
|
||||||
|
seconds) = 1 / frequency Here, the OSR is simply 1.
|
||||||
|
|
||||||
|
So at 1khz sample rate, let's say I decide I only need a 0.125 second sample
|
||||||
|
window and I want to clean up some noisy data. We can now use this equation:
|
||||||
|
OSR = frequency * sample window (in seconds)
|
||||||
|
|
||||||
|
All of these have ignored real world slowdowns like the time it takes to do
|
||||||
|
math on lower end hardware, interrupts slightly delaying the math, etc
|
||||||
|
Without wanting to do some hard analysis on whatever hardware we're using, I
|
||||||
|
usually take my frequency and half it to ensure timing requirements are met,
|
||||||
|
like so:
|
||||||
|
|
||||||
|
OSR = frequency / 2 * sample window
|
||||||
|
|
||||||
|
Please note: In applications that require real real-time data, this is not a
|
||||||
|
good way of doing things.
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "p_cbuffer.h"
|
||||||
|
|
||||||
|
DEFINE_CBUFFER(int);
|
||||||
|
#define CBUFFER_SIZE (6)
|
||||||
|
|
||||||
|
void display(cbuffer_int_t* cbuffer)
|
||||||
|
{
|
||||||
|
printf("Cbuffer:\n");
|
||||||
|
for (int ind = 0; ind < cbuffer->msize; ind++)
|
||||||
|
{
|
||||||
|
printf("[%d]: %d\n", ind, cbuffer->buffer[ind]);
|
||||||
|
}
|
||||||
|
printf("\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
int main(int argc, char** argv)
|
||||||
|
{
|
||||||
|
cbuffer_int_t cbuffer;
|
||||||
|
int data[CBUFFER_SIZE];
|
||||||
|
|
||||||
|
cbuffer_int_init(&cbuffer, data, CBUFFER_SIZE);
|
||||||
|
cbuffer.push(&cbuffer, 1);
|
||||||
|
cbuffer.push(&cbuffer, 2);
|
||||||
|
cbuffer.push(&cbuffer, 3);
|
||||||
|
cbuffer.push(&cbuffer, 4);
|
||||||
|
cbuffer.push(&cbuffer, 5);
|
||||||
|
printf("Is cbuffer filled? %s\n",
|
||||||
|
cbuffer.is_filled(&cbuffer) ? "Yes" : "No");
|
||||||
|
cbuffer.push(&cbuffer, 6);
|
||||||
|
printf("Is cbuffer filled? %s\n",
|
||||||
|
cbuffer.is_filled(&cbuffer) ? "Yes" : "No");
|
||||||
|
display(&cbuffer);
|
||||||
|
cbuffer.push(&cbuffer, 7);
|
||||||
|
display(&cbuffer);
|
||||||
|
printf("Emptying cbuffer...\n");
|
||||||
|
printf("Is cbuffer filled? %s\n",
|
||||||
|
cbuffer.is_filled(&cbuffer) ? "Yes" : "No");
|
||||||
|
cbuffer.empty(&cbuffer);
|
||||||
|
cbuffer.push(&cbuffer, 8);
|
||||||
|
display(&cbuffer);
|
||||||
|
printf("Is cbuffer filled? %s\n",
|
||||||
|
cbuffer.is_filled(&cbuffer) ? "Yes" : "No");
|
||||||
|
}
|
@ -0,0 +1,22 @@
|
|||||||
|
#ifndef __PBUFFER_H__
|
||||||
|
#define __PBUFFER_H__
|
||||||
|
|
||||||
|
#define MAX_QUEUE_SIZE (512)
|
||||||
|
#define MAX_CBUFFER_SIZE (512)
|
||||||
|
|
||||||
|
#include <stdbool.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <string.h>
|
||||||
|
|
||||||
|
typedef enum PB_STATUS
|
||||||
|
{
|
||||||
|
PB_GOOD = 0,
|
||||||
|
PB_BAD = 1,
|
||||||
|
PB_BAD_BUFFER_SIZE = 2,
|
||||||
|
PB_NULL_BUFFER = 3,
|
||||||
|
PB_NULL_DATA = 4,
|
||||||
|
PB_EMPTY_QUEUE = 5,
|
||||||
|
PB_FULL_QUEUE = 6
|
||||||
|
} PB_STATUS;
|
||||||
|
|
||||||
|
#endif
|
@ -0,0 +1,133 @@
|
|||||||
|
// Resource/Inspiration:
|
||||||
|
// https://embedjournal.com/implementing-circular-buffer-embedded-c/
|
||||||
|
|
||||||
|
/*
|
||||||
|
Penguin's Circular Buffer -- a simple floating queue designed for low memory
|
||||||
|
usage (mainly for embedded)
|
||||||
|
|
||||||
|
This is a ring buffer with limited capabilities. It is meant as a container
|
||||||
|
for moving data. Normally included features such as checks to see if the
|
||||||
|
buffer is full or empty have been omitted because this type of buffer is
|
||||||
|
being implemented mainly for sensor data usage. Data is almost never read
|
||||||
|
individually, and even if it is, it isn't meant to be cleared on read. It is
|
||||||
|
a simple moving buffer that automatically writes over old data for the
|
||||||
|
purpose of keeping track of the most up to date data.
|
||||||
|
|
||||||
|
Notes:
|
||||||
|
- Initialization is mandatory using p_cb_<type>_init.
|
||||||
|
- If a circular buffer is not initialized, you will run into a LOT of
|
||||||
|
hard-to-debug problems. Just initialize it.
|
||||||
|
- By default, all cb types should be defined as disabled. This is to save
|
||||||
|
on size. Enable the ones you want to use.
|
||||||
|
|
||||||
|
Behavior:
|
||||||
|
- Oldest data is overwritten
|
||||||
|
- popping of data isn't implemented. If this feature is required, use a
|
||||||
|
ring buffer.
|
||||||
|
|
||||||
|
Acronyms:
|
||||||
|
- p: penguin
|
||||||
|
- pb: penguin buffer
|
||||||
|
- cb: circular buffer
|
||||||
|
- uX, where X is bit size: unsigned X bit integer
|
||||||
|
- iX, where X is bit size: signed X bit integer
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef _PCIRCULARBUFFER_H_
|
||||||
|
#define _PCIRCULARBUFFER_H_
|
||||||
|
|
||||||
|
#include "p_buffer.h"
|
||||||
|
|
||||||
|
#define DEFINE_CBUFFER(type) \
|
||||||
|
typedef struct cbuffer_##type##_t \
|
||||||
|
{ \
|
||||||
|
type* buffer; \
|
||||||
|
int head; \
|
||||||
|
int csize; \
|
||||||
|
int msize; \
|
||||||
|
\
|
||||||
|
PB_STATUS (*push)(struct cbuffer_##type##_t * cbuffer, type value); \
|
||||||
|
PB_STATUS (*empty)(struct cbuffer_##type##_t * cbuffer); \
|
||||||
|
PB_STATUS (*is_filled)(struct cbuffer_##type##_t * cbuffer); \
|
||||||
|
} cbuffer_##type##_t; \
|
||||||
|
\
|
||||||
|
PB_STATUS cbuffer_##type##_is_filled(cbuffer_##type##_t* cbuffer) \
|
||||||
|
{ \
|
||||||
|
return (cbuffer->csize == cbuffer->msize); \
|
||||||
|
} \
|
||||||
|
\
|
||||||
|
PB_STATUS cbuffer_##type##_push(cbuffer_##type##_t* cbuffer, type value) \
|
||||||
|
{ \
|
||||||
|
PB_STATUS ret = PB_GOOD; \
|
||||||
|
\
|
||||||
|
if (!cbuffer) \
|
||||||
|
{ \
|
||||||
|
ret = PB_NULL_BUFFER; \
|
||||||
|
} \
|
||||||
|
else \
|
||||||
|
{ \
|
||||||
|
cbuffer->csize = \
|
||||||
|
(cbuffer->csize >= cbuffer->msize ? cbuffer->msize \
|
||||||
|
: cbuffer->csize + 1); \
|
||||||
|
cbuffer->buffer[cbuffer->head] = value; \
|
||||||
|
cbuffer->head = (cbuffer->head + 1) % cbuffer->msize; \
|
||||||
|
} \
|
||||||
|
\
|
||||||
|
return ret; \
|
||||||
|
} \
|
||||||
|
PB_STATUS cbuffer_##type##_empty(cbuffer_##type##_t* cbuffer) \
|
||||||
|
{ \
|
||||||
|
PB_STATUS ret = PB_GOOD; \
|
||||||
|
do \
|
||||||
|
{ \
|
||||||
|
if (!cbuffer) \
|
||||||
|
{ \
|
||||||
|
ret = PB_NULL_BUFFER; \
|
||||||
|
break; \
|
||||||
|
} \
|
||||||
|
\
|
||||||
|
if (!cbuffer->buffer) \
|
||||||
|
{ \
|
||||||
|
ret = PB_NULL_DATA; \
|
||||||
|
break; \
|
||||||
|
} \
|
||||||
|
memset(cbuffer->buffer, 0, sizeof(type) * cbuffer->msize); \
|
||||||
|
cbuffer->head = 0; \
|
||||||
|
cbuffer->csize = 0; \
|
||||||
|
} while (0); \
|
||||||
|
return ret; \
|
||||||
|
} \
|
||||||
|
\
|
||||||
|
PB_STATUS cbuffer_##type##_init(cbuffer_##type##_t* circ_buffer, \
|
||||||
|
type* buff, int total_length) \
|
||||||
|
{ \
|
||||||
|
PB_STATUS ret = PB_GOOD; \
|
||||||
|
do \
|
||||||
|
{ \
|
||||||
|
if (!buff) \
|
||||||
|
{ \
|
||||||
|
ret = PB_NULL_DATA; \
|
||||||
|
break; \
|
||||||
|
} \
|
||||||
|
\
|
||||||
|
if (total_length > MAX_CBUFFER_SIZE || total_length <= 0) \
|
||||||
|
{ \
|
||||||
|
ret = PB_BAD_BUFFER_SIZE; \
|
||||||
|
break; \
|
||||||
|
} \
|
||||||
|
memset(circ_buffer, 0, sizeof(cbuffer_##type##_t)); \
|
||||||
|
memset(buff, 0, sizeof(type) * total_length); \
|
||||||
|
circ_buffer->buffer = buff; \
|
||||||
|
circ_buffer->msize = total_length; \
|
||||||
|
circ_buffer->csize = 0; \
|
||||||
|
circ_buffer->head = 0; \
|
||||||
|
circ_buffer->push = cbuffer_##type##_push; \
|
||||||
|
circ_buffer->empty = cbuffer_##type##_empty; \
|
||||||
|
circ_buffer->is_filled = cbuffer_##type##_is_filled; \
|
||||||
|
circ_buffer->empty(circ_buffer); \
|
||||||
|
} while (0); \
|
||||||
|
\
|
||||||
|
return ret; \
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
Binary file not shown.
Loading…
Reference in New Issue