test_timespec.c 27 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820
  1. /*
  2. * Unit test for timespec's
  3. * Also for parse_uri_dest()
  4. *
  5. * This file is Copyright 2010 by the GPSD project
  6. * SPDX-License-Identifier: BSD-2-clause
  7. *
  8. */
  9. /* first so the #defs work */
  10. #include "../include/gpsd_config.h"
  11. #include <math.h>
  12. #include <stdbool.h>
  13. #include <stdint.h> /* required by C99, for int32_t */
  14. #include <stdio.h>
  15. #include <stdlib.h>
  16. #include <stdlib.h>
  17. #include <string.h>
  18. #include <unistd.h>
  19. #include "../include/compiler.h" // for FALLTHROUGH
  20. #include "../include/gpsd.h"
  21. #define TS_ZERO {0,0}
  22. #define TS_ZERO_ONE {0,1}
  23. #define TS_ZERO_ONEM {0,1000000}
  24. #define TS_ZERO_TWO {0,2}
  25. #define TS_ZERO_TWOM {0,2000000}
  26. #define TS_ZERO_TREES {0,333333333}
  27. #define TS_ZERO_SIXS7 {0,666666667}
  28. #define TS_ZERO_NINES {0,999999999}
  29. #define TS_ONE {1,0}
  30. #define TS_ONE_ONE {1,1}
  31. #define TS_TWO {2,0}
  32. #define TS_N_ZERO_ONE {0,-1}
  33. #define TS_N_ZERO_TWO {0,-2}
  34. #define TS_N_ZERO_TREES {0,-333333333}
  35. #define TS_N_ZERO_NINES {0,-999999999}
  36. #define TS_N_ONE {-1,0}
  37. /* minutes, hours, days */
  38. #define TS_ONEM {60,0} /* one minute */
  39. #define TS_ONEM_TREES {60,333333333} /* one minute, threes */
  40. #define TS_ONEM_NINES {60,999999999} /* one minute, nines */
  41. #define TS_ONEH {3600,0} /* one hour */
  42. #define TS_ONEH_TREES {3600,333333333} /* one hour, threes */
  43. #define TS_ONEH_NINES {3600,999999999} /* one hour, nines */
  44. #define TS_ONED {86400,0} /* one day */
  45. #define TS_ONED_TREES {86400,333333333} /* one day, threes */
  46. #define TS_ONED_NINES {86400,999999999} /* one day, nines */
  47. #define TS_N_ONEM {-60,0} /* negative one minute */
  48. #define TS_N_ONEH {-3600,0} /* negative one hour */
  49. #define TS_N_ONED {-86400,0} /* negative one day */
  50. /* Dec 31, 23:59 2037 GMT */
  51. #define TS_2037 {2145916799, 0}
  52. #define TS_2037_ONE {2145916799, 1}
  53. #define TS_2037_TWO {2145916799, 2}
  54. #define TS_2037_X {2145916799, 123456789}
  55. #define TS_2037_TREES {2145916799, 333333333}
  56. #define TS_2037_SIXS7 {2145916799, 666666667}
  57. #define TS_2037_NINES {2145916799, 999999999}
  58. #define TS_N_2037_TREES {-2145916799, -333333333}
  59. #define TS_N_2037_NINES {-2145916799, -999999999}
  60. /* a 32 bit copy of timespec_diff_ns() to force a 32 bit int */
  61. /* used to demonstrate how 32 bit longs can not work */
  62. #define timespec_diff_ns32(x, y) \
  63. (int32_t)((int32_t)(((x).tv_sec-(y).tv_sec)*NS_IN_SEC)+(x).tv_nsec-(y).tv_nsec)
  64. /* a 64 bit copy of timespec_diff_ns() to force a 64 bit int */
  65. /* used to demonstrate how 64 bit long longs can work */
  66. #define timespec_diff_ns64(x, y) \
  67. (int64_t)((int64_t)(((x).tv_sec-(y).tv_sec)*NS_IN_SEC)+(x).tv_nsec-(y).tv_nsec)
  68. /* convert long long ns to a timespec */
  69. #define ns_to_timespec(ts, ns) \
  70. (ts).tv_sec = ns / NS_IN_SEC; \
  71. (ts).tv_nsec = ns % NS_IN_SEC;
  72. /* convert double to a timespec */
  73. static inline void d_str( const double d, char *buf, size_t buf_size)
  74. {
  75. /* convert to string */
  76. if ( 0 <= d ) {
  77. (void) snprintf( buf, buf_size, " %.9f", d);
  78. } else {
  79. (void) snprintf( buf, buf_size, "%.9f", d);
  80. }
  81. }
  82. /* a - b should be c */
  83. struct subtract_test {
  84. struct timespec a;
  85. struct timespec b;
  86. struct timespec c;
  87. bool last; /* last test marker */
  88. };
  89. struct subtract_test subtract_tests[] = {
  90. { TS_ZERO, TS_ZERO, TS_ZERO, 0},
  91. { TS_ONE, TS_ONE, TS_ZERO, 0},
  92. { TS_ZERO_ONE, TS_ZERO_ONE, TS_ZERO, 0},
  93. { TS_ONE_ONE, TS_ONE_ONE, TS_ZERO, 0},
  94. { TS_N_ONE, TS_N_ONE, TS_ZERO, 0},
  95. { TS_N_ZERO_ONE, TS_N_ZERO_ONE, TS_ZERO, 0},
  96. { TS_ZERO_TREES, TS_ZERO_TREES, TS_ZERO, 0},
  97. { TS_ZERO_NINES, TS_ZERO_NINES, TS_ZERO, 0},
  98. { TS_ZERO_TREES, TS_ZERO, TS_ZERO_TREES, 0},
  99. { TS_ZERO, TS_N_ONE, TS_ONE, 0},
  100. { TS_ONE, TS_ZERO, TS_ONE, 0},
  101. { TS_TWO, TS_ONE, TS_ONE, 0},
  102. { TS_ONE_ONE, TS_ONE, TS_ZERO_ONE, 0},
  103. { TS_ONE, TS_ZERO_TREES, TS_ZERO_SIXS7, 0},
  104. { TS_ONE, TS_ZERO_NINES, TS_ZERO_ONE, 0},
  105. { TS_ZERO_TWO, TS_ZERO_ONE, TS_ZERO_ONE, 0},
  106. { TS_2037_ONE, TS_2037, TS_ZERO_ONE, 0},
  107. { TS_ONE_ONE, TS_ZERO_NINES, TS_ZERO_TWO, 0},
  108. { TS_ONEM, TS_ZERO, TS_ONEM, 0},
  109. { TS_ONEM_TREES, TS_ZERO, TS_ONEM_TREES, 0},
  110. { TS_ONEM_NINES, TS_ZERO, TS_ONEM_NINES, 0},
  111. { TS_ZERO, TS_ONEM, TS_N_ONEM, 0},
  112. { TS_ONEH, TS_ZERO, TS_ONEH, 0},
  113. { TS_ONEH_TREES, TS_ZERO, TS_ONEH_TREES, 0},
  114. { TS_ONEH_NINES, TS_ZERO, TS_ONEH_NINES, 0},
  115. { TS_ZERO, TS_ONEH, TS_N_ONEH, 0},
  116. { TS_ONED, TS_ZERO, TS_ONED, 0},
  117. { TS_ONED_TREES, TS_ZERO, TS_ONED_TREES, 0},
  118. { TS_ONED_NINES, TS_ZERO, TS_ONED_NINES, 0},
  119. { TS_ZERO, TS_ONED, TS_N_ONED, 0},
  120. { TS_2037_NINES, TS_2037, TS_ZERO_NINES, 0},
  121. { TS_2037_TREES, TS_ZERO, TS_2037_TREES, 0},
  122. { TS_2037_SIXS7, TS_2037, TS_ZERO_SIXS7, 0},
  123. { TS_2037_TREES, TS_2037, TS_ZERO_TREES, 0},
  124. { TS_2037_NINES, TS_ZERO, TS_2037_NINES, 0},
  125. { TS_ZERO, TS_ONE, TS_N_ONE, 0},
  126. { TS_ONE, TS_TWO, TS_N_ONE, 0},
  127. { TS_ZERO, TS_ZERO_ONE, TS_N_ZERO_ONE, 0},
  128. { TS_ONE, TS_ONE_ONE, TS_N_ZERO_ONE, 0},
  129. { TS_ZERO_ONE, TS_ZERO_TWO, TS_N_ZERO_ONE, 0},
  130. { TS_2037, TS_2037_ONE, TS_N_ZERO_ONE, 0},
  131. { TS_ZERO_NINES, TS_ONE_ONE, TS_N_ZERO_TWO, 0},
  132. { TS_2037, TS_2037_NINES, TS_N_ZERO_NINES, 0},
  133. { TS_ZERO, TS_2037_NINES, TS_N_2037_NINES, 1},
  134. };
  135. typedef struct ts_to_ms_test {
  136. struct timespec input;
  137. int64_t expected;
  138. bool last;
  139. } ts_to_ms_test_t;
  140. struct ts_to_ms_test ts_to_ms_tests[] = {
  141. { TS_ZERO, 0, 0},
  142. { TS_ZERO_ONE, 0, 0},
  143. { TS_ZERO_ONEM, 1, 0},
  144. { TS_ZERO_TWO, 0, 0},
  145. { TS_ZERO_TWOM, 2, 0},
  146. { TS_ZERO_NINES, 999, 0},
  147. { TS_ONE, 1000, 0},
  148. { TS_ONE_ONE, 1000, 0},
  149. { TS_TWO, 2000, 0},
  150. { TS_N_ZERO_ONE, 0, 0},
  151. { TS_N_ZERO_TWO, 0, 0},
  152. { TS_N_ZERO_NINES, -999, 0},
  153. { TS_N_ONE, -1000, 0},
  154. { TS_ONEM, 60000, 0},
  155. { TS_ONEM_TREES, 60333, 0},
  156. { TS_ONEH, 3600000, 0},
  157. { TS_ONEH_TREES, 3600333, 0},
  158. { TS_ONED, 86400000, 0},
  159. { TS_ONED_TREES, 86400333, 0},
  160. { TS_N_ONEM, -60000, 0},
  161. { TS_N_ONEH, -3600000, 0},
  162. { TS_N_ONED, -86400000, 0},
  163. { { -1, MS_IN_NS}, -999, 0},
  164. { { -1, -MS_IN_NS}, -1001, 0},
  165. // Note no (extra) loss of precision on the following
  166. { TS_2037, 2145916799000ULL, 0},
  167. { TS_2037_ONE, 2145916799000ULL, 0},
  168. { TS_2037_TREES, 2145916799333ULL, 0},
  169. { TS_2037_NINES, 2145916799999ULL, 1},
  170. };
  171. /*
  172. * test timespec_t to int64_t of milli seconds: TSTOMS()
  173. *
  174. */
  175. static int test_ts_to_ms(int verbose)
  176. {
  177. struct ts_to_ms_test *p = ts_to_ms_tests;
  178. int fail_count = 0;
  179. while (1) {
  180. char buf_i[TIMESPEC_LEN];
  181. int64_t result;
  182. result = TSTOMS(&p->input);
  183. timespec_str(&p->input, buf_i, sizeof(buf_i));
  184. if (p->expected != result) {
  185. printf("%21s = %lld, FAIL s/b %lld\n",
  186. buf_i, (long long) result, (long long) p->expected);
  187. fail_count++;
  188. } else if ( verbose ) {
  189. printf("%21s = %lld\n", buf_i, (long long) result);
  190. }
  191. if (p->last) {
  192. break;
  193. }
  194. p++;
  195. };
  196. if (fail_count) {
  197. printf("timespec subtract test failed %d tests\n", fail_count );
  198. } else {
  199. puts("timespec subtract test succeeded\n");
  200. }
  201. return fail_count;
  202. }
  203. /*
  204. * test subtractions using native timespec math: TS_SUB()
  205. *
  206. */
  207. static int test_ts_subtract(int verbose)
  208. {
  209. struct subtract_test *p = subtract_tests;
  210. int fail_count = 0;
  211. while (1) {
  212. char buf_a[TIMESPEC_LEN];
  213. char buf_b[TIMESPEC_LEN];
  214. char buf_c[TIMESPEC_LEN];
  215. char buf_r[TIMESPEC_LEN];
  216. struct timespec r;
  217. TS_SUB(&r, &p->a, &p->b);
  218. timespec_str( &p->a, buf_a, sizeof(buf_a) );
  219. timespec_str( &p->b, buf_b, sizeof(buf_b) );
  220. timespec_str( &p->c, buf_c, sizeof(buf_c) );
  221. timespec_str( &r, buf_r, sizeof(buf_r) );
  222. if ((p->c.tv_sec != r.tv_sec) || (p->c.tv_nsec != r.tv_nsec)) {
  223. printf("%21s - %21s = %21s, FAIL s/b %21s\n",
  224. buf_a, buf_b, buf_r, buf_c);
  225. fail_count++;
  226. } else if ( verbose ) {
  227. printf("%21s - %21s = %21s\n", buf_a, buf_b, buf_r);
  228. }
  229. if (p->last) {
  230. break;
  231. }
  232. p++;
  233. };
  234. if (fail_count) {
  235. printf("timespec subtract test failed %d tests\n", fail_count );
  236. } else {
  237. puts("timespec subtract test succeeded\n");
  238. }
  239. return fail_count;
  240. }
  241. /*
  242. * test subtractions using timespec_diff_ns()
  243. *
  244. */
  245. static int test_ns_subtract( int verbose )
  246. {
  247. struct subtract_test *p = subtract_tests;
  248. int fail_count = 0;
  249. while ( 1 ) {
  250. char buf_a[TIMESPEC_LEN];
  251. char buf_b[TIMESPEC_LEN];
  252. char buf_c[TIMESPEC_LEN];
  253. char buf_r[TIMESPEC_LEN];
  254. struct timespec r;
  255. long long r_ns;
  256. r_ns = timespec_diff_ns(p->a, p->b);
  257. timespec_str( &p->a, buf_a, sizeof(buf_a) );
  258. timespec_str( &p->b, buf_b, sizeof(buf_b) );
  259. timespec_str( &p->c, buf_c, sizeof(buf_c) );
  260. ns_to_timespec( r, r_ns);
  261. timespec_str( &r, buf_r, sizeof(buf_r) );
  262. if ( (p->c.tv_sec != r.tv_sec) || (p->c.tv_nsec != r.tv_nsec) ) {
  263. printf("%21s - %21s = %21s, FAIL s/b %21s\n",
  264. buf_a, buf_b, buf_r, buf_c);
  265. fail_count++;
  266. } else if ( verbose ) {
  267. printf("%21s - %21s = %21s\n", buf_a, buf_b, buf_r);
  268. }
  269. if ( p->last ) {
  270. break;
  271. }
  272. p++;
  273. };
  274. if ( fail_count ) {
  275. printf("ns subtract test failed %d tests\n", fail_count );
  276. } else {
  277. puts("ns subtract test succeeded\n");
  278. }
  279. return fail_count;
  280. }
  281. typedef struct format_test {
  282. struct timespec input;
  283. char *expected;
  284. bool last;
  285. } format_test_t;
  286. struct format_test format_tests[] = {
  287. { TS_ZERO, " 0.000000000", 0},
  288. { TS_ZERO_ONE, " 0.000000001", 0},
  289. { TS_ZERO_TWO, " 0.000000002", 0},
  290. { TS_ZERO_NINES, " 0.999999999", 0},
  291. { TS_ONE, " 1.000000000", 0},
  292. { TS_ONE_ONE, " 1.000000001", 0},
  293. { TS_TWO, " 2.000000000", 0},
  294. { TS_N_ZERO_ONE, "-0.000000001", 0},
  295. { TS_N_ZERO_TWO, "-0.000000002", 0},
  296. { TS_N_ZERO_NINES, "-0.999999999", 0},
  297. { TS_N_ONE, "-1.000000000", 0},
  298. { TS_ONEM, " 60.000000000", 0},
  299. { TS_ONEM_TREES, " 60.333333333", 0},
  300. { TS_ONEH, " 3600.000000000", 0},
  301. { TS_ONEH_TREES, " 3600.333333333", 0},
  302. { TS_ONED, " 86400.000000000", 0},
  303. { TS_ONED_TREES, " 86400.333333333", 0},
  304. { TS_N_ONEM, "-60.000000000", 0},
  305. { TS_N_ONEH, "-3600.000000000", 0},
  306. { TS_N_ONED, "-86400.000000000", 0},
  307. { { -1, 1}, "-1.000000001", 0},
  308. { { -1, -1}, "-1.000000001", 0},
  309. { TS_2037, " 2145916799.000000000", 0},
  310. { TS_2037_ONE, " 2145916799.000000001", 0},
  311. { TS_2037_TREES, " 2145916799.333333333", 0},
  312. { TS_2037_NINES, " 2145916799.999999999", 1},
  313. };
  314. static int test_format(int verbose )
  315. {
  316. format_test_t *p = format_tests;
  317. int fail_count = 0;
  318. while ( 1 ) {
  319. char buf[TIMESPEC_LEN];
  320. int fail;
  321. timespec_str( &p->input, buf, sizeof(buf) );
  322. fail = strncmp( buf, p->expected, TIMESPEC_LEN);
  323. if ( fail ) {
  324. printf("%21s, FAIL s/b: %21s\n", buf, p->expected);
  325. fail_count++;
  326. } else if ( verbose ) {
  327. printf("%21s\n", buf);
  328. }
  329. if ( p->last ) {
  330. break;
  331. }
  332. p++;
  333. };
  334. if ( fail_count ) {
  335. printf("timespec_str test failed %d tests\n", fail_count );
  336. } else {
  337. puts("timespec_str test succeeded\n");
  338. }
  339. return fail_count;
  340. }
  341. typedef struct {
  342. unsigned short week;
  343. int leap_seconds;
  344. timespec_t ts_tow;
  345. timespec_t ts_exp; // expected result
  346. char *exp_s; // expected string
  347. bool last;
  348. } gpstime_test_t;
  349. gpstime_test_t gpstime_tests[] = {
  350. // GPS time zero
  351. {0, 0, TS_ZERO, {315964800, 000000000}, "1980-01-06T00:00:00.000Z", 0},
  352. // GPS first roll over
  353. {1024, 7, TS_ZERO, {935279993, 000000000}, "1999-08-21T23:59:53.000Z", 0},
  354. // GPS first roll over
  355. {2048, 18, TS_ZERO, {1554595182, 000000000}, "2019-04-06T23:59:42.000Z", 0},
  356. {2076, 18, {239910, 100000000}, {1571769492, 100000000},
  357. "2019-10-22T18:38:12.100Z", 1},
  358. };
  359. static int test_gpsd_gpstime_resolv(int verbose)
  360. {
  361. char res_s[128];
  362. char buf[20];
  363. int fail_count = 0;
  364. struct gps_device_t session;
  365. struct gps_context_t context;
  366. timespec_t ts_res;
  367. gpstime_test_t *p = gpstime_tests;
  368. memset(&session, 0, sizeof(session));
  369. memset(&context, 0, sizeof(context));
  370. session.context = &context;
  371. context.errout.debug = 0; // a handle to change debug level
  372. while ( 1 ) {
  373. /* setup preconditions */
  374. context.gps_week = p->week;
  375. context.leap_seconds = p->leap_seconds;
  376. ts_res = gpsd_gpstime_resolv(&session, p->week, p->ts_tow);
  377. (void)timespec_to_iso8601(ts_res, res_s, sizeof(res_s));
  378. if (p->ts_exp.tv_sec != ts_res.tv_sec ||
  379. p->ts_exp.tv_nsec != ts_res.tv_nsec ||
  380. strcmp(res_s, p->exp_s) ) {
  381. // long long for 32-bit OS
  382. printf("FAIL %s s/b: %s\n"
  383. " %s s/b %s\n",
  384. timespec_str(&ts_res, buf, sizeof(buf)),
  385. timespec_str(&p->ts_exp, buf, sizeof(buf)),
  386. res_s, p->exp_s);
  387. fail_count++;
  388. } else if ( verbose ) {
  389. printf("%s (%s)\n",
  390. timespec_str(&p->ts_exp, buf, sizeof(buf)),
  391. p->exp_s);
  392. }
  393. if ( p->last ) {
  394. break;
  395. }
  396. p++;
  397. }
  398. if ( fail_count ) {
  399. printf("test_gpsd_gpstime_resolv test failed %d tests\n", fail_count);
  400. } else {
  401. puts("test_gpsd_gpstime_resolv test succeeded\n");
  402. }
  403. return fail_count;
  404. }
  405. static int ex_subtract_float(void)
  406. {
  407. struct subtract_test *p = subtract_tests;
  408. int fail_count = 0;
  409. printf( "\n\nsubtract test examples using doubles,floats,longs:\n"
  410. " ts: TS_SUB()\n"
  411. " l: timespec_to_ns() math\n"
  412. " l32: timespec_to_ns() math with 32 bit long\n"
  413. " l64: timespec_to_ns() math with 64 bit long\n"
  414. " f: float math\n"
  415. " d: double float math\n"
  416. "\n");
  417. while ( 1 ) {
  418. char buf_a[TIMESPEC_LEN];
  419. char buf_b[TIMESPEC_LEN];
  420. char buf_c[TIMESPEC_LEN];
  421. char buf_r[TIMESPEC_LEN];
  422. char buf_l[TIMESPEC_LEN];
  423. char buf_l32[TIMESPEC_LEN];
  424. char buf_l64[TIMESPEC_LEN];
  425. char buf_f[TIMESPEC_LEN];
  426. char buf_d[TIMESPEC_LEN];
  427. struct timespec ts_r;
  428. struct timespec ts_l;
  429. struct timespec ts_l32;
  430. struct timespec ts_l64;
  431. float f_a, f_b, f_r;
  432. double d_a, d_b, d_r;
  433. long long l;
  434. int32_t l32; /* simulate a 32 bit long */
  435. int64_t l64; /* simulate a 64 bit long */
  436. const char *fail_ts = "";
  437. const char *fail_l = "";
  438. const char *fail_l32 = "";
  439. const char *fail_l64 = "";
  440. const char *fail_f = "";
  441. const char *fail_d = "";
  442. /* timespec math */
  443. TS_SUB(&ts_r, &p->a, &p->b);
  444. /* float math */
  445. f_a = TSTONS( &p->a );
  446. f_b = TSTONS( &p->b );
  447. f_r = f_a - f_b;
  448. /* double float math */
  449. d_a = TSTONS( &p->a );
  450. d_b = TSTONS( &p->b );
  451. d_r = d_a - d_b;
  452. /* long math */
  453. l = timespec_diff_ns( p->a, p->b);
  454. l32 = timespec_diff_ns32( p->a, p->b);
  455. l64 = timespec_diff_ns64( p->a, p->b);
  456. /* now convert to strings */
  457. timespec_str( &p->a, buf_a, sizeof(buf_a) );
  458. timespec_str( &p->b, buf_b, sizeof(buf_b) );
  459. timespec_str( &p->c, buf_c, sizeof(buf_c) );
  460. timespec_str( &ts_r, buf_r, sizeof(buf_r) );
  461. ns_to_timespec( ts_l, l );
  462. timespec_str( &ts_l, buf_l, sizeof(buf_l) );
  463. ns_to_timespec( ts_l32, l32 );
  464. timespec_str( &ts_l32, buf_l32, sizeof(buf_l32) );
  465. ns_to_timespec( ts_l64, l64);
  466. timespec_str( &ts_l64, buf_l64, sizeof(buf_l64) );
  467. d_str( f_r, buf_f, sizeof(buf_f) );
  468. d_str( d_r, buf_d, sizeof(buf_d) );
  469. /* test strings */
  470. if ( strcmp( buf_r, buf_c) ) {
  471. fail_ts = "FAIL";
  472. fail_count++;
  473. }
  474. if ( strcmp( buf_l, buf_c) ) {
  475. fail_l = "FAIL";
  476. fail_count++;
  477. }
  478. if ( strcmp( buf_l32, buf_c) ) {
  479. fail_l32 = "FAIL";
  480. fail_count++;
  481. }
  482. if ( strcmp( buf_l64, buf_c) ) {
  483. fail_l64 = "FAIL";
  484. fail_count++;
  485. }
  486. if ( strcmp( buf_f, buf_c) ) {
  487. fail_f = "FAIL";
  488. fail_count++;
  489. }
  490. if ( strcmp( buf_d, buf_c) ) {
  491. fail_d = "FAIL";
  492. fail_count++;
  493. }
  494. printf("ts: %21s - %21s = %21s %s\n"
  495. "l; %21s - %21s = %21lld %s\n"
  496. "l32; %21s - %21s = %21lld %s\n"
  497. "l64; %21s - %21s = %21lld %s\n"
  498. "f; %21.9f - %21.9f = %21.9f %s\n"
  499. "d; %21.9f - %21.9f = %21.9f %s\n"
  500. "\n",
  501. buf_a, buf_b, buf_r, fail_ts,
  502. buf_a, buf_b, l, fail_l,
  503. buf_a, buf_b, (long long)l32, fail_l32,
  504. buf_a, buf_b, (long long)l64, fail_l64,
  505. f_a, f_b, f_r, fail_f,
  506. d_a, d_b, d_r, fail_d);
  507. if (p->last) {
  508. break;
  509. }
  510. p++;
  511. };
  512. if ( fail_count ) {
  513. printf("subtract test failed %d tests\n", fail_count );
  514. } else {
  515. puts("subtract test succeeded\n");
  516. }
  517. return fail_count;
  518. }
  519. /*
  520. * show examples of how integers and floats fail
  521. *
  522. */
  523. static void ex_precision(void)
  524. {
  525. format_test_t *p = format_tests;
  526. puts( "\n\n Simple conversion examples\n\n"
  527. "ts: timespec\n"
  528. "l32: 32 bit long\n"
  529. "l64: 64 bit long\n"
  530. "f: float\n"
  531. "d: double\n\n");
  532. while ( 1 ) {
  533. float f;
  534. double d;
  535. int32_t l32;
  536. int64_t l64;
  537. char buf_ts[TIMESPEC_LEN];
  538. char buf_l32[TIMESPEC_LEN];
  539. char buf_l64[TIMESPEC_LEN];
  540. char buf_f[TIMESPEC_LEN];
  541. char buf_d[TIMESPEC_LEN];
  542. const char *fail_ts = "";
  543. const char *fail_l32 = "";
  544. const char *fail_l64 = "";
  545. const char *fail_f = "";
  546. const char *fail_d = "";
  547. struct timespec *v = &(p->input);
  548. struct timespec ts_l32;
  549. struct timespec ts_l64;
  550. /* convert to test size */
  551. l32 = (int32_t)(v->tv_sec * NS_IN_SEC)+(int32_t)v->tv_nsec;
  552. l64 = (int64_t)(v->tv_sec * NS_IN_SEC)+(int64_t)v->tv_nsec;
  553. f = (float)TSTONS( v );
  554. d = TSTONS( v );
  555. /* now convert to strings */
  556. timespec_str( v, buf_ts, sizeof(buf_ts) );
  557. ns_to_timespec( ts_l32, l32);
  558. timespec_str( &ts_l32, buf_l32, sizeof(buf_l32) );
  559. ns_to_timespec( ts_l64, l64);
  560. timespec_str( &ts_l64, buf_l64, sizeof(buf_l64) );
  561. d_str( f, buf_f, sizeof(buf_f) );
  562. d_str( d, buf_d, sizeof(buf_d) );
  563. /* test strings */
  564. if ( strcmp( buf_ts, p->expected) ) {
  565. fail_ts = "FAIL";
  566. }
  567. if ( strcmp( buf_l32, p->expected) ) {
  568. fail_l32 = "FAIL";
  569. }
  570. if ( strcmp( buf_l64, p->expected) ) {
  571. fail_l64 = "FAIL";
  572. }
  573. if ( strcmp( buf_f, p->expected) ) {
  574. fail_f = "FAIL";
  575. }
  576. if ( strcmp( buf_d, p->expected) ) {
  577. fail_d = "FAIL";
  578. }
  579. printf( "ts: %21s %s\n"
  580. "l32: %21lld %s\n"
  581. "l64: %21lld %s\n"
  582. "f: %21.9f %s\n"
  583. "d: %21.9f %s\n\n",
  584. buf_ts, fail_ts,
  585. (long long)l32, fail_l32,
  586. (long long)l64, fail_l64,
  587. f, fail_f,
  588. d, fail_d);
  589. if ( p->last ) {
  590. break;
  591. }
  592. p++;
  593. }
  594. printf( "\n\nSubtraction examples:\n");
  595. ex_subtract_float();
  596. }
  597. struct test_parse_uri_dest_t {
  598. char *uri;
  599. char *host;
  600. char *service;
  601. char *device;
  602. } test_parse_uri_dest_t;
  603. // prefixed with gpsd:// in situ
  604. struct test_parse_uri_dest_t tests_parse_uri_dest[] = {
  605. {"localhost", "localhost", NULL, NULL},
  606. {"localhost/", "localhost", NULL, NULL},
  607. {"localhost:", "localhost", NULL, NULL},
  608. {"localhost::", "localhost", NULL, NULL},
  609. {"localhost::/dev/ttyAMA0", "localhost", NULL, "/dev/ttyAMA0"},
  610. {"localhost:2947:/dev/ttyAMA0", "localhost", "2947", "/dev/ttyAMA0"},
  611. {"localhost:2947", "localhost", "2947", NULL},
  612. {"localhost:2947/", "localhost", "2947", NULL},
  613. {"localhost:gpsd", "localhost", "gpsd", NULL},
  614. {"localhost:gpsd/", "localhost", "gpsd", NULL},
  615. {"gpsd.io", "gpsd.io", NULL, NULL},
  616. {"gpsd.io/", "gpsd.io", NULL, NULL},
  617. {"gpsd.io:", "gpsd.io", NULL, NULL},
  618. {"gpsd.io::", "gpsd.io", NULL, NULL},
  619. {"gpsd.io::/dev/ttyAMA0", "gpsd.io", NULL, "/dev/ttyAMA0"},
  620. {"gpsd.io:2947:/dev/ttyAMA0", "gpsd.io", "2947", "/dev/ttyAMA0"},
  621. {"gpsd.io:2947", "gpsd.io", "2947", NULL},
  622. {"gpsd.io:2947/", "gpsd.io", "2947", NULL},
  623. {"gpsd.io:gpsd", "gpsd.io", "gpsd", NULL},
  624. {"gpsd.io:gpsd/", "gpsd.io", "gpsd", NULL},
  625. {"127.0.0.1", "127.0.0.1", NULL, NULL},
  626. {"127.0.0.1/", "127.0.0.1", NULL, NULL},
  627. {"127.0.0.1:", "127.0.0.1", NULL, NULL},
  628. {"127.0.0.1::", "127.0.0.1", NULL, NULL},
  629. {"127.0.0.1::/dev/ttyAMA0", "127.0.0.1", NULL, "/dev/ttyAMA0"},
  630. {"127.0.0.1:2947", "127.0.0.1", "2947", NULL},
  631. {"127.0.0.1:2947/", "127.0.0.1", "2947", NULL},
  632. {"127.0.0.1:gpsd", "127.0.0.1", "gpsd", NULL},
  633. {"127.0.0.1:gpsd/", "127.0.0.1", "gpsd", NULL},
  634. {"[fe80::1]", "fe80::1", NULL, NULL},
  635. {"[fe80::1]/", "fe80::1", NULL, NULL},
  636. {"[fe80::1]:", "fe80::1", NULL, NULL},
  637. {"[fe80::1]::", "fe80::1", NULL, NULL},
  638. {"[fe80::1]::/dev/ttyAMA0", "fe80::1", NULL, "/dev/ttyAMA0"},
  639. {"[fe80::1]:2947", "fe80::1", "2947", NULL},
  640. {"[fe80::1]:2947/", "fe80::1", "2947", NULL},
  641. {"[fe80::1]:gpsd", "fe80::1", "gpsd", NULL},
  642. {"[fe80::1]:gpsd/", "fe80::1", "gpsd", NULL},
  643. {NULL, NULL, NULL, NULL},
  644. };
  645. static int test_parse_uri_dest(int verbose)
  646. {
  647. int fail_count = 0;
  648. char *host, *service, *device;
  649. struct test_parse_uri_dest_t *p = tests_parse_uri_dest;
  650. char uri[40];
  651. printf("\n\nTest parse_uri_dest()\n");
  652. while(NULL != p->uri) {
  653. int result;
  654. // parse_uri_dest() is destructive, so make a copy
  655. strlcpy(uri, p->uri, sizeof(uri));
  656. result = parse_uri_dest(uri, &host, &service, &device);
  657. if (0 != strcmp(p->host, host)) {
  658. result = 1;
  659. }
  660. if (NULL == service || NULL == p->service) {
  661. if (service != p->service) {
  662. result = 2;
  663. }
  664. } else if (0 != strcmp(p->service, service)) {
  665. result = 3;
  666. }
  667. if (NULL == device || NULL == p->device) {
  668. if (device != p->device) {
  669. result = 4;
  670. }
  671. } else if (0 != strcmp(p->device, device)) {
  672. result = 5;
  673. }
  674. if (0 != result) {
  675. printf("parse_uri_dest(%s, %s, %s, %s) failed %d\n", p->uri, host,
  676. service ? service : "NULL",
  677. device ? device : "NULL",
  678. result);
  679. printf(" s/b parse_uri_dest(%s, %s, %s, %s) = 0\n", p->uri,
  680. p->host,
  681. p->service ? p->service : "NULL",
  682. p->device ? p->device : "NULL");
  683. fail_count++;
  684. } else if (verbose) {
  685. printf("parse_uri_dest(%s, %s, %s, %s)\n", p->uri, host,
  686. service ? service : "NULL",
  687. device ? device : "NULL");
  688. }
  689. p++;
  690. }
  691. if (fail_count) {
  692. printf("parse_uri_dest() test failed %d tests\n", fail_count);
  693. } else {
  694. puts("parse_uri_dest() test succeeded\n");
  695. }
  696. return fail_count;
  697. }
  698. int main(int argc, char *argv[])
  699. {
  700. int fail_count = 0;
  701. int verbose = 0;
  702. int option;
  703. while ((option = getopt(argc, argv, "h?vV")) != -1) {
  704. switch (option) {
  705. default:
  706. fail_count = 1;
  707. FALLTHROUGH
  708. case '?':
  709. FALLTHROUGH
  710. case 'h':
  711. (void)fputs("usage: test_timespec [-v] [-V]\n", stderr);
  712. exit(fail_count);
  713. case 'V':
  714. (void)fprintf( stderr, "test_timespec %s\n", VERSION);
  715. exit(EXIT_SUCCESS);
  716. case 'v':
  717. verbose = 1;
  718. break;
  719. }
  720. }
  721. fail_count = test_format(verbose);
  722. fail_count += test_ts_to_ms(verbose);
  723. fail_count += test_ts_subtract(verbose);
  724. fail_count += test_ns_subtract(verbose);
  725. fail_count += test_gpsd_gpstime_resolv(verbose);
  726. fail_count += test_parse_uri_dest(verbose);
  727. if ( fail_count ) {
  728. printf("timespec tests failed %d tests\n", fail_count );
  729. exit(1);
  730. }
  731. printf("timespec tests succeeded\n");
  732. if (verbose) {
  733. ex_precision();
  734. }
  735. exit(0);
  736. }
  737. // vim: set expandtab shiftwidth=4