Blame gnulib-tests/test-timespec.c

Packit Service fdd496
/* Test timespec functions.
Packit Service fdd496
   Copyright 2015-2017 Free Software Foundation, Inc.
Packit Service fdd496
Packit Service fdd496
   This program is free software: you can redistribute it and/or modify
Packit Service fdd496
   it under the terms of the GNU General Public License as published by
Packit Service fdd496
   the Free Software Foundation; either version 3 of the License, or
Packit Service fdd496
   (at your option) any later version.
Packit Service fdd496
Packit Service fdd496
   This program is distributed in the hope that it will be useful,
Packit Service fdd496
   but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit Service fdd496
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
Packit Service fdd496
   GNU General Public License for more details.
Packit Service fdd496
Packit Service fdd496
   You should have received a copy of the GNU General Public License
Packit Service fdd496
   along with this program.  If not, see <http://www.gnu.org/licenses/>.  */
Packit Service fdd496
Packit Service fdd496
/* Written by Paul Eggert.  */
Packit Service fdd496
Packit Service fdd496
#include <config.h>
Packit Service fdd496
Packit Service fdd496
#include "timespec.h"
Packit Service fdd496
Packit Service fdd496
#include "intprops.h"
Packit Service fdd496
#include "macros.h"
Packit Service fdd496
Packit Service fdd496
#include <stdbool.h>
Packit Service fdd496
#include <limits.h>
Packit Service fdd496
Packit Service fdd496
static struct { int s; int ns; } const prototype[] =
Packit Service fdd496
  {
Packit Service fdd496
    { INT_MIN, 0 },
Packit Service fdd496
    { INT_MIN, 1 },
Packit Service fdd496
    { INT_MIN, TIMESPEC_RESOLUTION - 1 },
Packit Service fdd496
    { INT_MIN + 1, 0 },
Packit Service fdd496
    { INT_MIN + 1, 1 },
Packit Service fdd496
    { INT_MIN + 1, TIMESPEC_RESOLUTION - 1 },
Packit Service fdd496
    { -1, 0 },
Packit Service fdd496
    { -1, 1 },
Packit Service fdd496
    { -1, TIMESPEC_RESOLUTION - 1 },
Packit Service fdd496
    { 0, 0 },
Packit Service fdd496
    { 0, 1 },
Packit Service fdd496
    { 0, TIMESPEC_RESOLUTION - 1 },
Packit Service fdd496
    { 1, 0 },
Packit Service fdd496
    { 1, 1 },
Packit Service fdd496
    { 1, TIMESPEC_RESOLUTION - 1 },
Packit Service fdd496
    { 1234567890, 0 },
Packit Service fdd496
    { 1234567890, 1 },
Packit Service fdd496
    { 1234567890, TIMESPEC_RESOLUTION - 1 },
Packit Service fdd496
    { INT_MAX - 1, 0 },
Packit Service fdd496
    { INT_MAX - 1, 1 },
Packit Service fdd496
    { INT_MAX - 1, TIMESPEC_RESOLUTION - 1 },
Packit Service fdd496
    { INT_MAX, 0 },
Packit Service fdd496
    { INT_MAX, 1 },
Packit Service fdd496
    { INT_MAX, TIMESPEC_RESOLUTION - 1 },
Packit Service fdd496
    { INT_MAX, 2 * TIMESPEC_RESOLUTION }
Packit Service fdd496
  };
