-
Notifications
You must be signed in to change notification settings - Fork 9
/
Copy pathringbuffer.hpp
129 lines (103 loc) · 3.44 KB
/
ringbuffer.hpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
// Copyright (c) 2014, Christian Riggenbach
// All rights reserved.
//
// Redistribution and use in source and binary forms, with or without
// modification, are permitted provided that the following conditions are met:
//
// Redistributions of source code must retain the above copyright notice,
// this list of conditions and the following disclaimer.
// Redistributions in binary form must reproduce the above copyright
// notice, this list of conditions and the following disclaimer in the
// documentation and/or other materials provided with the distribution.
// Neither the name of Magahugu.ch nor the names of its contributors may be
// used to endorse or promote products derived from this software without
// specific prior written permission.
//
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
// AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
// IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
// ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
// LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
// CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
// SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
// INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
// CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
// ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
// POSSIBILITY OF SUCH DAMAGE.
#pragma once
#include <stddef.h>
#include <stdint.h>
#include <main.hpp>
template< class T, size_t Size, class size_type = uint16_t >
class RingBuffer {
private:
// Pointer zum Puffer
T Buffer[Size];
// gespeicherte Daten
volatile size_type dataInBuffer;
// Lese- und Schreibe-Pointer
volatile T* readPointer;
volatile T* writePointer;
public:
RingBuffer()
: Buffer(), dataInBuffer( 0 ), readPointer( Buffer ), writePointer( Buffer ) {}
bool isEmpty() { return dataInBuffer == 0; }
size_type sizeOfBuffer() { return dataInBuffer; }
void addToBufferBack( const T c ) {
TCritSect critical();
*writePointer++ = c;
// bei Überlauf auch readPointer inkrementieren
if( dataInBuffer < ( Size * sizeof( T ) ) ) {
dataInBuffer++;
} else {
if( ++readPointer >= &Buffer[Size] ) {
readPointer = Buffer;
}
}
// Überlauf
if( writePointer >= &( Buffer[Size] ) ) {
writePointer = Buffer;
}
}
void operator>>( T& c ) { c = takeFront(); }
size_type operator<<( const T c ) {
addToBufferBack( c );
return dataInBuffer;
}
T takeFront() {
T c = 0;
TCritSect critical();
// auf leeren Puffer prüfen
if( dataInBuffer ) {
// hole das Zeichen
c = *readPointer++;
// dataInBuffer ändern
dataInBuffer--;
// Überlauf
if( readPointer >= &( Buffer[Size] ) ) {
readPointer = Buffer;
}
}
return c;
}
T& viewFront() { return *readPointer; }
void incrementFront() {
TCritSect critical();
// auf leeren Puffer prüfen
if( dataInBuffer ) {
// hole das Zeichen
readPointer++;
// dataInBuffer ändern
dataInBuffer--;
// Überlauf
if( readPointer >= &( Buffer[Size] ) ) {
readPointer = Buffer;
}
}
}
void flush() {
TCritSect critical();
readPointer = writePointer;
dataInBuffer = 0;
}
};