From d38a52f0650b6d213f0a72eb66c722acd8f6312b Mon Sep 17 00:00:00 2001
From: David Wehr <wehrdo@users.noreply.github.com>
Date: Fri, 6 Jan 2017 17:59:14 -0600
Subject: [PATCH] Added circular buffer for async UART

---
 quad/sw/comm_dev/src/circ_buffer.c | 76 ++++++++++++++++++++++++++++++
 quad/sw/comm_dev/src/circ_buffer.h | 42 +++++++++++++++++
 2 files changed, 118 insertions(+)
 create mode 100644 quad/sw/comm_dev/src/circ_buffer.c
 create mode 100644 quad/sw/comm_dev/src/circ_buffer.h

diff --git a/quad/sw/comm_dev/src/circ_buffer.c b/quad/sw/comm_dev/src/circ_buffer.c
new file mode 100644
index 000000000..ef90c1acf
--- /dev/null
+++ b/quad/sw/comm_dev/src/circ_buffer.c
@@ -0,0 +1,76 @@
+//
+// Created by dawehr on 10/24/2016.
+//
+
+#include "circ_buffer.h"
+
+#define circ_buf_min(a,b) \
+   ({ __typeof__ (a) _a = (a); \
+       __typeof__ (b) _b = (b); \
+     _a < _b ? _a : _b; })
+
+
+unsigned char buffer[CIRC_BUFFER_SIZE];
+size_t buf_start = 0;
+size_t buf_end = 0;
+size_t size = 0;
+
+
+struct data_chunk getChunk() {
+    size_t chunk_end = buf_end;
+    if (buf_end < buf_start) {
+        chunk_end = CIRC_BUFFER_SIZE;
+    }
+    size_t ret_size = chunk_end - buf_start;
+    struct data_chunk c = {
+            ret_size,
+             &buffer[buf_start]
+    };
+    return c;
+}
+
+
+void markConsumed(size_t n_consumed) {
+    size_t chunk_end = buf_start + n_consumed;
+    buf_start = chunk_end;
+    size -= n_consumed;
+    if (buf_start == CIRC_BUFFER_SIZE) {
+        buf_start = 0;
+    }
+
+}
+
+size_t putChunk(struct data_chunk c) {
+    // The amount that will be placed into the buffer
+    size_t will_place = circ_buf_min(CIRC_BUFFER_SIZE - size, c.length);
+    // Actual amount placed so far
+    size_t placed = 0;
+    while (placed < will_place) {
+        // Available room without wrapping
+        size_t to_copy;
+        if (buf_end  >= buf_start) {
+            // If buffer is not wrapped, we can copy until the end of the buffer
+            to_copy = circ_buf_min(will_place, CIRC_BUFFER_SIZE - buf_end);
+        } else {
+            // Otherwise, remaining space in buffer is contiguous
+            to_copy = will_place - placed;
+        }
+        memcpy(buffer + buf_end, c.data + placed, to_copy);
+        // Update buffer endpoint
+        buf_end += to_copy;
+        if (buf_end > CIRC_BUFFER_SIZE) {
+            printf("Error: buf_end: %d\n", buf_end);
+        }
+        // If we copied to the end
+        if (buf_end == CIRC_BUFFER_SIZE) {
+            buf_end = 0;
+        }
+        placed += to_copy;
+    }
+    size += placed;
+    return placed;
+}
+
+size_t get_buffer_size() {
+    return size;
+}
diff --git a/quad/sw/comm_dev/src/circ_buffer.h b/quad/sw/comm_dev/src/circ_buffer.h
new file mode 100644
index 000000000..493543112
--- /dev/null
+++ b/quad/sw/comm_dev/src/circ_buffer.h
@@ -0,0 +1,42 @@
+//
+// Created by dawehr on 10/24/2016.
+//
+
+#ifndef CIRC_BUFFER_H
+#define CIRC_BUFFER_H
+
+#define CIRC_BUFFER_SIZE 2048
+#include <unistd.h>
+#include <memory.h>
+#include <stdio.h>
+
+struct data_chunk {
+    size_t length;
+    unsigned char* data;
+};
+
+/*
+ * Returns the largest contiguous chunk from the buffer
+ * Does not move the buffer forward. You must call markConsumed to
+ * mark the area as free.
+ */
+struct data_chunk getChunk();
+
+/*
+ * Marks the n_consumed bytes as used, and that
+ * area of the buffer can be used for new data now
+ */
+void markConsumed(size_t n_consumed);
+
+/*
+ * Places data into the circular buffer. Returns the number of bytes stored.
+ * Will store the entire chunk, unless there is not enough remaining size in the buffer
+ */
+size_t putChunk(struct data_chunk);
+
+/*
+ * Returns the remaining size in the buffer
+ */
+size_t get_buffer_size();
+
+#endif //CIRC_BUFFER_H
-- 
GitLab