1 /* 2 * ntp_calgps.c - calendar for GPS/GNSS based clocks 3 * 4 * Written by Juergen Perlinger (perlinger@ntp.org) for the NTP project. 5 * The contents of 'html/copyright.html' apply. 6 * 7 * -------------------------------------------------------------------- 8 * 9 * This module implements stuff often used with GPS/GNSS receivers 10 */ 11 12 #include <config.h> 13 #include <sys/types.h> 14 15 #include "ntp_types.h" 16 #include "ntp_calendar.h" 17 #include "ntp_calgps.h" 18 #include "ntp_stdlib.h" 19 #include "ntp_unixtime.h" 20 21 #include "ntp_fp.h" 22 #include "ntpd.h" 23 #include "vint64ops.h" 24 25 /* ==================================================================== 26 * misc. helpers -- might go elsewhere sometime? 27 * ==================================================================== 28 */ 29 30 l_fp 31 ntpfp_with_fudge( 32 l_fp lfp, 33 double ofs 34 ) 35 { 36 l_fp fpo; 37 /* calculate 'lfp - ofs' as '(l_fp)(-ofs) + lfp': negating a 38 * double is cheap, as it only flips one bit... 39 */ 40 ofs = -ofs; 41 DTOLFP(ofs, &fpo); 42 L_ADD(&fpo, &lfp); 43 return fpo; 44 } 45 46 47 /* ==================================================================== 48 * GPS calendar functions 49 * ==================================================================== 50 */ 51 52 /* -------------------------------------------------------------------- 53 * normalization functions for day/time and week/time representations. 54 * Since we only use moderate offsets (leap second corrections and 55 * alike) it does not really pay off to do a floor-corrected division 56 * here. We use compare/decrement/increment loops instead. 57 * -------------------------------------------------------------------- 58 */ 59 static void 60 _norm_ntp_datum( 61 TNtpDatum * datum 62 ) 63 { 64 static const int32_t limit = SECSPERDAY; 65 66 if (datum->secs >= limit) { 67 do 68 ++datum->days; 69 while ((datum->secs -= limit) >= limit); 70 } else if (datum->secs < 0) { 71 do 72 --datum->days; 73 while ((datum->secs += limit) < 0); 74 } 75 } 76 77 static void 78 _norm_gps_datum( 79 TGpsDatum * datum 80 ) 81 { 82 static const int32_t limit = 7 * SECSPERDAY; 83 84 if (datum->wsecs >= limit) { 85 do 86 ++datum->weeks; 87 while ((datum->wsecs -= limit) >= limit); 88 } else if (datum->wsecs < 0) { 89 do 90 --datum->weeks; 91 while ((datum->wsecs += limit) < 0); 92 } 93 } 94 95 /* -------------------------------------------------------------------- 96 * Add an offset to a day/time and week/time representation. 97 * 98 * !!Attention!! the offset should be small, compared to the time period 99 * (either a day or a week). 100 * -------------------------------------------------------------------- 101 */ 102 void 103 gpsntp_add_offset( 104 TNtpDatum * datum, 105 l_fp offset 106 ) 107 { 108 /* fraction can be added easily */ 109 datum->frac += offset.l_uf; 110 datum->secs += (datum->frac < offset.l_uf); 111 112 /* avoid integer overflow on the seconds */ 113 if (offset.l_ui >= INT32_MAX) 114 datum->secs -= (int32_t)~offset.l_ui + 1; 115 else 116 datum->secs += (int32_t)offset.l_ui; 117 _norm_ntp_datum(datum); 118 } 119 120 void 121 gpscal_add_offset( 122 TGpsDatum * datum, 123 l_fp offset 124 ) 125 { 126 /* fraction can be added easily */ 127 datum->frac += offset.l_uf; 128 datum->wsecs += (datum->frac < offset.l_uf); 129 130 131 /* avoid integer overflow on the seconds */ 132 if (offset.l_ui >= INT32_MAX) 133 datum->wsecs -= (int32_t)~offset.l_ui + 1; 134 else 135 datum->wsecs += (int32_t)offset.l_ui; 136 _norm_gps_datum(datum); 137 } 138 139 /* ------------------------------------------------------------------- 140 * API functions civil calendar and NTP datum 141 * ------------------------------------------------------------------- 142 */ 143 144 static TNtpDatum 145 _gpsntp_fix_gps_era( 146 TcNtpDatum * in 147 ) 148 { 149 /* force result in basedate era 150 * 151 * When calculating this directly in days, we have to execute a 152 * real modulus calculation, since we're obviously not doing a 153 * modulus by a power of 2. Executing this as true floor mod 154 * needs some care and is done under explicit usage of one's 155 * complement and masking to get mostly branchless code. 156 */ 157 static uint32_t const clen = 7*1024; 158 159 uint32_t base, days, sign; 160 TNtpDatum out = *in; 161 162 /* Get base in NTP day scale. No overflows here. */ 163 base = (basedate_get_gpsweek() + GPSNTP_WSHIFT) * 7 164 - GPSNTP_DSHIFT; 165 days = out.days; 166 167 sign = (uint32_t)-(days < base); 168 days = sign ^ (days - base); 169 days %= clen; 170 days = base + (sign & clen) + (sign ^ days); 171 172 out.days = days; 173 return out; 174 } 175 176 TNtpDatum 177 gpsntp_fix_gps_era( 178 TcNtpDatum * in 179 ) 180 { 181 TNtpDatum out = *in; 182 _norm_ntp_datum(&out); 183 return _gpsntp_fix_gps_era(&out); 184 } 185 186 /* ----------------------------------------------------------------- */ 187 static TNtpDatum 188 _gpsntp_from_daytime( 189 TcCivilDate * jd, 190 l_fp fofs, 191 TcNtpDatum * pivot, 192 int warp 193 ) 194 { 195 static const int32_t shift = SECSPERDAY / 2; 196 197 TNtpDatum retv; 198 199 /* set result based on pivot -- ops order is important here */ 200 ZERO(retv); 201 retv.secs = ntpcal_date_to_daysec(jd); 202 gpsntp_add_offset(&retv, fofs); /* result is normalized */ 203 retv.days = pivot->days; 204 205 /* Manual periodic extension without division: */ 206 if (pivot->secs < shift) { 207 int32_t lim = pivot->secs + shift; 208 retv.days -= (retv.secs > lim || 209 (retv.secs == lim && retv.frac >= pivot->frac)); 210 } else { 211 int32_t lim = pivot->secs - shift; 212 retv.days += (retv.secs < lim || 213 (retv.secs == lim && retv.frac < pivot->frac)); 214 } 215 return warp ? _gpsntp_fix_gps_era(&retv) : retv; 216 } 217 218 /* ----------------------------------------------------------------- 219 * Given the time-of-day part of a civil datum and an additional 220 * (fractional) offset, calculate a full time stamp around a given pivot 221 * time so that the difference between the pivot and the resulting time 222 * stamp is less or equal to 12 hours absolute. 223 */ 224 TNtpDatum 225 gpsntp_from_daytime2_ex( 226 TcCivilDate * jd, 227 l_fp fofs, 228 TcNtpDatum * pivot, 229 int/*BOOL*/ warp 230 ) 231 { 232 TNtpDatum dpiv = *pivot; 233 _norm_ntp_datum(&dpiv); 234 return _gpsntp_from_daytime(jd, fofs, &dpiv, warp); 235 } 236 237 /* ----------------------------------------------------------------- 238 * This works similar to 'gpsntp_from_daytime1()' and actually even uses 239 * it, but the pivot is calculated from the pivot given as 'l_fp' in NTP 240 * time scale. This is in turn expanded around the current system time, 241 * and the resulting absolute pivot is then used to calculate the full 242 * NTP time stamp. 243 */ 244 TNtpDatum 245 gpsntp_from_daytime1_ex( 246 TcCivilDate * jd, 247 l_fp fofs, 248 l_fp pivot, 249 int/*BOOL*/ warp 250 ) 251 { 252 vint64 pvi64; 253 TNtpDatum dpiv; 254 ntpcal_split split; 255 256 pvi64 = ntpcal_ntp_to_ntp(pivot.l_ui, NULL); 257 split = ntpcal_daysplit(&pvi64); 258 dpiv.days = split.hi; 259 dpiv.secs = split.lo; 260 dpiv.frac = pivot.l_uf; 261 return _gpsntp_from_daytime(jd, fofs, &dpiv, warp); 262 } 263 264 /* ----------------------------------------------------------------- 265 * Given a calendar date, zap it into a GPS time format and then convert 266 * that one into the NTP time scale. 267 */ 268 TNtpDatum 269 gpsntp_from_calendar_ex( 270 TcCivilDate * jd, 271 l_fp fofs, 272 int/*BOOL*/ warp 273 ) 274 { 275 TGpsDatum gps; 276 gps = gpscal_from_calendar_ex(jd, fofs, warp); 277 return gpsntp_from_gpscal_ex(&gps, FALSE); 278 } 279 280 /* ----------------------------------------------------------------- 281 * create a civil calendar datum from a NTP date representation 282 */ 283 void 284 gpsntp_to_calendar( 285 TCivilDate * cd, 286 TcNtpDatum * nd 287 ) 288 { 289 memset(cd, 0, sizeof(*cd)); 290 ntpcal_rd_to_date( 291 cd, 292 nd->days + DAY_NTP_STARTS + ntpcal_daysec_to_date( 293 cd, nd->secs)); 294 } 295 296 /* ----------------------------------------------------------------- 297 * get day/tod representation from week/tow datum 298 */ 299 TNtpDatum 300 gpsntp_from_gpscal_ex( 301 TcGpsDatum * gd, 302 int/*BOOL*/ warp 303 ) 304 { 305 TNtpDatum retv; 306 vint64 ts64; 307 ntpcal_split split; 308 TGpsDatum date = *gd; 309 310 if (warp) { 311 uint32_t base = basedate_get_gpsweek() + GPSNTP_WSHIFT; 312 _norm_gps_datum(&date); 313 date.weeks = ((date.weeks - base) & 1023u) + base; 314 } 315 316 ts64 = ntpcal_weekjoin(date.weeks, date.wsecs); 317 ts64 = subv64u32(&ts64, (GPSNTP_DSHIFT * SECSPERDAY)); 318 split = ntpcal_daysplit(&ts64); 319 320 retv.frac = gd->frac; 321 retv.secs = split.lo; 322 retv.days = split.hi; 323 return retv; 324 } 325 326 /* ----------------------------------------------------------------- 327 * get LFP from ntp datum 328 */ 329 l_fp 330 ntpfp_from_ntpdatum( 331 TcNtpDatum * nd 332 ) 333 { 334 l_fp retv; 335 336 retv.l_uf = nd->frac; 337 retv.l_ui = nd->days * (uint32_t)SECSPERDAY 338 + nd->secs; 339 return retv; 340 } 341 342 /* ------------------------------------------------------------------- 343 * API functions GPS week calendar 344 * 345 * Here we use a calendar base of 1899-12-31, so the NTP epoch has 346 * { 0, 86400.0 } in this representation. 347 * ------------------------------------------------------------------- 348 */ 349 350 static TGpsDatum 351 _gpscal_fix_gps_era( 352 TcGpsDatum * in 353 ) 354 { 355 /* force result in basedate era 356 * 357 * This is based on calculating the modulus to a power of two, 358 * so signed integer overflow does not affect the result. Which 359 * in turn makes for a very compact calculation... 360 */ 361 uint32_t base, week; 362 TGpsDatum out = *in; 363 364 week = out.weeks; 365 base = basedate_get_gpsweek() + GPSNTP_WSHIFT; 366 week = base + ((week - base) & (GPSWEEKS - 1)); 367 out.weeks = week; 368 return out; 369 } 370 371 TGpsDatum 372 gpscal_fix_gps_era( 373 TcGpsDatum * in 374 ) 375 { 376 TGpsDatum out = *in; 377 _norm_gps_datum(&out); 378 return _gpscal_fix_gps_era(&out); 379 } 380 381 /* ----------------------------------------------------------------- 382 * Given a calendar date, zap it into a GPS time format and the do a 383 * proper era mapping in the GPS time scale, based on the GPS base date, 384 * if so requested. 385 * 386 * This function also augments the century if just a 2-digit year 387 * (0..99) is provided on input. 388 * 389 * This is a fail-safe against GPS receivers with an unknown starting 390 * point for their internal calendar calculation and therefore 391 * unpredictable (but reproducible!) rollover behavior. While there 392 * *are* receivers that create a full date in the proper way, many 393 * others just don't. The overall damage is minimized by simply not 394 * trusting the era mapping of the receiver and doing the era assignment 395 * with a configurable base date *inside* ntpd. 396 */ 397 TGpsDatum 398 gpscal_from_calendar_ex( 399 TcCivilDate * jd, 400 l_fp fofs, 401 int/*BOOL*/ warp 402 ) 403 { 404 /* (-DAY_GPS_STARTS) (mod 7*1024) -- complement of cycle shift */ 405 static const uint32_t s_compl_shift = 406 (7 * 1024) - DAY_GPS_STARTS % (7 * 1024); 407 408 TGpsDatum gps; 409 TCivilDate cal; 410 int32_t days, week; 411 412 /* if needed, convert from 2-digit year to full year 413 * !!NOTE!! works only between 1980 and 2079! 414 */ 415 cal = *jd; 416 if (cal.year < 80) 417 cal.year += 2000; 418 else if (cal.year < 100) 419 cal.year += 1900; 420 421 /* get RDN from date, possibly adjusting the century */ 422 again: if (cal.month && cal.monthday) { /* use Y/M/D civil date */ 423 days = ntpcal_date_to_rd(&cal); 424 } else { /* using Y/DoY date */ 425 days = ntpcal_year_to_ystart(cal.year) 426 + (int32_t)cal.yearday 427 - 1; /* both RDN and yearday start with '1'. */ 428 } 429 430 /* Rebase to days after the GPS epoch. 'days' is positive here, 431 * but it might be less than the GPS epoch start. Depending on 432 * the input, we have to do different things to get the desired 433 * result. (Since we want to remap the era anyway, we only have 434 * to retain congruential identities....) 435 */ 436 437 if (days >= DAY_GPS_STARTS) { 438 /* simply shift to days since GPS epoch */ 439 days -= DAY_GPS_STARTS; 440 } else if (jd->year < 100) { 441 /* Two-digit year on input: add another century and 442 * retry. This can happen only if the century expansion 443 * yielded a date between 1980-01-01 and 1980-01-05, 444 * both inclusive. We have at most one retry here. 445 */ 446 cal.year += 100; 447 goto again; 448 } else { 449 /* A very bad date before the GPS epoch. There's not 450 * much we can do, except to add the complement of 451 * DAY_GPS_STARTS % (7 * 1024) here, that is, use a 452 * congruential identity: Add the complement instead of 453 * subtracting the value gives a value with the same 454 * modulus. But of course, now we MUST to go through a 455 * cycle fix... because the date was obviously wrong! 456 */ 457 warp = TRUE; 458 days += s_compl_shift; 459 } 460 461 /* Splitting to weeks is simple now: */ 462 week = days / 7; 463 days -= week * 7; 464 465 /* re-base on start of NTP with weeks mapped to 1024 weeks 466 * starting with the GPS base day set in the calendar. 467 */ 468 gps.weeks = week + GPSNTP_WSHIFT; 469 gps.wsecs = days * SECSPERDAY + ntpcal_date_to_daysec(&cal); 470 gps.frac = 0; 471 gpscal_add_offset(&gps, fofs); 472 return warp ? _gpscal_fix_gps_era(&gps) : gps; 473 } 474 475 /* ----------------------------------------------------------------- 476 * get civil date from week/tow representation 477 */ 478 void 479 gpscal_to_calendar( 480 TCivilDate * cd, 481 TcGpsDatum * wd 482 ) 483 { 484 TNtpDatum nd; 485 486 memset(cd, 0, sizeof(*cd)); 487 nd = gpsntp_from_gpscal_ex(wd, FALSE); 488 gpsntp_to_calendar(cd, &nd); 489 } 490 491 /* ----------------------------------------------------------------- 492 * Given the week and seconds in week, as well as the fraction/offset 493 * (which should/could include the leap seconds offset), unfold the 494 * weeks (which are assumed to have just 10 bits) into expanded weeks 495 * based on the GPS base date derived from the build date (default) or 496 * set by the configuration. 497 * 498 * !NOTE! This function takes RAW GPS weeks, aligned to the GPS start 499 * (1980-01-06) on input. The output weeks will be aligned to NTPD's 500 * week calendar start (1899-12-31)! 501 */ 502 TGpsDatum 503 gpscal_from_gpsweek( 504 uint16_t week, 505 int32_t secs, 506 l_fp fofs 507 ) 508 { 509 TGpsDatum retv; 510 511 retv.frac = 0; 512 retv.wsecs = secs; 513 retv.weeks = week + GPSNTP_WSHIFT; 514 gpscal_add_offset(&retv, fofs); 515 return _gpscal_fix_gps_era(&retv); 516 } 517 518 /* ----------------------------------------------------------------- 519 * internal work horse for time-of-week expansion 520 */ 521 static TGpsDatum 522 _gpscal_from_weektime( 523 int32_t wsecs, 524 l_fp fofs, 525 TcGpsDatum * pivot 526 ) 527 { 528 static const int32_t shift = SECSPERWEEK / 2; 529 530 TGpsDatum retv; 531 532 /* set result based on pivot -- ops order is important here */ 533 ZERO(retv); 534 retv.wsecs = wsecs; 535 gpscal_add_offset(&retv, fofs); /* result is normalized */ 536 retv.weeks = pivot->weeks; 537 538 /* Manual periodic extension without division: */ 539 if (pivot->wsecs < shift) { 540 int32_t lim = pivot->wsecs + shift; 541 retv.weeks -= (retv.wsecs > lim || 542 (retv.wsecs == lim && retv.frac >= pivot->frac)); 543 } else { 544 int32_t lim = pivot->wsecs - shift; 545 retv.weeks += (retv.wsecs < lim || 546 (retv.wsecs == lim && retv.frac < pivot->frac)); 547 } 548 return _gpscal_fix_gps_era(&retv); 549 } 550 551 /* ----------------------------------------------------------------- 552 * expand a time-of-week around a pivot given as week datum 553 */ 554 TGpsDatum 555 gpscal_from_weektime2( 556 int32_t wsecs, 557 l_fp fofs, 558 TcGpsDatum * pivot 559 ) 560 { 561 TGpsDatum wpiv = * pivot; 562 _norm_gps_datum(&wpiv); 563 return _gpscal_from_weektime(wsecs, fofs, &wpiv); 564 } 565 566 /* ----------------------------------------------------------------- 567 * epand a time-of-week around an pivot given as LFP, which in turn 568 * is expanded around the current system time and then converted 569 * into a week datum. 570 */ 571 TGpsDatum 572 gpscal_from_weektime1( 573 int32_t wsecs, 574 l_fp fofs, 575 l_fp pivot 576 ) 577 { 578 vint64 pvi64; 579 TGpsDatum wpiv; 580 ntpcal_split split; 581 582 /* get 64-bit pivot in NTP epoch */ 583 pvi64 = ntpcal_ntp_to_ntp(pivot.l_ui, NULL); 584 585 /* convert to weeks since 1899-12-31 and seconds in week */ 586 pvi64 = addv64u32(&pvi64, (GPSNTP_DSHIFT * SECSPERDAY)); 587 split = ntpcal_weeksplit(&pvi64); 588 589 wpiv.weeks = split.hi; 590 wpiv.wsecs = split.lo; 591 wpiv.frac = pivot.l_uf; 592 return _gpscal_from_weektime(wsecs, fofs, &wpiv); 593 } 594 595 /* ----------------------------------------------------------------- 596 * get week/tow representation from day/tod datum 597 */ 598 TGpsDatum 599 gpscal_from_gpsntp( 600 TcNtpDatum * gd 601 ) 602 { 603 TGpsDatum retv; 604 vint64 ts64; 605 ntpcal_split split; 606 607 ts64 = ntpcal_dayjoin(gd->days, gd->secs); 608 ts64 = addv64u32(&ts64, (GPSNTP_DSHIFT * SECSPERDAY)); 609 split = ntpcal_weeksplit(&ts64); 610 611 retv.frac = gd->frac; 612 retv.wsecs = split.lo; 613 retv.weeks = split.hi; 614 return retv; 615 } 616 617 /* ----------------------------------------------------------------- 618 * convert week/tow to LFP stamp 619 */ 620 l_fp 621 ntpfp_from_gpsdatum( 622 TcGpsDatum * gd 623 ) 624 { 625 l_fp retv; 626 627 retv.l_uf = gd->frac; 628 retv.l_ui = gd->weeks * (uint32_t)SECSPERWEEK 629 + (uint32_t)gd->wsecs 630 - (uint32_t)SECSPERDAY * GPSNTP_DSHIFT; 631 return retv; 632 } 633 634 /* -*-EOF-*- */ 635