00001
00002
00003
00004
00005
00006
00007
00008
00009
00010
00011
00012
00013
00014
00015
00016
00017
00018
00019
00020
00021
00022
00023
00024
00025
00026
00027
00028
00029
00030
00031
00032
00033
00034
00035
00036
00037
00038
00039
00040
00041
00042
00043
00044
00045
00046
00047
00048
00049
00050
00051
00052
00053
00054
00055
00056
00057
00058
00059
00060
00061
00062
00063
00064
00065
00066
00067
00068
00069
00070
00071
00072
00073
00074
00075
00076
00077
00078
00079
00080
00081
00082
00083
00084
00085
00086
00087
00088
00089
00090
00091
00092
00093
00094
00095
00096
00097
00098
00099
00100
00101
00102
00103
00104
00105
00106
00107
00108
00109
00110
00111
00112
00113
00114
00115
00116
00117
00118
00119
00120 #include <lal/LALRCSID.h>
00121
00122 NRCSID (GPSTOUTCC, "$Id: GPStoUTC.c,v 1.32 2008/04/29 21:31:12 kipp Exp $");
00123
00124 #include <lal/LALStdio.h>
00125 #include <lal/Date.h>
00126 #include "date_value.h"
00127 #include <lal/XLALError.h>
00128
00129 #define INFOSTR_LEN 256
00130
00131 struct tm *gmtime_r( const time_t *, struct tm * );
00132 char *asctime_r( const struct tm *, char * );
00133
00134
00135
00136
00137
00138
00139
00140
00141
00142
00143
00144
00145
00146 static const INT4 maxtestedGPS = 851644813;
00147
00148
00149
00150
00151
00152 void
00153 LALGPStoUTC (LALStatus *status,
00154 LALDate *p_utcDate,
00155 const LIGOTimeGPS *p_gpsTime,
00156 const LALLeapSecAccuracy *p_accuracy)
00157
00158
00159
00160 {
00161
00162
00163
00164
00165
00166
00167
00168
00169
00170
00171
00172
00173
00174 static const time_t leaps[] = {
00175 0,
00176 33350400,
00177 63072000,
00178 78796800,
00179 94694400,
00180 126230400,
00181 157766400,
00182 189302400,
00183 220924800,
00184 252460800,
00185 283996800,
00186 315532800,
00187 362793600,
00188 394329600,
00189 425865600,
00190 489024000,
00191 567993600,
00192 631152000,
00193 662688000,
00194 709948800,
00195 741484800,
00196 773020800,
00197 820454400,
00198 867715200,
00199 915148800,
00200 1136073600,
00201 };
00202
00203
00204 static const INT4 numleaps = sizeof(leaps)/sizeof(time_t);
00205 time_t unixTime;
00206 time_t tmptime;
00207 LALUnixDate tmputc;
00208 char tmpstamp[32];
00209 CHAR infostr[INFOSTR_LEN];
00210 INT4 i;
00211
00212 INITSTATUS (status, "LALGPStoUTC", GPSTOUTCC);
00213
00214 ASSERT (p_gpsTime != (LIGOTimeGPS *)NULL, status,
00215 DATEH_ENULLINPUT, DATEH_MSGENULLINPUT);
00216
00217 ASSERT (p_gpsTime->gpsSeconds >= 0, status,
00218 DATEH_ERANGEGPSABS, DATEH_MSGERANGEGPSABS);
00219
00220 ASSERT (p_accuracy != (LALLeapSecAccuracy *)NULL, status,
00221 DATEH_ENULLINPUT, DATEH_MSGENULLINPUT);
00222
00223 ASSERT ((*p_accuracy == LALLEAPSEC_STRICT ||
00224 *p_accuracy == LALLEAPSEC_LOOSE), status,
00225 DATEH_EACCPARAMOUTOFRANGE, DATEH_MSGEACCPARAMOUTOFRANGE);
00226
00227 ASSERT (p_utcDate != (LALDate *)NULL, status,
00228 DATEH_ENULLOUTPUT, DATEH_MSGENULLOUTPUT);
00229
00230 XLALPrintDeprecationWarning("LALGPStoUTC", "XLALGPSToUTC");
00231
00232
00233 unixTime = p_gpsTime->gpsSeconds + UNIXGPS;
00234
00235 if (lalDebugLevel & LALINFO)
00236 {
00237 LALSnprintf(infostr, INFOSTR_LEN, "Max. tested GPS is %d\n", maxtestedGPS);
00238 LALInfo(status, infostr);
00239 }
00240
00241
00242 tmptime = (22*365 + 7*366)* SECS_PER_DAY + 23;
00243 gmtime_r(&tmptime, &tmputc);
00244
00245 if (lalDebugLevel & LALINFO)
00246 {
00247 asctime_r(&tmputc, tmpstamp);
00248 LALSnprintf(infostr, INFOSTR_LEN, "tmputc = %s\n", tmpstamp);
00249 LALInfo(status, infostr);
00250 }
00251
00252
00253
00254
00255
00256
00257
00258
00259
00260 if (p_gpsTime->gpsSeconds > maxtestedGPS)
00261 {
00262
00263 if (*p_accuracy == LALLEAPSEC_STRICT)
00264 {
00265 ABORT(status, DATEH_ERANGEGPSABS, DATEH_MSGERANGEGPSABS);
00266 }
00267 else if (*p_accuracy == LALLEAPSEC_LOOSE)
00268 {
00269 LALWarning(status, "may be missing leap seconds");
00270 }
00271 else
00272 {
00273 LALWarning(status, "may be missing leap seconds");
00274 }
00275 }
00276
00277
00278
00279
00280
00281 if (tmputc.tm_sec == 60)
00282 {
00283 LALInfo(status, "gmtime_r() takes leap seconds into account");
00284
00285
00286
00287
00288
00289
00290
00291
00292
00293
00294 gmtime_r(&unixTime, &tmputc);
00295 p_utcDate->unixDate.tm_sec = tmputc.tm_sec;
00296 p_utcDate->unixDate.tm_min = tmputc.tm_min;
00297 p_utcDate->unixDate.tm_hour = tmputc.tm_hour;
00298 p_utcDate->unixDate.tm_mday = tmputc.tm_mday;
00299 p_utcDate->unixDate.tm_mon = tmputc.tm_mon;
00300 p_utcDate->unixDate.tm_year = tmputc.tm_year;
00301 p_utcDate->unixDate.tm_wday = tmputc.tm_wday;
00302 p_utcDate->unixDate.tm_yday = tmputc.tm_yday;
00303 p_utcDate->unixDate.tm_isdst = 0;
00304 }
00305 else
00306 {
00307 LALInfo(status, "gmtime_r() does not figure in leap seconds");
00308
00309
00310 i = 0;
00311 while (i < numleaps && leaps[i] + i - 1 < unixTime)
00312 ++i;
00313
00314 if (lalDebugLevel & LALINFO)
00315 {
00316 LALSnprintf(infostr, INFOSTR_LEN, "unixTime = %ld; leaps[%d] = %ld",
00317 unixTime, i, leaps[i]);
00318 LALInfo(status, infostr);
00319 }
00320
00321 if (unixTime == (leaps[i] + i - 1))
00322 {
00323 unixTime -= i;
00324 gmtime_r(&unixTime, &tmputc);
00325 p_utcDate->unixDate.tm_sec = 60;
00326 p_utcDate->unixDate.tm_min = tmputc.tm_min;
00327 p_utcDate->unixDate.tm_hour = tmputc.tm_hour;
00328 p_utcDate->unixDate.tm_mday = tmputc.tm_mday;
00329 p_utcDate->unixDate.tm_mon = tmputc.tm_mon;
00330 p_utcDate->unixDate.tm_year = tmputc.tm_year;
00331 p_utcDate->unixDate.tm_wday = tmputc.tm_wday;
00332 p_utcDate->unixDate.tm_yday = tmputc.tm_yday;
00333 p_utcDate->unixDate.tm_isdst = 0;
00334 }
00335 else
00336 {
00337 unixTime -= (i - 1);
00338 gmtime_r(&unixTime, &tmputc);
00339 p_utcDate->unixDate.tm_sec = tmputc.tm_sec;
00340 p_utcDate->unixDate.tm_min = tmputc.tm_min;
00341 p_utcDate->unixDate.tm_hour = tmputc.tm_hour;
00342 p_utcDate->unixDate.tm_mday = tmputc.tm_mday;
00343 p_utcDate->unixDate.tm_mon = tmputc.tm_mon;
00344 p_utcDate->unixDate.tm_year = tmputc.tm_year;
00345 p_utcDate->unixDate.tm_wday = tmputc.tm_wday;
00346 p_utcDate->unixDate.tm_yday = tmputc.tm_yday;
00347 p_utcDate->unixDate.tm_isdst = 0;
00348 }
00349 }
00350
00351
00352 p_utcDate->residualNanoSeconds = p_gpsTime->gpsNanoSeconds;
00353
00354 RETURN (status);
00355 }
00356
00357
00358
00359
00360
00361 static int days_in_year(const LALDate *p_utcDate)
00362 {
00363 int year = p_utcDate->unixDate.tm_year + 1900;
00364
00365
00366 if (year % 100 == 0)
00367 {
00368 if (year % 400 == 0)
00369 return 366;
00370 else
00371 return 365;
00372 }
00373
00374
00375 if (year % 4 == 0)
00376 return 366;
00377
00378 return 365;
00379 }
00380
00381
00382
00383
00384 static int days_in_month(const LALDate *p_utcDate)
00385 {
00386 int month = p_utcDate->unixDate.tm_mon;
00387
00388 switch (month) {
00389 case LALMONTH_JAN:
00390 case LALMONTH_MAR:
00391 case LALMONTH_MAY:
00392 case LALMONTH_JUL:
00393 case LALMONTH_AUG:
00394 case LALMONTH_OCT:
00395 case LALMONTH_DEC:
00396 return 31;
00397
00398 case LALMONTH_APR:
00399 case LALMONTH_JUN:
00400 case LALMONTH_SEP:
00401 case LALMONTH_NOV:
00402 return 30;
00403
00404 case 1:
00405 if (days_in_year(p_utcDate) == 366)
00406 return 29;
00407 else
00408 return 28;
00409 }
00410
00411 return -1;
00412 }
00413
00414
00415
00416
00417 typedef struct leap_sec
00418 {
00419 int year;
00420 int mon;
00421 INT4 leapsec;
00422 }
00423 leap_sec_t;
00424
00425
00426 void
00427 LALUTCtoGPS (LALStatus *status,
00428 LIGOTimeGPS *p_gpsTime,
00429 const LALDate *p_utcDate,
00430 const LALLeapSecAccuracy *p_accuracy)
00431
00432
00433
00434 {
00435
00436
00437
00438
00439
00440
00441
00442 static leap_sec_t leap_sec_data[] =
00443 {
00444 {72, LALMONTH_JAN, 1},
00445 {73, LALMONTH_JAN, 1},
00446 {74, LALMONTH_JAN, 1},
00447 {75, LALMONTH_JAN, 1},
00448 {76, LALMONTH_JAN, 1},
00449 {77, LALMONTH_JAN, 1},
00450 {78, LALMONTH_JAN, 1},
00451 {79, LALMONTH_JAN, 1},
00452 {80, LALMONTH_JAN, 1},
00453 {81, LALMONTH_JUL, 1},
00454 {82, LALMONTH_JUL, 1},
00455 {83, LALMONTH_JUL, 1},
00456 {85, LALMONTH_JUL, 1},
00457 {88, LALMONTH_JAN, 1},
00458 {90, LALMONTH_JAN, 1},
00459 {91, LALMONTH_JAN, 1},
00460 {92, LALMONTH_JUL, 1},
00461 {93, LALMONTH_JUL, 1},
00462 {94, LALMONTH_JUL, 1},
00463 {96, LALMONTH_JAN, 1},
00464 {97, LALMONTH_JUL, 1},
00465 {99, LALMONTH_JAN, 1},
00466 {106, LALMONTH_JAN, 1},
00467 };
00468
00469
00470 int ddays = 0;
00471 int dsecs = 0;
00472 LALDate tmpdate;
00473 static LALDate gpsref;
00474 int i = 0;
00475 char infostr[256];
00476 static const int nleaps = sizeof(leap_sec_data)/sizeof(leap_sec_t);
00477
00478 XLALPrintDeprecationWarning("LALUTCtoGPS", "XLALUTCToGPS");
00479
00480
00481 gpsref.unixDate.tm_sec = 0;
00482 gpsref.unixDate.tm_min = 0;
00483 gpsref.unixDate.tm_hour = 0;
00484 gpsref.unixDate.tm_mday = 6;
00485 gpsref.unixDate.tm_mon = LALMONTH_JAN;
00486 gpsref.unixDate.tm_year = 80;
00487 gpsref.unixDate.tm_wday = 0;
00488 gpsref.unixDate.tm_yday = 0;
00489
00490 if (lalDebugLevel & LALINFO)
00491 {
00492 LALSnprintf(infostr, INFOSTR_LEN, "Date given: %d-%d-%d %d:%d:%d %d\n",
00493 p_utcDate->unixDate.tm_year+1900, p_utcDate->unixDate.tm_mon+1,
00494 p_utcDate->unixDate.tm_mday, p_utcDate->unixDate.tm_hour,
00495 p_utcDate->unixDate.tm_min, p_utcDate->unixDate.tm_sec,
00496 p_utcDate->residualNanoSeconds);
00497
00498 LALInfo(status, infostr);
00499 }
00500
00501 INITSTATUS(status, "LALUTCtoGPS", GPSTOUTCC);
00502
00503 ASSERT (p_gpsTime != (LIGOTimeGPS *)NULL, status,
00504 DATEH_ENULLOUTPUT, DATEH_MSGENULLOUTPUT);
00505
00506 ASSERT (p_accuracy != (LALLeapSecAccuracy *)NULL, status,
00507 DATEH_ENULLINPUT, DATEH_MSGENULLINPUT);
00508
00509 ASSERT ((*p_accuracy == LALLEAPSEC_STRICT ||
00510 *p_accuracy == LALLEAPSEC_LOOSE), status,
00511 DATEH_EACCPARAMOUTOFRANGE, DATEH_MSGEACCPARAMOUTOFRANGE);
00512
00513 ASSERT (p_utcDate != (LALDate *)NULL, status,
00514 DATEH_ENULLINPUT, DATEH_MSGENULLINPUT);
00515
00516
00517 ASSERT (p_utcDate->unixDate.tm_year > 80 ||
00518 (p_utcDate->unixDate.tm_year == 80 &&
00519 (p_utcDate->unixDate.tm_mon > LALMONTH_JAN ||
00520 (p_utcDate->unixDate.tm_mon == LALMONTH_JAN &&
00521 p_utcDate->unixDate.tm_mday >= 6))), status,
00522 DATEH_EGPSDATETOOEARLY, DATEH_MSGEGPSDATETOOEARLY);
00523
00524
00525
00526
00527
00528
00529
00530
00531
00532
00533
00534
00535
00536
00537
00538
00539
00540 if (p_utcDate->unixDate.tm_year > 106 ||
00541 (p_utcDate->unixDate.tm_year == 106 &&
00542 (p_utcDate->unixDate.tm_mon > LALMONTH_DEC ||
00543 (p_utcDate->unixDate.tm_mon == LALMONTH_DEC &&
00544 p_utcDate->unixDate.tm_mday == 31 &&
00545 p_utcDate->unixDate.tm_hour == 23 &&
00546 p_utcDate->unixDate.tm_min == 59 &&
00547 p_utcDate->unixDate.tm_sec > 59))))
00548 {
00549
00550 if (*p_accuracy == LALLEAPSEC_STRICT)
00551 {
00552 ABORT(status, DATEH_ERANGEGPSABS, DATEH_MSGERANGEGPSABS);
00553 }
00554 else if (*p_accuracy == LALLEAPSEC_LOOSE)
00555 {
00556 LALWarning(status, "may be missing leap seconds");
00557 }
00558 else
00559 {
00560 LALWarning(status, "may be missing leap seconds");
00561 }
00562 }
00563
00564
00565
00566 tmpdate.unixDate.tm_year = 80;
00567 tmpdate.unixDate.tm_mon = LALMONTH_JAN;
00568 tmpdate.unixDate.tm_mday = 6;
00569 tmpdate.unixDate.tm_hour = 0;
00570 tmpdate.unixDate.tm_min = 0;
00571 tmpdate.unixDate.tm_sec = 0;
00572 tmpdate.residualNanoSeconds = 0;
00573
00574 while (tmpdate.unixDate.tm_year < p_utcDate->unixDate.tm_year)
00575 {
00576 ddays += days_in_year(&tmpdate);
00577 tmpdate.unixDate.tm_year++;
00578 }
00579 ddays -= 5;
00580
00581 while (tmpdate.unixDate.tm_mon < p_utcDate->unixDate.tm_mon)
00582 {
00583 ddays += days_in_month(&tmpdate);
00584 tmpdate.unixDate.tm_mon++;
00585 }
00586
00587 ddays += p_utcDate->unixDate.tm_mday - 1;
00588 dsecs = ddays * SECS_PER_DAY;
00589
00590 dsecs += p_utcDate->unixDate.tm_hour * SECS_PER_HOUR +
00591 p_utcDate->unixDate.tm_min * SECS_PER_MIN +
00592 p_utcDate->unixDate.tm_sec;
00593
00594
00595 i = 9;
00596 while (i < nleaps)
00597 {
00598 if (leap_sec_data[i].year < p_utcDate->unixDate.tm_year)
00599 dsecs++;
00600 else if (leap_sec_data[i].year == p_utcDate->unixDate.tm_year &&
00601 leap_sec_data[i].mon <= p_utcDate->unixDate.tm_mon)
00602 dsecs++;
00603
00604 ++i;
00605 }
00606
00607 p_gpsTime->gpsSeconds = dsecs;
00608 p_gpsTime->gpsNanoSeconds = p_utcDate->residualNanoSeconds;
00609
00610
00611 RETURN (status);
00612 }
00613
00614
00615
00616
00617
00618
00619
00620
00621
00622
00623
00624
00625
00626
00627
00628
00629
00630
00631
00632
00633
00634
00635
00636
00637
00638
00639
00640
00641
00642
00643
00644
00645
00646
00647
00648
00649
00650
00651
00652
00653
00654
00655
00656
00657
00658
00659
00660
00661
00662
00663
00664
00665 typedef struct gps_leap_sec {
00666 time_t gps;
00667 INT4 tai_utc;
00668 } gps_leap_sec_t;
00669
00670
00671 void
00672 LALLeapSecs (LALStatus *status,
00673 INT4 *p_leapSecs,
00674
00675
00676
00677
00678 const LIGOTimeGPS *p_gpsTime,
00679 const LALLeapSecFormatAndAcc *p_formatAndAcc)
00680
00681 {
00682
00683
00684
00685
00686 static const gps_leap_sec_t gpsLeaps[] =
00687 {
00688 {0, 19},
00689 {46828801, 20},
00690 {78364802, 21},
00691 {109900803, 22},
00692 {173059204, 23},
00693 {252028805, 24},
00694 {315187206, 25},
00695 {346723207, 26},
00696 {393984008, 27},
00697 {425520009, 28},
00698 {457056010, 29},
00699 {504489611, 30},
00700 {551750412, 31},
00701 {599184013, 32},
00702 {820108814, 33},
00703 };
00704
00705
00706 static const INT4 numleaps = sizeof(gpsLeaps)/sizeof(gps_leap_sec_t);
00707 char infostr[256];
00708 INT4 i;
00709
00710
00711 INITSTATUS (status, "LALLeapSecs", GPSTOUTCC);
00712
00713 ASSERT (p_leapSecs != (INT4 *)NULL, status,
00714 DATEH_ENULLOUTPUT, DATEH_MSGENULLOUTPUT);
00715
00716 ASSERT (p_gpsTime != (LIGOTimeGPS *)NULL, status,
00717 DATEH_ENULLINPUT, DATEH_MSGENULLINPUT);
00718
00719 ASSERT (p_gpsTime->gpsSeconds >= 0, status,
00720 DATEH_ERANGEGPSABS, DATEH_MSGERANGEGPSABS);
00721
00722 ASSERT (p_formatAndAcc != (LALLeapSecFormatAndAcc *)NULL, status,
00723 DATEH_ENULLINPUT, DATEH_MSGENULLINPUT);
00724
00725 ASSERT ((p_formatAndAcc->format == LALLEAPSEC_TAIUTC ||
00726 p_formatAndAcc->format == LALLEAPSEC_GPSUTC),
00727 status, DATEH_EFORMATPARAMOUTOFRANGE,
00728 DATEH_MSGEFORMATPARAMOUTOFRANGE);
00729
00730 ASSERT ((p_formatAndAcc->accuracy == LALLEAPSEC_LOOSE ||
00731 p_formatAndAcc->accuracy == LALLEAPSEC_STRICT),
00732 status, DATEH_EACCPARAMOUTOFRANGE,
00733 DATEH_MSGEACCPARAMOUTOFRANGE);
00734
00735 XLALPrintDeprecationWarning("LALLeapSecs", "XLALLeapSeconds");
00736
00737 if (lalDebugLevel & LALINFO)
00738 {
00739 LALSnprintf(infostr, INFOSTR_LEN, "Max. tested GPS is %d\n", maxtestedGPS);
00740 LALInfo(status, infostr);
00741 }
00742
00743
00744
00745
00746
00747
00748
00749
00750
00751 if (p_gpsTime->gpsSeconds > maxtestedGPS)
00752 {
00753
00754 if (p_formatAndAcc->accuracy == LALLEAPSEC_STRICT)
00755 {
00756 ABORT(status, DATEH_ERANGEGPSABS, DATEH_MSGERANGEGPSABS);
00757 }
00758 else if (p_formatAndAcc->accuracy == LALLEAPSEC_LOOSE)
00759 {
00760 LALWarning(status, "may be missing leap seconds");
00761 }
00762 else
00763 {
00764 LALWarning(status, "may be missing leap seconds");
00765 }
00766 }
00767
00768 if (p_gpsTime->gpsSeconds == 0)
00769 {
00770 *p_leapSecs = gpsLeaps[0].tai_utc;
00771 }
00772 else
00773 {
00774 i = 1;
00775 while (p_gpsTime->gpsSeconds > gpsLeaps[i].gps &&
00776 i < numleaps)
00777 ++i;
00778
00779 *p_leapSecs = gpsLeaps[i-1].tai_utc;
00780 }
00781
00782 if (lalDebugLevel & LALINFO)
00783 {
00784 LALSnprintf(infostr, INFOSTR_LEN, "Format = %d\n", p_formatAndAcc->format);
00785 LALInfo(status, infostr);
00786 }
00787
00788 if (p_formatAndAcc->format == LALLEAPSEC_GPSUTC)
00789 {
00790 *p_leapSecs -= 19;
00791 }
00792
00793 RETURN(status);
00794 }