Time scale transformation: Coordinated Universal Time, UTC, to International Atomic Time, TAI.
Status: canonical.
UTC1+UTC2 is quasi Julian Date (see Note 2), apportioned in any convenient way between the two arguments, for example where UTC1 is the Julian Day Number and UTC2 is the fraction of a day.
JD cannot unambiguously represent UTC during a leap second unless special measures are taken. The convention in the present routine is that the JD day represents UTC days whether the length is 86399, 86400 or 86401 SI seconds. In the 1960-1972 era there were smaller jumps (in either direction) each time the linear UTC(TAI) expression was changed, and these "mini-leaps" are also included in the SOFA convention.
The warning status "dubious year" flags UTCs that predate the introduction of the time scale or that are too far in the future to be trusted. See DAT for further details.
The routine DTF2D converts from calendar date and time of day into 2-part Julian Date, and in the case of UTC implements the leap-second-ambiguity convention described above.
The returned TAI1,TAI2 are such that their sum is the TAI Julian Date.
McCarthy, D. D., Petit, G. (eds.), IERS Conventions (2003), IERS Technical Note No. 32, BKG (2004)
Explanatory Supplement to the Astronomical Almanac, P. Kenneth Seidelmann (ed), University Science Books (1992)
Type | Intent | Optional | Attributes | Name | ||
---|---|---|---|---|---|---|
real(kind=wp), | intent(in) | :: | utc1 | UTC as a 2-part quasi Julian Date (Notes 1-4) |
||
real(kind=wp), | intent(in) | :: | utc2 | UTC as a 2-part quasi Julian Date (Notes 1-4) |
||
real(kind=wp), | intent(out) | :: | tai1 | TAI as a 2-part Julian Date (Note 5) |
||
real(kind=wp), | intent(out) | :: | tai2 | TAI as a 2-part Julian Date (Note 5) |
||
integer, | intent(out) | :: | j |
|
subroutine UTCTAI ( utc1, utc2, tai1, tai2, j )
implicit none
real(wp),intent(in) :: utc1 !! UTC as a 2-part quasi Julian Date (Notes 1-4)
real(wp),intent(in) :: utc2 !! UTC as a 2-part quasi Julian Date (Notes 1-4)
real(wp),intent(out) :: tai1 !! TAI as a 2-part Julian Date (Note 5)
real(wp),intent(out) :: tai2 !! TAI as a 2-part Julian Date (Note 5)
integer,intent(out) :: j !! status:
!! * +1 = dubious year (Note 3)
!! * 0 = OK
!! * -1 = unacceptable date
logical :: big1
integer :: iy, im, id, js, iyt, imt, idt
real(wp) :: u1, u2, fd, dat0, dat12, w, dat24, dlod, dleap, &
z1, z2, a2
! Put the two parts of the UTC into big-first order.
big1 = abs(utc1) >= abs(utc2)
if ( big1 ) then
u1 = utc1
u2 = utc2
else
u1 = utc2
u2 = utc1
end if
! Get TAI-UTC at 0h today.
call JD2CAL ( u1, u2, iy, im, id, fd, js )
if ( js==0 ) then
call DAT ( iy, im, id, 0.0_wp, dat0, js )
if ( js>=0 ) then
! Get TAI-UTC at 12h today (to detect drift).
call DAT ( iy, im, id, 0.5_wp, dat12, js )
if ( js>=0 ) then
! Get TAI-UTC at 0h tomorrow (to detect jumps).
call JD2CAL ( u1+1.5_wp, u2-fd, iyt, imt, idt, w, js )
if ( js==0 ) then
call DAT ( iyt, imt, idt, 0.0_wp, dat24, js )
if ( js>=0 ) then
! Separate TAI-UTC change into per-day (DLOD) and any jump (DLEAP).
dlod = 2.0_wp * ( dat12 - dat0 )
dleap = dat24 - ( dat0 + dlod )
! Remove any scaling applied to spread leap into preceding day.
fd = fd * (d2s+dleap)/d2s
! Scale from (pre-1972) UTC seconds to SI seconds.
fd = fd * (d2s+dlod)/d2s
! Today's calendar date to 2-part JD.
call CAL2JD ( iy, im, id, z1, z2, js )
if ( js==0 ) then
! Assemble the TAI result, preserving the UTC split and order.
a2 = z1 - u1
a2 = ( a2 + z2 ) + ( fd + dat0/d2s )
if ( big1 ) then
tai1 = u1
tai2 = a2
else
tai1 = a2
tai2 = u1
end if
end if
end if
end if
end if
end if
end if
! Status.
j = js
end subroutine UTCTAI