latlong.c 29 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048
  1. //
  2. // This file is part of Dire Wolf, an amateur radio packet TNC.
  3. //
  4. // Copyright (C) 2013, 2014, 2015 John Langner, WB2OSZ
  5. //
  6. // This program is free software: you can redistribute it and/or modify
  7. // it under the terms of the GNU General Public License as published by
  8. // the Free Software Foundation, either version 2 of the License, or
  9. // (at your option) any later version.
  10. //
  11. // This program is distributed in the hope that it will be useful,
  12. // but WITHOUT ANY WARRANTY; without even the implied warranty of
  13. // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  14. // GNU General Public License for more details.
  15. //
  16. // You should have received a copy of the GNU General Public License
  17. // along with this program. If not, see <http://www.gnu.org/licenses/>.
  18. //
  19. /*------------------------------------------------------------------
  20. *
  21. * Module: latlong.c
  22. *
  23. * Purpose: Various functions for dealing with latitude and longitude.
  24. *
  25. * Description: Originally, these were scattered around in many places.
  26. * Over time they might all be gathered into one place
  27. * for consistency, reuse, and easier maintenance.
  28. *
  29. *---------------------------------------------------------------*/
  30. #include "direwolf.h"
  31. #include <stdlib.h>
  32. #include <string.h>
  33. #include <stdio.h>
  34. #include <unistd.h>
  35. #include <ctype.h>
  36. #include <time.h>
  37. #include <math.h>
  38. #include <assert.h>
  39. #include "latlong.h"
  40. //#include "textcolor.h"
  41. /*------------------------------------------------------------------
  42. *
  43. * Name: latitude_to_str
  44. *
  45. * Purpose: Convert numeric latitude to string for transmission.
  46. *
  47. * Inputs: dlat - Floating point degrees.
  48. * ambiguity - If 1, 2, 3, or 4, blank out that many trailing digits.
  49. *
  50. * Outputs: slat - String in format ddmm.mm[NS]
  51. * Should always be exactly 8 characters + NUL.
  52. *
  53. * Returns: None
  54. *
  55. * Idea for future:
  56. * Non zero ambiguity removes least significant digits without rounding.
  57. * Maybe we could use -1 and -2 to add extra digits using !DAO! as
  58. * documented in http://www.aprs.org/datum.txt
  59. *
  60. * For example, -1 adds one more human readable digit.
  61. * lat minutes 12.345 would produce "12.34" and !W5 !
  62. *
  63. * -2 would encode almost 2 digits in base 91.
  64. * lat minutes 10.0027 would produce "10.00" and !w: !
  65. *
  66. *----------------------------------------------------------------*/
  67. void latitude_to_str (double dlat, int ambiguity, char *slat)
  68. {
  69. char hemi; /* Hemisphere: N or S */
  70. int ideg; /* whole number of degrees. */
  71. double dmin; /* Minutes after removing degrees. */
  72. char smin[8]; /* Minutes in format mm.mm */
  73. if (dlat < -90.) {
  74. //text_color_set(DW_COLOR_ERROR);
  75. printf ("Latitude is less than -90. Changing to -90.n");
  76. dlat = -90.;
  77. }
  78. if (dlat > 90.) {
  79. //text_color_set(DW_COLOR_ERROR);
  80. printf ("Latitude is greater than 90. Changing to 90.n");
  81. dlat = 90.;
  82. }
  83. if (dlat < 0) {
  84. dlat = (- dlat);
  85. hemi = 'S';
  86. }
  87. else {
  88. hemi = 'N';
  89. }
  90. ideg = (int)dlat;
  91. dmin = (dlat - ideg) * 60.;
  92. snprintf (smin, sizeof(smin), "%05.2f", dmin);
  93. /* Due to roundoff, 59.9999 could come out as "60.00" */
  94. if (smin[0] == '6') {
  95. smin[0] = '0';
  96. ideg++;
  97. }
  98. sprintf (slat, "%02d%s%c", ideg, smin, hemi);
  99. if (ambiguity >= 1) {
  100. slat[6] = ' ';
  101. if (ambiguity >= 2) {
  102. slat[5] = ' ';
  103. if (ambiguity >= 3) {
  104. slat[3] = ' ';
  105. if (ambiguity >= 4) {
  106. slat[2] = ' ';
  107. }
  108. }
  109. }
  110. }
  111. } /* end latitude_to_str */
  112. /*------------------------------------------------------------------
  113. *
  114. * Name: longitude_to_str
  115. *
  116. * Purpose: Convert numeric longitude to string for transmission.
  117. *
  118. * Inputs: dlong - Floating point degrees.
  119. * ambiguity - If 1, 2, 3, or 4, blank out that many trailing digits.
  120. *
  121. * Outputs: slat - String in format dddmm.mm[NS]
  122. * Should always be exactly 9 characters + NUL.
  123. *
  124. * Returns: None
  125. *
  126. *----------------------------------------------------------------*/
  127. void longitude_to_str (double dlong, int ambiguity, char *slong)
  128. {
  129. char hemi; /* Hemisphere: N or S */
  130. int ideg; /* whole number of degrees. */
  131. double dmin; /* Minutes after removing degrees. */
  132. char smin[8]; /* Minutes in format mm.mm */
  133. if (dlong < -180.) {
  134. //text_color_set(DW_COLOR_ERROR);
  135. printf ("Longitude is less than -180. Changing to -180.n");
  136. dlong = -180.;
  137. }
  138. if (dlong > 180.) {
  139. //text_color_set(DW_COLOR_ERROR);
  140. printf ("Longitude is greater than 180. Changing to 180.n");
  141. dlong = 180.;
  142. }
  143. if (dlong < 0) {
  144. dlong = (- dlong);
  145. hemi = 'W';
  146. }
  147. else {
  148. hemi = 'E';
  149. }
  150. ideg = (int)dlong;
  151. dmin = (dlong - ideg) * 60.;
  152. snprintf (smin, sizeof(smin), "%05.2f", dmin);
  153. /* Due to roundoff, 59.9999 could come out as "60.00" */
  154. if (smin[0] == '6') {
  155. smin[0] = '0';
  156. ideg++;
  157. }
  158. sprintf (slong, "%03d%s%c", ideg, smin, hemi);
  159. /*
  160. * The spec says position ambiguity in latitude also
  161. * applies to longitude automatically.
  162. * Blanking longitude digits is not necessary but I do it
  163. * because it makes things clearer.
  164. */
  165. if (ambiguity >= 1) {
  166. slong[7] = ' ';
  167. if (ambiguity >= 2) {
  168. slong[6] = ' ';
  169. if (ambiguity >= 3) {
  170. slong[4] = ' ';
  171. if (ambiguity >= 4) {
  172. slong[3] = ' ';
  173. }
  174. }
  175. }
  176. }
  177. } /* end longitude_to_str */
  178. /*------------------------------------------------------------------
  179. *
  180. * Name: latitude_to_comp_str
  181. *
  182. * Purpose: Convert numeric latitude to compressed string for transmission.
  183. *
  184. * Inputs: dlat - Floating point degrees.
  185. *
  186. * Outputs: slat - String in format yyyy.
  187. * Exactly 4 bytes, no nul terminator.
  188. *
  189. *----------------------------------------------------------------*/
  190. void latitude_to_comp_str (double dlat, char *clat)
  191. {
  192. int y, y0, y1, y2, y3;
  193. if (dlat < -90.) {
  194. //text_color_set(DW_COLOR_ERROR);
  195. printf ("Latitude is less than -90. Changing to -90.n");
  196. dlat = -90.;
  197. }
  198. if (dlat > 90.) {
  199. //text_color_set(DW_COLOR_ERROR);
  200. printf ("Latitude is greater than 90. Changing to 90.n");
  201. dlat = 90.;
  202. }
  203. y = (int)round(380926. * (90. - dlat));
  204. y0 = y / (91*91*91);
  205. y -= y0 * (91*91*91);
  206. y1 = y / (91*91);
  207. y -= y1 * (91*91);
  208. y2 = y / (91);
  209. y -= y2 * (91);
  210. y3 = y;
  211. clat[0] = y0 + 33;
  212. clat[1] = y1 + 33;
  213. clat[2] = y2 + 33;
  214. clat[3] = y3 + 33;
  215. }
  216. /*------------------------------------------------------------------
  217. *
  218. * Name: longitude_to_comp_str
  219. *
  220. * Purpose: Convert numeric longitude to compressed string for transmission.
  221. *
  222. * Inputs: dlong - Floating point degrees.
  223. *
  224. * Outputs: slat - String in format xxxx.
  225. * Exactly 4 bytes, no nul terminator.
  226. *
  227. *----------------------------------------------------------------*/
  228. void longitude_to_comp_str (double dlong, char *clon)
  229. {
  230. int x, x0, x1, x2, x3;
  231. if (dlong < -180.) {
  232. //text_color_set(DW_COLOR_ERROR);
  233. printf ("Longitude is less than -180. Changing to -180.n");
  234. dlong = -180.;
  235. }
  236. if (dlong > 180.) {
  237. //text_color_set(DW_COLOR_ERROR);
  238. printf ("Longitude is greater than 180. Changing to 180.n");
  239. dlong = 180.;
  240. }
  241. x = (int)round(190463. * (180. + dlong));
  242. x0 = x / (91*91*91);
  243. x -= x0 * (91*91*91);
  244. x1 = x / (91*91);
  245. x -= x1 * (91*91);
  246. x2 = x / (91);
  247. x -= x2 * (91);
  248. x3 = x;
  249. clon[0] = x0 + 33;
  250. clon[1] = x1 + 33;
  251. clon[2] = x2 + 33;
  252. clon[3] = x3 + 33;
  253. }
  254. /*------------------------------------------------------------------
  255. *
  256. * Name: latitude_to_nmea
  257. *
  258. * Purpose: Convert numeric latitude to strings for NMEA sentence.
  259. *
  260. * Inputs: dlat - Floating point degrees.
  261. *
  262. * Outputs: slat - String in format ddmm.mmmm
  263. * hemi - Hemisphere or empty string.
  264. *
  265. * Returns: None
  266. *
  267. *----------------------------------------------------------------*/
  268. void latitude_to_nmea (double dlat, char *slat, char *hemi)
  269. {
  270. int ideg; /* whole number of degrees. */
  271. double dmin; /* Minutes after removing degrees. */
  272. char smin[10]; /* Minutes in format mm.mmmm */
  273. if (dlat == G_UNKNOWN) {
  274. strcpy (slat, "");
  275. strcpy (hemi, "");
  276. return;
  277. }
  278. if (dlat < -90.) {
  279. //text_color_set(DW_COLOR_ERROR);
  280. printf ("Latitude is less than -90. Changing to -90.n");
  281. dlat = -90.;
  282. }
  283. if (dlat > 90.) {
  284. //text_color_set(DW_COLOR_ERROR);
  285. printf ("Latitude is greater than 90. Changing to 90.n");
  286. dlat = 90.;
  287. }
  288. if (dlat < 0) {
  289. dlat = (- dlat);
  290. strcpy (hemi, "S");
  291. }
  292. else {
  293. strcpy (hemi, "N");
  294. }
  295. ideg = (int)dlat;
  296. dmin = (dlat - ideg) * 60.;
  297. snprintf (smin, sizeof(smin), "%07.4f", dmin);
  298. /* Due to roundoff, 59.99999 could come out as "60.0000" */
  299. if (smin[0] == '6') {
  300. smin[0] = '0';
  301. ideg++;
  302. }
  303. sprintf (slat, "%02d%s", ideg, smin);
  304. } /* end latitude_to_str */
  305. /*------------------------------------------------------------------
  306. *
  307. * Name: longitude_to_nmea
  308. *
  309. * Purpose: Convert numeric longitude to strings for NMEA sentence.
  310. *
  311. * Inputs: dlong - Floating point degrees.
  312. *
  313. * Outputs: slong - String in format dddmm.mmmm
  314. * hemi - Hemisphere or empty string.
  315. *
  316. * Returns: None
  317. *
  318. *----------------------------------------------------------------*/
  319. void longitude_to_nmea (double dlong, char *slong, char *hemi)
  320. {
  321. int ideg; /* whole number of degrees. */
  322. double dmin; /* Minutes after removing degrees. */
  323. char smin[10]; /* Minutes in format mm.mmmm */
  324. if (dlong == G_UNKNOWN) {
  325. strcpy (slong, "");
  326. strcpy (hemi, "");
  327. return;
  328. }
  329. if (dlong < -180.) {
  330. //text_color_set(DW_COLOR_ERROR);
  331. printf ("longitude is less than -180. Changing to -180.n");
  332. dlong = -180.;
  333. }
  334. if (dlong > 180.) {
  335. //text_color_set(DW_COLOR_ERROR);
  336. printf ("longitude is greater than 180. Changing to 180.n");
  337. dlong = 180.;
  338. }
  339. if (dlong < 0) {
  340. dlong = (- dlong);
  341. strcpy (hemi, "W");
  342. }
  343. else {
  344. strcpy (hemi, "E");
  345. }
  346. ideg = (int)dlong;
  347. dmin = (dlong - ideg) * 60.;
  348. snprintf (smin, sizeof(smin), "%07.4f", dmin);
  349. /* Due to roundoff, 59.99999 could come out as "60.0000" */
  350. if (smin[0] == '6') {
  351. smin[0] = '0';
  352. ideg++;
  353. }
  354. sprintf (slong, "%03d%s", ideg, smin);
  355. } /* end longitude_to_nmea */
  356. /*------------------------------------------------------------------
  357. *
  358. * Function: latitude_from_nmea
  359. *
  360. * Purpose: Convert NMEA latitude encoding to degrees.
  361. *
  362. * Inputs: pstr - Pointer to numeric string.
  363. * phemi - Pointer to following field. Should be N or S.
  364. *
  365. * Returns: Double precision value in degrees. Negative for South.
  366. *
  367. * Description: Latitude field has
  368. * 2 digits for degrees
  369. * 2 digits for minutes
  370. * period
  371. * Variable number of fractional digits for minutes.
  372. * I've seen 2, 3, and 4 fractional digits.
  373. *
  374. *
  375. * Bugs: Very little validation of data.
  376. *
  377. * Errors: Return constant G_UNKNOWN for any type of error.
  378. *
  379. *------------------------------------------------------------------*/
  380. double latitude_from_nmea (char *pstr, char *phemi)
  381. {
  382. double lat;
  383. if ( ! isdigit((uint)(pstr[0]))) return (G_UNKNOWN);
  384. if (pstr[4] != '.') return (G_UNKNOWN);
  385. lat = (pstr[0] - '0') * 10 + (pstr[1] - '0') + atof(pstr+2) / 60.0;
  386. if (lat < 0 || lat > 90) {
  387. //text_color_set(DW_COLOR_ERROR);
  388. printf("Error: Latitude not in range of 0 to 90.\n");
  389. }
  390. // Saw this one time:
  391. // $GPRMC,000000,V,0000.0000,0,00000.0000,0,000,000,000000,,*01
  392. // If location is unknown, I think the hemisphere should be
  393. // an empty string. TODO: Check on this.
  394. // 'V' means void, so sentence should be discarded rather than
  395. // trying to extract any data from it.
  396. if (*phemi != 'N' && *phemi != 'S' && *phemi != '\0') {
  397. //text_color_set(DW_COLOR_ERROR);
  398. printf("Error: Latitude hemisphere should be N or S.\n");
  399. }
  400. if (*phemi == 'S') lat = ( - lat);
  401. return (lat);
  402. }
  403. /*------------------------------------------------------------------
  404. *
  405. * Function: longitude_from_nmea
  406. *
  407. * Purpose: Convert NMEA longitude encoding to degrees.
  408. *
  409. * Inputs: pstr - Pointer to numeric string.
  410. * phemi - Pointer to following field. Should be E or W.
  411. *
  412. * Returns: Double precision value in degrees. Negative for West.
  413. *
  414. * Description: Longitude field has
  415. * 3 digits for degrees
  416. * 2 digits for minutes
  417. * period
  418. * Variable number of fractional digits for minutes
  419. *
  420. *
  421. * Bugs: Very little validation of data.
  422. *
  423. * Errors: Return constant G_UNKNOWN for any type of error.
  424. *
  425. *------------------------------------------------------------------*/
  426. double longitude_from_nmea (char *pstr, char *phemi)
  427. {
  428. double lon;
  429. if ( ! isdigit((uint)(pstr[0]))) return (G_UNKNOWN);
  430. if (pstr[5] != '.') return (G_UNKNOWN);
  431. lon = (pstr[0] - '0') * 100 + (pstr[1] - '0') * 10 + (pstr[2] - '0') + atof(pstr+3) / 60.0;
  432. if (lon < 0 || lon > 180) {
  433. //text_color_set(DW_COLOR_ERROR);
  434. printf("Error: Longitude not in range of 0 to 180.\n");
  435. }
  436. if (*phemi != 'E' && *phemi != 'W' && *phemi != '\0') {
  437. //text_color_set(DW_COLOR_ERROR);
  438. printf("Error: Longitude hemisphere should be E or W.\n");
  439. }
  440. if (*phemi == 'W') lon = ( - lon);
  441. return (lon);
  442. }
  443. /*------------------------------------------------------------------
  444. *
  445. * Function: ll_distance_km
  446. *
  447. * Purpose: Calculate distance between two locations.
  448. *
  449. * Inputs: lat1, lon1 - One location, in degrees.
  450. * lat2, lon2 - other location
  451. *
  452. * Returns: Distance in km.
  453. *
  454. * Description: The Ubiquitous Haversine formula.
  455. *
  456. *------------------------------------------------------------------*/
  457. #define R 6371
  458. double ll_distance_km (double lat1, double lon1, double lat2, double lon2)
  459. {
  460. double a;
  461. lat1 *= M_PI / 180;
  462. lon1 *= M_PI / 180;
  463. lat2 *= M_PI / 180;
  464. lon2 *= M_PI / 180;
  465. a = pow(sin((lat2-lat1)/2),2) + cos(lat1) * cos(lat2) * pow(sin((lon2-lon1)/2),2);
  466. return (R * 2 *atan2(sqrt(a), sqrt(1-a)));
  467. }
  468. /*------------------------------------------------------------------
  469. *
  470. * Function: ll_bearing_deg
  471. *
  472. * Purpose: Calculate bearing between two locations.
  473. *
  474. * Inputs: lat1, lon1 - starting location, in degrees.
  475. * lat2, lon2 - destination location
  476. *
  477. * Returns: Initial Bearing, in degrees.
  478. * The calculation produces Range +- 180 degrees.
  479. * But I think that 0 - 360 would be more customary?
  480. *
  481. *------------------------------------------------------------------*/
  482. double ll_bearing_deg (double lat1, double lon1, double lat2, double lon2)
  483. {
  484. double b;
  485. lat1 *= M_PI / 180;
  486. lon1 *= M_PI / 180;
  487. lat2 *= M_PI / 180;
  488. lon2 *= M_PI / 180;
  489. b = atan2 (sin(lon2-lon1) * cos(lat2),
  490. cos(lat1) * sin(lat2) - sin(lat1) * cos(lat2) * cos(lon2-lon1));
  491. b *= 180 / M_PI;
  492. if (b < 0) b += 360;
  493. return (b);
  494. }
  495. /*------------------------------------------------------------------
  496. *
  497. * Function: ll_dest_lat
  498. * ll_dest_lon
  499. *
  500. * Purpose: Calculate the destination location given a starting point,
  501. * distance, and bearing,
  502. *
  503. * Inputs: lat1, lon1 - starting location, in degrees.
  504. * dist - distance in km.
  505. * bearing - direction in degrees. Shouldn't matter
  506. * if it is in +- 180 or 0 to 360 range.
  507. *
  508. * Returns: New latitude or longitude.
  509. *
  510. *------------------------------------------------------------------*/
  511. double ll_dest_lat (double lat1, double lon1, double dist, double bearing)
  512. {
  513. double lat2;
  514. lat1 *= M_PI / 180; // Everything to radians.
  515. lon1 *= M_PI / 180;
  516. bearing *= M_PI / 180;
  517. lat2 = asin(sin(lat1) * cos(dist/R) + cos(lat1) * sin(dist/R) * cos(bearing));
  518. lat2 *= 180 / M_PI; // Back to degrees.
  519. return (lat2);
  520. }
  521. double ll_dest_lon (double lat1, double lon1, double dist, double bearing)
  522. {
  523. double lon2;
  524. double lat2;
  525. lat1 *= M_PI / 180; // Everything to radians.
  526. lon1 *= M_PI / 180;
  527. bearing *= M_PI / 180;
  528. lat2 = asin(sin(lat1) * cos(dist/R) + cos(lat1) * sin(dist/R) * cos(bearing));
  529. lon2 = lon1 + atan2(sin(bearing) * sin(dist/R) * cos(lat1), cos(dist/R) - sin(lat1) * sin(lat2));
  530. lon2 *= 180 / M_PI; // Back to degrees.
  531. return (lon2);
  532. }
  533. /*------------------------------------------------------------------
  534. *
  535. * Function: ll_from_grid_square
  536. *
  537. * Purpose: Convert Maidenhead locator to latitude and longitude.
  538. *
  539. * Inputs: maidenhead - 2, 4, 6, 8, 10, or 12 character grid square locator.
  540. *
  541. * Outputs: dlat, dlon - Latitude and longitude.
  542. * Original values unchanged if error.
  543. *
  544. * Returns: 1 for success, 0 if error.
  545. *
  546. * Reference: A good converter for spot checking. Only handles 4 or 6 characters :-(
  547. * http://home.arcor.de/waldemar.kebsch/The_Makrothen_Contest/fmaidenhead.html
  548. *
  549. * Rambling: What sort of resolution does this provide?
  550. * For 8 character form, each latitude unit is 0.25 minute.
  551. * (Longitude can be up to twice that around the equator.)
  552. * 6371 km * 2 * pi * 0.25 / 60 / 360 = 0.463 km. Is that right?
  553. *
  554. * Using this calculator, http://www.earthpoint.us/Convert.aspx
  555. * It gives lower left corner of square rather than the middle. :-(
  556. *
  557. * FN42MA00 --> 19T 334361mE 4651711mN
  558. * FN42MA11 --> 19T 335062mE 4652157mN
  559. * ------ -------
  560. * 701 446 meters difference.
  561. *
  562. * With another two pairs, we are down around 2 meters for latitude.
  563. *
  564. *------------------------------------------------------------------*/
  565. #define MH_MIN_PAIR 1
  566. #define MH_MAX_PAIR 6
  567. #define MH_UNITS ( 18 * 10 * 24 * 10 * 24 * 10 * 2 )
  568. static const struct {
  569. char *position;
  570. char min_ch;
  571. char max_ch;
  572. int value;
  573. } mh_pair[MH_MAX_PAIR] = {
  574. { "first", 'A', 'R', 10 * 24 * 10 * 24 * 10 * 2 },
  575. { "second", '0', '9', 24 * 10 * 24 * 10 * 2 },
  576. { "third", 'A', 'X', 10 * 24 * 10 * 2 },
  577. { "fourth", '0', '9', 24 * 10 * 2 },
  578. { "fifth", 'A', 'X', 10 * 2 },
  579. { "sixth", '0', '9', 2 } }; // Even so we can get center of square.
  580. #if 1
  581. int ll_from_grid_square (char *maidenhead, double *dlat, double *dlon)
  582. {
  583. char mh[16]; /* Local copy, changed to upper case. */
  584. int ilat = 0, ilon = 0; /* In units in table above. */
  585. char *p;
  586. int n;
  587. int np = strlen(maidenhead) / 2; /* Number of pairs of characters. */
  588. if (strlen(maidenhead) %2 != 0 || np < MH_MIN_PAIR || np > MH_MAX_PAIR) {
  589. //text_color_set(DW_COLOR_ERROR);
  590. printf("Maidenhead locator \"%s\" must from 1 to %d pairs of characters.\n", maidenhead, MH_MAX_PAIR);
  591. return (0);
  592. }
  593. strlcpy (mh, maidenhead, sizeof(mh));
  594. for (p = mh; *p != '\0'; p++) {
  595. if (islower((uint)*p)) *p = toupper((uint)*p);
  596. }
  597. for (n = 0; n < np; n++) {
  598. if (mh[2*n] < mh_pair[n].min_ch || mh[2*n] > mh_pair[n].max_ch ||
  599. mh[2*n+1] < mh_pair[n].min_ch || mh[2*n+1] > mh_pair[n].max_ch) {
  600. //text_color_set(DW_COLOR_ERROR);
  601. printf("The %s pair of characters in Maidenhead locator \"%s\" must be in range of %c thru %c.\n",
  602. mh_pair[n].position, maidenhead, mh_pair[n].min_ch, mh_pair[n].max_ch);
  603. return (0);
  604. }
  605. ilon += ( mh[2*n] - mh_pair[n].min_ch ) * mh_pair[n].value;
  606. ilat += ( mh[2*n+1] - mh_pair[n].min_ch ) * mh_pair[n].value;
  607. if (n == np-1) { // If last pair, take center of square.
  608. ilon += mh_pair[n].value / 2;
  609. ilat += mh_pair[n].value / 2;
  610. }
  611. }
  612. *dlat = (double)ilat / MH_UNITS * 180. - 90.;
  613. *dlon = (double)ilon / MH_UNITS * 360. - 180.;
  614. ////text_color_set(DW_COLOR_DEBUG);
  615. //printf("DEBUG: Maidenhead conversion \"%s\" -> %.6f %.6f\n", maidenhead, *dlat, *dlon);
  616. return (1);
  617. }
  618. #else
  619. int ll_from_grid_square (char *maidenhead, double *dlat, double *dlon)
  620. {
  621. double lat, lon;
  622. char mh[16];
  623. if (strlen(maidenhead) != 2 && strlen(maidenhead) != 4 && strlen(maidenhead) != 6 && strlen(maidenhead) != 8) {
  624. //text_color_set(DW_COLOR_ERROR);
  625. printf("Maidenhead locator \"%s\" must 2, 4, 6, or 8 characters.\n", maidenhead);
  626. return (0);
  627. }
  628. strcpy (mh, maidenhead);
  629. if (islower((uint)mh[0])) mh[0] = toupper((uint)mh[0]);
  630. if (islower((uint)mh[1])) mh[1] = toupper((uint)mh[1]);
  631. if (mh[0] < 'A' || mh[0] > 'R' || mh[1] < 'A' || mh[1] > 'R') {
  632. //text_color_set(DW_COLOR_ERROR);
  633. printf("The first pair of characters in Maidenhead locator \"%s\" must be in range of A thru R.\n", maidenhead);
  634. return (0);
  635. }
  636. /* Lon: 360 deg / 18 squares = 20 deg / square */
  637. /* Lat: 180 deg / 18 squares = 10 deg / square */
  638. lon = (mh[0] - 'A') * 20 - 180;
  639. lat = (mh[1] - 'A') * 10 - 90;
  640. if (strlen(mh) >= 4) {
  641. if ( ! isdigit((uint)mh[2]) || ! isdigit((uint)mh[3]) ) {
  642. //text_color_set(DW_COLOR_ERROR);
  643. printf("The second pair of characters in Maidenhead locator \"%s\" must be digits.\n", maidenhead);
  644. return (0);
  645. }
  646. /* Lon: 20 deg / 10 squares = 2 deg / square */
  647. /* Lat: 10 deg / 10 squares = 1 deg / square */
  648. lon += (mh[2] - '0') * 2;
  649. lat += (mh[3] - '0');
  650. if (strlen(mh) >=6) {
  651. if (islower((uint)mh[4])) mh[4] = toupper((uint)mh[4]);
  652. if (islower((uint)mh[5])) mh[5] = toupper((uint)mh[5]);
  653. if (mh[4] < 'A' || mh[4] > 'X' || mh[5] < 'A' || mh[5] > 'X') {
  654. //text_color_set(DW_COLOR_ERROR);
  655. printf("The third pair of characters in Maidenhead locator \"%s\" must be in range of A thru X.\n", maidenhead);
  656. return (0);
  657. }
  658. /* Lon: 2 deg / 24 squares = 5 minutes / square */
  659. /* Lat: 1 deg / 24 squares = 2.5 minutes / square */
  660. lon += (mh[4] - 'A') * 5.0 / 60.0;
  661. lat += (mh[5] - 'A') * 2.5 / 60.0;
  662. if (strlen(mh) >= 8) {
  663. if ( ! isdigit((uint)mh[6]) || ! isdigit((uint)mh[7]) ) {
  664. //text_color_set(DW_COLOR_ERROR);
  665. printf("The fourth pair of characters in Maidenhead locator \"%s\" must be digits.\n", maidenhead);
  666. return (0);
  667. }
  668. /* Lon: 5 min / 10 squares = 0.5 minutes / square */
  669. /* Lat: 2.5 min / 10 squares = 0.25 minutes / square */
  670. lon += (mh[6] - '0') * 0.50 / 60.0;
  671. lat += (mh[7] - '0') * 0.25 / 60.0;
  672. lon += 0.250 / 60.0; /* Move from corner to center of square */
  673. lat += 0.125 / 60.0;
  674. }
  675. else {
  676. lon += 2.5 / 60.0; /* Move from corner to center of square */
  677. lat += 1.25 / 60.0;
  678. }
  679. }
  680. else {
  681. lon += 1.0; /* Move from corner to center of square */
  682. lat += 0.5;
  683. }
  684. }
  685. else {
  686. lon += 10; /* Move from corner to center of square */
  687. lat += 5;
  688. }
  689. ////text_color_set(DW_COLOR_DEBUG);
  690. //printf("DEBUG: Maidenhead conversion \"%s\" -> %.6f %.6f\n", maidenhead, lat, lon);
  691. *dlat = lat;
  692. *dlon = lon;
  693. return (1);
  694. }
  695. #endif
  696. /* end ll_from_grid_square */
  697. #if LLTEST
  698. /* gcc -o lltest -DLLTEST latlong.c textcolor.o misc.a && lltest */
  699. int main (int argc, char *argv[])
  700. {
  701. char result[20];
  702. int errors = 0;
  703. int ok;
  704. double dlat, dlon;
  705. double d, b;
  706. /* Latitude to APRS format. */
  707. latitude_to_str (45.25, 0, result);
  708. if (strcmp(result, "4515.00N") != 0) { errors++; printf ("Error 1.1: Did not expect \"%s\"\n", result); }
  709. latitude_to_str (-45.25, 0, result);
  710. if (strcmp(result, "4515.00S") != 0) { errors++; printf ("Error 1.2: Did not expect \"%s\"\n", result); }
  711. latitude_to_str (45.999830, 0, result);
  712. if (strcmp(result, "4559.99N") != 0) { errors++; printf ("Error 1.3: Did not expect \"%s\"\n", result); }
  713. latitude_to_str (45.99999, 0, result);
  714. if (strcmp(result, "4600.00N") != 0) { errors++; printf ("Error 1.4: Did not expect \"%s\"\n", result); }
  715. latitude_to_str (45.999830, 1, result);
  716. if (strcmp(result, "4559.9 N") != 0) { errors++; printf ("Error 1.5: Did not expect \"%s\"\n", result); }
  717. latitude_to_str (45.999830, 2, result);
  718. if (strcmp(result, "4559. N") != 0) { errors++; printf ("Error 1.6: Did not expect \"%s\"\n", result); }
  719. latitude_to_str (45.999830, 3, result);
  720. if (strcmp(result, "455 . N") != 0) { errors++; printf ("Error 1.7: Did not expect \"%s\"\n", result); }
  721. latitude_to_str (45.999830, 4, result);
  722. if (strcmp(result, "45 . N") != 0) { errors++; printf ("Error 1.8: Did not expect \"%s\"\n", result); }
  723. /* Longitude to APRS format. */
  724. longitude_to_str (45.25, 0, result);
  725. if (strcmp(result, "04515.00E") != 0) { errors++; printf ("Error 2.1: Did not expect \"%s\"\n", result); }
  726. longitude_to_str (-45.25, 0, result);
  727. if (strcmp(result, "04515.00W") != 0) { errors++; printf ("Error 2.2: Did not expect \"%s\"\n", result); }
  728. longitude_to_str (45.999830, 0, result);
  729. if (strcmp(result, "04559.99E") != 0) { errors++; printf ("Error 2.3: Did not expect \"%s\"\n", result); }
  730. longitude_to_str (45.99999, 0, result);
  731. if (strcmp(result, "04600.00E") != 0) { errors++; printf ("Error 2.4: Did not expect \"%s\"\n", result); }
  732. longitude_to_str (45.999830, 1, result);
  733. if (strcmp(result, "04559.9 E") != 0) { errors++; printf ("Error 2.5: Did not expect \"%s\"\n", result); }
  734. longitude_to_str (45.999830, 2, result);
  735. if (strcmp(result, "04559. E") != 0) { errors++; printf ("Error 2.6: Did not expect \"%s\"\n", result); }
  736. longitude_to_str (45.999830, 3, result);
  737. if (strcmp(result, "0455 . E") != 0) { errors++; printf ("Error 2.7: Did not expect \"%s\"\n", result); }
  738. longitude_to_str (45.999830, 4, result);
  739. if (strcmp(result, "045 . E") != 0) { errors++; printf ("Error 2.8: Did not expect \"%s\"\n", result); }
  740. /* Compressed format. */
  741. /* Protocol spec example has <*e7 but I got <*e8 due to rounding rather than truncation to integer. */
  742. memset(result, 0, sizeof(result));
  743. latitude_to_comp_str (-90.0, result);
  744. if (strcmp(result, "{{!!") != 0) { errors++; printf ("Error 3.1: Did not expect \"%s\"\n", result); }
  745. latitude_to_comp_str (49.5, result);
  746. if (strcmp(result, "5L!!") != 0) { errors++; printf ("Error 3.2: Did not expect \"%s\"\n", result); }
  747. latitude_to_comp_str (90.0, result);
  748. if (strcmp(result, "!!!!") != 0) { errors++; printf ("Error 3.3: Did not expect \"%s\"\n", result); }
  749. longitude_to_comp_str (-180.0, result);
  750. if (strcmp(result, "!!!!") != 0) { errors++; printf ("Error 3.4: Did not expect \"%s\"\n", result); }
  751. longitude_to_comp_str (-72.75, result);
  752. if (strcmp(result, "<*e8") != 0) { errors++; printf ("Error 3.5: Did not expect \"%s\"\n", result); }
  753. longitude_to_comp_str (180.0, result);
  754. if (strcmp(result, "{{!!") != 0) { errors++; printf ("Error 3.6: Did not expect \"%s\"\n", result); }
  755. // to be continued for others... NMEA...
  756. /* Distance & bearing - Take a couple examples from other places and see if we get similar results. */
  757. // http://www.movable-type.co.uk/scripts/latlong.html
  758. d = ll_distance_km (35., 45., 35., 135.);
  759. b = ll_bearing_deg (35., 45., 35., 135.);
  760. if (d < 7862 || d > 7882) { errors++; printf ("Error 5.1: Did not expect distance %.1f\n", d); }
  761. if (b < 59.7 || b > 60.3) { errors++; printf ("Error 5.2: Did not expect bearing %.1f\n", b); }
  762. // Sydney to Kinsale. https://woodshole.er.usgs.gov/staffpages/cpolloni/manitou/ccal.htm
  763. d = ll_distance_km (-33.8688, 151.2093, 51.7059, -8.5222);
  764. b = ll_bearing_deg (-33.8688, 151.2093, 51.7059, -8.5222);
  765. if (d < 17435 || d > 17455) { errors++; printf ("Error 5.3: Did not expect distance %.1f\n", d); }
  766. if (b < 327-1 || b > 327+1) { errors++; printf ("Error 5.4: Did not expect bearing %.1f\n", b); }
  767. /*
  768. * More distance and bearing.
  769. * Here we will start at some location1 (lat1,lon1) and go some distance (d1) at some bearing (b1).
  770. * This results in a new location2 (lat2, lon2).
  771. * We then calculate the distance and bearing from location1 to location2 and compare with the intention.
  772. */
  773. int lat1, lon1, d1 = 10, b1;
  774. double lat2, lon2, d2, b2;
  775. for (lat1 = -60; lat1 <= 60; lat1 += 30) {
  776. for (lon1 = -180; lon1 <= 180; lon1 +=30) {
  777. for (b1 = 0; b1 < 360; b1 += 15) {
  778. lat2 = ll_dest_lat ((double)lat1, (double)lon1, (double)d1, (double)b1);
  779. lon2 = ll_dest_lon ((double)lat1, (double)lon1, (double)d1, (double)b1);
  780. d2 = ll_distance_km ((double)lat1, (double)lon1, lat2, lon2);
  781. b2 = ll_bearing_deg ((double)lat1, (double)lon1, lat2, lon2);
  782. if (b2 > 359.9 && b2 < 360.1) b2 = 0;
  783. // must be within 0.1% of distance and 0.1 degree.
  784. if (d2 < 0.999 * d1 || d2 > 1.001 * d1) { errors++; printf ("Error 5.8: lat1=%d, lon2=%d, d1=%d, b1=%d, d2=%.2f\n", lat1, lon1, d1, b1, d2); }
  785. if (b2 < b1 - 0.1 || b2 > b1 + 0.1) { errors++; printf ("Error 5.9: lat1=%d, lon2=%d, d1=%d, b1=%d, b2=%.2f\n", lat1, lon1, d1, b1, b2); }
  786. }
  787. }
  788. }
  789. /* Maidenhead locator to lat/long. */
  790. ok = ll_from_grid_square ("BL11", &dlat, &dlon);
  791. if (!ok || dlat < 20.4999999 || dlat > 21.5000001 || dlon < -157.0000001 || dlon > -156.9999999) { errors++; printf ("Error 7.1: Did not expect %.6f %.6f\n", dlat, dlon); }
  792. ok = ll_from_grid_square ("BL11BH", &dlat, &dlon);
  793. if (!ok || dlat < 21.31249 || dlat > 21.31251 || dlon < -157.87501 || dlon > -157.87499) { errors++; printf ("Error 7.2: Did not expect %.6f %.6f\n", dlat, dlon); }
  794. #if 0 // TODO: add more test cases after comparing results with other cconverters.
  795. // Many other converters are limited to smaller number of characters,
  796. // or return corner rather than center of square, or return 3 decimal places for degrees.
  797. ok = ll_from_grid_square ("BL11BH16", &dlat, &dlon);
  798. if (!ok || dlat < 21.? || dlat > 21.? || dlon < -157.? || dlon > -157.?) { errors++; printf ("Error 7.3: Did not expect %.6f %.6f\n", dlat, dlon); }
  799. ok = ll_from_grid_square ("BL11BH16oo", &dlat, &dlon);
  800. if (!ok || dlat < 21.? || dlat > 21.? || dlon < -157.? || dlon > -157.?) { errors++; printf ("Error 7.4: Did not expect %.6f %.6f\n", dlat, dlon); }
  801. ok = ll_from_grid_square ("BL11BH16oo66", &dlat, &dlon);
  802. if (!ok || dlat < 21.? || dlat > 21.? || dlon < -157.? || dlon > -157.?) { errors++; printf ("Error 7.5: Did not expect %.6f %.6f\n", dlat, dlon); }
  803. #endif
  804. if (errors > 0) {
  805. //text_color_set (DW_COLOR_ERROR);
  806. printf ("\nLocation Coordinate Conversion Test - FAILED!\n");
  807. exit (EXIT_FAILURE);
  808. }
  809. //text_color_set (DW_COLOR_REC);
  810. printf ("\nLocation Coordinate Conversion Test - SUCCESS!\n");
  811. exit (EXIT_SUCCESS);
  812. }
  813. #endif
  814. /* end latlong.c */