gps.c 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286
  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, FuriMessageQueue* event_queue) {
  116. furi_assert(event_queue);
  117. PluginEvent event = {.type = EventTypeKey, .input = *input_event};
  118. furi_message_queue_put(event_queue, &event, FuriWaitForever);
  119. }
  120. int32_t gps_app(void* p) {
  121. UNUSED(p);
  122. // Disable expansion protocol to avoid interference with UART Handle
  123. Expansion* expansion = furi_record_open(RECORD_EXPANSION);
  124. expansion_disable(expansion);
  125. FuriMessageQueue* event_queue = furi_message_queue_alloc(8, sizeof(PluginEvent));
  126. GpsUart* gps_uart = gps_uart_enable();
  127. gps_uart->mutex = furi_mutex_alloc(FuriMutexTypeNormal);
  128. if(!gps_uart->mutex) {
  129. FURI_LOG_E("GPS", "cannot create mutex\r\n");
  130. free(gps_uart);
  131. // Return previous state of expansion
  132. expansion_enable(expansion);
  133. furi_record_close(RECORD_EXPANSION);
  134. return 255;
  135. }
  136. uint8_t attempts = 0;
  137. bool otg_was_enabled = furi_hal_power_is_otg_enabled();
  138. while(!furi_hal_power_is_otg_enabled() && attempts++ < 5) {
  139. furi_hal_power_enable_otg();
  140. furi_delay_ms(10);
  141. }
  142. furi_delay_ms(200);
  143. // set system callbacks
  144. ViewPort* view_port = view_port_alloc();
  145. view_port_draw_callback_set(view_port, render_callback, gps_uart);
  146. view_port_input_callback_set(view_port, input_callback, event_queue);
  147. // open GUI and register view_port
  148. Gui* gui = furi_record_open(RECORD_GUI);
  149. gui_add_view_port(gui, view_port, GuiLayerFullscreen);
  150. PluginEvent event;
  151. for(bool processing = true; processing;) {
  152. FuriStatus event_status = furi_message_queue_get(event_queue, &event, 100);
  153. furi_mutex_acquire(gps_uart->mutex, FuriWaitForever);
  154. if(event_status == FuriStatusOk) {
  155. // press events
  156. if(event.type == EventTypeKey) {
  157. if(event.input.type == InputTypeShort) {
  158. switch(event.input.key) {
  159. case InputKeyBack:
  160. processing = false;
  161. break;
  162. case InputKeyOk:
  163. if(!gps_uart->backlight_on) {
  164. notification_message_block(
  165. gps_uart->notifications, &sequence_display_backlight_enforce_on);
  166. gps_uart->backlight_on = true;
  167. } else {
  168. notification_message_block(
  169. gps_uart->notifications, &sequence_display_backlight_enforce_auto);
  170. notification_message(
  171. gps_uart->notifications, &sequence_display_backlight_off);
  172. gps_uart->backlight_on = false;
  173. }
  174. gps_uart->view_state = CHANGE_BACKLIGHT;
  175. furi_mutex_release(gps_uart->mutex);
  176. view_port_update(view_port);
  177. furi_delay_ms(1000);
  178. gps_uart->view_state = NORMAL;
  179. break;
  180. default:
  181. break;
  182. }
  183. } else if(event.input.type == InputTypeLong) {
  184. switch(event.input.key) {
  185. case InputKeyUp:
  186. gps_uart_deinit_thread(gps_uart);
  187. const int baudrate_length =
  188. sizeof(gps_baudrates) / sizeof(gps_baudrates[0]);
  189. current_gps_baudrate++;
  190. if(current_gps_baudrate >= baudrate_length) {
  191. current_gps_baudrate = 0;
  192. }
  193. gps_uart->baudrate = gps_baudrates[current_gps_baudrate];
  194. gps_uart_init_thread(gps_uart);
  195. gps_uart->view_state = CHANGE_BAUDRATE;
  196. furi_mutex_release(gps_uart->mutex);
  197. view_port_update(view_port);
  198. furi_delay_ms(1000);
  199. gps_uart->view_state = NORMAL;
  200. break;
  201. case InputKeyRight:
  202. gps_uart->speed_units++;
  203. if(gps_uart->speed_units == INVALID) {
  204. gps_uart->speed_units = KNOTS;
  205. }
  206. gps_uart->view_state = CHANGE_SPEEDUNIT;
  207. furi_mutex_release(gps_uart->mutex);
  208. view_port_update(view_port);
  209. furi_delay_ms(1000);
  210. gps_uart->view_state = NORMAL;
  211. break;
  212. case InputKeyDown:
  213. gps_uart->view_state = CHANGE_DEEPSLEEP;
  214. gps_uart->deep_sleep_enabled = !gps_uart->deep_sleep_enabled;
  215. // tested on Telit SE868-A and SL871L-S
  216. furi_hal_serial_tx(
  217. gps_uart->serial_handle,
  218. (uint8_t*)"$PMTK161,0*28\r\n",
  219. strlen("$PMTK161,0*28\r\n"));
  220. furi_mutex_release(gps_uart->mutex);
  221. view_port_update(view_port);
  222. furi_delay_ms(1000);
  223. gps_uart->view_state = NORMAL;
  224. break;
  225. case InputKeyBack:
  226. processing = false;
  227. break;
  228. default:
  229. break;
  230. }
  231. }
  232. }
  233. }
  234. if(gps_uart->view_state == NORMAL) {
  235. furi_mutex_release(gps_uart->mutex);
  236. view_port_update(view_port);
  237. }
  238. }
  239. notification_message_block(gps_uart->notifications, &sequence_display_backlight_enforce_auto);
  240. view_port_enabled_set(view_port, false);
  241. gui_remove_view_port(gui, view_port);
  242. furi_record_close(RECORD_GUI);
  243. view_port_free(view_port);
  244. furi_message_queue_free(event_queue);
  245. furi_mutex_free(gps_uart->mutex);
  246. gps_uart_disable(gps_uart);
  247. if(furi_hal_power_is_otg_enabled() && !otg_was_enabled) {
  248. furi_hal_power_disable_otg();
  249. }
  250. // Return previous state of expansion
  251. expansion_enable(expansion);
  252. furi_record_close(RECORD_EXPANSION);
  253. return 0;
  254. }