| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196 |
- #include <string.h>
- #include "minmea.h"
- #include "gps_uart.h"
- typedef enum
- {
- WorkerEvtStop = (1 << 0),
- WorkerEvtRxDone = (1 << 1),
- } WorkerEvtFlags;
- #define WORKER_ALL_RX_EVENTS (WorkerEvtStop | WorkerEvtRxDone)
- static void gps_uart_on_irq_cb(UartIrqEvent ev, uint8_t data, void* context)
- {
- GpsUart* gps_uart = (GpsUart*)context;
- if (ev == UartIrqEventRXNE)
- {
- furi_stream_buffer_send(gps_uart->rx_stream, &data, 1, 0);
- furi_thread_flags_set(furi_thread_get_id(gps_uart->thread), WorkerEvtRxDone);
- }
- }
- static void gps_uart_serial_init(GpsUart* gps_uart)
- {
- furi_hal_console_disable();
- furi_hal_uart_set_irq_cb(FuriHalUartIdUSART1, gps_uart_on_irq_cb, gps_uart);
- furi_hal_uart_set_br(FuriHalUartIdUSART1, GPS_BAUDRATE);
- }
- static void gps_uart_serial_deinit(GpsUart* gps_uart)
- {
- UNUSED(gps_uart);
- furi_hal_uart_set_irq_cb(FuriHalUartIdUSART1, NULL, NULL);
- furi_hal_console_enable();
- }
- static void gps_uart_parse_nmea(GpsUart* gps_uart, char* line)
- {
- switch (minmea_sentence_id(line, false))
- {
- case MINMEA_SENTENCE_RMC:
- {
- struct minmea_sentence_rmc frame;
- if (minmea_parse_rmc(&frame, line))
- {
- gps_uart->status.valid = frame.valid;
- gps_uart->status.latitude = minmea_tocoord(&frame.latitude);
- gps_uart->status.longitude = minmea_tocoord(&frame.longitude);
- gps_uart->status.speed = minmea_tofloat(&frame.speed);
- gps_uart->status.course = minmea_tofloat(&frame.course);
- gps_uart->status.time_hours = frame.time.hours;
- gps_uart->status.time_minutes = frame.time.minutes;
- gps_uart->status.time_seconds = frame.time.seconds;
- notification_message_block(gps_uart->notifications, &sequence_blink_green_10);
- }
- } break;
- case MINMEA_SENTENCE_GGA:
- {
- struct minmea_sentence_gga frame;
- if (minmea_parse_gga(&frame, line))
- {
- gps_uart->status.latitude = minmea_tocoord(&frame.latitude);
- gps_uart->status.longitude = minmea_tocoord(&frame.longitude);
- gps_uart->status.altitude = minmea_tofloat(&frame.altitude);
- gps_uart->status.altitude_units = frame.altitude_units;
- gps_uart->status.fix_quality = frame.fix_quality;
- gps_uart->status.satellites_tracked = frame.satellites_tracked;
- gps_uart->status.time_hours = frame.time.hours;
- gps_uart->status.time_minutes = frame.time.minutes;
- gps_uart->status.time_seconds = frame.time.seconds;
- notification_message_block(gps_uart->notifications, &sequence_blink_magenta_10);
- }
- } break;
- default:
- break;
- }
- }
- static int32_t gps_uart_worker(void* context)
- {
- GpsUart* gps_uart = (GpsUart*)context;
- gps_uart->rx_stream = furi_stream_buffer_alloc(RX_BUF_SIZE * 5, 1);
- size_t rx_offset = 0;
- gps_uart_serial_init(gps_uart);
- while (1)
- {
- uint32_t events =
- furi_thread_flags_wait(WORKER_ALL_RX_EVENTS, FuriFlagWaitAny, FuriWaitForever);
- furi_check((events & FuriFlagError) == 0);
- if (events & WorkerEvtStop)
- {
- break;
- }
-
- if (events & WorkerEvtRxDone)
- {
- size_t len = 0;
- do
- {
- len = furi_stream_buffer_receive(gps_uart->rx_stream, gps_uart->rx_buf + rx_offset, RX_BUF_SIZE - 1 - rx_offset,
- 0);
- if (len > 0)
- {
- rx_offset += len;
- gps_uart->rx_buf[rx_offset] = '\0';
- char * line_current = (char *)gps_uart->rx_buf;
- while (1)
- {
- while (*line_current == '\0' && line_current < (char *)gps_uart->rx_buf + rx_offset - 1)
- {
- line_current++;
- }
- char * newline = strchr(line_current, '\n');
- if (newline)
- {
- *newline = '\0';
- gps_uart_parse_nmea(gps_uart, line_current);
- line_current = newline + 1;
- }
- else
- {
- if (line_current > (char *)gps_uart->rx_buf)
- {
- rx_offset = 0;
- while (*line_current)
- {
- gps_uart->rx_buf[rx_offset++] = *(line_current++);
- }
- }
- break;
- }
- }
- }
- }
- while (len > 0);
- }
- }
- gps_uart_serial_deinit(gps_uart);
- furi_stream_buffer_free(gps_uart->rx_stream);
- return 0;
- }
- GpsUart* gps_uart_enable()
- {
- GpsUart* gps_uart = malloc(sizeof(GpsUart));
- gps_uart->status.valid = false;
- gps_uart->status.latitude = 0.0;
- gps_uart->status.longitude = 0.0;
- gps_uart->status.speed = 0.0;
- gps_uart->status.course = 0.0;
- gps_uart->status.altitude = 0.0;
- gps_uart->status.altitude_units = ' ';
- gps_uart->status.fix_quality = 0;
- gps_uart->status.satellites_tracked = 0;
- gps_uart->status.time_hours = 0;
- gps_uart->status.time_minutes = 0;
- gps_uart->status.time_seconds = 0;
- gps_uart->notifications = furi_record_open(RECORD_NOTIFICATION);
- gps_uart->thread = furi_thread_alloc();
- furi_thread_set_name(gps_uart->thread, "GpsUartWorker");
- furi_thread_set_stack_size(gps_uart->thread, 1024);
- furi_thread_set_context(gps_uart->thread, gps_uart);
- furi_thread_set_callback(gps_uart->thread, gps_uart_worker);
- furi_thread_start(gps_uart->thread);
- return gps_uart;
- }
- void gps_uart_disable(GpsUart* gps_uart)
- {
- furi_assert(gps_uart);
- furi_thread_flags_set(furi_thread_get_id(gps_uart->thread), WorkerEvtStop);
- furi_thread_join(gps_uart->thread);
- furi_thread_free(gps_uart->thread);
- furi_record_close(RECORD_NOTIFICATION);
- free(gps_uart);
- }
|