Blame gnulib-tests/test-float.c

Packit 33f14e
/* Test of <float.h> substitute.
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 Bruno Haible <bruno@clisp.org>, 2011.  */
Packit 33f14e
Packit 33f14e
#include <config.h>
Packit 33f14e
Packit 33f14e
#include <float.h>
Packit 33f14e
Packit 33f14e
#include "fpucw.h"
Packit 33f14e
#include "macros.h"
Packit 33f14e
Packit 33f14e
/* Check that FLT_RADIX is a constant expression.  */
Packit 33f14e
int a[] = { FLT_RADIX };
Packit 33f14e
Packit 33f14e
#if FLT_RADIX == 2
Packit 33f14e
Packit 33f14e
/* Return 2^n.  */
Packit 33f14e
static float
Packit 33f14e
pow2f (int n)
Packit 33f14e
{
Packit 33f14e
  int k = n;
Packit 33f14e
  volatile float x = 1;
Packit 33f14e
  volatile float y = 2;
Packit 33f14e
  /* Invariant: 2^n == x * y^k.  */
Packit 33f14e
  if (k < 0)
Packit 33f14e
    {
Packit 33f14e
      y = 0.5f;
Packit 33f14e
      k = - k;
Packit 33f14e
    }
Packit 33f14e
  while (k > 0)
Packit 33f14e
    {
Packit 33f14e
      if (k != 2 * (k / 2))
Packit 33f14e
        {
Packit 33f14e
          x = x * y;
Packit 33f14e
          k = k - 1;
Packit 33f14e
        }
Packit 33f14e
      if (k == 0)
Packit 33f14e
        break;
Packit 33f14e
      y = y * y;
Packit 33f14e
      k = k / 2;
Packit 33f14e
    }
Packit 33f14e
  /* Now k == 0, hence x == 2^n.  */
Packit 33f14e
  return x;
Packit 33f14e
}
Packit 33f14e
Packit 33f14e
/* Return 2^n.  */
Packit 33f14e
static double
Packit 33f14e
pow2d (int n)
Packit 33f14e
{
Packit 33f14e
  int k = n;
Packit 33f14e
  volatile double x = 1;
Packit 33f14e
  volatile double y = 2;
Packit 33f14e
  /* Invariant: 2^n == x * y^k.  */
Packit 33f14e
  if (k < 0)
Packit 33f14e
    {
Packit 33f14e
      y = 0.5;
Packit 33f14e
      k = - k;
Packit 33f14e
    }
Packit 33f14e
  while (k > 0)
Packit 33f14e
    {
Packit 33f14e
      if (k != 2 * (k / 2))
Packit 33f14e
        {
Packit 33f14e
          x = x * y;
Packit 33f14e
          k = k - 1;
Packit 33f14e
        }
Packit 33f14e
      if (k == 0)
Packit 33f14e
        break;
Packit 33f14e
      y = y * y;
Packit 33f14e
      k = k / 2;
Packit 33f14e
    }
Packit 33f14e
  /* Now k == 0, hence x == 2^n.  */
Packit 33f14e
  return x;
Packit 33f14e
}
Packit 33f14e
Packit 33f14e
/* Return 2^n.  */
Packit 33f14e
static long double
Packit 33f14e
pow2l (int n)
Packit 33f14e
{
Packit 33f14e
  int k = n;
Packit 33f14e
  volatile long double x = 1;
Packit 33f14e
  volatile long double y = 2;
Packit 33f14e
  /* Invariant: 2^n == x * y^k.  */
Packit 33f14e
  if (k < 0)
Packit 33f14e
    {
Packit 33f14e
      y = 0.5L;
Packit 33f14e
      k = - k;
Packit 33f14e
    }
Packit 33f14e
  while (k > 0)
Packit 33f14e
    {
Packit 33f14e
      if (k != 2 * (k / 2))
Packit 33f14e
        {
Packit 33f14e
          x = x * y;
Packit 33f14e
          k = k - 1;
Packit 33f14e
        }
Packit 33f14e
      if (k == 0)
Packit 33f14e
        break;
Packit 33f14e
      y = y * y;
Packit 33f14e
      k = k / 2;
Packit 33f14e
    }
Packit 33f14e
  /* Now k == 0, hence x == 2^n.  */
Packit 33f14e
  return x;
Packit 33f14e
}
Packit 33f14e
Packit 33f14e
/* ----------------------- Check macros for 'float' ----------------------- */
Packit 33f14e
Packit 33f14e
/* Check that the FLT_* macros expand to constant expressions.  */
Packit 33f14e
int fb[] =
Packit 33f14e
  {
Packit 33f14e
    FLT_MANT_DIG, FLT_MIN_EXP, FLT_MAX_EXP,
Packit 33f14e
    FLT_DIG, FLT_MIN_10_EXP, FLT_MAX_10_EXP
Packit 33f14e
  };
