Blame lib/printf-frexp.c

Packit 8f70b4
/* Split a double into fraction and mantissa, for hexadecimal printf.
Packit 8f70b4
   Copyright (C) 2007, 2009-2018 Free Software Foundation, Inc.
Packit 8f70b4
Packit 8f70b4
   This program is free software: you can redistribute it and/or modify
Packit 8f70b4
   it under the terms of the GNU General Public License as published by
Packit 8f70b4
   the Free Software Foundation; either version 3 of the License, or
Packit 8f70b4
   (at your option) any later version.
Packit 8f70b4
Packit 8f70b4
   This program is distributed in the hope that it will be useful,
Packit 8f70b4
   but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 8f70b4
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
Packit 8f70b4
   GNU General Public License for more details.
Packit 8f70b4
Packit 8f70b4
   You should have received a copy of the GNU General Public License
Packit 8f70b4
   along with this program.  If not, see <https://www.gnu.org/licenses/>.  */
Packit 8f70b4
Packit 8f70b4
#if ! defined USE_LONG_DOUBLE
Packit 8f70b4
# include <config.h>
Packit 8f70b4
#endif
Packit 8f70b4
Packit 8f70b4
/* Specification.  */
Packit 8f70b4
#ifdef USE_LONG_DOUBLE
Packit 8f70b4
# include "printf-frexpl.h"
Packit 8f70b4
#else
Packit 8f70b4
# include "printf-frexp.h"
Packit 8f70b4
#endif
Packit 8f70b4
Packit 8f70b4
#include <float.h>
Packit 8f70b4
#include <math.h>
Packit 8f70b4
#ifdef USE_LONG_DOUBLE
Packit 8f70b4
# include "fpucw.h"
Packit 8f70b4
#endif
Packit 8f70b4
Packit 8f70b4
/* This file assumes FLT_RADIX = 2.  If FLT_RADIX is a power of 2 greater
Packit 8f70b4
   than 2, or not even a power of 2, some rounding errors can occur, so that
Packit 8f70b4
   then the returned mantissa is only guaranteed to be <= 2.0, not < 2.0.  */