Packit Service fdd496
enum { nprototypes = sizeof prototype / sizeof *prototype };
Packit Service fdd496
Packit Service fdd496
static bool
Packit Service fdd496
valid (struct timespec a)
Packit Service fdd496
{
Packit Service fdd496
  return 0 <= a.tv_nsec && a.tv_nsec < TIMESPEC_RESOLUTION;
Packit Service fdd496
}
Packit Service fdd496
Packit Service fdd496
static int
Packit Service fdd496
sign (int i)
Packit Service fdd496
{
Packit Service fdd496
  return i < 0 ? -1 : 0 < i;
Packit Service fdd496
}
Packit Service fdd496
Packit Service fdd496
static int
Packit Service fdd496
cmp (struct timespec a, struct timespec b)
Packit Service fdd496
{
Packit Service fdd496
  return sign (timespec_cmp (a, b));
Packit Service fdd496
}
Packit Service fdd496
Packit Service fdd496
static bool
Packit Service fdd496
eq (struct timespec a, struct timespec b)
Packit Service fdd496
{
Packit Service fdd496
  return timespec_cmp (a, b) == 0;
Packit Service fdd496
}
Packit Service fdd496
Packit Service fdd496
static bool
Packit Service fdd496
extremal (struct timespec a)
Packit Service fdd496
{
Packit Service fdd496
  return ((a.tv_sec == TYPE_MINIMUM (time_t) && a.tv_nsec == 0)
Packit Service fdd496
	  || (a.tv_sec == TYPE_MAXIMUM (time_t)
Packit Service fdd496
	      && a.tv_nsec == TIMESPEC_RESOLUTION - 1));
Packit Service fdd496
}
Packit Service fdd496
Packit Service fdd496
int
Packit Service fdd496
main (void)
Packit Service fdd496
{
Packit Service fdd496
  int i, j, k;
Packit Service fdd496
  struct timespec test[nprototypes + 1];
Packit Service fdd496
  int ntests;
Packit Service fdd496
  int computed_resolution = 1;
Packit Service fdd496
  struct timespec prevroundtrip;
Packit Service fdd496
Packit Service fdd496
  test[0] = make_timespec (TYPE_MINIMUM (time_t), -1);
Packit Service fdd496
  ntests = 1;
Packit Service fdd496
  for (i = 0; i < nprototypes; i++)
Packit Service fdd496
    {
Packit Service fdd496
      int s = prototype[i].s;
Packit Service fdd496
      if (TYPE_SIGNED (time_t) || 0 <= s)
Packit Service fdd496
	{
Packit Service fdd496
	  time_t t = (s <= INT_MIN + 1 ? s - INT_MIN + TYPE_MINIMUM (time_t)
Packit Service fdd496
		      : INT_MAX - 1 <= s ? s - INT_MAX + TYPE_MAXIMUM (time_t)
Packit Service fdd496
		      : s);
Packit Service fdd496
	  test[ntests++] = make_timespec (t, prototype[i].ns);
Packit Service fdd496
	}
Packit Service fdd496
    }
Packit Service fdd496
Packit Service fdd496
  for (i = 0; i < LOG10_TIMESPEC_RESOLUTION; i++)
Packit Service fdd496
    computed_resolution *= 10;
Packit Service fdd496
  ASSERT (computed_resolution == TIMESPEC_RESOLUTION);
Packit Service fdd496
Packit Service fdd496
  for (i = 0; i < ntests; i++)
Packit Service fdd496
    {
Packit Service fdd496
      struct timespec a = test[i];
Packit Service fdd496
Packit Service fdd496
      struct timespec roundtrip = dtotimespec (timespectod (a));
Packit Service fdd496
      if (i != 0)
Packit Service fdd496
        ASSERT (cmp (prevroundtrip, roundtrip) <= 0);
Packit Service fdd496
      prevroundtrip = roundtrip;
Packit Service fdd496
Packit Service fdd496
      ASSERT (sign (timespec_sign (a)) == cmp (a, make_timespec (0, 0)));
Packit Service fdd496
Packit Service fdd496
      if (valid (a))
Packit Service fdd496
	for (j = 0; j < ntests; j++)
Packit Service fdd496
	  {
Packit Service fdd496
	    struct timespec b = test[j];
Packit Service fdd496
	    if (valid (b))
Packit Service fdd496
	      {
Packit Service fdd496
		struct timespec sum = timespec_add (a, b);
Packit Service fdd496
		struct timespec diff = timespec_sub (a, b);
Packit Service fdd496
		struct timespec rdiff = timespec_sub (b, a);
Packit Service fdd496
		ASSERT (cmp (a, b) == sign (i - j));
Packit Service fdd496
		ASSERT (eq (sum, timespec_add (b, a)));
Packit Service fdd496
		if (! extremal (sum))
Packit Service fdd496
		  {
Packit Service fdd496
		    ASSERT (eq (a, timespec_sub (sum, b)));
Packit Service fdd496
		    ASSERT (eq (b, timespec_sub (sum, a)));
Packit Service fdd496
Packit Service fdd496
		    for (k = 0; k < ntests; k++)
Packit Service fdd496
		      {
Packit Service fdd496
			struct timespec c = test[k];
Packit Service fdd496
			if (valid (c))
Packit Service fdd496
			  {
Packit Service fdd496
			    struct timespec sumbc = timespec_add (b, c);
Packit Service fdd496
			    if (! extremal (sumbc))
Packit Service fdd496
			      ASSERT (eq (timespec_add (a, sumbc),
Packit Service fdd496
					  timespec_add (sum, c)));
Packit Service fdd496
			  }
Packit Service fdd496
		      }
Packit Service fdd496
		  }
Packit Service fdd496
		if (! extremal (diff))
Packit Service fdd496
		  ASSERT (eq (a, timespec_add (diff, b)));
Packit Service fdd496
		if (! extremal (rdiff))
Packit Service fdd496
		  ASSERT (eq (b, timespec_add (rdiff, a)));
Packit Service fdd496
Packit Service fdd496
	      }
Packit Service fdd496
	  }
Packit Service fdd496
    }
Packit Service fdd496
Packit Service fdd496
  return 0;
Packit Service fdd496
}