gps.c 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287
  1. #include "gps_uart.h"
  2. #include "constants.h"
  3. #include <furi.h>
  4. #include <furi_hal_power.h>
  5. #include <gui/gui.h>
  6. #include <string.h>
  7. #include <expansion/expansion.h>
  8. typedef enum {
  9. EventTypeTick,
  10. EventTypeKey,
  11. } EventType;
  12. typedef struct {
  13. EventType type;
  14. InputEvent input;
  15. } PluginEvent;
  16. static void render_callback(Canvas* const canvas, void* context) {
  17. furi_assert(context);
  18. GpsUart* gps_uart = context;
  19. furi_mutex_acquire(gps_uart->mutex, FuriWaitForever);
  20. char buffer[64];
  21. switch(gps_uart->view_state) {
  22. case CHANGE_BAUDRATE:
  23. canvas_set_font(canvas, FontPrimary);
  24. canvas_draw_str_aligned(canvas, 64, 32, AlignCenter, AlignBottom, "Baudrate set to:");
  25. snprintf(buffer, 64, "%ld baud", gps_uart->baudrate);
  26. canvas_draw_str_aligned(canvas, 64, 47, AlignCenter, AlignBottom, buffer);
  27. break;
  28. case CHANGE_BACKLIGHT:
  29. canvas_set_font(canvas, FontPrimary);
  30. canvas_draw_str_aligned(
  31. canvas,
  32. 64,
  33. 32,
  34. AlignCenter,
  35. AlignBottom,
  36. gps_uart->backlight_on ? "Backlight enabled" : "Backlight disabled");
  37. break;
  38. case CHANGE_DEEPSLEEP:
  39. canvas_set_font(canvas, FontPrimary);
  40. canvas_draw_str_aligned(
  41. canvas,
  42. 64,
  43. 32,
  44. AlignCenter,
  45. AlignBottom,
  46. gps_uart->deep_sleep_enabled ? "Deep sleep enabled" : "Deep sleep disabled");
  47. break;
  48. case CHANGE_SPEEDUNIT:
  49. canvas_set_font(canvas, FontPrimary);
  50. canvas_draw_str_aligned(canvas, 64, 32, AlignCenter, AlignBottom, "Speed unit set to:");
  51. switch(gps_uart->speed_units) {
  52. case KPH:
  53. canvas_draw_str_aligned(canvas, 64, 47, AlignCenter, AlignBottom, "km/h");
  54. break;
  55. case MPH:
  56. canvas_draw_str_aligned(canvas, 64, 47, AlignCenter, AlignBottom, "mi/h");
  57. break;
  58. case KNOTS:
  59. default:
  60. canvas_draw_str_aligned(canvas, 64, 47, AlignCenter, AlignBottom, "kn");
  61. break;
  62. }
  63. break;
  64. case NORMAL:
  65. default:
  66. canvas_set_font(canvas, FontPrimary);
  67. canvas_draw_str_aligned(canvas, 32, 8, AlignCenter, AlignBottom, "Latitude");
  68. canvas_draw_str_aligned(canvas, 96, 8, AlignCenter, AlignBottom, "Longitude");
  69. canvas_draw_str_aligned(canvas, 21, 30, AlignCenter, AlignBottom, "Course");
  70. canvas_draw_str_aligned(canvas, 64, 30, AlignCenter, AlignBottom, "Speed");
  71. canvas_draw_str_aligned(canvas, 107, 30, AlignCenter, AlignBottom, "Altitude");
  72. canvas_draw_str_aligned(canvas, 32, 52, AlignCenter, AlignBottom, "Satellites");
  73. canvas_draw_str_aligned(canvas, 96, 52, AlignCenter, AlignBottom, "Last Fix");
  74. canvas_set_font(canvas, FontSecondary);
  75. snprintf(buffer, 64, "%f", (double)gps_uart->status.latitude);
  76. canvas_draw_str_aligned(canvas, 32, 18, AlignCenter, AlignBottom, buffer);
  77. snprintf(buffer, 64, "%f", (double)gps_uart->status.longitude);
  78. canvas_draw_str_aligned(canvas, 96, 18, AlignCenter, AlignBottom, buffer);
  79. snprintf(buffer, 64, "%.1f", (double)gps_uart->status.course);
  80. canvas_draw_str_aligned(canvas, 21, 40, AlignCenter, AlignBottom, buffer);
  81. switch(gps_uart->speed_units) {
  82. case KPH:
  83. snprintf(buffer, 64, "%.2f km/h", (double)(gps_uart->status.speed * KNOTS_TO_KPH));
  84. break;
  85. case MPH:
  86. snprintf(buffer, 64, "%.2f mi/h", (double)(gps_uart->status.speed * KNOTS_TO_MPH));
  87. break;
  88. case KNOTS:
  89. default:
  90. snprintf(buffer, 64, "%.2f kn", (double)gps_uart->status.speed);
  91. break;
  92. }
  93. canvas_draw_str_aligned(canvas, 64, 40, AlignCenter, AlignBottom, buffer);
  94. snprintf(
  95. buffer,
  96. 64,
  97. "%.1f %c",
  98. (double)gps_uart->status.altitude,
  99. tolower(gps_uart->status.altitude_units));
  100. canvas_draw_str_aligned(canvas, 107, 40, AlignCenter, AlignBottom, buffer);
  101. snprintf(buffer, 64, "%d", gps_uart->status.satellites_tracked);
  102. canvas_draw_str_aligned(canvas, 32, 62, AlignCenter, AlignBottom, buffer);
  103. snprintf(
  104. buffer,
  105. 64,
  106. "%02d:%02d:%02d UTC",
  107. gps_uart->status.time_hours,
  108. gps_uart->status.time_minutes,
  109. gps_uart->status.time_seconds);
  110. canvas_draw_str_aligned(canvas, 96, 62, AlignCenter, AlignBottom, buffer);
  111. break;
  112. }
  113. furi_mutex_release(gps_uart->mutex);
  114. }
  115. static void input_callback(InputEvent* input_event, void* ctx) {
  116. furi_assert(ctx);
  117. FuriMessageQueue* event_queue = ctx;
  118. PluginEvent event = {.type = EventTypeKey, .input = *input_event};
  119. furi_message_queue_put(event_queue, &event, FuriWaitForever);
  120. }
  121. int32_t gps_app(void* p) {
  122. UNUSED(p);
  123. // Disable expansion protocol to avoid interference with UART Handle
  124. Expansion* expansion = furi_record_open(RECORD_EXPANSION);
  125. expansion_disable(expansion);
  126. FuriMessageQueue* event_queue = furi_message_queue_alloc(8, sizeof(PluginEvent));
  127. GpsUart* gps_uart = gps_uart_enable();
  128. gps_uart->mutex = furi_mutex_alloc(FuriMutexTypeNormal);
  129. if(!gps_uart->mutex) {
  130. FURI_LOG_E("GPS", "cannot create mutex\r\n");
  131. free(gps_uart);
  132. // Return previous state of expansion
  133. expansion_enable(expansion);
  134. furi_record_close(RECORD_EXPANSION);
  135. return 255;
  136. }
  137. uint8_t attempts = 0;
  138. bool otg_was_enabled = furi_hal_power_is_otg_enabled();
  139. while(!furi_hal_power_is_otg_enabled() && attempts++ < 5) {
  140. furi_hal_power_enable_otg();
  141. furi_delay_ms(10);
  142. }
  143. furi_delay_ms(200);
  144. // set system callbacks
  145. ViewPort* view_port = view_port_alloc();
  146. view_port_draw_callback_set(view_port, render_callback, gps_uart);
  147. view_port_input_callback_set(view_port, input_callback, event_queue);
  148. // open GUI and register view_port
  149. Gui* gui = furi_record_open(RECORD_GUI);
  150. gui_add_view_port(gui, view_port, GuiLayerFullscreen);
  151. PluginEvent event;
  152. for(bool processing = true; processing;) {
  153. FuriStatus event_status = furi_message_queue_get(event_queue, &event, 100);
  154. furi_mutex_acquire(gps_uart->mutex, FuriWaitForever);
  155. if(event_status == FuriStatusOk) {
  156. // press events
  157. if(event.type == EventTypeKey) {
  158. if(event.input.type == InputTypeShort) {
  159. switch(event.input.key) {
  160. case InputKeyBack:
  161. processing = false;
  162. break;
  163. case InputKeyOk:
  164. if(!gps_uart->backlight_on) {
  165. notification_message_block(
  166. gps_uart->notifications, &sequence_display_backlight_enforce_on);
  167. gps_uart->backlight_on = true;
  168. } else {
  169. notification_message_block(
  170. gps_uart->notifications, &sequence_display_backlight_enforce_auto);
  171. notification_message(
  172. gps_uart->notifications, &sequence_display_backlight_off);
  173. gps_uart->backlight_on = false;
  174. }
  175. gps_uart->view_state = CHANGE_BACKLIGHT;
  176. furi_mutex_release(gps_uart->mutex);
  177. view_port_update(view_port);
  178. furi_delay_ms(1000);
  179. gps_uart->view_state = NORMAL;
  180. break;
  181. default:
  182. break;
  183. }
  184. } else if(event.input.type == InputTypeLong) {
  185. switch(event.input.key) {
  186. case InputKeyUp:
  187. gps_uart_deinit_thread(gps_uart);
  188. const int baudrate_length =
  189. sizeof(gps_baudrates) / sizeof(gps_baudrates[0]);
  190. current_gps_baudrate++;
  191. if(current_gps_baudrate >= baudrate_length) {
  192. current_gps_baudrate = 0;
  193. }
  194. gps_uart->baudrate = gps_baudrates[current_gps_baudrate];
  195. gps_uart_init_thread(gps_uart);
  196. gps_uart->view_state = CHANGE_BAUDRATE;
  197. furi_mutex_release(gps_uart->mutex);
  198. view_port_update(view_port);
  199. furi_delay_ms(1000);
  200. gps_uart->view_state = NORMAL;
  201. break;
  202. case InputKeyRight:
  203. gps_uart->speed_units++;
  204. if(gps_uart->speed_units == INVALID) {
  205. gps_uart->speed_units = KNOTS;
  206. }
  207. gps_uart->view_state = CHANGE_SPEEDUNIT;
  208. furi_mutex_release(gps_uart->mutex);
  209. view_port_update(view_port);
  210. furi_delay_ms(1000);
  211. gps_uart->view_state = NORMAL;
  212. break;
  213. case InputKeyDown:
  214. gps_uart->view_state = CHANGE_DEEPSLEEP;
  215. gps_uart->deep_sleep_enabled = !gps_uart->deep_sleep_enabled;
  216. // tested on Telit SE868-A and SL871L-S
  217. furi_hal_serial_tx(
  218. gps_uart->serial_handle,
  219. (uint8_t*)"$PMTK161,0*28\r\n",
  220. strlen("$PMTK161,0*28\r\n"));
  221. furi_mutex_release(gps_uart->mutex);
  222. view_port_update(view_port);
  223. furi_delay_ms(1000);
  224. gps_uart->view_state = NORMAL;
  225. break;
  226. case InputKeyBack:
  227. processing = false;
  228. break;
  229. default:
  230. break;
  231. }
  232. }
  233. }
  234. }
  235. if(gps_uart->view_state == NORMAL) {
  236. furi_mutex_release(gps_uart->mutex);
  237. view_port_update(view_port);
  238. }
  239. }
  240. notification_message_block(gps_uart->notifications, &sequence_display_backlight_enforce_auto);
  241. view_port_enabled_set(view_port, false);
  242. gui_remove_view_port(gui, view_port);
  243. furi_record_close(RECORD_GUI);
  244. view_port_free(view_port);
  245. furi_message_queue_free(event_queue);
  246. furi_mutex_free(gps_uart->mutex);
  247. gps_uart_disable(gps_uart);
  248. if(furi_hal_power_is_otg_enabled() && !otg_was_enabled) {
  249. furi_hal_power_disable_otg();
  250. }
  251. // Return previous state of expansion
  252. expansion_enable(expansion);
  253. furi_record_close(RECORD_EXPANSION);
  254. return 0;
  255. }