Packit 33f14e
float fc[] = { FLT_EPSILON, FLT_MIN, FLT_MAX };
Packit 33f14e
Packit 33f14e
static void
Packit 33f14e
test_float (void)
Packit 33f14e
{
Packit 33f14e
  /* Check that the value of FLT_MIN_EXP is well parenthesized.  */
Packit 33f14e
  ASSERT ((FLT_MIN_EXP % 101111) == (FLT_MIN_EXP) % 101111);
Packit 33f14e
Packit 33f14e
  /* Check that the value of DBL_MIN_10_EXP is well parenthesized.  */
Packit 33f14e
  ASSERT ((FLT_MIN_10_EXP % 101111) == (FLT_MIN_10_EXP) % 101111);
Packit 33f14e
Packit 33f14e
  /* Check that 'float' is as specified in IEEE 754.  */
Packit 33f14e
  ASSERT (FLT_MANT_DIG == 24);
Packit 33f14e
  ASSERT (FLT_MIN_EXP == -125);
Packit 33f14e
  ASSERT (FLT_MAX_EXP == 128);
Packit 33f14e
Packit 33f14e
  /* Check the value of FLT_MIN_10_EXP.  */
Packit 33f14e
  ASSERT (FLT_MIN_10_EXP == - (int) (- (FLT_MIN_EXP - 1) * 0.30103));
Packit 33f14e
Packit 33f14e
  /* Check the value of FLT_DIG.  */
Packit 33f14e
  ASSERT (FLT_DIG == (int) ((FLT_MANT_DIG - 1) * 0.30103));
Packit 33f14e
Packit 33f14e
  /* Check the value of FLT_MIN_10_EXP.  */
Packit 33f14e
  ASSERT (FLT_MIN_10_EXP == - (int) (- (FLT_MIN_EXP - 1) * 0.30103));
Packit 33f14e
Packit 33f14e
  /* Check the value of FLT_MAX_10_EXP.  */
Packit 33f14e
  ASSERT (FLT_MAX_10_EXP == (int) (FLT_MAX_EXP * 0.30103));
Packit 33f14e
Packit 33f14e
  /* Check the value of FLT_MAX.  */
Packit 33f14e
  {
Packit 33f14e
    volatile float m = FLT_MAX;
Packit 33f14e
    int n;
Packit 33f14e
Packit 33f14e
    ASSERT (m + m > m);
Packit 33f14e
    for (n = 0; n <= 2 * FLT_MANT_DIG; n++)
Packit 33f14e
      {
Packit 33f14e
        volatile float pow2_n = pow2f (n); /* 2^n */
Packit 33f14e
        volatile float x = m + (m / pow2_n);
Packit 33f14e
        if (x > m)
Packit 33f14e
          ASSERT (x + x == x);
Packit 33f14e
        else
Packit 33f14e
          ASSERT (!(x + x == x));
Packit 33f14e
      }
Packit 33f14e
  }
Packit 33f14e
Packit 33f14e
  /* Check the value of FLT_MIN.  */
Packit 33f14e
  {
Packit 33f14e
    volatile float m = FLT_MIN;
Packit 33f14e
    volatile float x = pow2f (FLT_MIN_EXP - 1);
Packit 33f14e
    ASSERT (m == x);
Packit 33f14e
  }
Packit 33f14e
Packit 33f14e
  /* Check the value of FLT_EPSILON.  */
Packit 33f14e
  {
Packit 33f14e
    volatile float e = FLT_EPSILON;
Packit 33f14e
    volatile float me;
Packit 33f14e
    int n;
Packit 33f14e
Packit 33f14e
    me = 1.0f + e;
Packit 33f14e
    ASSERT (me > 1.0f);
Packit 33f14e
    ASSERT (me - 1.0f == e);
Packit 33f14e
    for (n = 0; n <= 2 * FLT_MANT_DIG; n++)
Packit 33f14e
      {
Packit 33f14e
        volatile float half_n = pow2f (- n); /* 2^-n */
Packit 33f14e
        volatile float x = me - half_n;
Packit 33f14e
        if (x < me)
Packit 33f14e
          ASSERT (x <= 1.0f);
Packit 33f14e
      }
Packit 33f14e
  }
Packit 33f14e
}
Packit 33f14e
Packit 33f14e
/* ----------------------- Check macros for 'double' ----------------------- */
Packit 33f14e
Packit 33f14e
/* Check that the DBL_* macros expand to constant expressions.  */
Packit 33f14e
int db[] =
Packit 33f14e
  {
Packit 33f14e
    DBL_MANT_DIG, DBL_MIN_EXP, DBL_MAX_EXP,
Packit 33f14e
    DBL_DIG, DBL_MIN_10_EXP, DBL_MAX_10_EXP
Packit 33f14e
  };
