#include "lint.h"
#include "base_conversion.h"
#include <sys/types.h>
#include "libc.h"
#include <locale.h>
static void
__k_gconvert(int ndigits, decimal_record *pd, int trailing, char *buf)
{
char *p;
int i;
char decpt = *(localeconv()->decimal_point);
p = buf;
if (pd->sign)
*(p++) = '-';
switch (pd->fpclass) {
case fp_zero:
*(p++) = '0';
if (trailing != 0) {
*(p++) = decpt;
for (i = 0; i < ndigits - 1; i++)
*(p++) = '0';
}
*p++ = 0;
break;
case fp_subnormal:
case fp_normal:
if ((pd->exponent > 0) || (pd->exponent < -(ndigits + 3))) {
char estring[4];
int n;
*(p++) = pd->ds[0];
*(p++) = decpt;
for (i = 1; pd->ds[i] != 0; )
*(p++) = pd->ds[i++];
if (trailing == 0) {
p--;
while (*p == '0')
p--;
if (*p != decpt)
p++;
}
*(p++) = 'e';
n = pd->exponent + i - 1;
if (n >= 0)
*(p++) = '+';
else {
*(p++) = '-';
n = -n;
}
__four_digits_quick((unsigned short) n, estring);
for (i = 0; estring[i] == '0'; i++)
;
if (i > 2)
i = 2;
for (; i <= 3; )
*(p++) = estring[i++];
} else {
if (pd->exponent >= (1 - ndigits)) {
for (i = 0; i < (ndigits + pd->exponent); )
*(p++) = pd->ds[i++];
*(p++) = decpt;
if (pd->ds[i] != 0) {
for (; i < ndigits; )
*(p++) = pd->ds[i++];
}
} else {
*(p++) = '0';
*(p++) = decpt;
for (i = 0; i < -(pd->exponent + ndigits); i++)
*(p++) = '0';
for (i = 0; pd->ds[i] != 0; )
*(p++) = pd->ds[i++];
}
if (trailing == 0) {
p--;
while (*p == '0')
p--;
if (*p != decpt)
p++;
}
}
*(p++) = 0;
break;
default:
__infnanstring(pd->fpclass, ndigits, p);
break;
}
}
char *
gconvert(double number, int ndigits, int trailing, char *buf)
{
decimal_mode dm;
decimal_record dr;
fp_exception_field_type fef;
#if defined(__sparc)
dm.rd = _QgetRD();
#elif defined(__i386) || defined(__amd64)
dm.rd = __xgetRD();
#else
#error Unknown architecture
#endif
dm.df = floating_form;
if (ndigits < 0)
ndigits = 6;
else if (ndigits == 0)
ndigits = 1;
else if (ndigits >= DECIMAL_STRING_LENGTH)
ndigits = DECIMAL_STRING_LENGTH - 1;
dm.ndigits = ndigits;
double_to_decimal(&number, &dm, &dr, &fef);
__k_gconvert(ndigits, &dr, trailing, buf);
return (buf);
}
char *
sgconvert(single *number, int ndigits, int trailing, char *buf)
{
decimal_mode dm;
decimal_record dr;
fp_exception_field_type fef;
#if defined(__sparc)
dm.rd = _QgetRD();
#elif defined(__i386) || defined(__amd64)
dm.rd = __xgetRD();
#else
#error Unknown architecture
#endif
dm.df = floating_form;
if (ndigits < 0)
ndigits = 6;
else if (ndigits == 0)
ndigits = 1;
else if (ndigits >= DECIMAL_STRING_LENGTH)
ndigits = DECIMAL_STRING_LENGTH - 1;
dm.ndigits = ndigits;
single_to_decimal(number, &dm, &dr, &fef);
__k_gconvert(ndigits, &dr, trailing, buf);
return (buf);
}
char *
qgconvert(quadruple *number, int ndigits, int trailing, char *buf)
{
decimal_mode dm;
decimal_record dr;
fp_exception_field_type fef;
#if defined(__sparc)
dm.rd = _QgetRD();
#elif defined(__i386) || defined(__amd64)
dm.rd = __xgetRD();
#else
#error Unknown architecture
#endif
dm.df = floating_form;
if (ndigits < 0)
ndigits = 6;
else if (ndigits == 0)
ndigits = 1;
else if (ndigits >= DECIMAL_STRING_LENGTH)
ndigits = DECIMAL_STRING_LENGTH - 1;
dm.ndigits = ndigits;
#if defined(__sparc)
quadruple_to_decimal(number, &dm, &dr, &fef);
#elif defined(__i386) || defined(__amd64)
extended_to_decimal((extended *)number, &dm, &dr, &fef);
#else
#error Unknown architecture
#endif
__k_gconvert(ndigits, &dr, trailing, buf);
return (buf);
}