TimeOffset.java

  1. /* Copyright 2022-2025 Luc Maisonobe
  2.  * Licensed to CS GROUP (CS) under one or more
  3.  * contributor license agreements.  See the NOTICE file distributed with
  4.  * this work for additional information regarding copyright ownership.
  5.  * CS licenses this file to You under the Apache License, Version 2.0
  6.  * (the "License"); you may not use this file except in compliance with
  7.  * the License.  You may obtain a copy of the License at
  8.  *
  9.  *   http://www.apache.org/licenses/LICENSE-2.0
  10.  *
  11.  * Unless required by applicable law or agreed to in writing, software
  12.  * distributed under the License is distributed on an "AS IS" BASIS,
  13.  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  14.  * See the License for the specific language governing permissions and
  15.  * limitations under the License.
  16.  */
  17. package org.orekit.time;

  18. import org.hipparchus.exception.LocalizedCoreFormats;
  19. import org.hipparchus.util.FastMath;
  20. import org.orekit.errors.OrekitException;
  21. import org.orekit.errors.OrekitMessages;

  22. import java.io.Serializable;
  23. import java.util.concurrent.TimeUnit;

  24. /** This class represents a time range split into seconds and attoseconds.
  25.  * <p>
  26.  * Instances of this class may either be interpreted as offsets from a reference
  27.  * date, or they may be interpreted as durations. Negative values represent
  28.  * dates earlier than the reference date in the first interpretation, and
  29.  * negative durations in the second interpretation.
  30.  * </p>
  31.  * <p>
  32.  * The whole number of seconds is stored as signed primitive long, so the range
  33.  * of dates that can be represented is ±292 billion years. The fractional part
  34.  * within the second is stored as non-negative primitive long with fixed precision
  35.  * at a resolution of one attosecond (10⁻¹⁸s). The choice of attoseconds allows
  36.  * to represent exactly all important offsets (between TT and TAI, or between UTC
  37.  * and TAI during the linear eras), as well as all times converted from standard
  38.  * Java Instant, Date or TimeUnit classes. It also allows simple computation as
  39.  * adding or subtracting a few values in attoseconds that are less than one second
  40.  * does not overflow (a primitive long could hold any values between ±9.22s in
  41.  * attoseconds so simple additions and subtractions followed by handling a carry
  42.  * to bring the value back between 0 and 10¹⁸ is straightforward). There are also
  43.  * special encodings (internally using negative longs in the fractional part) to
  44.  * represent {@link #NaN}, {@link #POSITIVE_INFINITY} and {@link #NEGATIVE_INFINITY}.
  45.  * </p>
  46.  * @author Luc Maisonobe
  47.  * @see AbsoluteDate
  48.  * @see FieldAbsoluteDate
  49.  * @since 13.0
  50.  */
  51. public class TimeOffset
  52.     implements Comparable<TimeOffset>, Serializable {

  53.     /** Split time representing 0. */
  54.     public static final TimeOffset ZERO = new TimeOffset(0L, 0L);

  55.     /** Split time representing 1 attosecond. */
  56.     public static final TimeOffset ATTOSECOND = new TimeOffset(0L, 1L);

  57.     /** Split time representing 1 femtosecond. */
  58.     public static final TimeOffset FEMTOSECOND = new TimeOffset(0L, 1000L);

  59.     /** Split time representing 1 picosecond. */
  60.     public static final TimeOffset PICOSECOND = new TimeOffset(0L, 1000000L);

  61.     /** Split time representing 1 nanosecond. */
  62.     public static final TimeOffset NANOSECOND = new TimeOffset(0L, 1000000000L);

  63.     /** Split time representing 1 microsecond. */
  64.     public static final TimeOffset MICROSECOND = new TimeOffset(0L, 1000000000000L);

  65.     /** Split time representing 1 millisecond. */
  66.     public static final TimeOffset MILLISECOND = new TimeOffset(0L, 1000000000000000L);

  67.     /** Split time representing 1 second. */
  68.     public static final TimeOffset SECOND = new TimeOffset(1L, 0L);

  69.     /** Split time representing 1 minute. */
  70.     public static final TimeOffset MINUTE = new TimeOffset(60L, 0L);

  71.     /** Split time representing 1 hour. */
  72.     public static final TimeOffset HOUR = new TimeOffset(3600L, 0L);

  73.     /** Split time representing 1 day. */
  74.     public static final TimeOffset DAY = new TimeOffset(86400L, 0L);

  75.     /** Split time representing 1 day that includes an additional leap second. */
  76.     public static final TimeOffset DAY_WITH_POSITIVE_LEAP = new TimeOffset(86401L, 0L);

  77.     // CHECKSTYLE: stop ConstantName
  78.     /** Split time representing a NaN. */
  79.     public static final TimeOffset NaN = new TimeOffset(Double.NaN);
  80.     // CHECKSTYLE: resume ConstantName

  81.     /** Split time representing negative infinity. */
  82.     public static final TimeOffset NEGATIVE_INFINITY = new TimeOffset(Double.NEGATIVE_INFINITY);

  83.     /** Split time representing positive infinity. */
  84.     public static final TimeOffset POSITIVE_INFINITY = new TimeOffset(Double.POSITIVE_INFINITY);

  85.     /** Indicator for NaN time (bits pattern arbitrarily selected to avoid hashcode collisions). */
  86.     private static final long NAN_INDICATOR      = -0XFFL;

  87.     /** Indicator for positive infinite time (bits pattern arbitrarily selected to avoid hashcode collisions). */
  88.     private static final long POSITIVE_INFINITY_INDICATOR = -0XFF00L;

  89.     /** Indicator for negative infinite time (bits pattern arbitrarily selected to avoid hashcode collisions). */
  90.     private static final long NEGATIVE_INFINITY_INDICATOR = -0XFF0000L;

  91.     /** Milliseconds in one second. */
  92.     private static final long MILLIS_IN_SECOND = 1000L;

  93.     /** Microseconds in one second. */
  94.     private static final long MICROS_IN_SECOND = 1000000L;

  95.     /** Nanoseconds in one second. */
  96.     private static final long NANOS_IN_SECOND = 1000000000L;

  97.     /** Attoseconds in one second. */
  98.     private static final long ATTOS_IN_SECOND = 1000000000000000000L;

  99.     /** Attoseconds in one half-second. */
  100.     private static final long ATTOS_IN_HALF_SECOND = 500000000000000000L;

  101.     /** Factor to split long for multiplications.
  102.      * <p>
  103.      * It is important that SPLIT * SPLIT = ATTOS_IN_SECOND.
  104.      * </p>
  105.      */
  106.     private static final long SPLIT = 1000000000L;

  107.     /** Number of digits after separator for attoseconds. */
  108.     private static final int DIGITS_ATTOS = 18;

  109.     /** Multipliers for parsing partial strings. */
  110.     // CHECKSTYLE: stop Indentation check
  111.     private static final long[] MULTIPLIERS = new long[] {
  112.                          1L,
  113.                         10L,
  114.                        100L,
  115.                       1000L,
  116.                      10000L,
  117.                     100000L,
  118.                    1000000L,
  119.                   10000000L,
  120.                  100000000L,
  121.                 1000000000L,
  122.                10000000000L,
  123.               100000000000L,
  124.              1000000000000L,
  125.             10000000000000L,
  126.            100000000000000L,
  127.           1000000000000000L,
  128.          10000000000000000L,
  129.         100000000000000000L,
  130.        1000000000000000000L
  131.     };
  132.     // CHECKSTYLE: resume Indentation check

  133.     /** Serializable UID. */
  134.     private static final long serialVersionUID = 20240711L;

  135.     /** Seconds part. */
  136.     private final long seconds;

  137.     /** AttoSeconds part. */
  138.     private final long attoSeconds;

  139.     /**
  140.      * Build a time by adding several times.
  141.      * @param times times to add
  142.      */
  143.     public TimeOffset(final TimeOffset... times) {
  144.         final RunningSum runningSum = new RunningSum();
  145.         for (final TimeOffset time : times) {
  146.             runningSum.add(time);
  147.         }
  148.         final TimeOffset sum = runningSum.normalize();
  149.         this.seconds     = sum.getSeconds();
  150.         this.attoSeconds = sum.getAttoSeconds();
  151.     }

  152.     /**
  153.      * Build a time from its components.
  154.      * <p>
  155.      * The components will be normalized so that {@link #getAttoSeconds()}
  156.      * returns a value between {@code 0L} and {1000000000000000000L}
  157.      * </p>
  158.      * @param seconds seconds part
  159.      * @param attoSeconds attoseconds part
  160.      */
  161.     public TimeOffset(final long seconds, final long attoSeconds) {
  162.         final long qAtto = attoSeconds / ATTOS_IN_SECOND;
  163.         final long rAtto = attoSeconds - qAtto * ATTOS_IN_SECOND;
  164.         if (rAtto < 0L) {
  165.             this.seconds     = seconds + qAtto - 1L;
  166.             this.attoSeconds = ATTOS_IN_SECOND + rAtto;
  167.         } else {
  168.             this.seconds     = seconds + qAtto;
  169.             this.attoSeconds = rAtto;
  170.         }
  171.     }

  172.     /**
  173.      * Build a time from a value in seconds.
  174.      *
  175.      * @param time time
  176.      */
  177.     public TimeOffset(final double time) {
  178.         if (Double.isNaN(time)) {
  179.             seconds     = 0L;
  180.             attoSeconds = NAN_INDICATOR;
  181.         } else if (time < Long.MIN_VALUE || time > Long.MAX_VALUE) {
  182.             if (time < 0L) {
  183.                 seconds     = Long.MIN_VALUE;
  184.                 attoSeconds = NEGATIVE_INFINITY_INDICATOR;
  185.             } else {
  186.                 seconds     = Long.MAX_VALUE;
  187.                 attoSeconds = POSITIVE_INFINITY_INDICATOR;
  188.             }
  189.         } else {
  190.             final double tiSeconds  = FastMath.rint(time);
  191.             final double subSeconds = time - tiSeconds;
  192.             if (subSeconds < 0L) {
  193.                 seconds     = (long) tiSeconds - 1L;
  194.                 attoSeconds = FastMath.round(subSeconds * ATTOS_IN_SECOND) + ATTOS_IN_SECOND;
  195.             } else {
  196.                 seconds     = (long) tiSeconds;
  197.                 attoSeconds = FastMath.round(subSeconds * ATTOS_IN_SECOND);
  198.             }
  199.         }
  200.     }

  201.     /**
  202.      * Multiplicative constructor.
  203.      * <p>
  204.      * This constructors builds a split time corresponding to {@code factor} ⨉ {@code time}
  205.      * </p>
  206.      * @param factor multiplicative factor (negative values allowed here, contrary to {@link #multiply(long)})
  207.      * @param time base time
  208.      */
  209.     public TimeOffset(final long factor, final TimeOffset time) {
  210.         this(factor < 0 ? time.multiply(-factor).negate() : time.multiply(factor));
  211.     }

  212.     /**
  213.      * Linear combination constructor.
  214.      * <p>
  215.      * This constructors builds a split time corresponding to
  216.      * {@code f1} ⨉ {@code t1} + {@code f2} ⨉ {@code t2}
  217.      * </p>
  218.      * @param f1 first multiplicative factor (negative values allowed here, contrary to {@link #multiply(long)})
  219.      * @param t1 first base time
  220.      * @param f2 second multiplicative factor (negative values allowed here, contrary to {@link #multiply(long)})
  221.      * @param t2 second base time
  222.      */
  223.     public TimeOffset(final long f1, final TimeOffset t1,
  224.                       final long f2, final TimeOffset t2) {
  225.         this(new TimeOffset(f1, t1).add(new TimeOffset(f2, t2)));
  226.     }

  227.     /**
  228.      * Linear combination constructor.
  229.      * <p>
  230.      * This constructors builds a split time corresponding to
  231.      * {@code f1} ⨉ {@code t1} + {@code f2} ⨉ {@code t2} + {@code f3} ⨉ {@code t3}
  232.      * </p>
  233.      * @param f1 first multiplicative factor (negative values allowed here, contrary to {@link #multiply(long)})
  234.      * @param t1 first base time
  235.      * @param f2 second multiplicative factor (negative values allowed here, contrary to {@link #multiply(long)})
  236.      * @param t2 second base time
  237.      * @param f3 third multiplicative factor (negative values allowed here, contrary to {@link #multiply(long)})
  238.      * @param t3 third base time
  239.      */
  240.     public TimeOffset(final long f1, final TimeOffset t1,
  241.                       final long f2, final TimeOffset t2,
  242.                       final long f3, final TimeOffset t3) {
  243.         this(new TimeOffset(f1, t1).add(new TimeOffset(f2, t2)).add(new TimeOffset(f3, t3)));
  244.     }

  245.     /**
  246.      * Linear combination constructor.
  247.      * <p>
  248.      * This constructors builds a split time corresponding to
  249.      * {@code f1} ⨉ {@code t1} + {@code f2} ⨉ {@code t2} + {@code f3} ⨉ {@code t3} + {@code f4} ⨉ {@code t4}
  250.      * </p>
  251.      * @param f1 first multiplicative factor (negative values allowed here, contrary to {@link #multiply(long)})
  252.      * @param t1 first base time
  253.      * @param f2 second multiplicative factor (negative values allowed here, contrary to {@link #multiply(long)})
  254.      * @param t2 second base time
  255.      * @param f3 third multiplicative factor (negative values allowed here, contrary to {@link #multiply(long)})
  256.      * @param t3 third base time
  257.      * @param f4 fourth multiplicative factor (negative values allowed here, contrary to {@link #multiply(long)})
  258.      * @param t4 fourth base time
  259.      */
  260.     public TimeOffset(final long f1, final TimeOffset t1,
  261.                       final long f2, final TimeOffset t2,
  262.                       final long f3, final TimeOffset t3,
  263.                       final long f4, final TimeOffset t4) {
  264.         this(new TimeOffset(f1, t1).
  265.              add(new TimeOffset(f2, t2)).
  266.              add(new TimeOffset(f3, t3)).
  267.              add(new TimeOffset(f4, t4)));
  268.     }

  269.     /**
  270.      * Linear combination constructor.
  271.      * <p>
  272.      * This constructors builds a split time corresponding to
  273.      * {@code f1} ⨉ {@code t1} + {@code f2} ⨉ {@code t2} + {@code f3} ⨉ {@code t3} + {@code f4} ⨉ {@code t4} + {@code f5} ⨉ {@code t5}
  274.      * </p>
  275.      * @param f1 first multiplicative factor (negative values allowed here, contrary to {@link #multiply(long)})
  276.      * @param t1 first base time
  277.      * @param f2 second multiplicative factor (negative values allowed here, contrary to {@link #multiply(long)})
  278.      * @param t2 second base time
  279.      * @param f3 third multiplicative factor (negative values allowed here, contrary to {@link #multiply(long)})
  280.      * @param t3 third base time
  281.      * @param f4 fourth multiplicative factor (negative values allowed here, contrary to {@link #multiply(long)})
  282.      * @param t4 fourth base time
  283.      * @param f5 fifth multiplicative factor (negative values allowed here, contrary to {@link #multiply(long)})
  284.      * @param t5 fifth base time
  285.      */
  286.     public TimeOffset(final long f1, final TimeOffset t1,
  287.                       final long f2, final TimeOffset t2,
  288.                       final long f3, final TimeOffset t3,
  289.                       final long f4, final TimeOffset t4,
  290.                       final long f5, final TimeOffset t5) {
  291.         this(new TimeOffset(f1, t1).
  292.              add(new TimeOffset(f2, t2)).
  293.              add(new TimeOffset(f3, t3)).
  294.              add(new TimeOffset(f4, t4)).
  295.              add(new TimeOffset(f5, t5)));
  296.     }

  297.     /**
  298.      * Build a time from a value defined in some time unit.
  299.      *
  300.      * @param time time
  301.      * @param unit   time unit in which {@code time} is expressed
  302.      */
  303.     public TimeOffset(final long time, final TimeUnit unit) {
  304.         switch (unit) {
  305.             case DAYS: {
  306.                 final long limit = (Long.MAX_VALUE - DAY.seconds / 2) / DAY.seconds;
  307.                 if (time < -limit) {
  308.                     seconds     = Long.MIN_VALUE;
  309.                     attoSeconds = NEGATIVE_INFINITY_INDICATOR;
  310.                 } else if (time > limit) {
  311.                     seconds     = Long.MAX_VALUE;
  312.                     attoSeconds = POSITIVE_INFINITY_INDICATOR;
  313.                 } else {
  314.                     seconds = time * DAY.seconds;
  315.                     attoSeconds = 0L;
  316.                 }
  317.                 break;
  318.             }
  319.             case HOURS: {
  320.                 final long limit = (Long.MAX_VALUE - HOUR.seconds / 2) / HOUR.seconds;
  321.                 if (time < -limit) {
  322.                     seconds     = Long.MIN_VALUE;
  323.                     attoSeconds = NEGATIVE_INFINITY_INDICATOR;
  324.                 } else if (time > limit) {
  325.                     seconds     = Long.MAX_VALUE;
  326.                     attoSeconds = POSITIVE_INFINITY_INDICATOR;
  327.                 } else {
  328.                     seconds     = time * HOUR.seconds;
  329.                     attoSeconds = 0L;
  330.                 }
  331.                 break;
  332.             }
  333.             case MINUTES: {
  334.                 final long limit = (Long.MAX_VALUE - MINUTE.seconds / 2) / MINUTE.seconds;
  335.                 if (time < -limit) {
  336.                     seconds     = Long.MIN_VALUE;
  337.                     attoSeconds = NEGATIVE_INFINITY_INDICATOR;
  338.                 } else if (time > limit) {
  339.                     seconds     = Long.MAX_VALUE;
  340.                     attoSeconds = POSITIVE_INFINITY_INDICATOR;
  341.                 } else {
  342.                     seconds     = time * MINUTE.seconds;
  343.                     attoSeconds = 0L;
  344.                 }
  345.                 break;
  346.             }
  347.             case SECONDS:
  348.                 seconds     = time;
  349.                 attoSeconds = 0L;
  350.                 break;
  351.             case MILLISECONDS: {
  352.                 final long s = time / MILLIS_IN_SECOND;
  353.                 final long r = (time - s * MILLIS_IN_SECOND) * MILLISECOND.attoSeconds;
  354.                 if (r < 0L) {
  355.                     seconds     = s - 1L;
  356.                     attoSeconds = ATTOS_IN_SECOND + r;
  357.                 } else {
  358.                     seconds     = s;
  359.                     attoSeconds = r;
  360.                 }
  361.                 break;
  362.             }
  363.             case MICROSECONDS: {
  364.                 final long s = time / MICROS_IN_SECOND;
  365.                 final long r = (time - s * MICROS_IN_SECOND) * MICROSECOND.attoSeconds;
  366.                 if (r < 0L) {
  367.                     seconds     = s - 1L;
  368.                     attoSeconds = ATTOS_IN_SECOND + r;
  369.                 } else {
  370.                     seconds     = s;
  371.                     attoSeconds = r;
  372.                 }
  373.                 break;
  374.             }
  375.             case NANOSECONDS: {
  376.                 final long s = time / NANOS_IN_SECOND;
  377.                 final long r = (time - s * NANOS_IN_SECOND) * NANOSECOND.attoSeconds;
  378.                 if (r < 0L) {
  379.                     seconds     = s - 1L;
  380.                     attoSeconds = ATTOS_IN_SECOND + r;
  381.                 } else {
  382.                     seconds     = s;
  383.                     attoSeconds = r;
  384.                 }
  385.                 break;
  386.             }
  387.             default:
  388.                 throw new OrekitException(OrekitMessages.UNKNOWN_UNIT, unit.name());
  389.         }
  390.     }

  391.     /** Copy constructor, for internal use only.
  392.      * @param time time to copy
  393.      */
  394.     private TimeOffset(final TimeOffset time) {
  395.         seconds     = time.seconds;
  396.         attoSeconds = time.attoSeconds;
  397.     }

  398.     /** check if the time is zero.
  399.      * @return true if the time is zero
  400.      */
  401.     public boolean isZero() {
  402.         return seconds == 0L && attoSeconds == 0L;
  403.     }

  404.     /** Check if time is finite (i.e. neither {@link #isNaN() NaN} nor {@link #isInfinite() infinite)}.
  405.      * @return true if time is finite
  406.      * @see #isNaN()
  407.      * @see #isInfinite()
  408.      * @see #isNegativeInfinity()
  409.      * @see #isPositiveInfinity()
  410.      */
  411.     public boolean isFinite() {
  412.         return attoSeconds >= 0L;
  413.     }

  414.     /** Check if time is NaN.
  415.      * @return true if time is NaN
  416.      * @see #isFinite()
  417.      * @see #isInfinite()
  418.      * @see #isNegativeInfinity()
  419.      * @see #isPositiveInfinity()
  420.      */
  421.     public boolean isNaN() {
  422.         return attoSeconds == NAN_INDICATOR;
  423.     }

  424.     /** Check if time is infinity.
  425.      * @return true if time is infinity
  426.      * @see #isFinite()
  427.      * @see #isNaN()
  428.      * @see #isNegativeInfinity()
  429.      * @see #isPositiveInfinity()
  430.      */
  431.     public boolean isInfinite() {
  432.         return isPositiveInfinity() || isNegativeInfinity();
  433.     }

  434.     /** Check if time is positive infinity.
  435.      * @return true if time is positive infinity
  436.      * @see #isFinite()
  437.      * @see #isNaN()
  438.      * @see #isInfinite()
  439.      * @see #isNegativeInfinity()
  440.      */
  441.     public boolean isPositiveInfinity() {
  442.         return attoSeconds == POSITIVE_INFINITY_INDICATOR;
  443.     }

  444.     /** Check if time is negative infinity.
  445.      * @return true if time is negative infinity
  446.      * @see #isFinite()
  447.      * @see #isNaN()
  448.      * @see #isInfinite()
  449.      * @see #isPositiveInfinity()
  450.      */
  451.     public boolean isNegativeInfinity() {
  452.         return attoSeconds == NEGATIVE_INFINITY_INDICATOR;
  453.     }

  454.     /** Build a time by adding two times.
  455.      * @param t time to add
  456.      * @return this+t
  457.      */
  458.     public TimeOffset add(final TimeOffset t) {
  459.         final RunningSum runningSum = new RunningSum();
  460.         runningSum.add(this);
  461.         runningSum.add(t);
  462.         return runningSum.normalize();
  463.     }

  464.     /** Build a time by subtracting one time from the instance.
  465.      * @param t time to subtract
  466.      * @return this-t
  467.      */
  468.     public TimeOffset subtract(final TimeOffset t) {
  469.         if (attoSeconds < 0 || t.attoSeconds < 0) {
  470.             // gather all special cases in one big check to avoid rare multiple tests
  471.             if (isNaN() ||
  472.                 t.isNaN() ||
  473.                 isPositiveInfinity() && t.isPositiveInfinity() ||
  474.                 isNegativeInfinity() && t.isNegativeInfinity()) {
  475.                 return NaN;
  476.             } else if (isInfinite()) {
  477.                 // t is either a finite time or the infinity opposite to this
  478.                 return this;
  479.             } else {
  480.                 // this is either a finite time or the infinity opposite to t
  481.                 return t.isPositiveInfinity() ? NEGATIVE_INFINITY : POSITIVE_INFINITY;
  482.             }
  483.         } else {
  484.             // regular subtraction between two finite times
  485.             return new TimeOffset(seconds - t.seconds, attoSeconds - t.attoSeconds);
  486.         }
  487.     }

  488.     /** Multiply the instance by a positive or zero constant.
  489.      * @param p multiplication factor (must be positive)
  490.      * @return this ⨉ p
  491.      */
  492.     public TimeOffset multiply(final long p) {
  493.         if (p < 0) {
  494.             throw new OrekitException(OrekitMessages.NOT_POSITIVE, p);
  495.         }
  496.         if (isFinite()) {
  497.             final TimeOffset abs   = seconds < 0 ? negate() : this;
  498.             final long pHigh   = p / SPLIT;
  499.             final long pLow    = p - pHigh * SPLIT;
  500.             final long sHigh   = abs.seconds / SPLIT;
  501.             final long sLow    = abs.seconds - sHigh * SPLIT;
  502.             final long aHigh   = abs.attoSeconds / SPLIT;
  503.             final long aLow    = abs.attoSeconds - aHigh * SPLIT;
  504.             final long ps1     = pHigh * sLow + pLow * sHigh;
  505.             final long ps0     = pLow * sLow;
  506.             final long pa2     = pHigh * aHigh;
  507.             final long pa1     = pHigh * aLow + pLow * aHigh;
  508.             final long pa1High = pa1 / SPLIT;
  509.             final long pa1Low  = pa1 - pa1High * SPLIT;
  510.             final long pa0     = pLow * aLow;

  511.             // check for overflow
  512.             if (pHigh * sHigh != 0 || ps1 / SPLIT != 0) {
  513.                 throw new OrekitException(LocalizedCoreFormats.OVERFLOW_IN_MULTIPLICATION, abs.seconds, p);
  514.             }

  515.             // here we use the fact that SPLIT * SPLIT = ATTOS_IN_SECOND
  516.             final TimeOffset mul = new TimeOffset(SPLIT * ps1 + ps0 + pa2 + pa1High, SPLIT * pa1Low + pa0);
  517.             return seconds < 0 ? mul.negate() : mul;
  518.         } else {
  519.             // already NaN, +∞ or -∞, unchanged except 0 ⨉ ±∞ = NaN
  520.             return p == 0 ? TimeOffset.NaN : this;
  521.         }
  522.     }

  523.     /** Divide the instance by a positive constant.
  524.      * @param q division factor (must be strictly positive)
  525.      * @return this ÷ q
  526.      */
  527.     public TimeOffset divide(final int q) {
  528.         if (q <= 0) {
  529.             throw new OrekitException(OrekitMessages.NOT_STRICTLY_POSITIVE, q);
  530.         }
  531.         if (isFinite()) {
  532.             final long      sSec  = seconds         / q;
  533.             final long      rSec  = seconds         - sSec * q;
  534.             final long      sK    = ATTOS_IN_SECOND / q;
  535.             final long      rK    = ATTOS_IN_SECOND - sK * q;
  536.             final TimeOffset tsSec = new TimeOffset(0L, sSec);
  537.             final TimeOffset trSec = new TimeOffset(0L, rSec);
  538.             return new TimeOffset(tsSec.multiply(sK).multiply(q),
  539.                                   tsSec.multiply(rK),
  540.                                   trSec.multiply(sK),
  541.                                   // here, we use the fact q is a positive int (not a long!)
  542.                                   // hence rSec * rK < q² does not overflow
  543.                                   new TimeOffset(0L, (attoSeconds + rSec * rK) / q));
  544.         } else {
  545.             // already NaN, +∞ or -∞, unchanged as q > 0
  546.             return this;
  547.         }
  548.     }

  549.     /** Negate the instance.
  550.      * @return new instance corresponding to opposite time
  551.      */
  552.     public TimeOffset negate() {
  553.         // handle special cases
  554.         if (attoSeconds < 0) {
  555.             // gather all special cases in one big check to avoid rare multiple tests
  556.             return isNaN() ? this : (seconds < 0 ? POSITIVE_INFINITY : NEGATIVE_INFINITY);
  557.         } else {
  558.             // the negative number of attoseconds will be normalized back to positive by the constructor
  559.             return new TimeOffset(-seconds, -attoSeconds);
  560.         }
  561.     }

  562.     /** Get the time in some unit.
  563.      * @param unit time unit
  564.      * @return time in this unit, rounded to the closest long,
  565.      * returns arbitrarily {@link Long#MAX_VALUE} for {@link #isNaN() NaN times}
  566.      */
  567.     public long getRoundedTime(final TimeUnit unit) {

  568.         // handle special cases
  569.         if (attoSeconds < 0) {
  570.             // gather all special cases in one big check to avoid rare multiple tests
  571.             return (isNaN() || seconds >= 0) ? Long.MAX_VALUE : Long.MIN_VALUE;
  572.         }

  573.         final long sign = seconds < 0L ? -1L : 1L;
  574.         switch (unit) {
  575.             case DAYS:
  576.                 return sign * ((sign * seconds + DAY.seconds / 2) / DAY.seconds);
  577.             case HOURS:
  578.                 return sign * ((sign * seconds + HOUR.seconds / 2) / HOUR.seconds);
  579.             case MINUTES:
  580.                 return sign * ((sign * seconds + MINUTE.seconds / 2) / MINUTE.seconds);
  581.             case SECONDS:
  582.                 return seconds + ((attoSeconds >= ATTOS_IN_SECOND / 2) ? 1 : 0);
  583.             case MILLISECONDS:
  584.                 return seconds * MILLIS_IN_SECOND +
  585.                        (attoSeconds + MILLISECOND.attoSeconds / 2) / MILLISECOND.attoSeconds;
  586.             case MICROSECONDS:
  587.                 return seconds * MICROS_IN_SECOND +
  588.                        (attoSeconds + MICROSECOND.attoSeconds / 2) / MICROSECOND.attoSeconds;
  589.             case NANOSECONDS:
  590.                 return seconds * NANOS_IN_SECOND +
  591.                        (attoSeconds + NANOSECOND.attoSeconds / 2) / NANOSECOND.attoSeconds;
  592.             default:
  593.                 throw new OrekitException(OrekitMessages.UNKNOWN_UNIT, unit.name());
  594.         }
  595.     }

  596.     /** Get the normalized seconds part of the time.
  597.      * @return normalized seconds part of the time (may be negative)
  598.      */
  599.     public long getSeconds() {
  600.         return seconds;
  601.     }

  602.     /** Get the normalized attoseconds part of the time.
  603.      * <p>
  604.      * The normalized attoseconds is always between {@code 0L} and
  605.      * {@code 1000000000000000000L} for <em>finite</em> ranges. Note that it
  606.      * may reach {@code 1000000000000000000L} if for example the time is less
  607.      * than 1 attosecond <em>before</em> a whole second. It is negative
  608.      * for {@link #isNaN() NaN} or {@link #isInfinite() infinite} times.
  609.      * </p>
  610.      * @return normalized attoseconds part of the time
  611.      */
  612.     public long getAttoSeconds() {
  613.         return attoSeconds;
  614.     }

  615.     /** Get the time collapsed into a single double.
  616.      * <p>
  617.      * Beware that lots of accuracy is lost when combining {@link #getSeconds()} and {@link #getAttoSeconds()}
  618.      * into a single double.
  619.      * </p>
  620.      * @return time as a single double
  621.      */
  622.     public double toDouble() {
  623.         if (isFinite()) {
  624.             // regular value
  625.             long closeSeconds      = seconds;
  626.             long signedAttoSeconds = attoSeconds;
  627.             if (attoSeconds > ATTOS_IN_HALF_SECOND) {
  628.                 // we are closer to next second than to previous one
  629.                 // take this into account in the computation
  630.                 // in order to avoid losing precision
  631.                 closeSeconds++;
  632.                 signedAttoSeconds -= ATTOS_IN_SECOND;
  633.             }
  634.             return closeSeconds + ((double) signedAttoSeconds) / ATTOS_IN_SECOND;
  635.         } else {
  636.             // special values
  637.             return isNaN() ? Double.NaN : FastMath.copySign(Double.POSITIVE_INFINITY, seconds);
  638.         }
  639.     }

  640.     /** Parse a string to produce an accurate split time.
  641.      * <p>
  642.      * This method is more accurate than parsing the string as a double and then
  643.      * calling {@link TimeOffset#TimeOffset(double)} because it reads the before
  644.      * separator and after separator parts in decimal, hence avoiding problems like
  645.      * for example 0.1 not being an exact IEEE754 number.
  646.      * </p>
  647.      * @param s string to parse
  648.      * @return parsed split time
  649.      */
  650.     public static TimeOffset parse(final String s) {

  651.         // decompose the string
  652.         // we use neither Long.parseLong nor Integer.parseInt because we want to avoid
  653.         // performing several loops over the characters as we need to keep track of
  654.         // delimiters decimal point and exponent marker positions
  655.         final int length = s.length();
  656.         long significandSign = 1L;
  657.         int  exponentSign    = 1;
  658.         int  separatorIndex  = length;
  659.         int  exponentIndex   = length;
  660.         long beforeSeparator = 0L;
  661.         long afterSeparator  = 0L;
  662.         int  exponent        = 0;
  663.         int  digitsBefore    = 0;
  664.         int  digitsAfter     = 0;
  665.         int  digitsExponent  = 0;
  666.         int index = 0;
  667.         while (index < length) {

  668.             // current character
  669.             final char c = s.charAt(index);

  670.             if (Character.isDigit(c)) {
  671.                 if (separatorIndex == length) {
  672.                     // we are parsing the part before separator
  673.                     ++digitsBefore;
  674.                     beforeSeparator = beforeSeparator * 10 + c - '0';
  675.                     if (digitsBefore > 19 || beforeSeparator < 0) {
  676.                         // overflow occurred
  677.                         break;
  678.                     }
  679.                 } else if (exponentIndex == length) {
  680.                     // we are parsing the part between separator and exponent
  681.                     if (digitsAfter < DIGITS_ATTOS) {
  682.                         // we never overflow here, we just ignore extra digits
  683.                         afterSeparator = afterSeparator * 10 + c - '0';
  684.                         ++digitsAfter;
  685.                     }
  686.                 } else {
  687.                     // we are parsing the exponent
  688.                     ++digitsExponent;
  689.                     exponent = exponent * 10 + c - '0';
  690.                     if (digitsExponent > 10 || exponent < 0) {
  691.                         // overflow occurred
  692.                         break;
  693.                     }
  694.                 }
  695.             } else if (c == '.' && separatorIndex == length) {
  696.                 separatorIndex = index;
  697.             } else if ((c == 'e' || c == 'E') && exponentIndex == length) {
  698.                 if (separatorIndex == length) {
  699.                     separatorIndex = index;
  700.                 }
  701.                 exponentIndex = index;
  702.             } else if (c == '-') {
  703.                 if (index == 0) {
  704.                     significandSign = -1L;
  705.                 } else if (index == exponentIndex + 1) {
  706.                     exponentSign = -1;
  707.                 } else {
  708.                     break;
  709.                 }
  710.             } else if (c == '+') {
  711.                 if (index == 0) {
  712.                     significandSign = 1L;
  713.                 } else if (index == exponentIndex + 1) {
  714.                     exponentSign = 1;
  715.                 } else {
  716.                     break;
  717.                 }
  718.             } else {
  719.                 break;
  720.             }

  721.             ++index;

  722.         }

  723.         if (length == 0 || index < length) {
  724.             // decomposition failed, either it is a special case or an unparsable string
  725.             if (s.equals("-∞")) {
  726.                 return TimeOffset.NEGATIVE_INFINITY;
  727.             } else if (s.equals("+∞")) {
  728.                 return TimeOffset.POSITIVE_INFINITY;
  729.             } else if (s.equalsIgnoreCase("NaN")) {
  730.                 return TimeOffset.NaN;
  731.             } else {
  732.                 throw new OrekitException(OrekitMessages.CANNOT_PARSE_DATA, s);
  733.             }
  734.         }

  735.         // decomposition was successful, build the split time
  736.         long seconds;
  737.         long attoseconds;
  738.         if (exponentSign < 0) {
  739.             // the part before separator must be split into seconds and attoseconds
  740.             if (exponent >= MULTIPLIERS.length) {
  741.                 seconds = 0L;
  742.                 if (exponent - DIGITS_ATTOS >= MULTIPLIERS.length) {
  743.                     // underflow
  744.                     attoseconds = 0L;
  745.                 } else {
  746.                     attoseconds = beforeSeparator / MULTIPLIERS[exponent - DIGITS_ATTOS];
  747.                 }
  748.             } else {
  749.                 final long secondsMultiplier    = MULTIPLIERS[exponent];
  750.                 final long attoBeforeMultiplier = MULTIPLIERS[DIGITS_ATTOS - exponent];
  751.                 seconds     = beforeSeparator / secondsMultiplier;
  752.                 attoseconds = (beforeSeparator - seconds * secondsMultiplier) * attoBeforeMultiplier;
  753.                 while (digitsAfter + exponent > DIGITS_ATTOS) {
  754.                     // drop least significant digits below one attosecond
  755.                     afterSeparator /= 10;
  756.                     digitsAfter--;
  757.                 }
  758.                 final long attoAfterMultiplier = MULTIPLIERS[DIGITS_ATTOS - exponent - digitsAfter];
  759.                 attoseconds += afterSeparator * attoAfterMultiplier;
  760.             }
  761.         } else {
  762.             // the part after separator must be split into seconds and attoseconds
  763.             if (exponent >= MULTIPLIERS.length) {
  764.                 if (beforeSeparator == 0L && afterSeparator == 0L) {
  765.                     return TimeOffset.ZERO;
  766.                 } else if (significandSign < 0) {
  767.                     return TimeOffset.NEGATIVE_INFINITY;
  768.                 } else {
  769.                     return TimeOffset.POSITIVE_INFINITY;
  770.                 }
  771.             } else {
  772.                 final long secondsMultiplier = MULTIPLIERS[exponent];
  773.                 seconds = beforeSeparator * secondsMultiplier;
  774.                 if (exponent > digitsAfter) {
  775.                     seconds += afterSeparator * MULTIPLIERS[exponent - digitsAfter];
  776.                     attoseconds = 0L;
  777.                 } else {
  778.                     final long q = afterSeparator / MULTIPLIERS[digitsAfter - exponent];
  779.                     seconds    += q;
  780.                     attoseconds = (afterSeparator - q * MULTIPLIERS[digitsAfter - exponent]) *
  781.                                   MULTIPLIERS[DIGITS_ATTOS - digitsAfter + exponent];
  782.                 }
  783.             }
  784.         }

  785.         return new TimeOffset(significandSign * seconds, significandSign * attoseconds);

  786.     }

  787.     /** Compare the instance with another one.
  788.      * <p>
  789.      * Not that in order to be consistent with {@code Double#compareTo(Double)},
  790.      * NaN is considered equal to itself and greater than positive infinity.
  791.      * </p>
  792.      * @param other other time to compare the instance to
  793.      * @return a negative integer, zero, or a positive integer if applying this time
  794.      * to reference date would result in a date being before, simultaneous, or after
  795.      * the date obtained by applying the other time to the same reference date.
  796.      */
  797.     public int compareTo(final TimeOffset other) {
  798.         if (isFinite()) {
  799.             if (other.isFinite()) {
  800.                 return seconds == other.seconds ?
  801.                        Long.compare(attoSeconds, other.attoSeconds) :
  802.                        Long.compare(seconds, other.seconds);
  803.             } else {
  804.                 // if other is ±∞ or NaN, and NaN is considered larger than +∞
  805.                 return other.isNegativeInfinity() ? 1 : -1;
  806.             }
  807.         } else {
  808.             // instance is ±∞ or NaN, and NaN is considered larger than +∞
  809.             if (isNaN()) {
  810.                 // for consistency with Double.compareTo, NaN is considered equal to itself
  811.                 return other.isNaN() ? 0 : 1;
  812.             } else if (other.isNaN()) {
  813.                 return -1;
  814.             } else {
  815.                 // instance is ±∞, other is either finite or ±∞ but not NaN
  816.                 // at infinity, seconds are set to either Long.MIN_VALUE or Long.MAX_VALUE
  817.                 return Long.compare(seconds, other.seconds);
  818.             }
  819.         }
  820.     }

  821.     /** {@inheritDoc} */
  822.     @Override
  823.     public boolean equals(final Object o) {
  824.         if (this == o) {
  825.             return true;
  826.         }
  827.         if (o == null || o.getClass() != this.getClass()) {
  828.             return false;
  829.         }
  830.         final TimeOffset timeOffset = (TimeOffset) o;
  831.         return seconds == timeOffset.seconds && attoSeconds == timeOffset.attoSeconds;
  832.     }

  833.     /** {@inheritDoc} */
  834.     @Override
  835.     public int hashCode() {
  836.         return Long.hashCode(seconds) ^ Long.hashCode(attoSeconds);
  837.     }

  838.     /** Local class for summing several instances. */
  839.     private static class RunningSum {

  840.         /** Number of terms that can be added before normalization is needed. */
  841.         private static final int COUNT_DOWN_MAX = 9;

  842.         /** Seconds part. */
  843.         private long seconds;

  844.         /** AttoSeconds part. */
  845.         private long attoSeconds;

  846.         /** Indicator for NaN presence. */
  847.         private boolean addedNaN;

  848.         /** Indicator for +∞ presence. */
  849.         private boolean addedPositiveInfinity;

  850.         /** Indicator for -∞ presence. */
  851.         private boolean addedNegativeInfinity;

  852.         /** Countdown for checking carry. */
  853.         private int countDown;

  854.         /** Simple constructor.
  855.          */
  856.         RunningSum() {
  857.             countDown = COUNT_DOWN_MAX;
  858.         }

  859.         /** Add one term.
  860.          * @param term term to add
  861.          */
  862.         public void add(final TimeOffset term) {
  863.             if (term.isFinite()) {
  864.                 // regular addition
  865.                 seconds     += term.seconds;
  866.                 attoSeconds += term.attoSeconds;
  867.                 if (--countDown == 0) {
  868.                     // we have added several terms, we should normalize
  869.                     // the fields before attoseconds overflow (it may overflow after 9 additions)
  870.                     normalize();
  871.                 }
  872.             } else if (term.isNegativeInfinity()) {
  873.                 addedNegativeInfinity = true;
  874.             } else if (term.isPositiveInfinity()) {
  875.                 addedPositiveInfinity = true;
  876.             } else {
  877.                 addedNaN = true;
  878.             }
  879.         }

  880.         /** Normalize current running sum.
  881.          * @return normalized value
  882.          */
  883.         public TimeOffset normalize() {

  884.             // after normalization, we will have the equivalent of one entry processed
  885.             countDown = COUNT_DOWN_MAX - 1;

  886.             if (addedNaN || addedNegativeInfinity && addedPositiveInfinity) {
  887.                 // we have built a NaN
  888.                 seconds     = NaN.seconds;
  889.                 attoSeconds = NaN.attoSeconds;
  890.                 return NaN;
  891.             } else if (addedNegativeInfinity) {
  892.                 // we have built -∞
  893.                 seconds     = NEGATIVE_INFINITY.seconds;
  894.                 attoSeconds = NEGATIVE_INFINITY.attoSeconds;
  895.                 return NEGATIVE_INFINITY;
  896.             } else if (addedPositiveInfinity) {
  897.                 // we have built +∞
  898.                 seconds     = POSITIVE_INFINITY.seconds;
  899.                 attoSeconds = POSITIVE_INFINITY.attoSeconds;
  900.                 return POSITIVE_INFINITY;
  901.             } else {
  902.                 // this is a regular time
  903.                 final TimeOffset regular = new TimeOffset(seconds, attoSeconds);
  904.                 seconds     = regular.seconds;
  905.                 attoSeconds = regular.attoSeconds;
  906.                 return regular;
  907.             }
  908.         }

  909.     }

  910. }