Packit 33f14e
double dc[] = { DBL_EPSILON, DBL_MIN, DBL_MAX };
Packit 33f14e
Packit 33f14e
static void
Packit 33f14e
test_double (void)
Packit 33f14e
{
Packit 33f14e
  /* Check that the value of DBL_MIN_EXP is well parenthesized.  */
Packit 33f14e
  ASSERT ((DBL_MIN_EXP % 101111) == (DBL_MIN_EXP) % 101111);
Packit 33f14e
Packit 33f14e
  /* Check that the value of DBL_MIN_10_EXP is well parenthesized.  */
Packit 33f14e
  ASSERT ((DBL_MIN_10_EXP % 101111) == (DBL_MIN_10_EXP) % 101111);
Packit 33f14e
Packit 33f14e
  /* Check that 'double' is as specified in IEEE 754.  */
Packit 33f14e
  ASSERT (DBL_MANT_DIG == 53);
Packit 33f14e
  ASSERT (DBL_MIN_EXP == -1021);
Packit 33f14e
  ASSERT (DBL_MAX_EXP == 1024);
Packit 33f14e
Packit 33f14e
  /* Check the value of DBL_MIN_10_EXP.  */
Packit 33f14e
  ASSERT (DBL_MIN_10_EXP == - (int) (- (DBL_MIN_EXP - 1) * 0.30103));
Packit 33f14e
Packit 33f14e
  /* Check the value of DBL_DIG.  */
Packit 33f14e
  ASSERT (DBL_DIG == (int) ((DBL_MANT_DIG - 1) * 0.30103));
Packit 33f14e
Packit 33f14e
  /* Check the value of DBL_MIN_10_EXP.  */
Packit 33f14e
  ASSERT (DBL_MIN_10_EXP == - (int) (- (DBL_MIN_EXP - 1) * 0.30103));
Packit 33f14e
Packit 33f14e
  /* Check the value of DBL_MAX_10_EXP.  */
Packit 33f14e
  ASSERT (DBL_MAX_10_EXP == (int) (DBL_MAX_EXP * 0.30103));
Packit 33f14e
Packit 33f14e
  /* Check the value of DBL_MAX.  */
Packit 33f14e
  {
Packit 33f14e
    volatile double m = DBL_MAX;
Packit 33f14e
    int n;
Packit 33f14e
Packit 33f14e
    ASSERT (m + m > m);
Packit 33f14e
    for (n = 0; n <= 2 * DBL_MANT_DIG; n++)
Packit 33f14e
      {
Packit 33f14e
        volatile double pow2_n = pow2d (n); /* 2^n */
Packit 33f14e
        volatile double x = m + (m / pow2_n);
Packit 33f14e
        if (x > m)
Packit 33f14e
          ASSERT (x + x == x);
Packit 33f14e
        else
Packit 33f14e
          ASSERT (!(x + x == x));
Packit 33f14e
      }
Packit 33f14e
  }
Packit 33f14e
Packit 33f14e
  /* Check the value of DBL_MIN.  */
Packit 33f14e
  {
Packit 33f14e
    volatile double m = DBL_MIN;
Packit 33f14e
    volatile double x = pow2d (DBL_MIN_EXP - 1);
Packit 33f14e
    ASSERT (m == x);
Packit 33f14e
  }
Packit 33f14e
Packit 33f14e
  /* Check the value of DBL_EPSILON.  */
Packit 33f14e
  {
Packit 33f14e
    volatile double e = DBL_EPSILON;
Packit 33f14e
    volatile double me;
Packit 33f14e
    int n;
Packit 33f14e
Packit 33f14e
    me = 1.0 + e;
Packit 33f14e
    ASSERT (me > 1.0);
Packit 33f14e
    ASSERT (me - 1.0 == e);
Packit 33f14e
    for (n = 0; n <= 2 * DBL_MANT_DIG; n++)
Packit 33f14e
      {
Packit 33f14e
        volatile double half_n = pow2d (- n); /* 2^-n */
Packit 33f14e
        volatile double x = me - half_n;
Packit 33f14e
        if (x < me)
Packit 33f14e
          ASSERT (x <= 1.0);
Packit 33f14e
      }
Packit 33f14e
  }
Packit 33f14e
}
Packit 33f14e
Packit 33f14e
/* -------------------- Check macros for 'long double' -------------------- */
Packit 33f14e
Packit 33f14e
/* Check that the LDBL_* macros expand to constant expressions.  */
Packit 33f14e
int lb[] =
Packit 33f14e
  {
Packit 33f14e
    LDBL_MANT_DIG, LDBL_MIN_EXP, LDBL_MAX_EXP,
Packit 33f14e
    LDBL_DIG, LDBL_MIN_10_EXP, LDBL_MAX_10_EXP
Packit 33f14e
  };
