aboutsummaryrefslogtreecommitdiffstats
path: root/firmware/x300/lib/wb_pkt_iface64.c
diff options
context:
space:
mode:
Diffstat (limited to 'firmware/x300/lib/wb_pkt_iface64.c')
-rw-r--r--firmware/x300/lib/wb_pkt_iface64.c87
1 files changed, 87 insertions, 0 deletions
diff --git a/firmware/x300/lib/wb_pkt_iface64.c b/firmware/x300/lib/wb_pkt_iface64.c
new file mode 100644
index 000000000..3c0774bdd
--- /dev/null
+++ b/firmware/x300/lib/wb_pkt_iface64.c
@@ -0,0 +1,87 @@
+// Copyright 2012 Ettus Research LLC
+
+#include <wb_pkt_iface64.h>
+#include <wb_utils.h>
+#include <printf.h>
+
+#define NUM_BYTES_MASK 0x1fff
+
+static uint32_t get_status(wb_pkt_iface64_config_t *config)
+{
+ return wb_peek32(config->config_addr);
+}
+
+static void set_control(wb_pkt_iface64_config_t *config)
+{
+ wb_poke32(config->config_addr, config->ctrl);
+}
+
+wb_pkt_iface64_config_t wb_pkt_iface64_init(const uint32_t base, const size_t ctrl_offset)
+{
+ wb_pkt_iface64_config_t config;
+ config.base = base;
+ config.ctrl = 0;
+ config.config_addr = base + ctrl_offset;
+ set_control(&config);
+ wb_pkt_iface64_rx_release(&config); //always release, in case left in a filled state
+ return config;
+}
+
+const void *wb_pkt_iface64_rx_try_claim(wb_pkt_iface64_config_t *config, size_t *num_bytes)
+{
+ const uint32_t status = get_status(config);
+ const uint32_t rx_state_flag = (status >> 31) & 0x1;
+ *num_bytes = (status & NUM_BYTES_MASK);
+ //if (*num_bytes & 0x7) *num_bytes -= 8; //adjust for tuser
+ if (rx_state_flag == 0) return NULL;
+ return (void *)config->base;
+}
+
+void wb_pkt_iface64_rx_release(wb_pkt_iface64_config_t *config)
+{
+ config->ctrl |= 1ul << 31; //does a release
+ set_control(config);
+ while (true)
+ {
+ const uint32_t status = get_status(config);
+ const uint32_t rx_state_flag = (status >> 31) & 0x1;
+ if (rx_state_flag == 0)
+ {
+ config->ctrl &= ~(1ul << 31); //allows for next claim
+ set_control(config);
+ return;
+ }
+ }
+}
+
+void *wb_pkt_iface64_tx_claim(wb_pkt_iface64_config_t *config)
+{
+ while (true)
+ {
+ const uint32_t status = get_status(config);
+ const uint32_t tx_state_flag = (status >> 30) & 0x1;
+ if (tx_state_flag == 1) break;
+ }
+ return (void *)config->base;
+}
+
+void wb_pkt_iface64_tx_submit(wb_pkt_iface64_config_t *config, size_t num_bytes)
+{
+ config->ctrl |= (1ul << 30); //allows for next claim
+ config->ctrl &= ~(NUM_BYTES_MASK); //clear num bytes
+ if (num_bytes & 0x7) num_bytes += 8; //adjust for tuser
+ config->ctrl |= num_bytes & NUM_BYTES_MASK; //set num bytes
+ set_control(config);
+
+ //wait for state machine to transition
+ while (true)
+ {
+ const uint32_t status = get_status(config);
+ const uint32_t tx_state_flag = (status >> 30) & 0x1;
+ if (tx_state_flag == 0) break;
+ }
+
+ config->ctrl &= ~(1ul << 30); //release
+ set_control(config);
+
+}