2010-12-07 00:06:48 +01:00
|
|
|
/* GCC Quad-Precision Math Library
|
2011-01-17 10:14:41 +01:00
|
|
|
Copyright (C) 2010, 2011 Free Software Foundation, Inc.
|
2010-12-07 00:06:48 +01:00
|
|
|
Written by Francois-Xavier Coudert <fxcoudert@gcc.gnu.org>
|
|
|
|
|
2011-01-17 10:14:41 +01:00
|
|
|
This file is part of the libquadmath library.
|
|
|
|
Libquadmath is free software; you can redistribute it and/or
|
2010-12-07 00:06:48 +01:00
|
|
|
modify it under the terms of the GNU Library General Public
|
|
|
|
License as published by the Free Software Foundation; either
|
|
|
|
version 2 of the License, or (at your option) any later version.
|
|
|
|
|
2011-01-17 10:14:41 +01:00
|
|
|
Libquadmath is distributed in the hope that it will be useful,
|
2010-12-07 00:06:48 +01:00
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
Library General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU Library General Public
|
2011-01-17 10:14:41 +01:00
|
|
|
License along with libquadmath; see the file COPYING.LIB. If
|
2010-12-07 00:06:48 +01:00
|
|
|
not, write to the Free Software Foundation, Inc., 51 Franklin Street - Fifth Floor,
|
|
|
|
Boston, MA 02110-1301, USA. */
|
|
|
|
|
2010-11-16 21:23:19 +00:00
|
|
|
#include "quadmath.h"
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
#define ABS(x) ((x) >= 0 ? (x) : -(x))
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
format (char * res, const __float128 x, size_t n)
|
|
|
|
{
|
|
|
|
char buffer[1024];
|
|
|
|
char *p;
|
|
|
|
|
|
|
|
memset (buffer, 0, sizeof(buffer));
|
|
|
|
|
|
|
|
g_Qfmt (buffer, &x, n + 1, sizeof(buffer) - 3);
|
|
|
|
p = buffer + (*buffer == '-' ? 1 : 0);
|
|
|
|
|
2011-01-17 10:14:41 +01:00
|
|
|
/* The sign is the easiest part. */
|
2010-11-16 21:23:19 +00:00
|
|
|
res[0] = (signbitq (x) ? '-' : '+');
|
|
|
|
|
|
|
|
if (*p == '.')
|
|
|
|
{
|
2011-01-17 10:14:41 +01:00
|
|
|
/* We have a number smaller than 1, without exponent. */
|
2010-11-16 21:23:19 +00:00
|
|
|
int exp = 0;
|
|
|
|
char *c;
|
|
|
|
|
|
|
|
for (c = p+1; *c == '0'; c++)
|
|
|
|
exp++;
|
|
|
|
|
2011-01-17 10:14:41 +01:00
|
|
|
/* We move the string "exp" characters left. */
|
2010-11-16 21:23:19 +00:00
|
|
|
size_t l = strlen (p+1+exp);
|
|
|
|
memcpy (res + 2, p + 1 + exp, l);
|
|
|
|
memset (res + 2 + l, '0', n - l + 1);
|
|
|
|
sprintf (res + n + 3, "e-%02d", exp + 1);
|
|
|
|
|
|
|
|
res[1] = res[2];
|
|
|
|
res[2] = '.';
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2011-01-17 10:14:41 +01:00
|
|
|
/* Now, do we already have an exponent. */
|
2010-11-16 21:23:19 +00:00
|
|
|
char *c;
|
|
|
|
for (c = p; *c && *c != 'e'; c++)
|
|
|
|
;
|
|
|
|
if (*c)
|
|
|
|
{
|
|
|
|
int exp = strtol (c + 1, NULL, 10);
|
|
|
|
|
|
|
|
size_t l = c - p;
|
|
|
|
|
|
|
|
memcpy (res + 1, p, l);
|
|
|
|
if (l <= n + 1)
|
|
|
|
memset (res + 1 + l, '0', (int) n - l + 2);
|
|
|
|
|
|
|
|
sprintf (res + n + 3, "e%c%02d", exp >= 0 ? '+' : '-', ABS(exp));
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2011-01-17 10:14:41 +01:00
|
|
|
/* If we have no exponent, normalize and add the exponent. */
|
2010-11-16 21:23:19 +00:00
|
|
|
for (c = p; *c && *c != '.'; c++)
|
|
|
|
;
|
|
|
|
|
|
|
|
res[1] = *p;
|
|
|
|
res[2] = '.';
|
|
|
|
|
|
|
|
size_t l = c - p;
|
|
|
|
memcpy (res + 3, p + 1, l);
|
|
|
|
size_t l2 = strlen (c + 1);
|
|
|
|
memcpy (res + 2 + l, c + 1, l2);
|
|
|
|
memset (res + 2 + l + l2, '0', n - (l + l2) + 1);
|
|
|
|
sprintf (res + n + 3, "e+%02d", l - 1);
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void
|
re PR fortran/46625 (libquadmath: Mangle internal symbols; rename __float128 <-> string functions)
PR fortran/46625
* quadmath.map (QUADMATH_1.0): Remove quadmath_strtopQ
and quadmath_dtoaq. Add strtoflt128 and quadmath_flt128tostr.
* quadmath_weak.h (quadmath_strtopQ, quadmath_dtoaq): Remove.
(strtoflt128, quadmath_flt128tostr): Add.
* gdtoa/strtopQ.c (quadmath_strtopQ): Rename to...
(strtoflt128): ... this. Return __float128, instead of writing
to memory pointed by last argument.
* quadmath.h: Use C style comments instead of C++ style.
(quadmath_strtopQ, quadmath_dtoaq): Remove prototypes.
(strtoflt128, quadmath_flt128tostr): Add prototypes.
* libquadmath.texi (quadmath_dtoaq): Rename to quadmath_flt128tostr.
(quadmath_strtopQ): Rename to strtoflt128. Adjust prototype,
adjust examples.
* quadmath_io.c (quadmath_dtoaq): Rename to...
(quadmath_flt128tostr): ... this.
libgfortran/
* io/write_float.def (DTOAQ): Use quadmath_flt128tostr
instead of quadmath_dtoa.
* io/transfer128.c (tmp1, tmp2): New variables, bring in
strtoflt128 and quadmath_flt128tostr.
(transfer_real128, transfer_real128_write, transfer_complex128,
transfer_complex128_write): Remove tmp1/tmp2 variables.
* io/read.c (convert_real): Use strtoflt128 instead of
quadmath_strtopQ, adjust for the changed arguments and return
value.
From-SVN: r168856
2011-01-16 17:55:27 +01:00
|
|
|
quadmath_flt128tostr (char *s, size_t size, size_t n, __float128 x)
|
2010-11-16 21:23:19 +00:00
|
|
|
{
|
|
|
|
char buffer[1024];
|
|
|
|
memset (buffer, 0, sizeof(buffer));
|
|
|
|
format (buffer, x, n);
|
|
|
|
memcpy (s, buffer, size);
|
|
|
|
}
|