Packit 33f14e
long double lc1 = LDBL_EPSILON;
Packit 33f14e
long double lc2 = LDBL_MIN;
Packit 33f14e
#if 0 /* LDBL_MAX is not a constant expression on some platforms.  */
Packit 33f14e
long double lc3 = LDBL_MAX;
Packit 33f14e
#endif
Packit 33f14e
Packit 33f14e
static void
Packit 33f14e
test_long_double (void)
Packit 33f14e
{
Packit 33f14e
  /* Check that the value of LDBL_MIN_EXP is well parenthesized.  */
Packit 33f14e
  ASSERT ((LDBL_MIN_EXP % 101111) == (LDBL_MIN_EXP) % 101111);
Packit 33f14e
Packit 33f14e
  /* Check that the value of LDBL_MIN_10_EXP is well parenthesized.  */
Packit 33f14e
  ASSERT ((LDBL_MIN_10_EXP % 101111) == (LDBL_MIN_10_EXP) % 101111);
Packit 33f14e
Packit 33f14e
  /* Check that 'long double' is at least as wide as 'double'.  */
Packit 33f14e
  ASSERT (LDBL_MANT_DIG >= DBL_MANT_DIG);
Packit 33f14e
  ASSERT (LDBL_MIN_EXP - LDBL_MANT_DIG <= DBL_MIN_EXP - DBL_MANT_DIG);
Packit 33f14e
  ASSERT (LDBL_MAX_EXP >= DBL_MAX_EXP);
Packit 33f14e
Packit 33f14e
  /* Check the value of LDBL_DIG.  */
Packit 33f14e
  ASSERT (LDBL_DIG == (int)((LDBL_MANT_DIG - 1) * 0.30103));
Packit 33f14e
Packit 33f14e
  /* Check the value of LDBL_MIN_10_EXP.  */
Packit 33f14e
  ASSERT (LDBL_MIN_10_EXP == - (int) (- (LDBL_MIN_EXP - 1) * 0.30103));
Packit 33f14e
Packit 33f14e
  /* Check the value of LDBL_MAX_10_EXP.  */
Packit 33f14e
  ASSERT (LDBL_MAX_10_EXP == (int) (LDBL_MAX_EXP * 0.30103));
Packit 33f14e
Packit 33f14e
  /* Check the value of LDBL_MAX.  */
Packit 33f14e
  {
Packit 33f14e
    volatile long double m = LDBL_MAX;
Packit 33f14e
    int n;
Packit 33f14e
Packit 33f14e
    ASSERT (m + m > m);
Packit 33f14e
    for (n = 0; n <= 2 * LDBL_MANT_DIG; n++)
Packit 33f14e
      {
Packit 33f14e
        volatile long double pow2_n = pow2l (n); /* 2^n */
Packit 33f14e
        volatile long double x = m + (m / pow2_n);
Packit 33f14e
        if (x > m)
Packit 33f14e
          ASSERT (x + x == x);
Packit 33f14e
        else
Packit 33f14e
          ASSERT (!(x + x == x));
Packit 33f14e
      }
Packit 33f14e
  }
Packit 33f14e
Packit 33f14e
  /* Check the value of LDBL_MIN.  */
Packit 33f14e
  {
Packit 33f14e
    volatile long double m = LDBL_MIN;
Packit 33f14e
    volatile long double x = pow2l (LDBL_MIN_EXP - 1);
Packit 33f14e
    ASSERT (m == x);
Packit 33f14e
  }
Packit 33f14e
Packit 33f14e
  /* Check the value of LDBL_EPSILON.  */
Packit 33f14e
  {
Packit 33f14e
    volatile long double e = LDBL_EPSILON;
Packit 33f14e
    volatile long double me;
Packit 33f14e
    int n;
Packit 33f14e
Packit 33f14e
    me = 1.0L + e;
Packit 33f14e
    ASSERT (me > 1.0L);
Packit 33f14e
    ASSERT (me - 1.0L == e);
Packit 33f14e
    for (n = 0; n <= 2 * LDBL_MANT_DIG; n++)
Packit 33f14e
      {
Packit 33f14e
        volatile long double half_n = pow2l (- n); /* 2^-n */
Packit 33f14e
        volatile long double x = me - half_n;
Packit 33f14e
        if (x < me)
Packit 33f14e
          ASSERT (x <= 1.0L);
Packit 33f14e
      }
Packit 33f14e
  }
Packit 33f14e
}
Packit 33f14e
Packit 33f14e
int
Packit 33f14e
main ()
Packit 33f14e
{
Packit 33f14e
  test_float ();
Packit 33f14e
  test_double ();
Packit 33f14e
Packit 33f14e
  {
Packit 33f14e
    DECL_LONG_DOUBLE_ROUNDING
Packit 33f14e
Packit 33f14e
    BEGIN_LONG_DOUBLE_ROUNDING ();
Packit 33f14e
Packit 33f14e
    test_long_double ();
Packit 33f14e
Packit 33f14e
    END_LONG_DOUBLE_ROUNDING ();
Packit 33f14e
  }
Packit 33f14e
Packit 33f14e
  return 0;
Packit 33f14e
}
Packit 33f14e
Packit 33f14e
#else
Packit 33f14e
Packit 33f14e
int
Packit 33f14e
main ()
Packit 33f14e
{
Packit 33f14e
  fprintf (stderr, "Skipping test: FLT_RADIX is not 2.\n");
Packit 33f14e
  return 77;
Packit 33f14e
}
Packit 33f14e
Packit 33f14e
#endif