Blame lib/printf-frexp.c

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