aboutsummaryrefslogtreecommitdiff
path: root/embassy-sync/src/ring_buffer.rs
diff options
context:
space:
mode:
Diffstat (limited to 'embassy-sync/src/ring_buffer.rs')
-rw-r--r--embassy-sync/src/ring_buffer.rs146
1 files changed, 146 insertions, 0 deletions
diff --git a/embassy-sync/src/ring_buffer.rs b/embassy-sync/src/ring_buffer.rs
new file mode 100644
index 000000000..521084024
--- /dev/null
+++ b/embassy-sync/src/ring_buffer.rs
@@ -0,0 +1,146 @@
1pub struct RingBuffer<const N: usize> {
2 buf: [u8; N],
3 start: usize,
4 end: usize,
5 empty: bool,
6}
7
8impl<const N: usize> RingBuffer<N> {
9 pub const fn new() -> Self {
10 Self {
11 buf: [0; N],
12 start: 0,
13 end: 0,
14 empty: true,
15 }
16 }
17
18 pub fn push_buf(&mut self) -> &mut [u8] {
19 if self.start == self.end && !self.empty {
20 trace!(" ringbuf: push_buf empty");
21 return &mut self.buf[..0];
22 }
23
24 let n = if self.start <= self.end {
25 self.buf.len() - self.end
26 } else {
27 self.start - self.end
28 };
29
30 trace!(" ringbuf: push_buf {:?}..{:?}", self.end, self.end + n);
31 &mut self.buf[self.end..self.end + n]
32 }
33
34 pub fn push(&mut self, n: usize) {
35 trace!(" ringbuf: push {:?}", n);
36 if n == 0 {
37 return;
38 }
39
40 self.end = self.wrap(self.end + n);
41 self.empty = false;
42 }
43
44 pub fn pop_buf(&mut self) -> &mut [u8] {
45 if self.empty {
46 trace!(" ringbuf: pop_buf empty");
47 return &mut self.buf[..0];
48 }
49
50 let n = if self.end <= self.start {
51 self.buf.len() - self.start
52 } else {
53 self.end - self.start
54 };
55
56 trace!(" ringbuf: pop_buf {:?}..{:?}", self.start, self.start + n);
57 &mut self.buf[self.start..self.start + n]
58 }
59
60 pub fn pop(&mut self, n: usize) {
61 trace!(" ringbuf: pop {:?}", n);
62 if n == 0 {
63 return;
64 }
65
66 self.start = self.wrap(self.start + n);
67 self.empty = self.start == self.end;
68 }
69
70 pub fn is_full(&self) -> bool {
71 self.start == self.end && !self.empty
72 }
73
74 pub fn is_empty(&self) -> bool {
75 self.empty
76 }
77
78 #[allow(unused)]
79 pub fn len(&self) -> usize {
80 if self.empty {
81 0
82 } else if self.start < self.end {
83 self.end - self.start
84 } else {
85 N + self.end - self.start
86 }
87 }
88
89 pub fn clear(&mut self) {
90 self.start = 0;
91 self.end = 0;
92 self.empty = true;
93 }
94
95 fn wrap(&self, n: usize) -> usize {
96 assert!(n <= self.buf.len());
97 if n == self.buf.len() {
98 0
99 } else {
100 n
101 }
102 }
103}
104
105#[cfg(test)]
106mod tests {
107 use super::*;
108
109 #[test]
110 fn push_pop() {
111 let mut rb: RingBuffer<4> = RingBuffer::new();
112 let buf = rb.push_buf();
113 assert_eq!(4, buf.len());
114 buf[0] = 1;
115 buf[1] = 2;
116 buf[2] = 3;
117 buf[3] = 4;
118 rb.push(4);
119
120 let buf = rb.pop_buf();
121 assert_eq!(4, buf.len());
122 assert_eq!(1, buf[0]);
123 rb.pop(1);
124
125 let buf = rb.pop_buf();
126 assert_eq!(3, buf.len());
127 assert_eq!(2, buf[0]);
128 rb.pop(1);
129
130 let buf = rb.pop_buf();
131 assert_eq!(2, buf.len());
132 assert_eq!(3, buf[0]);
133 rb.pop(1);
134
135 let buf = rb.pop_buf();
136 assert_eq!(1, buf.len());
137 assert_eq!(4, buf[0]);
138 rb.pop(1);
139
140 let buf = rb.pop_buf();
141 assert_eq!(0, buf.len());
142
143 let buf = rb.push_buf();
144 assert_eq!(4, buf.len());
145 }
146}