ublox_worker.c 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661
  1. #include "ublox_worker_i.h"
  2. #define TAG "UbloxWorker"
  3. UbloxWorker* ublox_worker_alloc() {
  4. UbloxWorker* ublox_worker = malloc(sizeof(UbloxWorker));
  5. ublox_worker->thread =
  6. furi_thread_alloc_ex("UbloxWorker", 2 * 1024, ublox_worker_task, ublox_worker);
  7. ublox_worker->callback = NULL;
  8. ublox_worker->context = NULL;
  9. ublox_worker_change_state(ublox_worker, UbloxWorkerStateReady);
  10. return ublox_worker;
  11. }
  12. void ublox_worker_free(UbloxWorker* ublox_worker) {
  13. furi_assert(ublox_worker);
  14. furi_thread_free(ublox_worker->thread);
  15. free(ublox_worker);
  16. }
  17. UbloxWorkerState ublox_worker_get_state(UbloxWorker* ublox_worker) {
  18. return ublox_worker->state;
  19. }
  20. void ublox_worker_start(
  21. UbloxWorker* ublox_worker,
  22. UbloxWorkerState state,
  23. UbloxWorkerCallback callback,
  24. void* context) {
  25. furi_assert(ublox_worker);
  26. ublox_worker->callback = callback;
  27. ublox_worker->context = context;
  28. ublox_worker_change_state(ublox_worker, state);
  29. furi_thread_start(ublox_worker->thread);
  30. }
  31. void ublox_worker_stop(UbloxWorker* ublox_worker) {
  32. furi_assert(ublox_worker);
  33. furi_assert(ublox_worker->thread);
  34. Ublox* ublox = ublox_worker->context;
  35. furi_assert(ublox);
  36. //FURI_LOG_I(TAG, "worker_stop: %p", ublox);
  37. /*FuriThreadState state = furi_thread_get_state(ublox_worker->thread);
  38. if (state == FuriThreadStateStopped) {
  39. FURI_LOG_I(TAG, "worker state stopped");
  40. } else if (state == FuriThreadStateStarting) {
  41. FURI_LOG_I(TAG, "worker state starting");
  42. } else if (state == FuriThreadStateRunning) {
  43. FURI_LOG_I(TAG, "worker state running");
  44. }*/
  45. // close the logfile if currently logging
  46. //FURI_LOG_I(TAG, "log state: %d", ublox->log_state);
  47. if(ublox->log_state == UbloxLogStateLogging) {
  48. FURI_LOG_I(TAG, "closing log file on worker stop");
  49. ublox->log_state = UbloxLogStateNone;
  50. if (!kml_close_file(&(ublox->kmlfile))) {
  51. FURI_LOG_E(TAG, "failed to close KML file!");
  52. }
  53. }
  54. if(furi_thread_get_state(ublox_worker->thread) != FuriThreadStateStopped) {
  55. ublox_worker_change_state(ublox_worker, UbloxWorkerStateStop);
  56. furi_thread_join(ublox_worker->thread);
  57. }
  58. }
  59. void ublox_worker_change_state(UbloxWorker* ublox_worker, UbloxWorkerState state) {
  60. ublox_worker->state = state;
  61. }
  62. void clear_ublox_data() {
  63. int fails = 0;
  64. if (!furi_hal_i2c_is_device_ready(
  65. &furi_hal_i2c_handle_external,
  66. UBLOX_I2C_ADDRESS << 1,
  67. furi_ms_to_ticks(I2C_TIMEOUT_MS))) {
  68. FURI_LOG_E(TAG, "clear_ublox_data(): device not ready");
  69. return;
  70. }
  71. uint8_t tx[] = {0xff};
  72. uint8_t response = 0;
  73. while(response != 0xff && fails < 30) {
  74. if(!furi_hal_i2c_trx(
  75. &furi_hal_i2c_handle_external,
  76. UBLOX_I2C_ADDRESS << 1,
  77. tx, 1,
  78. &response, 1,
  79. furi_ms_to_ticks(I2C_TIMEOUT_MS))) {
  80. // if the GPS is disconnected during this loop, this will
  81. // loop forever, we must make that not happen. 30 loops is
  82. // plenty, and if the clearing doesn't work, the requisite
  83. // error will be generated by the caller on the next
  84. // actual attempt to reach the GPS.
  85. fails++;
  86. FURI_LOG_E(TAG, "clear_ublox_data(): error clearing ublox data");
  87. }
  88. }
  89. }
  90. int32_t ublox_worker_task(void* context) {
  91. UbloxWorker* ublox_worker = context;
  92. furi_hal_i2c_acquire(&furi_hal_i2c_handle_external);
  93. if(ublox_worker->state == UbloxWorkerStateRead) {
  94. ublox_worker_read_nav_messages(context);
  95. } else if (ublox_worker->state == UbloxWorkerStateSyncTime) {
  96. FURI_LOG_I(TAG, "sync time");
  97. ublox_worker_sync_to_gps_time(ublox_worker);
  98. } else if(ublox_worker->state == UbloxWorkerStateResetOdometer) {
  99. ublox_worker_reset_odo(ublox_worker);
  100. } else if(ublox_worker->state == UbloxWorkerStateStop) {
  101. FURI_LOG_D(TAG, "state stop");
  102. }
  103. ublox_worker_change_state(ublox_worker, UbloxWorkerStateStop);
  104. furi_hal_i2c_release(&furi_hal_i2c_handle_external);
  105. return 0;
  106. }
  107. void ublox_worker_read_nav_messages(void* context) {
  108. // this function is fairly complicated: it inits the GPS, handles
  109. // logging states, and reads data from the GPS to push it to the
  110. // main app struct.
  111. // IMPORTANT NOTE: we don't use a timer that continually respawns
  112. // the thread because that causes a memory leak.
  113. UbloxWorker* ublox_worker = context;
  114. Ublox* ublox = ublox_worker->context;
  115. // We only start logging at the same time we restart the worker.
  116. if(ublox->log_state == UbloxLogStateStartLogging) {
  117. FURI_LOG_I(TAG, "start logging");
  118. // assemble full logfile pathname
  119. FuriString* fullname = furi_string_alloc();
  120. path_concat(furi_string_get_cstr(ublox->logfile_folder), ublox->text_store, fullname);
  121. FURI_LOG_I(TAG, "fullname is %s", furi_string_get_cstr(fullname));
  122. if (!kml_open_file(ublox->storage, &(ublox->kmlfile), furi_string_get_cstr(fullname))) {
  123. FURI_LOG_E(TAG, "failed to open KML file %s!", furi_string_get_cstr(fullname));
  124. ublox->log_state = UbloxLogStateNone;
  125. }
  126. ublox->log_state = UbloxLogStateLogging;
  127. furi_string_free(fullname);
  128. ublox_worker->callback(UbloxWorkerEventLogStateChanged, ublox_worker->context);
  129. }
  130. while(!ublox->gps_initted) {
  131. if(ublox_worker->state != UbloxWorkerStateRead) {
  132. return;
  133. }
  134. // have to clear right before init to make retrying init work
  135. clear_ublox_data();
  136. if(ublox_worker_init_gps(ublox_worker)) {
  137. ublox->gps_initted = true;
  138. break;
  139. } else {
  140. ublox_worker->callback(UbloxWorkerEventFailed, ublox_worker->context);
  141. FURI_LOG_E(TAG, "init GPS failed, try again");
  142. }
  143. // don't try constantly, no reason to
  144. furi_delay_ms(500);
  145. }
  146. // clear data so we don't an error on startup
  147. clear_ublox_data();
  148. // break the loop when the thread state changes
  149. while(ublox_worker->state == UbloxWorkerStateRead) {
  150. // we interrupt with checking the state to help reduce
  151. // lag. it's not perfect, but it does overall improve things.
  152. bool pvt = ublox_worker_read_pvt(ublox_worker);
  153. if (ublox_worker->state != UbloxWorkerStateRead) break;
  154. // clearing makes the second read much faster
  155. clear_ublox_data();
  156. if (ublox_worker->state != UbloxWorkerStateRead) break;
  157. bool odo = ublox_worker_read_odo(ublox_worker);
  158. if (pvt && odo) {
  159. ublox_worker->callback(UbloxWorkerEventDataReady, ublox_worker->context);
  160. if (ublox->log_state == UbloxLogStateLogging) {
  161. if (!kml_add_path_point(&(ublox->kmlfile),
  162. // ublox returns values as floats * 1e7 in int form
  163. (double)(ublox->nav_pvt.lat) / (double)1e7,
  164. (double)(ublox->nav_pvt.lon) / (double)1e7,
  165. ublox->nav_pvt.hMSL / 1e3)) { // convert altitude to meters
  166. FURI_LOG_E(TAG, "failed to write line to file");
  167. }
  168. }
  169. } else {
  170. ublox_worker->callback(UbloxWorkerEventFailed, ublox_worker->context);
  171. }
  172. uint32_t ticks = furi_get_tick();
  173. while(furi_get_tick() - ticks < furi_ms_to_ticks(((ublox->data_display_state).refresh_rate * 1000))) {
  174. // putting this here (should) make the logging response faster
  175. if(ublox->log_state == UbloxLogStateStopLogging) {
  176. FURI_LOG_I(TAG, "stop logging");
  177. if (!kml_close_file(&(ublox->kmlfile))) {
  178. FURI_LOG_E(TAG, "failed to close KML file!");
  179. }
  180. ublox->log_state = UbloxLogStateNone;
  181. ublox_worker->callback(UbloxWorkerEventLogStateChanged, ublox_worker->context);
  182. }
  183. if(ublox_worker->state != UbloxWorkerStateRead) {
  184. return;
  185. }
  186. }
  187. }
  188. }
  189. void ublox_worker_sync_to_gps_time(void* context) {
  190. UbloxWorker* ublox_worker = context;
  191. Ublox* ublox = ublox_worker->context;
  192. UbloxFrame frame_tx;
  193. frame_tx.class = UBX_NAV_CLASS;
  194. frame_tx.id = UBX_NAV_TIMEUTC_MESSAGE;
  195. frame_tx.len = 0;
  196. frame_tx.payload = NULL;
  197. UbloxMessage* message_tx = ublox_frame_to_bytes(&frame_tx);
  198. UbloxMessage* message_rx = ublox_worker_i2c_transfer(message_tx, UBX_NAV_TIMEUTC_MESSAGE_LENGTH);
  199. if(message_rx == NULL) {
  200. FURI_LOG_E(TAG, "get_gps_time transfer failed");
  201. ublox_worker_change_state(ublox_worker, UbloxWorkerStateStop);
  202. ublox_worker->callback(UbloxWorkerEventFailed, ublox_worker->context);
  203. return;
  204. }
  205. FURI_LOG_I(TAG, "got message");
  206. UbloxFrame* frame_rx = ublox_bytes_to_frame(message_rx);
  207. ublox_message_free(message_rx);
  208. if(frame_rx == NULL) {
  209. FURI_LOG_E(TAG, "NULL pointer, something wrong with NAV-TIMEUTC message!");
  210. ublox_worker_change_state(ublox_worker, UbloxWorkerStateStop);
  211. ublox_worker->callback(UbloxWorkerEventFailed, ublox_worker->context);
  212. return;
  213. } else {
  214. Ublox_NAV_TIMEUTC_Message nav_timeutc = {
  215. .iTOW = (frame_rx->payload[0]) | (frame_rx->payload[1] << 8) |
  216. (frame_rx->payload[2] << 16) | (frame_rx->payload[3] << 24),
  217. .tAcc = (frame_rx->payload[4]) | (frame_rx->payload[5] << 8) |
  218. (frame_rx->payload[6] << 16) | (frame_rx->payload[7] << 24),
  219. .nano = (frame_rx->payload[8]) | (frame_rx->payload[9] << 8) |
  220. (frame_rx->payload[10] << 16) | (frame_rx->payload[11] << 24),
  221. .year = (frame_rx->payload[12]) | (frame_rx->payload[13] << 8),
  222. .month = frame_rx->payload[14],
  223. .day = frame_rx->payload[15],
  224. .hour = frame_rx->payload[16],
  225. .min = frame_rx->payload[17],
  226. .sec = frame_rx->payload[18],
  227. .valid = frame_rx->payload[19],
  228. };
  229. ublox->nav_timeutc = nav_timeutc;
  230. ublox_frame_free(frame_rx);
  231. ublox_worker->callback(UbloxWorkerEventDataReady, ublox_worker->context);
  232. }
  233. }
  234. FuriString* print_uint8_array(uint8_t* array, int length) {
  235. FuriString* s = furi_string_alloc();
  236. for(int i = 0; i < length - 1; i++) {
  237. furi_string_cat_printf(s, "%x, ", array[i]);
  238. }
  239. furi_string_cat_printf(s, "%x", array[length - 1]);
  240. return s;
  241. }
  242. UbloxMessage* ublox_worker_i2c_transfer(UbloxMessage* message_tx, uint8_t read_length) {
  243. if (!furi_hal_i2c_is_device_ready(
  244. &furi_hal_i2c_handle_external,
  245. UBLOX_I2C_ADDRESS << 1,
  246. furi_ms_to_ticks(I2C_TIMEOUT_MS))) {
  247. FURI_LOG_E(TAG, "device not ready");
  248. return NULL;
  249. }
  250. // Either our I2C implementation is broken or the GPS's is, so we
  251. // end up reading a lot more data than we need to. That means that
  252. // the I2C comm code for this app is a little bit of a hack, but
  253. // it works fine and is fast enough, so I don't really care. It
  254. // certainly doesn't break the GPS.
  255. if(!furi_hal_i2c_tx(
  256. &furi_hal_i2c_handle_external,
  257. UBLOX_I2C_ADDRESS << 1,
  258. message_tx->message, message_tx->length,
  259. furi_ms_to_ticks(I2C_TIMEOUT_MS))) {
  260. FURI_LOG_E(TAG, "error writing message to GPS");
  261. return NULL;
  262. }
  263. uint8_t* response = malloc((size_t)read_length);
  264. // The GPS sends 0xff until it has a complete message to respond
  265. // with. We have to wait until it stops sending that. (Why this
  266. // works is a little bit...uh, well, I don't know. Shouldn't reading
  267. // more bytes make it so that the data is completely read out and no
  268. // longer available?)
  269. //FURI_LOG_I(TAG, "start ticks at %lu", furi_get_tick()); // returns ms
  270. while(true) {
  271. if(!furi_hal_i2c_rx(
  272. &furi_hal_i2c_handle_external,
  273. UBLOX_I2C_ADDRESS << 1,
  274. response, 1,
  275. furi_ms_to_ticks(I2C_TIMEOUT_MS))) {
  276. FURI_LOG_E(TAG, "error reading first byte of response");
  277. free(response);
  278. return NULL;
  279. }
  280. // checking with 0xb5 prevents strange bursts of junk data from becoming an issue.
  281. if(response[0] != 0xff && response[0] == 0xb5) {
  282. //FURI_LOG_I(TAG, "read rest of message at %lu", furi_get_tick());
  283. if(!furi_hal_i2c_rx(
  284. &furi_hal_i2c_handle_external,
  285. UBLOX_I2C_ADDRESS << 1,
  286. &(response[1]), read_length - 1, // first byte already read
  287. furi_ms_to_ticks(I2C_TIMEOUT_MS))) {
  288. FURI_LOG_E(TAG, "error reading rest of response");
  289. free(response);
  290. return NULL;
  291. }
  292. break;
  293. }
  294. furi_delay_ms(1);
  295. }
  296. UbloxMessage* message_rx = malloc(sizeof(UbloxMessage));
  297. message_rx->message = response;
  298. message_rx->length = read_length;
  299. return message_rx; // message_rx->message needs to be freed later
  300. }
  301. bool ublox_worker_read_pvt(UbloxWorker* ublox_worker) {
  302. //FURI_LOG_I(TAG, "mem free before PVT read: %u", memmgr_get_free_heap());
  303. Ublox* ublox = ublox_worker->context;
  304. // Read NAV-PVT by sending NAV-PVT with no payload
  305. UbloxFrame* frame_tx = malloc(sizeof(UbloxFrame));
  306. frame_tx->class = UBX_NAV_CLASS;
  307. frame_tx->id = UBX_NAV_PVT_MESSAGE;
  308. frame_tx->len = 0;
  309. frame_tx->payload = NULL;
  310. UbloxMessage* message_tx = ublox_frame_to_bytes(frame_tx);
  311. ublox_frame_free(frame_tx);
  312. UbloxMessage* message_rx = ublox_worker_i2c_transfer(message_tx, UBX_NAV_PVT_MESSAGE_LENGTH);
  313. ublox_message_free(message_tx);
  314. if(message_rx == NULL) {
  315. FURI_LOG_E(TAG, "read_pvt transfer failed");
  316. //ublox_worker_change_state(ublox_worker, UbloxWorkerStateStop);
  317. return false;
  318. }
  319. UbloxFrame* frame_rx = ublox_bytes_to_frame(message_rx);
  320. ublox_message_free(message_rx);
  321. if(frame_rx == NULL) {
  322. FURI_LOG_E(TAG, "NULL pointer, something wrong with NAV-PVT message!");
  323. //ublox_worker_change_state(ublox_worker, UbloxWorkerStateStop);
  324. return false;
  325. } else {
  326. // build nav-pvt struct. this is very ugly and there's not much I can do about it.
  327. Ublox_NAV_PVT_Message nav_pvt = {
  328. .iTOW = (frame_rx->payload[0]) | (frame_rx->payload[1] << 8) |
  329. (frame_rx->payload[2] << 16) | (frame_rx->payload[3] << 24),
  330. .year = (frame_rx->payload[4]) | (frame_rx->payload[5] << 8),
  331. .month = frame_rx->payload[6],
  332. .day = frame_rx->payload[7],
  333. .hour = frame_rx->payload[8],
  334. .min = frame_rx->payload[9],
  335. .sec = frame_rx->payload[10],
  336. .valid = frame_rx->payload[11],
  337. .tAcc = (frame_rx->payload[12]) | (frame_rx->payload[13] << 8) |
  338. (frame_rx->payload[14] << 16) | (frame_rx->payload[15] << 24),
  339. .nano = (frame_rx->payload[16]) | (frame_rx->payload[17] << 8) |
  340. (frame_rx->payload[18] << 16) | (frame_rx->payload[19] << 24),
  341. .fixType = frame_rx->payload[20],
  342. .flags = frame_rx->payload[21],
  343. .flags2 = frame_rx->payload[22],
  344. .numSV = frame_rx->payload[23],
  345. .lon = (frame_rx->payload[24]) | (frame_rx->payload[25] << 8) |
  346. (frame_rx->payload[26] << 16) | (frame_rx->payload[27] << 24),
  347. .lat = (frame_rx->payload[28]) | (frame_rx->payload[29] << 8) |
  348. (frame_rx->payload[30] << 16) | (frame_rx->payload[31] << 24),
  349. .height = (frame_rx->payload[32]) | (frame_rx->payload[33] << 8) |
  350. (frame_rx->payload[34] << 16) | (frame_rx->payload[35] << 24),
  351. .hMSL = (frame_rx->payload[36]) | (frame_rx->payload[37] << 8) |
  352. (frame_rx->payload[38] << 16) | (frame_rx->payload[39] << 24),
  353. .hAcc = (frame_rx->payload[40]) | (frame_rx->payload[41] << 8) |
  354. (frame_rx->payload[42] << 16) | (frame_rx->payload[43] << 24),
  355. .vAcc = (frame_rx->payload[44]) | (frame_rx->payload[45] << 8) |
  356. (frame_rx->payload[46] << 16) | (frame_rx->payload[47] << 24),
  357. .velN = (frame_rx->payload[48]) | (frame_rx->payload[49] << 8) |
  358. (frame_rx->payload[50] << 16) | (frame_rx->payload[51] << 24),
  359. .velE = (frame_rx->payload[52]) | (frame_rx->payload[53] << 8) |
  360. (frame_rx->payload[54] << 16) | (frame_rx->payload[55] << 24),
  361. .velD = (frame_rx->payload[56]) | (frame_rx->payload[57] << 8) |
  362. (frame_rx->payload[58] << 16) | (frame_rx->payload[59] << 24),
  363. .gSpeed = (frame_rx->payload[60]) | (frame_rx->payload[61] << 8) |
  364. (frame_rx->payload[62] << 16) | (frame_rx->payload[63] << 24),
  365. .headMot = (frame_rx->payload[64]) | (frame_rx->payload[65] << 8) |
  366. (frame_rx->payload[66] << 16) | (frame_rx->payload[67] << 24),
  367. .sAcc = (frame_rx->payload[68]) | (frame_rx->payload[69] << 8) |
  368. (frame_rx->payload[70] << 16) | (frame_rx->payload[71] << 24),
  369. .headAcc = (frame_rx->payload[72]) | (frame_rx->payload[73] << 8) |
  370. (frame_rx->payload[74] << 16) | (frame_rx->payload[75] << 24),
  371. .pDOP = (frame_rx->payload[76]) | (frame_rx->payload[77] << 8),
  372. .flags3 = (frame_rx->payload[78]) | (frame_rx->payload[79] << 8),
  373. .reserved1 = frame_rx->payload[80],
  374. .reserved2 = frame_rx->payload[81],
  375. .reserved3 = frame_rx->payload[82],
  376. .reserved4 = frame_rx->payload[83],
  377. .headVeh = (frame_rx->payload[84]) | (frame_rx->payload[85] << 8) |
  378. (frame_rx->payload[86] << 16) | (frame_rx->payload[87] << 24),
  379. .magDec = (frame_rx->payload[88]) | (frame_rx->payload[89] << 8),
  380. .magAcc = (frame_rx->payload[90]) | (frame_rx->payload[91] << 8),
  381. };
  382. // Using a local variable for nav_pvt is fine, because nav_pvt in
  383. // the Ublox struct is also not a pointer, so this assignment
  384. // effectively compiles to a memcpy.
  385. ublox->nav_pvt = nav_pvt;
  386. ublox_frame_free(frame_rx);
  387. return true;
  388. }
  389. return false;
  390. }
  391. bool ublox_worker_read_odo(UbloxWorker* ublox_worker) {
  392. Ublox* ublox = ublox_worker->context;
  393. UbloxFrame* frame_tx = malloc(sizeof(UbloxFrame));
  394. frame_tx->class = UBX_NAV_CLASS;
  395. frame_tx->id = UBX_NAV_ODO_MESSAGE;
  396. frame_tx->len = 0;
  397. frame_tx->payload = NULL;
  398. UbloxMessage* message_tx = ublox_frame_to_bytes(frame_tx);
  399. ublox_frame_free(frame_tx);
  400. UbloxMessage* message_rx = ublox_worker_i2c_transfer(message_tx, UBX_NAV_ODO_MESSAGE_LENGTH);
  401. ublox_message_free(message_tx);
  402. if(message_rx == NULL) {
  403. FURI_LOG_E(TAG, "read_odo transfer failed");
  404. return false;
  405. }
  406. UbloxFrame* frame_rx = ublox_bytes_to_frame(message_rx);
  407. ublox_message_free(message_rx);
  408. if(frame_rx == NULL) {
  409. FURI_LOG_E(TAG, "NULL pointer, something wrong with NAV-ODO message!");
  410. return false;
  411. } else {
  412. Ublox_NAV_ODO_Message nav_odo = {
  413. .version = frame_rx->payload[0],
  414. .reserved1 = frame_rx->payload[1],
  415. .reserved2 = frame_rx->payload[2],
  416. .reserved3 = frame_rx->payload[3],
  417. .iTOW = (frame_rx->payload[4]) | (frame_rx->payload[5] << 8) |
  418. (frame_rx->payload[6] << 16) | (frame_rx->payload[7] << 24),
  419. .distance = (frame_rx->payload[8]) | (frame_rx->payload[9] << 8) |
  420. (frame_rx->payload[10] << 16) | (frame_rx->payload[11] << 24),
  421. .totalDistance = (frame_rx->payload[12]) | (frame_rx->payload[13] << 8) |
  422. (frame_rx->payload[14] << 16) | (frame_rx->payload[15] << 24),
  423. .distanceStd = (frame_rx->payload[16]) | (frame_rx->payload[17] << 8) |
  424. (frame_rx->payload[18] << 16) | (frame_rx->payload[19] << 24),
  425. };
  426. FURI_LOG_I(TAG, "odo (m): %lu", nav_odo.distance);
  427. ublox->nav_odo = nav_odo;
  428. ublox_frame_free(frame_rx);
  429. return true;
  430. }
  431. }
  432. /**
  433. * Set the power mode to "Balanced", enable the odometer, and
  434. * configure odometer and dynamic platform model according to user
  435. * settings.
  436. */
  437. bool ublox_worker_init_gps(UbloxWorker* ublox_worker) {
  438. Ublox* ublox = ublox_worker->context;
  439. // Set power mode
  440. /*** read initial cfg-pms configuration first ***/
  441. UbloxFrame pms_frame_tx;
  442. pms_frame_tx.class = UBX_CFG_CLASS;
  443. pms_frame_tx.id = UBX_CFG_PMS_MESSAGE;
  444. pms_frame_tx.len = 0;
  445. pms_frame_tx.payload = NULL;
  446. UbloxMessage* pms_message_tx = ublox_frame_to_bytes(&pms_frame_tx);
  447. UbloxMessage* pms_message_rx =
  448. ublox_worker_i2c_transfer(pms_message_tx, UBX_CFG_PMS_MESSAGE_LENGTH);
  449. ublox_message_free(pms_message_tx);
  450. if(pms_message_rx == NULL) {
  451. FURI_LOG_E(TAG, "CFG-PMS read transfer failed");
  452. return false;
  453. }
  454. // set power setup value to "balanced"
  455. pms_message_rx->message[6 + 1] = 0x01;
  456. pms_frame_tx.class = UBX_CFG_CLASS;
  457. pms_frame_tx.id = UBX_CFG_PMS_MESSAGE;
  458. pms_frame_tx.len = 8;
  459. pms_frame_tx.payload = pms_message_rx->message;
  460. pms_message_tx = ublox_frame_to_bytes(&pms_frame_tx);
  461. UbloxMessage* ack = ublox_worker_i2c_transfer(pms_message_tx, UBX_ACK_ACK_MESSAGE_LENGTH);
  462. if(ack == NULL) {
  463. FURI_LOG_E(TAG, "ACK after CFG-PMS set transfer failed");
  464. return false;
  465. }
  466. FURI_LOG_I(
  467. TAG, "CFG-PMS ack: id = %u, type = %s", ack->message[3], ack->message[3] ? "ACK" : "NAK");
  468. ublox_message_free(pms_message_tx);
  469. ublox_message_free(pms_message_rx);
  470. ublox_message_free(ack);
  471. /***** Odometer *****/
  472. // Enable odometer by changing CFG-ODO.
  473. UbloxFrame odo_frame_tx;
  474. odo_frame_tx.class = UBX_CFG_CLASS;
  475. odo_frame_tx.id = UBX_CFG_ODO_MESSAGE;
  476. odo_frame_tx.len = 0;
  477. odo_frame_tx.payload = NULL;
  478. UbloxMessage* odo_message_tx = ublox_frame_to_bytes(&odo_frame_tx);
  479. UbloxMessage* odo_message_rx =
  480. ublox_worker_i2c_transfer(odo_message_tx, UBX_CFG_ODO_MESSAGE_LENGTH);
  481. ublox_message_free(odo_message_tx);
  482. if(odo_message_rx == NULL) {
  483. FURI_LOG_E(TAG, "CFG-ODO transfer failed");
  484. return false;
  485. }
  486. odo_frame_tx.class = UBX_CFG_CLASS;
  487. odo_frame_tx.id = UBX_CFG_ODO_MESSAGE;
  488. odo_frame_tx.len = 20;
  489. odo_frame_tx.payload = odo_message_rx->message;
  490. // TODO: low-pass filters in settings?
  491. // enable useODO bit in flags
  492. odo_frame_tx.payload[4] |= 1;
  493. odo_frame_tx.payload[5] = (ublox->device_state).odometer_mode;
  494. odo_message_tx = ublox_frame_to_bytes(&odo_frame_tx);
  495. ack = ublox_worker_i2c_transfer(odo_message_tx, UBX_ACK_ACK_MESSAGE_LENGTH);
  496. if(ack == NULL) {
  497. FURI_LOG_E(TAG, "ACK after CFG-ODO set transfer failed");
  498. return false;
  499. }
  500. FURI_LOG_I(
  501. TAG, "CFG-ODO ack: id = %u, type = %s", ack->message[3], ack->message[3] ? "ACK" : "NAK");
  502. ublox_message_free(odo_message_tx);
  503. ublox_message_free(odo_message_rx);
  504. ublox_message_free(ack);
  505. // finally configure the navigation engine
  506. UbloxFrame nav5_frame_tx;
  507. nav5_frame_tx.class = UBX_CFG_CLASS;
  508. nav5_frame_tx.id = UBX_CFG_NAV5_MESSAGE;
  509. nav5_frame_tx.len = 0;
  510. nav5_frame_tx.payload = NULL;
  511. UbloxMessage* nav5_message_tx = ublox_frame_to_bytes(&nav5_frame_tx);
  512. UbloxMessage* nav5_message_rx =
  513. ublox_worker_i2c_transfer(nav5_message_tx, UBX_CFG_NAV5_MESSAGE_LENGTH);
  514. ublox_message_free(nav5_message_tx);
  515. if(nav5_message_rx == NULL) {
  516. FURI_LOG_E(TAG, "CFG-NAV5 transfer failed");
  517. return false;
  518. }
  519. // first two bytes tell the GPS what changes to apply, setting this
  520. // bit tells it to apply the dynamic platfrom model settings.
  521. nav5_frame_tx.class = UBX_CFG_CLASS;
  522. nav5_frame_tx.id = UBX_CFG_NAV5_MESSAGE;
  523. nav5_frame_tx.len = 36;
  524. nav5_frame_tx.payload = nav5_message_rx->message;
  525. nav5_frame_tx.payload[0] = 1; // tell GPS to apply only the platform model settings
  526. nav5_frame_tx.payload[2] = (ublox->device_state).platform_model;
  527. nav5_message_tx = ublox_frame_to_bytes(&nav5_frame_tx);
  528. ack = ublox_worker_i2c_transfer(nav5_message_tx, UBX_ACK_ACK_MESSAGE_LENGTH);
  529. if(ack == NULL) {
  530. FURI_LOG_E(TAG, "ACK after CFG-NAV5 set transfer failed");
  531. return false;
  532. }
  533. FURI_LOG_I(
  534. TAG, "CFG-NAV5 ack: id = %u, type = %s", ack->message[3], ack->message[3] ? "ACK" : "NAK");
  535. ublox_message_free(nav5_message_rx);
  536. ublox_message_free(ack);
  537. return true;
  538. }
  539. // this one is being kind of slow
  540. void ublox_worker_reset_odo(UbloxWorker* ublox_worker) {
  541. FURI_LOG_I(TAG, "ublox_worker_reset_odo");
  542. UbloxFrame odo_frame_tx;
  543. odo_frame_tx.class = UBX_NAV_CLASS;
  544. odo_frame_tx.id = UBX_NAV_RESETODO_MESSAGE;
  545. odo_frame_tx.len = 0;
  546. odo_frame_tx.payload = NULL;
  547. UbloxMessage* odo_message_tx = ublox_frame_to_bytes(&odo_frame_tx);
  548. UbloxMessage* ack = ublox_worker_i2c_transfer(odo_message_tx, UBX_ACK_ACK_MESSAGE_LENGTH);
  549. ublox_message_free(odo_message_tx);
  550. if(ack == NULL) {
  551. FURI_LOG_E(TAG, "ACK after NAV-RESETODO set transfer failed");
  552. ublox_worker->callback(UbloxWorkerEventFailed, ublox_worker->context);
  553. return;
  554. } else {
  555. FURI_LOG_I(
  556. TAG,
  557. "NAV-RESETODO ack: id = %u, type = %s",
  558. ack->message[3],
  559. ack->message[3] ? "ACK" : "NAK");
  560. ublox_message_free(ack);
  561. }
  562. ublox_worker->callback(UbloxWorkerEventOdoReset, ublox_worker->context);
  563. // no reason to trigger an event on success, the user will see that
  564. // the odometer has been reset on the next update.
  565. }