gps_uart.c 4.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127
  1. #include <string.h>
  2. #include "minmea.h"
  3. #include "gps_uart.h"
  4. typedef enum {
  5. WorkerEvtStop = (1 << 0),
  6. WorkerEvtRxDone = (1 << 1),
  7. } WorkerEvtFlags;
  8. #define WORKER_ALL_RX_EVENTS \
  9. (WorkerEvtStop | WorkerEvtRxDone)
  10. static void gps_uart_on_irq_cb(UartIrqEvent ev, uint8_t data, void* context) {
  11. GpsUart* gps_uart = (GpsUart*)context;
  12. BaseType_t xHigherPriorityTaskWoken = pdFALSE;
  13. if(ev == UartIrqEventRXNE) {
  14. xStreamBufferSendFromISR(gps_uart->rx_stream, &data, 1, &xHigherPriorityTaskWoken);
  15. furi_thread_flags_set(furi_thread_get_id(gps_uart->thread), WorkerEvtRxDone);
  16. portYIELD_FROM_ISR(xHigherPriorityTaskWoken);
  17. }
  18. }
  19. static void gps_uart_serial_init(GpsUart* gps_uart) {
  20. furi_hal_console_disable();
  21. furi_hal_uart_set_irq_cb(FuriHalUartIdUSART1, gps_uart_on_irq_cb, gps_uart);
  22. furi_hal_uart_set_br(FuriHalUartIdUSART1, GPS_BAUDRATE);
  23. }
  24. static void gps_uart_serial_deinit(GpsUart* gps_uart) {
  25. UNUSED(gps_uart);
  26. furi_hal_uart_set_irq_cb(FuriHalUartIdUSART1, NULL, NULL);
  27. furi_hal_console_enable();
  28. }
  29. static void gps_uart_parse_nmea(GpsUart* gps_uart, char* line)
  30. {
  31. switch(minmea_sentence_id(line, false)) {
  32. case MINMEA_SENTENCE_RMC: {
  33. struct minmea_sentence_rmc frame;
  34. if (minmea_parse_rmc(&frame, line)) {
  35. gps_uart->status.latitude = minmea_tocoord(&frame.latitude);
  36. gps_uart->status.longitude = minmea_tocoord(&frame.longitude);
  37. }
  38. } break;
  39. default: break;
  40. }
  41. }
  42. static int32_t gps_uart_worker(void* context) {
  43. GpsUart* gps_uart = (GpsUart*)context;
  44. gps_uart->rx_stream = xStreamBufferCreate(RX_BUF_SIZE * 5, 1);
  45. size_t rx_offset = 0;
  46. gps_uart_serial_init(gps_uart);
  47. while(1) {
  48. uint32_t events =
  49. furi_thread_flags_wait(WORKER_ALL_RX_EVENTS, FuriFlagWaitAny, FuriWaitForever);
  50. furi_check((events & FuriFlagError) == 0);
  51. if(events & WorkerEvtStop) break;
  52. if(events & WorkerEvtRxDone) {
  53. size_t len = 0;
  54. do {
  55. len = xStreamBufferReceive(gps_uart->rx_stream, gps_uart->rx_buf + rx_offset,
  56. RX_BUF_SIZE - 1 - rx_offset, 0);
  57. if(len > 0) {
  58. rx_offset += len;
  59. gps_uart->rx_buf[rx_offset] = '\0';
  60. char * line_current = (char *)gps_uart->rx_buf;
  61. while(1) {
  62. while (*line_current == '\0' && line_current < (char *)gps_uart->rx_buf + rx_offset - 1)
  63. line_current++;
  64. char * newline = strchr(line_current, '\n');
  65. if(newline) {
  66. *newline = '\0';
  67. gps_uart_parse_nmea(gps_uart, line_current);
  68. line_current = newline + 1;
  69. } else {
  70. if(line_current > (char *)gps_uart->rx_buf) {
  71. rx_offset = 0;
  72. while(*line_current) {
  73. gps_uart->rx_buf[rx_offset++] = *(line_current++);
  74. }
  75. }
  76. break;
  77. }
  78. }
  79. }
  80. } while(len > 0);
  81. }
  82. }
  83. gps_uart_serial_deinit(gps_uart);
  84. vStreamBufferDelete(gps_uart->rx_stream);
  85. return 0;
  86. }
  87. GpsUart* gps_uart_enable() {
  88. GpsUart* gps_uart = malloc(sizeof(GpsUart));
  89. gps_uart->status.latitude = 0.0;
  90. gps_uart->status.longitude = 0.0;
  91. gps_uart->thread = furi_thread_alloc();
  92. furi_thread_set_name(gps_uart->thread, "GpsUartWorker");
  93. furi_thread_set_stack_size(gps_uart->thread, 1024);
  94. furi_thread_set_context(gps_uart->thread, gps_uart);
  95. furi_thread_set_callback(gps_uart->thread, gps_uart_worker);
  96. furi_thread_start(gps_uart->thread);
  97. return gps_uart;
  98. }
  99. void gps_uart_disable(GpsUart* gps_uart) {
  100. furi_assert(gps_uart);
  101. furi_thread_flags_set(furi_thread_get_id(gps_uart->thread), WorkerEvtStop);
  102. furi_thread_join(gps_uart->thread);
  103. furi_thread_free(gps_uart->thread);
  104. free(gps_uart);
  105. }