Blame gnulib-tests/timespec-sub.c

Packit 33f14e
/* Subtract two struct timespec values.
Packit 33f14e
Packit 33f14e
   Copyright (C) 2011-2017 Free Software Foundation, Inc.
Packit 33f14e
Packit 33f14e
   This program is free software: you can redistribute it and/or modify
Packit 33f14e
   it under the terms of the GNU General Public License as published by
Packit 33f14e
   the Free Software Foundation; either version 3 of the License, or
Packit 33f14e
   (at your option) any later version.
Packit 33f14e
Packit 33f14e
   This program is distributed in the hope that it will be useful,
Packit 33f14e
   but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 33f14e
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
Packit 33f14e
   GNU General Public License for more details.
Packit 33f14e
Packit 33f14e
   You should have received a copy of the GNU General Public License
Packit 33f14e
   along with this program.  If not, see <http://www.gnu.org/licenses/>.  */
Packit 33f14e
Packit 33f14e
/* Written by Paul Eggert.  */
Packit 33f14e
Packit 33f14e
/* Return the difference between two timespec values A and B.  On
Packit 33f14e
   overflow, return an extremal value.  This assumes 0 <= tv_nsec <
Packit 33f14e
   TIMESPEC_RESOLUTION.  */
Packit 33f14e
Packit 33f14e
#include <config.h>
Packit 33f14e
#include "timespec.h"
Packit 33f14e
Packit 33f14e
#include "intprops.h"
Packit 33f14e
Packit 33f14e
struct timespec
Packit 33f14e
timespec_sub (struct timespec a, struct timespec b)
Packit 33f14e
{
Packit 33f14e
  time_t rs = a.tv_sec;
Packit 33f14e
  time_t bs = b.tv_sec;
Packit 33f14e
  int ns = a.tv_nsec - b.tv_nsec;
Packit 33f14e
  int rns = ns;
Packit 33f14e
  time_t tmin = TYPE_MINIMUM (time_t);
Packit 33f14e
  time_t tmax = TYPE_MAXIMUM (time_t);
Packit 33f14e
Packit 33f14e
  if (ns < 0)
Packit 33f14e
    {
Packit 33f14e
      rns = ns + TIMESPEC_RESOLUTION;
Packit 33f14e
      if (bs < tmax)
Packit 33f14e
        bs++;
Packit 33f14e
      else if (- TYPE_SIGNED (time_t) < rs)
Packit 33f14e
        rs--;
Packit 33f14e
      else
Packit 33f14e
        goto low_overflow;
Packit 33f14e
    }
Packit 33f14e
Packit 33f14e
  /* INT_SUBTRACT_WRAPV is not appropriate since time_t might be unsigned.
Packit 33f14e
     In theory time_t might be narrower than int, so plain
Packit 33f14e
     INT_SUBTRACT_OVERFLOW does not suffice.  */
Packit 33f14e
  if (! INT_SUBTRACT_OVERFLOW (rs, bs) && tmin <= rs - bs && rs - bs <= tmax)
Packit 33f14e
    rs -= bs;
Packit 33f14e
  else
Packit 33f14e
    {
Packit 33f14e
      if (rs < 0)
Packit 33f14e
        {
Packit 33f14e
        low_overflow:
Packit 33f14e
          rs = tmin;
Packit 33f14e
          rns = 0;
Packit 33f14e
        }
Packit 33f14e
      else
Packit 33f14e
        {
Packit 33f14e
          rs = tmax;
Packit 33f14e
          rns = TIMESPEC_RESOLUTION - 1;
Packit 33f14e
        }
Packit 33f14e
    }
Packit 33f14e
Packit 33f14e
  return make_timespec (rs, rns);
Packit 33f14e
}