ASR_BASE

Change-Id: Icf3719cc0afe3eeb3edc7fa80a2eb5199ca9dda1
diff --git a/external/subpack/utils/gl-mifi-mcu/src/module.c b/external/subpack/utils/gl-mifi-mcu/src/module.c
new file mode 100755
index 0000000..8f5a514
--- /dev/null
+++ b/external/subpack/utils/gl-mifi-mcu/src/module.c
@@ -0,0 +1,210 @@
+#include <linux/module.h>
+#include <linux/gpio.h>
+#include <linux/hrtimer.h>
+#include <linux/interrupt.h>
+#include <linux/ktime.h>
+#include <linux/proc_fs.h>
+#include <linux/seq_file.h>
+#include <linux/version.h>
+
+MODULE_LICENSE("GPL");
+MODULE_AUTHOR("Nuno Goncalves");
+MODULE_DESCRIPTION("GL-MiFi power monitoring MCU interface");
+MODULE_VERSION("0.1");
+
+static int gpio_tx = 19;
+static int gpio_rx = 8;
+static int baudrate = 1200;
+static int query_interval_sec = 4;
+
+static struct hrtimer timer_tx;
+static struct hrtimer timer_rx;
+static ktime_t period;
+static int rx_bit_index = -1;
+
+static unsigned read_buf_ready = 0;
+static unsigned read_buf_size = 0;
+static char read_buf[2][64] = {{0},{0}};
+
+static int proc_show(struct seq_file *m, void *v)
+{
+  seq_printf(m, "%s\n", read_buf[read_buf_ready]);
+  return 0;
+}
+
+static int proc_open(struct inode *inode, struct  file *file)
+{
+  return single_open(file, proc_show, NULL);
+}
+
+#if LINUX_VERSION_CODE >= KERNEL_VERSION(5,6,0)
+static const struct proc_ops hello_proc_ops = {
+  .proc_open = proc_open,
+  .proc_read = seq_read,
+  .proc_lseek = seq_lseek,
+  .proc_release = single_release,
+};
+#else
+static const struct file_operations hello_proc_ops = {
+  .owner = THIS_MODULE,
+  .open = proc_open,
+  .read = seq_read,
+  .llseek = seq_lseek,
+  .release = single_release,
+};
+#endif
+
+static irq_handler_t handle_rx_start(unsigned int irq, void* device, struct pt_regs* registers)
+{
+  if (rx_bit_index == -1)
+  {
+    hrtimer_start(&timer_rx, ktime_set(0, period / 2), HRTIMER_MODE_REL);
+  }
+  return (irq_handler_t) IRQ_HANDLED;
+}
+
+static enum hrtimer_restart handle_tx(struct hrtimer* timer)
+{
+  ktime_t current_time = ktime_get();
+  const unsigned char character = 'g';
+  static int bit_index = -1;
+
+  // Start bit.
+  if (bit_index == -1)
+  {
+      gpio_set_value(gpio_tx, 0);
+      bit_index++;
+  }
+
+  // Data bits.
+  else if (0 <= bit_index && bit_index < 8)
+  {
+    gpio_set_value(gpio_tx, 1 & (character >> bit_index));
+    bit_index++;
+  }
+
+  // Stop bit.
+  else if (bit_index == 8)
+  {
+    gpio_set_value(gpio_tx, 1);
+    bit_index = -1;
+  }
+
+  hrtimer_forward(&timer_tx, current_time, bit_index == 8
+    ? ktime_set(query_interval_sec, 0) //wait for next query cycle
+    : period); //wait for next bit period
+
+  return HRTIMER_RESTART;
+}
+
+void receive_character(unsigned char character)
+{
+  if(character == '{')
+    read_buf_size = 0;
+
+  if(read_buf_size < (sizeof(read_buf[0])-1) || character == '}')
+  {
+    read_buf[!read_buf_ready][read_buf_size++] = character;
+    if(character == '}')
+    {
+      read_buf[!read_buf_ready][read_buf_size] = '\0';
+      read_buf_ready = !read_buf_ready;
+      read_buf_size = 0;
+    }
+  }
+}
+
+static enum hrtimer_restart handle_rx(struct hrtimer* timer)
+{
+  ktime_t current_time = ktime_get();
+  static unsigned int character = 0;
+  int bit_value = gpio_get_value(gpio_rx);
+  enum hrtimer_restart result = HRTIMER_NORESTART;
+  bool must_restart_timer = false;
+
+  // Start bit.
+  if (rx_bit_index == -1)
+  {
+    rx_bit_index++;
+    character = 0;
+    must_restart_timer = true;
+  }
+
+  // Data bits.
+  else if (0 <= rx_bit_index && rx_bit_index < 8)
+  {
+    if (bit_value == 0)
+    {
+      character &= 0xfeff;
+    }
+    else
+    {
+      character |= 0x0100;
+    }
+
+    rx_bit_index++;
+    character >>= 1;
+    must_restart_timer = true;
+  }
+
+  // Stop bit.
+  else if (rx_bit_index == 8)
+  {
+    receive_character(character);
+    rx_bit_index = -1;
+  }
+
+  // Restarts the RX timer.
+  if (must_restart_timer)
+  {
+    hrtimer_forward(&timer_rx, current_time, period);
+    result = HRTIMER_RESTART;
+  }
+
+  return result;
+}
+
+static int __init init(void)
+{
+  bool success = true;
+
+  proc_create("gl_mifi_mcu", 0, NULL, &hello_proc_ops);
+
+  success &= gpio_request(gpio_tx, "soft_uart_tx") == 0;
+  success &= gpio_direction_output(gpio_tx, 1) == 0;
+  success &= gpio_request(gpio_rx, "soft_uart_rx") == 0;
+  success &= gpio_direction_input(gpio_rx) == 0;
+  success &= gpio_set_debounce(gpio_rx, 1000/baudrate/2);
+
+  success &= request_irq(
+    gpio_to_irq(gpio_rx),
+    (irq_handler_t) handle_rx_start,
+    IRQF_TRIGGER_FALLING,
+    "gl_mifi_mcu_irq_handler",
+    NULL) == 0;
+
+  hrtimer_init(&timer_tx, CLOCK_MONOTONIC, HRTIMER_MODE_REL);
+  timer_tx.function = &handle_tx;
+  hrtimer_init(&timer_rx, CLOCK_MONOTONIC, HRTIMER_MODE_REL);
+  timer_rx.function = &handle_rx;
+  period = ktime_set(0, 1000000000/baudrate);
+  hrtimer_start(&timer_tx, period, HRTIMER_MODE_REL);
+
+  return success;
+}
+
+static void __exit exit(void)
+{
+  disable_irq(gpio_to_irq(gpio_rx));
+  hrtimer_cancel(&timer_tx);
+  hrtimer_cancel(&timer_rx);
+  free_irq(gpio_to_irq(gpio_rx), NULL);
+  gpio_set_value(gpio_tx, 0);
+  gpio_free(gpio_tx);
+  gpio_free(gpio_rx);
+  remove_proc_entry("gl_mifi_mcu", NULL);
+}
+
+module_init(init);
+module_exit(exit);
+