Packit 8f70b4
Packit 8f70b4
#ifdef USE_LONG_DOUBLE
Packit 8f70b4
# define FUNC printf_frexpl
Packit 8f70b4
# define DOUBLE long double
Packit 8f70b4
# define MIN_EXP LDBL_MIN_EXP
Packit 8f70b4
# if HAVE_FREXPL_IN_LIBC && HAVE_LDEXPL_IN_LIBC
Packit 8f70b4
#  define USE_FREXP_LDEXP
Packit 8f70b4
#  define FREXP frexpl
Packit 8f70b4
#  define LDEXP ldexpl
Packit 8f70b4
# endif
Packit 8f70b4
# define DECL_ROUNDING DECL_LONG_DOUBLE_ROUNDING
Packit 8f70b4
# define BEGIN_ROUNDING() BEGIN_LONG_DOUBLE_ROUNDING ()
Packit 8f70b4
# define END_ROUNDING() END_LONG_DOUBLE_ROUNDING ()
Packit 8f70b4
# define L_(literal) literal##L
Packit 8f70b4
#else
Packit 8f70b4
# define FUNC printf_frexp
Packit 8f70b4
# define DOUBLE double
Packit 8f70b4
# define MIN_EXP DBL_MIN_EXP
Packit 8f70b4
# if HAVE_FREXP_IN_LIBC && HAVE_LDEXP_IN_LIBC
Packit 8f70b4
#  define USE_FREXP_LDEXP
Packit 8f70b4
#  define FREXP frexp
Packit 8f70b4
#  define LDEXP ldexp
Packit 8f70b4
# endif
Packit 8f70b4
# define DECL_ROUNDING
Packit 8f70b4
# define BEGIN_ROUNDING()
Packit 8f70b4
# define END_ROUNDING()
Packit 8f70b4
# define L_(literal) literal
Packit 8f70b4
#endif
Packit 8f70b4
Packit 8f70b4
DOUBLE
Packit 8f70b4
FUNC (DOUBLE x, int *expptr)
Packit 8f70b4
{
Packit 8f70b4
  int exponent;
Packit 8f70b4
  DECL_ROUNDING
Packit 8f70b4
Packit 8f70b4
  BEGIN_ROUNDING ();
Packit 8f70b4
Packit 8f70b4
#ifdef USE_FREXP_LDEXP
Packit 8f70b4
  /* frexp and ldexp are usually faster than the loop below.  */
Packit 8f70b4
  x = FREXP (x, &exponent);
Packit 8f70b4
Packit 8f70b4
  x = x + x;
Packit 8f70b4
  exponent -= 1;
Packit 8f70b4
Packit 8f70b4
  if (exponent < MIN_EXP - 1)
Packit 8f70b4
    {
Packit 8f70b4
      x = LDEXP (x, exponent - (MIN_EXP - 1));
Packit 8f70b4
      exponent = MIN_EXP - 1;
Packit 8f70b4
    }
Packit 8f70b4
#else
Packit 8f70b4
  {
Packit 8f70b4
    /* Since the exponent is an 'int', it fits in 64 bits.  Therefore the
Packit 8f70b4
       loops are executed no more than 64 times.  */
Packit 8f70b4
    DOUBLE pow2[64]; /* pow2[i] = 2^2^i */
Packit 8f70b4
    DOUBLE powh[64]; /* powh[i] = 2^-2^i */
Packit 8f70b4
    int i;
Packit 8f70b4
Packit 8f70b4
    exponent = 0;
Packit 8f70b4
    if (x >= L_(1.0))
Packit 8f70b4
      {
Packit 8f70b4
        /* A nonnegative exponent.  */
Packit 8f70b4
        {
Packit 8f70b4
          DOUBLE pow2_i; /* = pow2[i] */
Packit 8f70b4
          DOUBLE powh_i; /* = powh[i] */
Packit 8f70b4
Packit 8f70b4
          /* Invariants: pow2_i = 2^2^i, powh_i = 2^-2^i,
Packit 8f70b4
             x * 2^exponent = argument, x >= 1.0.  */
Packit 8f70b4
          for (i = 0, pow2_i = L_(2.0), powh_i = L_(0.5);
Packit 8f70b4
               ;
Packit 8f70b4
               i++, pow2_i = pow2_i * pow2_i, powh_i = powh_i * powh_i)
Packit 8f70b4
            {
Packit 8f70b4
              if (x >= pow2_i)
Packit 8f70b4
                {
Packit 8f70b4
                  exponent += (1 << i);
Packit 8f70b4
                  x *= powh_i;
Packit 8f70b4
                }
Packit 8f70b4
              else
Packit 8f70b4
                break;
Packit 8f70b4
Packit 8f70b4
              pow2[i] = pow2_i;
Packit 8f70b4
              powh[i] = powh_i;
Packit 8f70b4
            }
Packit 8f70b4
        }
Packit 8f70b4
        /* Here 1.0 <= x < 2^2^i.  */
Packit 8f70b4
      }
Packit 8f70b4
    else
Packit 8f70b4
      {
Packit 8f70b4
        /* A negative exponent.  */
Packit 8f70b4
        {
Packit 8f70b4
          DOUBLE pow2_i; /* = pow2[i] */
Packit 8f70b4
          DOUBLE powh_i; /* = powh[i] */
Packit 8f70b4
Packit 8f70b4
          /* Invariants: pow2_i = 2^2^i, powh_i = 2^-2^i,
Packit 8f70b4
             x * 2^exponent = argument, x < 1.0, exponent >= MIN_EXP - 1.  */
Packit 8f70b4
          for (i = 0, pow2_i = L_(2.0), powh_i = L_(0.5);
Packit 8f70b4
               ;
Packit 8f70b4
               i++, pow2_i = pow2_i * pow2_i, powh_i = powh_i * powh_i)
Packit 8f70b4
            {
Packit 8f70b4
              if (exponent - (1 << i) < MIN_EXP - 1)
Packit 8f70b4
                break;
Packit 8f70b4
Packit 8f70b4
              exponent -= (1 << i);
Packit 8f70b4
              x *= pow2_i;
Packit 8f70b4
              if (x >= L_(1.0))
Packit 8f70b4
                break;
Packit 8f70b4
Packit 8f70b4
              pow2[i] = pow2_i;
Packit 8f70b4
              powh[i] = powh_i;
Packit 8f70b4
            }
Packit 8f70b4
        }
Packit 8f70b4
        /* Here either x < 1.0 and exponent - 2^i < MIN_EXP - 1 <= exponent,
Packit 8f70b4
           or 1.0 <= x < 2^2^i and exponent >= MIN_EXP - 1.  */
Packit 8f70b4
Packit 8f70b4
        if (x < L_(1.0))
Packit 8f70b4
          /* Invariants: x * 2^exponent = argument, x < 1.0 and
Packit 8f70b4
             exponent - 2^i < MIN_EXP - 1 <= exponent.  */
Packit 8f70b4
          while (i > 0)
Packit 8f70b4
            {
Packit 8f70b4
              i--;
Packit 8f70b4
              if (exponent - (1 << i) >= MIN_EXP - 1)
Packit 8f70b4
                {
Packit 8f70b4
                  exponent -= (1 << i);
Packit 8f70b4
                  x *= pow2[i];
Packit 8f70b4
                  if (x >= L_(1.0))
Packit 8f70b4
                    break;
Packit 8f70b4
                }
Packit 8f70b4
            }
Packit 8f70b4
Packit 8f70b4
        /* Here either x < 1.0 and exponent = MIN_EXP - 1,
Packit 8f70b4
           or 1.0 <= x < 2^2^i and exponent >= MIN_EXP - 1.  */
Packit 8f70b4
      }
Packit 8f70b4
Packit 8f70b4
    /* Invariants: x * 2^exponent = argument, and
Packit 8f70b4
       either x < 1.0 and exponent = MIN_EXP - 1,
Packit 8f70b4
       or 1.0 <= x < 2^2^i and exponent >= MIN_EXP - 1.  */
Packit 8f70b4
    while (i > 0)
Packit 8f70b4
      {
Packit 8f70b4
        i--;
Packit 8f70b4
        if (x >= pow2[i])
Packit 8f70b4
          {
Packit 8f70b4
            exponent += (1 << i);
Packit 8f70b4
            x *= powh[i];
Packit 8f70b4
          }
Packit 8f70b4
      }
Packit 8f70b4
    /* Here either x < 1.0 and exponent = MIN_EXP - 1,
Packit 8f70b4
       or 1.0 <= x < 2.0 and exponent >= MIN_EXP - 1.  */
Packit 8f70b4
  }
Packit 8f70b4
#endif
Packit 8f70b4
Packit 8f70b4
  END_ROUNDING ();
Packit 8f70b4
Packit 8f70b4
  *expptr = exponent;
Packit 8f70b4
  return x;
Packit 8f70b4
}