monitor_sirf.c 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680
  1. /*
  2. * SiRF object for the GPS packet monitor.
  3. *
  4. * This file is Copyright 2010 by the GPSD project
  5. * SPDX-License-Identifier: BSD-2-clause
  6. *
  7. */
  8. #include "../include/gpsd_config.h" /* must be before all includes */
  9. #include <assert.h>
  10. #include <math.h>
  11. #include <stdbool.h>
  12. #include <stdlib.h> /* for labs() */
  13. #include <string.h>
  14. #include <sys/time.h>
  15. #include <time.h>
  16. #include "../include/gpsd.h"
  17. #include "../include/bits.h"
  18. #include "../include/gpsmon.h"
  19. #include "../include/strfuncs.h"
  20. #if defined(SIRF_ENABLE) && defined(BINARY_ENABLE)
  21. extern const struct gps_type_t driver_sirf;
  22. static WINDOW *mid2win, *mid4win, *mid6win, *mid7win, *mid9win, *mid13win;
  23. static WINDOW *mid19win, *mid27win;
  24. static bool dispmode = false, subframe_enabled = false, ppstime_enabled = false;
  25. static int leapseconds;
  26. static char *verbpat[] = {
  27. "#Time:",
  28. "@R Time:",
  29. "CSTD: New almanac for",
  30. "NOTICE: DOP Q Boost",
  31. "RTC not set",
  32. "numOfSVs = 0",
  33. "rtcaj tow ",
  34. NULL
  35. };
  36. static char *dgpsvec[] = {
  37. "None",
  38. "SBAS",
  39. "Serial",
  40. "Beacon",
  41. "Software",
  42. };
  43. /* check range of an unsigned quantity */
  44. #define CHECK_RANGE(vec, i) ((i) < sizeof(vec)/sizeof(vec[0]))
  45. /*****************************************************************************
  46. *
  47. * SiRF packet-decoding routines
  48. *
  49. *****************************************************************************/
  50. #define display (void)mvwprintw
  51. #define SIRF_CHANNELS 12 /* max channels allowed in SiRF format */
  52. static bool sirf_initialize(void)
  53. {
  54. unsigned int i;
  55. mid2win = subwin(devicewin, 6, 80, 1, 0);
  56. mid4win = subwin(devicewin, SIRF_CHANNELS + 3, 30, 7, 0);
  57. mid6win = subwin(devicewin, 3, 50, 7, 30);
  58. mid7win = subwin(devicewin, 4, 50, 13, 30);
  59. mid9win = subwin(devicewin, 3, 50, 10, 30);
  60. mid13win = subwin(devicewin, 3, 50, 17, 30);
  61. mid19win = newwin(16, 50, 7, 30);
  62. mid27win = subwin(devicewin, 3, 50, 20, 30);
  63. if (mid2win == NULL || mid4win == NULL || mid6win == NULL
  64. || mid9win == NULL || mid13win == NULL || mid19win == NULL
  65. || mid27win == NULL)
  66. return false;
  67. (void)syncok(mid2win, true);
  68. (void)syncok(mid4win, true);
  69. (void)syncok(mid6win, true);
  70. (void)syncok(mid7win, true);
  71. (void)syncok(mid9win, true);
  72. (void)syncok(mid13win, true);
  73. (void)syncok(mid27win, true);
  74. (void)wborder(mid2win, 0, 0, 0, 0, 0, 0, 0, 0),
  75. (void)wattrset(mid2win, A_BOLD);
  76. (void)wmove(mid2win, 0, 1);
  77. display(mid2win, 0, 12, " X ");
  78. display(mid2win, 0, 21, " Y ");
  79. display(mid2win, 0, 30, " Z ");
  80. display(mid2win, 0, 43, " North ");
  81. display(mid2win, 0, 54, " East ");
  82. display(mid2win, 0, 65, " Alt ");
  83. (void)wmove(mid2win, 1, 1);
  84. (void)wprintw(mid2win,
  85. "Pos: m m");
  86. (void)wmove(mid2win, 2, 1);
  87. (void)wprintw(mid2win,
  88. "Vel: m/s"
  89. " climb m/s");
  90. (void)wmove(mid2win, 3, 1);
  91. (void)wprintw(mid2win,
  92. "Time: Leap: Heading:"
  93. " speed m/s");
  94. (void)wmove(mid2win, 4, 1);
  95. (void)wprintw(mid2win,
  96. "Fix: HDOP:"
  97. " M1: M2: ");
  98. display(mid2win, 5, 24, " Packet type 2 (0x02) ");
  99. (void)wattrset(mid2win, A_NORMAL);
  100. (void)wborder(mid4win, 0, 0, 0, 0, 0, 0, 0, 0),
  101. (void)wattrset(mid4win, A_BOLD);
  102. display(mid4win, 1, 1, "Ch PRN Az El Stat C/N ? SF");
  103. for (i = 0; i < SIRF_CHANNELS; i++) {
  104. display(mid4win, (int)(i + 2), 1, "%2d", i);
  105. }
  106. display(mid4win, 0, 1, " Measured Tracker ");
  107. display(mid4win, 14, 4, " Packet Type 4 (0x04) ");
  108. (void)wattrset(mid4win, A_NORMAL);
  109. (void)wborder(mid19win, 0, 0, 0, 0, 0, 0, 0, 0),
  110. (void)wattrset(mid19win, A_BOLD);
  111. display(mid19win, 1, 1, "Alt. hold mode:");
  112. display(mid19win, 2, 1, "Alt. hold source:");
  113. display(mid19win, 3, 1, "Alt. source input:");
  114. display(mid19win, 4, 1, "Degraded timeout:");
  115. display(mid19win, 5, 1, "DR timeout:");
  116. display(mid19win, 6, 1, "Track smooth mode:");
  117. display(mid19win, 7, 1, "Static Navigation:");
  118. display(mid19win, 8, 1, "3SV Least Squares:");
  119. display(mid19win, 9, 1, "DOP Mask mode:");
  120. display(mid19win, 10, 1, "Nav. Elev. mask:");
  121. display(mid19win, 11, 1, "Nav. Power mask:");
  122. display(mid19win, 12, 1, "DGPS Source:");
  123. display(mid19win, 13, 1, "DGPS Mode:");
  124. display(mid19win, 14, 1, "DGPS Timeout:");
  125. display(mid19win, 1, 26, "LP Push-to-Fix:");
  126. display(mid19win, 2, 26, "LP On Time:");
  127. display(mid19win, 3, 26, "LP Interval:");
  128. display(mid19win, 4, 26, "U. Tasks Enab.:");
  129. display(mid19win, 5, 26, "U. Task Inter.:");
  130. display(mid19win, 6, 26, "LP Pwr Cyc En:");
  131. display(mid19win, 7, 26, "LP Max Acq Srch:");
  132. display(mid19win, 8, 26, "LP Max Off Time:");
  133. display(mid19win, 9, 26, "APM enabled:");
  134. display(mid19win, 10, 26, "# of Fixes:");
  135. display(mid19win, 11, 26, "Time btw Fixes:");
  136. display(mid19win, 12, 26, "H/V Error Max:");
  137. display(mid19win, 13, 26, "Rsp Time Max:");
  138. display(mid19win, 14, 26, "Time/Accu:");
  139. display(mid19win, 15, 8, " Packet type 19 (0x13) ");
  140. (void)wattrset(mid19win, A_NORMAL);
  141. (void)wborder(mid6win, 0, 0, 0, 0, 0, 0, 0, 0),
  142. (void)wattrset(mid6win, A_BOLD);
  143. display(mid6win, 0, 1, " Firmware Version ");
  144. display(mid6win, 2, 8, " Packet Type 6 (0x06) ");
  145. (void)wattrset(mid6win, A_NORMAL);
  146. (void)wborder(mid7win, 0, 0, 0, 0, 0, 0, 0, 0),
  147. (void)wattrset(mid7win, A_BOLD);
  148. display(mid7win, 0, 1, " Clock Status ");
  149. display(mid7win, 1, 1, "SVs: ");
  150. display(mid7win, 1, 9, "Drift: ");
  151. display(mid7win, 1, 23, "Bias: ");
  152. display(mid7win, 2, 1, "GPS Time: ");
  153. display(mid7win, 2, 23, "PPS: ");
  154. (void)syncok(mid7win, true);
  155. (void)mvwaddstr(mid7win, 2, 40, "N/A");
  156. display(mid7win, 3, 8, " Packet type 7 (0x07) ");
  157. (void)wattrset(mid7win, A_NORMAL);
  158. (void)wborder(mid9win, 0, 0, 0, 0, 0, 0, 0, 0),
  159. (void)wattrset(mid9win, A_BOLD);
  160. display(mid9win, 0, 1, " CPU Throughput ");
  161. display(mid9win, 1, 1, "Max: ");
  162. display(mid9win, 1, 13, "Lat: ");
  163. display(mid9win, 1, 25, "Time: ");
  164. display(mid9win, 1, 39, "MS: ");
  165. display(mid9win, 2, 8, " Packet type 9 (0x09) ");
  166. (void)wattrset(mid9win, A_NORMAL);
  167. (void)wborder(mid13win, 0, 0, 0, 0, 0, 0, 0, 0),
  168. (void)wattrset(mid13win, A_BOLD);
  169. display(mid13win, 0, 1, " Visible List ");
  170. display(mid13win, 2, 8, " Packet type 13 (0x0D) ");
  171. (void)wattrset(mid13win, A_NORMAL);
  172. (void)wborder(mid27win, 0, 0, 0, 0, 0, 0, 0, 0),
  173. (void)wattrset(mid27win, A_BOLD);
  174. display(mid27win, 0, 1, " DGPS Status ");
  175. display(mid27win, 2, 8, " Packet type 27 (0x1B) ");
  176. (void)wattrset(mid27win, A_NORMAL);
  177. /* probe for version */
  178. (void)monitor_control_send((unsigned char *)"\x84\x00", 2);
  179. /* initialize the GPS context's time fields */
  180. gpsd_time_init(session.context, time(NULL));
  181. return true;
  182. }
  183. static void decode_ecef(double x, double y, double z,
  184. double vx, double vy, double vz)
  185. {
  186. const double a = WGS84A;
  187. const double b = WGS84B;
  188. const double e2 = (a * a - b * b) / (a * a);
  189. const double e_2 = (a * a - b * b) / (b * b);
  190. double lambda, p, theta, phi, n, h, vnorth, veast, vup, speed, heading;
  191. lambda = atan2(y, x);
  192. p = sqrt(pow(x, 2) + pow(y, 2));
  193. theta = atan2(z * a, p * b);
  194. phi =
  195. atan2(z + e_2 * b * pow(sin(theta), 3),
  196. p - e2 * a * pow(cos(theta), 3));
  197. n = a / sqrt(1.0 - e2 * pow(sin(phi), 2));
  198. h = p / cos(phi) - n;
  199. h -= wgs84_separation((double)(RAD_2_DEG * phi),
  200. (double)(RAD_2_DEG * lambda));
  201. vnorth =
  202. -vx * sin(phi) * cos(lambda) - vy * sin(phi) * sin(lambda) +
  203. vz * cos(phi);
  204. veast = -vx * sin(lambda) + vy * cos(lambda);
  205. vup =
  206. vx * cos(phi) * cos(lambda) + vy * cos(phi) * sin(lambda) +
  207. vz * sin(phi);
  208. speed = sqrt(pow(vnorth, 2) + pow(veast, 2));
  209. heading = atan2(veast, vnorth);
  210. if (heading < 0)
  211. heading += 2 * GPS_PI;
  212. /* North and East position fields */
  213. (void)wattrset(mid2win, A_UNDERLINE);
  214. (void)wmove(mid2win, 1, 40);
  215. (void)wprintw(mid2win, "%9.5f %9.5f %9d",
  216. (double)(RAD_2_DEG * phi),
  217. (double)(RAD_2_DEG * lambda),
  218. (int)h);
  219. (void)mvwaddch(mid2win, 1, 49, ACS_DEGREE);
  220. (void)mvwaddch(mid2win, 1, 59, ACS_DEGREE);
  221. /* North and East velocity fields */
  222. (void)wmove(mid2win, 2, 40);
  223. (void)wprintw(mid2win, "%9.1f %9.1f %9.1f", vnorth, veast, vup);
  224. /* heading and speed fields */
  225. (void)wmove(mid2win, 3, 54);
  226. (void)wprintw(mid2win, "%5.1f %9.1f", (double)(RAD_2_DEG * heading), speed);
  227. (void)mvwaddch(mid2win, 3, 59, ACS_DEGREE);
  228. (void)wattrset(mid2win, A_NORMAL);
  229. }
  230. static void sirf_update(void)
  231. {
  232. int i, j, ch, sv;
  233. unsigned u, x;
  234. unsigned char *buf;
  235. size_t len;
  236. uint8_t dgps;
  237. char tbuf[JSON_DATE_MAX+1];
  238. buf = session.lexer.outbuffer + 4;
  239. len = session.lexer.outbuflen - 8;
  240. switch (buf[0]) {
  241. case 0x02: /* Measured Navigation Data */
  242. (void)wmove(mid2win, 1, 6); /* ECEF position */
  243. (void)wprintw(mid2win, "%8d %8d %8d", getbes32(buf, 1),
  244. getbes32(buf, 5), getbes32(buf, 9));
  245. (void)wmove(mid2win, 2, 6); /* ECEF velocity */
  246. (void)wprintw(mid2win, "%8.1f %8.1f %8.1f",
  247. (double)getbes16(buf, 13) / 8, (double)getbes16(buf,
  248. 15) / 8,
  249. (double)getbes16(buf, 17) / 8);
  250. decode_ecef((double)getbes32(buf, 1), (double)getbes32(buf, 5),
  251. (double)getbes32(buf, 9), (double)getbes16(buf, 13) / 8,
  252. (double)getbes16(buf, 15) / 8, (double)getbes16(buf,
  253. 17) / 8);
  254. /* line 3 */
  255. (void)wmove(mid2win, 3, 7);
  256. if (0 < session.gpsdata.fix.time.tv_sec) {
  257. (void)wprintw(mid2win, "%-24s",
  258. timespec_to_iso8601(session.gpsdata.fix.time, tbuf,
  259. sizeof(tbuf)));
  260. }
  261. (void)wmove(mid2win, 3, 38);
  262. (void)wattrset(mid2win, A_UNDERLINE);
  263. if (ppstime_enabled)
  264. (void)wprintw(mid2win, "%02d", leapseconds);
  265. else
  266. (void)wprintw(mid2win, "??");
  267. (void)wattrset(mid2win, A_NORMAL);
  268. /* line 4 */
  269. /* HDOP */
  270. (void)wmove(mid2win, 4, 59);
  271. (void)wprintw(mid2win, "%4.1f", (double)getub(buf, 20) / 5);
  272. /* Mode 1 */
  273. (void)wmove(mid2win, 4, 69);
  274. (void)wprintw(mid2win, "%02x", getub(buf, 19));
  275. /* Mode 2 */
  276. (void)wmove(mid2win, 4, 77);
  277. (void)wprintw(mid2win, "%02x", getub(buf, 21));
  278. /* SVs in fix */
  279. (void)wmove(mid2win, 4, 6);
  280. (void)wprintw(mid2win, "%2u = ",
  281. getub(buf, 28));
  282. /* SV list */
  283. (void)wmove(mid2win, 4, 10);
  284. x = getub(buf, 28);
  285. if (19 < x) {
  286. x = 19; // SiRF Star V max is 19
  287. }
  288. for (u = 0; u < x; u++)
  289. (void)wprintw(mid2win, " %2u", getub(buf, 29 + u));
  290. monitor_log("MND 0x02=");
  291. break;
  292. case 0x04: /* Measured Tracking Data */
  293. ch = (int)getub(buf, 7);
  294. for (i = 0; i < ch; i++) {
  295. int az, el, state, off;
  296. double cn;
  297. off = 8 + 15 * i;
  298. (void)wmove(mid4win, i + 2, 3);
  299. sv = (int)getub(buf, off);
  300. az = (int)getub(buf, off + 1) * 3 / 2;
  301. el = (int)getub(buf, off + 2) / 2;
  302. state = (int)getbeu16(buf, off + 3);
  303. cn = 0;
  304. for (j = 0; j < 10; j++)
  305. cn += (int)getub(buf, off + 5 + j);
  306. cn /= 10;
  307. (void)wprintw(mid4win, " %3d %3d %2d %04x %4.1f %c",
  308. sv, az, el, state, cn, state == 0xbf ? 'T' : ' ');
  309. }
  310. monitor_log("MTD 0x04=");
  311. break;
  312. #ifdef __UNUSED__
  313. case 0x05: /* raw track data */
  314. for (off = 1; off < len; off += 51) {
  315. ch = getbeu32(buf, off);
  316. (void)wmove(mid4win, ch + 2, 19);
  317. cn = 0;
  318. for (j = 0; j < 10; j++)
  319. cn += getub(buf, off + 34 + j);
  320. printw("%5.1f", (double)cn / 10);
  321. printw("%9d%3d%5d", getbeu32(buf, off + 8),
  322. (int)getbeu16(buf, off + 12), (int)getbeu16(buf, off + 14));
  323. printw("%8.5f %10.5f", (double)getbeu32(buf, off + 16) / 65536,
  324. (double)getbeu32(buf, off + 20) / 1024);
  325. }
  326. monitor_log("RTD 0x05=");
  327. break;
  328. #endif /* __UNUSED */
  329. case 0x06: /* firmware version */
  330. display(mid6win, 1, 1, "%s", buf + 1);
  331. monitor_log("FV 0x06=");
  332. break;
  333. case 0x07: /* Response - Clock Status Data */
  334. display(mid7win, 1, 5, "%2d", getub(buf, 7)); /* SVs */
  335. /* Clock ppstimes */
  336. display(mid7win, 1, 16, "%lu", (unsigned long)getbeu32(buf, 8));
  337. display(mid7win, 1, 29, "%lu", (unsigned long)getbeu32(buf, 12));
  338. /* Clock Bias */
  339. display(mid7win, 2, 11, "%lu", (unsigned long)getbeu32(buf, 16));
  340. /* Estimated Time */
  341. monitor_log("CSD 0x07=");
  342. break;
  343. case 0x08: /* 50 BPS data */
  344. ch = (int)getub(buf, 1);
  345. sv = (int)getub(buf, 2);
  346. display(mid4win, ch + 2, 27, "%2d", sv);
  347. subframe_enabled = true;
  348. monitor_log("50B 0x08=");
  349. break;
  350. case 0x09: /* Throughput */
  351. /*SegStatMax */
  352. display(mid9win, 1, 6, "%.3f", (double)getbeu16(buf, 1) / 186);
  353. /*SegStatLat */
  354. display(mid9win, 1, 18, "%.3f", (double)getbeu16(buf, 3) / 186);
  355. /*SegStatTime */
  356. display(mid9win, 1, 31, "%.3f", (double)getbeu16(buf, 5) / 186);
  357. /* Last Millisecond */
  358. display(mid9win, 1, 42, "%3d", (int)getbeu16(buf, 7));
  359. monitor_log("THR 0x09=");
  360. break;
  361. case 0x0b: /* Command Acknowledgement */
  362. monitor_log("ACK 0x0b=");
  363. break;
  364. case 0x0c: /* Command NAcknowledgement */
  365. monitor_log("NAK 0x0c=");
  366. break;
  367. case 0x0d: /* Visible List */
  368. display(mid13win, 1, 1, "%02d = ",
  369. getub(buf, 1));
  370. (void)wmove(mid13win, 1, 5);
  371. for (i = 0; i < (int)getub(buf, 1); i++)
  372. (void)wprintw(mid13win, " %d", getub(buf, 2 + 5 * i));
  373. monitor_log("VL 0x0d=");
  374. break;
  375. case 0x13:
  376. #define YESNO(n) (((int)getub(buf, n) != 0)?'Y':'N')
  377. display(mid19win, 1, 20, "%d", getub(buf, 5)); /* Alt. hold mode */
  378. display(mid19win, 2, 20, "%d", getub(buf, 6)); /* Alt. hold source */
  379. /* Alt. source input */
  380. display(mid19win, 3, 20, "%dm", (int)getbeu16(buf, 7));
  381. if (getub(buf, 9) != (uint8_t) '\0') {
  382. /* Degraded timeout */
  383. display(mid19win, 4, 20, "%dsec", getub(buf, 10));
  384. } else {
  385. display(mid19win, 4, 20, "N/A ");
  386. }
  387. display(mid19win, 5, 20, "%dsec", getub(buf, 11)); /* DR timeout */
  388. display(mid19win, 6, 20, "%c", YESNO(12)); /* Track smooth mode */
  389. display(mid19win, 7, 20, "%c", YESNO(13)); /* Static Nav. */
  390. display(mid19win, 8, 20, "0x%x", getub(buf, 14)); // 3SV Least Squares
  391. display(mid19win, 9, 20, "0x%x", getub(buf, 19)); /* DOP Mask mode */
  392. /* Nav. Elev. mask */
  393. display(mid19win, 10, 20, "0x%x", (int)getbeu16(buf, 20));
  394. /* Nav. Power mask */
  395. display(mid19win, 11, 20, "0x%x", getub(buf, 22));
  396. display(mid19win, 12, 20, "0x%x", getub(buf, 27)); /* DGPS Source */
  397. display(mid19win, 13, 20, "0x%x", getub(buf, 28)); /* DGPS Mode */
  398. display(mid19win, 14, 20, "%dsec", getub(buf, 29)); /* DGPS Timeout */
  399. display(mid19win, 1, 42, "%c", YESNO(34)); /* LP Push-to-Fix */
  400. display(mid19win, 2, 42, "%dms", getbeu32(buf, 35)); /* LP On Time */
  401. display(mid19win, 3, 42, "%d", getbeu32(buf, 39)); /* LP Interval */
  402. display(mid19win, 4, 42, "%c", YESNO(43)); /* User Tasks enabled */
  403. /* User Task Interval */
  404. display(mid19win, 5, 42, "%d", getbeu32(buf, 44));
  405. display(mid19win, 6, 42, "%c", YESNO(48)); // LP Power Cycling Enabled
  406. /* LP Max Acq Search Time */
  407. display(mid19win, 7, 42, "%d", getbeu32(buf, 49));
  408. /* LP Max Off Time */
  409. display(mid19win, 8, 42, "%d", getbeu32(buf, 53));
  410. display(mid19win, 9, 42, "%c", YESNO(57)); /* APM Enabled */
  411. /* # of fixes */
  412. display(mid19win, 10, 42, "%d", (int)getbeu16(buf, 58));
  413. /* Time Between fixes */
  414. display(mid19win, 11, 42, "%d", (int)getbeu16(buf, 60));
  415. display(mid19win, 12, 42, "%d", getub(buf, 62)); /* H/V Error Max */
  416. display(mid19win, 13, 42, "%d", getub(buf, 63)); // Response Time Max
  417. /* Time/Accu & Duty Cycle Priority */
  418. display(mid19win, 14, 42, "%d", getub(buf, 64));
  419. #undef YESNO
  420. monitor_log("NP 0x13=");
  421. break;
  422. case 0x1b:
  423. /******************************************************************
  424. Not actually documented in any published materials before the
  425. 1.6 version of the SiRF binary protocol manual.
  426. Here is what Chris Kuethe got from the SiRF folks,
  427. (plus some corrections from the GpsPaSsion forums):
  428. Start of message
  429. ----------------
  430. Message ID 1 byte 27
  431. Correction Source 1 byte 0=None, 1=SBAS, 2=Serial, 3=Beacon,
  432. 4=Software
  433. total: 2 bytes
  434. Middle part of message varies if using beacon or other:
  435. -------------------------------------------------------
  436. If Beacon:
  437. Receiver Freq Hz 4 bytes
  438. Bit rate BPS 1 byte
  439. Status bit map 1 byte 01=Signal Valid,
  440. 02=Auto frequency detect
  441. 04=Auto bit rate detect
  442. Signal Magnitude 4 bytes Note: in internal units
  443. Signal Strength dB 2 bytes derived from Signal Magnitude
  444. SNR dB 2 bytes
  445. total: 14 bytes
  446. If Not Beacon:
  447. Correction Age[12] 1 byte x 12 Age in seconds in same order as follows
  448. Reserved 2 bytes
  449. total: 14 bytes
  450. End of Message
  451. --------------
  452. Repeated 12 times (pad with 0 if less than 12 SV corrections):
  453. SVID 1 byte
  454. Correction (cm) 2 bytes (signed short)
  455. total 3 x 12 = 36 bytes
  456. ******************************************************************/
  457. dgps = getub(buf, 1);
  458. display(mid27win, 1, 1, "%8s = ",
  459. (CHECK_RANGE(dgpsvec, dgps) ? dgpsvec[dgps] : "???"));
  460. (void)wmove(mid27win, 1, 11);
  461. for (ch = 0; ch < SIRF_CHANNELS; ch++)
  462. if (getub(buf, 16 + 3 * ch) != '\0')
  463. (void)wprintw(mid27win, " %d", getub(buf, 16 + 3 * ch));
  464. monitor_log("DST 0x1b=");
  465. break;
  466. case 0x1c: /* NL Measurement Data */
  467. case 0x1d: /* NL DGPS Data */
  468. case 0x1e: /* NL SV State Data */
  469. case 0x1f: /* NL Initialized Data */
  470. subframe_enabled = true;
  471. monitor_log("NL 0x%02x=", buf[0]);
  472. break;
  473. case 0x29: /* Geodetic Navigation Data */
  474. monitor_log("GND 0x29=");
  475. break;
  476. case 0x32: /* SBAS Parameters */
  477. monitor_log("SBP 0x32=");
  478. break;
  479. case 0x34: /* PPS Time */
  480. ppstime_enabled = true;
  481. leapseconds = (int)getbeu16(buf, 8);
  482. monitor_log("PPS 0x34=");
  483. break;
  484. case 0xff: /* Development Data */
  485. while (len > 0 && buf[len - 1] == '\n')
  486. len--;
  487. while (len > 0 && buf[len - 1] == ' ')
  488. len--;
  489. buf[len] = '\0';
  490. j = 1;
  491. for (i = 0; verbpat[i] != NULL; i++)
  492. if (str_starts_with((char *)(buf + 1), verbpat[i])) {
  493. j = 0;
  494. break;
  495. }
  496. if (j != 0)
  497. monitor_log("%s\n", buf + 1);
  498. monitor_log("DD 0xff=");
  499. break;
  500. default:
  501. monitor_log("UNK 0x%02x=", buf[0]);
  502. break;
  503. }
  504. /* elicit navigation parameters */
  505. if (dispmode && (time(NULL) % 10 == 0)) {
  506. (void)monitor_control_send((unsigned char *)"\x98\x00", 2);
  507. }
  508. /* clear the 50bps data field every 6 seconds */
  509. if (subframe_enabled && (time(NULL) % 6 == 0)) {
  510. for (ch = 0; ch < SIRF_CHANNELS; ch++)
  511. display(mid4win, ch + 2, 27, " ");
  512. }
  513. if (dispmode) {
  514. (void)touchwin(mid19win);
  515. (void)wnoutrefresh(mid19win);
  516. }
  517. pps_update(mid7win, 2, 32);
  518. }
  519. static int sirf_command(char line[])
  520. {
  521. unsigned char buf[BUFSIZ];
  522. int v;
  523. switch (line[0]) {
  524. case 'A': /* toggle 50bps subframe data */
  525. (void)memset(buf, '\0', sizeof(buf));
  526. putbyte(buf, 0, 0x80);
  527. putbyte(buf, 23, 0x0c);
  528. putbyte(buf, 24, subframe_enabled ? 0x00 : 0x10);
  529. (void)monitor_control_send(buf, 25);
  530. /*
  531. * The subframe_enabled off because we're counting on receipt of
  532. * a subframe packet to turn it on if the command succeeds.
  533. */
  534. subframe_enabled = false;
  535. return COMMAND_MATCH;
  536. case 'M': /* static navigation */
  537. putbyte(buf, 0, 0x8f); /* id */
  538. putbyte(buf, 1, atoi(line + 1));
  539. (void)monitor_control_send(buf, 2);
  540. return COMMAND_MATCH;
  541. case 'D': /* MID 4 rate change (undocumented) */
  542. v = atoi(line + 1);
  543. if (v > 30)
  544. return COMMAND_MATCH;
  545. putbyte(buf, 0, 0xa6);
  546. putbyte(buf, 1, 0);
  547. putbyte(buf, 2, 4); /* satellite picture */
  548. putbyte(buf, 3, v);
  549. putbyte(buf, 4, 0);
  550. putbyte(buf, 5, 0);
  551. putbyte(buf, 6, 0);
  552. putbyte(buf, 7, 0);
  553. (void)monitor_control_send(buf, 8);
  554. return COMMAND_MATCH;
  555. case 'P': /* poll navigation params */
  556. dispmode = !dispmode;
  557. if (dispmode) {
  558. (void)syncok(mid6win, false);
  559. (void)syncok(mid7win, false);
  560. (void)syncok(mid9win, false);
  561. (void)syncok(mid13win, false);
  562. (void)syncok(mid27win, false);
  563. } else {
  564. (void)syncok(mid6win, true);
  565. (void)wsyncup(mid6win);
  566. (void)syncok(mid7win, true);
  567. (void)wsyncup(mid7win);
  568. (void)syncok(mid9win, true);
  569. (void)wsyncup(mid9win);
  570. (void)syncok(mid13win, true);
  571. (void)wsyncup(mid13win);
  572. (void)syncok(mid27win, true);
  573. (void)wsyncup(mid27win);
  574. }
  575. return COMMAND_MATCH;
  576. }
  577. return COMMAND_UNKNOWN; /* no match */
  578. }
  579. static void sirf_wrap(void)
  580. {
  581. (void)delwin(mid2win);
  582. (void)delwin(mid4win);
  583. (void)delwin(mid6win);
  584. (void)delwin(mid7win);
  585. (void)delwin(mid9win);
  586. (void)delwin(mid13win);
  587. (void)delwin(mid19win);
  588. (void)delwin(mid27win);
  589. }
  590. const struct monitor_object_t sirf_mmt = {
  591. .initialize = sirf_initialize,
  592. .update = sirf_update,
  593. .command = sirf_command,
  594. .wrap = sirf_wrap,
  595. .min_y = 22,.min_x = 80,
  596. .driver = &driver_sirf,
  597. };
  598. #endif /* defined(SIRF_ENABLE) && defined(BINARY_ENABLE) */
  599. /* sirfmon.c ends here */
  600. // vim: set expandtab shiftwidth=4