2020-09-30 16:53:55 +02:00

709 lines
18 KiB
C

/***
*cvt.c - C floating-point output conversions
*
* Copyright (c) 1983-2001, Microsoft Corporation. All rights reserved.
*
*Purpose:
* contains routines for performing %e, %f, and %g output conversions
* for printf, etc.
*
* routines include _cfltcvt(), _cftoe(), _cftof(), _cftog(),
* _fassign(), _positive(), _cropzeros(), _forcdecpt()
*
*Revision History:
* 04-18-84 RN author
* 01-15-87 BCM corrected processing of %g formats (to handle precision
* as the maximum number of signifcant digits displayed)
* 03-24-87 BCM Evaluation Issues: (fccvt.obj version for ?LIBFA)
* ------------------
* SDS - no problem
* GD/TS :
* char g_fmt = 0; (local, initialized)
* int g_magnitude =0; (local, initialized)
* char g_round_expansion = 0; (local, initialized)
* STRFLT g_pflt; (local, uninitialized)
* other INIT :
* ALTMATH __fpmath() initialization (perhaps)
* TERM - nothing
* 10-22-87 BCM changes for OS/2 Support Library -
* including elimination of g_... static variables
* in favor of stack-based variables & function arguments
* under MTHREAD switch; changed interfaces to _cfto? routines
* 01-15-88 BCM remove IBMC20 switches; use only memmove, not memcpy;
* use just MTHREAD switch, not SS_NEQ_DGROUP
* 06-13-88 WAJ Fixed %.1g processing for small x
* 08-02-88 WAJ Made changes to _fassign() for new input().
* 03-09-89 WAJ Added some long double support.
* 06-05-89 WAJ Made changes for C6. LDOUBLE => long double
* 06-12-89 WAJ Renamed this file from cvtn.c to cvt.c
* 11-02-89 WAJ Removed register.h
* 06-28-90 WAJ Removed fars.
* 11-15-90 WAJ Added _cdecl where needed. Also "pascal" => "_pascal".
* 09-12-91 GDP _cdecl=>_CALLTYPE2 _pascal=>_CALLTYPE5 near=>_NEAR
* 04-30-92 GDP Removed floating point code. Instead used S/W routines
* (_atodbl, _atoflt _atoldbl), so that to avoid
* generation of IEEE exceptions from the lib code.
* 03-11-93 JWM Added minimal support for _INTL decimal point - one byte only!
* 04-06-93 SKS Replace _CALLTYPE* with __cdecl
* 07-16-93 SRW ALPHA Merge
* 11-15-93 GJF Merged in NT SDK version ("ALPHA merge" stuff). Also,
* dropped support of Alpha acc compier, replaced i386
* with _M_IX86, replaced MTHREAD with _MT.
* 09-06-94 CFW Remove _INTL switch.
* 09-05-00 GB Changed the defination of fltout functions. Use DOUBLE
* instead of double
*
*******************************************************************************/
#include <ctype.h>
#include <string.h>
#include <math.h>
#include <cv.h>
#include <nlsint.h>
#ifdef _M_IX86
/* Uncomment this for enabling 10-byte long double string conversions */
/* #define LONG_DOUBLE */
#endif
/* this routine resides in the crt32 tree */
extern void _fptostr(char *buf, int digits, STRFLT pflt);
static void _CALLTYPE5 _shift( char *s, int dist );
#ifdef _MT
static char * _cftoe2( char * buf, int ndec, int caps, STRFLT pflt, char g_fmt );
static char * _cftof2( char * buf, int ndec, STRFLT pflt, char g_fmt );
#else /* not _MT */
static char * _cftoe_g( double * pvalue, char * buf, int ndec, int caps );
static char * _cftof_g( double * pvalue, char * buf, int ndec );
#endif /* not _MT */
/***
*_forcdecpt(buffer) - force a decimal point in floating-point output
*Purpose:
* force a decimal point in floating point output. we are only called if '#'
* flag is given and precision is 0; so we know the number has no '.'. insert
* the '.' and move everybody else back one position, until '\0' seen
*
* side effects: futzes around with the buffer, trying to insert a '.'
* after the initial string of digits. the first char can usually be
* skipped since it will be a digit or a '-'. but in the 0-precision case,
* the number could start with 'e' or 'E', so we'd want the '.' before the
* exponent in that case.
*
*Entry:
* buffer = (char *) pointer to buffer to modify
*
*Exit:
* returns : (void)
*
*Exceptions:
*******************************************************************************/
void __cdecl _forcdecpt( char * buffer )
{
char holdchar;
char nextchar;
if (tolower(*buffer) != 'e'){
do {
buffer++;
}
while (isdigit(*buffer));
}
holdchar = *buffer;
*buffer++ = *__decimal_point;
do {
nextchar = *buffer;
*buffer = holdchar;
holdchar = nextchar;
}
while(*buffer++);
}
/***
*_cropzeros(buffer) - removes trailing zeros from floating-point output
*Purpose:
* removes trailing zeros (after the '.') from floating-point output;
* called only when we're doing %g format, there's no '#' flag, and
* precision is non-zero. plays around with the buffer, looking for
* trailing zeros. when we find them, then we move everbody else forward
* so they overlay the zeros. if we eliminate the entire fraction part,
* then we overlay the decimal point ('.'), too.
*
* side effects: changes the buffer from
* [-] digit [digit...] [ . [digits...] [0...] ] [(exponent part)]
* to
* [-] digit [digit...] [ . digit [digits...] ] [(exponent part)]
* or
* [-] digit [digit...] [(exponent part)]
*
*Entry:
* buffer = (char *) pointer to buffer to modify
*
*Exit:
* returns : (void)
*
*Exceptions:
*******************************************************************************/
void __cdecl _cropzeros( char * buf )
{
char *stop;
while (*buf && *buf != *__decimal_point)
buf++;
if (*buf++) {
while (*buf && *buf != 'e' && *buf != 'E')
buf++;
stop = buf--;
while (*buf == '0')
buf--;
if (*buf == *__decimal_point)
buf--;
while( (*++buf = *stop++) != '\0' );
}
}
int __cdecl _positive( double * arg )
{
return( (*arg >= 0.0) );
}
void __cdecl _fassign( int flag, char * argument, char * number )
{
FLOAT floattemp;
DOUBLE doubletemp;
#ifdef LONG_DOUBLE
_LDOUBLE longtemp;
switch( flag ){
case 2:
_atoldbl( &longtemp, number );
*(_LDOUBLE UNALIGNED *)argument = longtemp;
break;
case 1:
_atodbl( &doubletemp, number );
*(DOUBLE UNALIGNED *)argument = doubletemp;
break;
default:
_atoflt( &floattemp, number );
*(FLOAT UNALIGNED *)argument = floattemp;
}
#else /* not LONG_DOUBLE */
if (flag) {
_atodbl( &doubletemp, number );
*(DOUBLE UNALIGNED *)argument = doubletemp;
} else {
_atoflt( &floattemp, number );
*(FLOAT UNALIGNED *)argument = floattemp;
}
#endif /* not LONG_DOUBLE */
}
#ifndef _MT
static char g_fmt = 0;
static int g_magnitude = 0;
static char g_round_expansion = 0;
static STRFLT g_pflt;
#endif
/*
* Function name: _cftoe
*
* Arguments: pvalue - double * pointer
* buf - char * pointer
* ndec - int
* caps - int
*
* Description: _cftoe converts the double pointed to by pvalue to a null
* terminated string of ASCII digits in the c language
* printf %e format, nad returns a pointer to the result.
* This format has the form [-]d.ddde(+/-)ddd, where there
* will be ndec digits following the decimal point. If
* ndec <= 0, no decimal point will appear. The low order
* digit is rounded. If caps is nonzero then the exponent
* will appear as E(+/-)ddd.
*
* Side Effects: the buffer 'buf' is assumed to have a minimum length
* of CVTBUFSIZE (defined in cvt.h) and the routines will
* not write over this size.
*
* Author: written R.K. Wyss, Microsoft, Sept. 9, 1983
*
* History:
*
*/
#ifdef _MT
static char * _cftoe2( char * buf, int ndec, int caps, STRFLT pflt, char g_fmt )
#else
char * __cdecl _cftoe( double * pvalue, char * buf, int ndec, int caps )
#endif
{
#ifndef _MT
STRFLT pflt;
DOUBLE *pdvalue = (DOUBLE *)pvalue;
#endif
char *p;
int exp;
/* first convert the value */
/* place the output in the buffer and round. Leave space in the buffer
* for the '-' sign (if any) and the decimal point (if any)
*/
if (g_fmt) {
#ifndef _MT
pflt = g_pflt;
#endif
/* shift it right one place if nec. for decimal point */
p = buf + (pflt->sign == '-');
_shift(p, (ndec > 0));
}
#ifndef _MT
else {
pflt = _fltout(*pdvalue);
_fptostr(buf + (pflt->sign == '-') + (ndec > 0), ndec + 1, pflt);
}
#endif
/* now fix the number up to be in e format */
p = buf;
/* put in negative sign if needed */
if (pflt->sign == '-')
*p++ = '-';
/* put in decimal point if needed. Copy the first digit to the place
* left for it and put the decimal point in its place
*/
if (ndec > 0) {
*p = *(p+1);
*(++p) = *__decimal_point;
}
/* find the end of the string and attach the exponent field */
p = strcpy(p+ndec+(!g_fmt), "e+000");
/* adjust exponent indicator according to caps flag and increment
* pointer to point to exponent sign
*/
if (caps)
*p = 'E';
p++;
/* if mantissa is zero, then the number is 0 and we are done; otherwise
* adjust the exponent sign (if necessary) and value.
*/
if (*pflt->mantissa != '0') {
/* check to see if exponent is negative; if so adjust exponent sign and
* exponent value.
*/
if( (exp = pflt->decpt - 1) < 0 ) {
exp = -exp;
*p = '-';
}
p++;
if (exp >= 100) {
*p += (char)(exp / 100);
exp %= 100;
}
p++;
if (exp >= 10) {
*p += (char)(exp / 10);
exp %= 10;
}
*++p += (char)exp;
}
return(buf);
}
#ifdef _MT
char * __cdecl _cftoe( double * pvalue, char * buf, int ndec, int caps )
{
struct _strflt retstrflt;
char resstr[21];
DOUBLE *pdvalue = (DOUBLE *)pvalue;
STRFLT pflt = &retstrflt;
_fltout2(*pdvalue, (struct _strflt *)&retstrflt,
(char *)resstr);
_fptostr(buf + (pflt->sign == '-') + (ndec > 0), ndec + 1, pflt);
_cftoe2(buf, ndec, caps, pflt, /* g_fmt = */ 0);
return( buf );
}
#else /* not _MT */
static char * _cftoe_g( double * pvalue, char * buf, int ndec, int caps )
{
char *res;
g_fmt = 1;
res = _cftoe(pvalue, buf, ndec, caps);
g_fmt = 0;
return (res);
}
#endif /* not _MT */
#ifdef _MT
static char * _cftof2( char * buf, int ndec, STRFLT pflt, char g_fmt )
#else
char * __cdecl _cftof( double * pvalue, char * buf, int ndec )
#endif
{
#ifndef _MT
STRFLT pflt;
DOUBLE *pdvalue = (DOUBLE *)pvalue;
#endif
char *p;
#ifdef _MT
int g_magnitude = pflt->decpt - 1;
#endif
/* first convert the value */
/* place the output in the users buffer and round. Save space for
* the minus sign now if it will be needed
*/
if (g_fmt) {
#ifndef _MT
pflt = g_pflt;
#endif
p = buf + (pflt->sign == '-');
if (g_magnitude == ndec) {
char *q = p + g_magnitude;
*q++ = '0';
*q = '\0';
/* allows for extra place-holding '0' in the exponent == precision
* case of the g format
*/
}
}
#ifndef _MT
else {
pflt = _fltout(*pdvalue);
_fptostr(buf+(pflt->sign == '-'), ndec + pflt->decpt, pflt);
}
#endif
/* now fix up the number to be in the correct f format */
p = buf;
/* put in negative sign, if necessary */
if (pflt->sign == '-')
*p++ = '-';
/* insert leading 0 for purely fractional values and position ourselves
* at the correct spot for inserting the decimal point
*/
if (pflt->decpt <= 0) {
_shift(p, 1);
*p++ = '0';
}
else
p += pflt->decpt;
/* put in decimal point if required and any zero padding needed */
if (ndec > 0) {
_shift(p, 1);
*p++ = *__decimal_point;
/* if the value is less than 1 then we may need to put 0's out in
* front of the first non-zero digit of the mantissa
*/
if (pflt->decpt < 0) {
if( g_fmt )
ndec = -pflt->decpt;
else
ndec = (ndec < -pflt->decpt ) ? ndec : -pflt->decpt;
_shift(p, ndec);
memset( p, '0', ndec);
}
}
return( buf);
}
/*
* Function name: _cftof
*
* Arguments: value - double * pointer
* buf - char * pointer
* ndec - int
*
* Description: _cftof converts the double pointed to by pvalue to a null
* terminated string of ASCII digits in the c language
* printf %f format, and returns a pointer to the result.
* This format has the form [-]ddddd.ddddd, where there will
* be ndec digits following the decimal point. If ndec <= 0,
* no decimal point will appear. The low order digit is
* rounded.
*
* Side Effects: the buffer 'buf' is assumed to have a minimum length
* of CVTBUFSIZE (defined in cvt.h) and the routines will
* not write over this size.
*
* Author: written R.K. Wyss, Microsoft, Sept. 9, 1983
*
* History:
*
*/
#ifdef _MT
char * __cdecl _cftof( double * pvalue, char * buf, int ndec )
{
struct _strflt retstrflt;
char resstr[21];
DOUBLE *pdvalue = (DOUBLE *)pvalue;
STRFLT pflt = &retstrflt;
_fltout2(*pdvalue, (struct _strflt *) &retstrflt,
(char *) resstr);
_fptostr(buf+(pflt->sign == '-'), ndec + pflt->decpt, pflt);
_cftof2(buf, ndec, pflt, /* g_fmt = */ 0);
return( buf );
}
#else /* not _MT */
static char * _cftof_g( double * pvalue, char * buf, int ndec )
{
char *res;
g_fmt = 1;
res = _cftof(pvalue, buf, ndec);
g_fmt = 0;
return (res);
}
#endif /* not _MT */
/*
* Function name: _cftog
*
* Arguments: value - double * pointer
* buf - char * pointer
* ndec - int
*
* Description: _cftog converts the double pointed to by pvalue to a null
* terminated string of ASCII digits in the c language
* printf %g format, and returns a pointer to the result.
* The form used depends on the value converted. The printf
* %e form will be used if the magnitude of valude is less
* than -4 or is greater than ndec, otherwise printf %f will
* be used. ndec always specifies the number of digits
* following the decimal point. The low order digit is
* appropriately rounded.
*
* Side Effects: the buffer 'buf' is assumed to have a minimum length
* of CVTBUFSIZE (defined in cvt.h) and the routines will
* not write over this size.
*
* Author: written R.K. Wyss, Microsoft, Sept. 9, 1983
*
* History:
*
*/
char * __cdecl _cftog( double * pvalue, char * buf, int ndec, int caps )
{
char *p;
DOUBLE *pdvalue = (DOUBLE *)pvalue;
#ifdef _MT
char g_round_expansion = 0;
STRFLT g_pflt;
int g_magnitude;
struct _strflt retstrflt;
char resstr[21];
/* first convert the number */
g_pflt = &retstrflt;
_fltout2(*pdvalue, (struct _strflt *)&retstrflt,
(char *)resstr);
#else /* not _MT */
/* first convert the number */
g_pflt = _fltout(*pdvalue);
#endif /* not _MT */
g_magnitude = g_pflt->decpt - 1;
p = buf + (g_pflt->sign == '-');
_fptostr(p, ndec, g_pflt);
g_round_expansion = (char)(g_magnitude < (g_pflt->decpt-1));
/* compute the magnitude of value */
g_magnitude = g_pflt->decpt - 1;
/* convert value to the c language g format */
if (g_magnitude < -4 || g_magnitude >= ndec){ /* use e format */
/* (g_round_expansion ==>
* extra digit will be overwritten by 'e+xxx')
*/
#ifdef _MT
return(_cftoe2(buf, ndec, caps, g_pflt, /* g_fmt = */ 1));
#else
return(_cftoe_g(pvalue, buf, ndec, caps));
#endif
}
else { /* use f format */
if (g_round_expansion) {
/* throw away extra final digit from expansion */
while (*p++);
*(p-2) = '\0';
}
#ifdef _MT
return(_cftof2(buf, ndec, g_pflt, /* g_fmt = */ 1));
#else
return(_cftof_g(pvalue, buf, ndec));
#endif
}
}
/***
*_cfltcvt(arg, buf, format, precision, caps) - convert floating-point output
*Purpose:
*
*Entry:
* arg = (double *) pointer to double-precision floating-point number
* buf = (char *) pointer to buffer into which to put the converted
* ASCII form of the number
* format = (int) 'e', 'f', or 'g'
* precision = (int) giving number of decimal places for %e and %f formats,
* and giving maximum number of significant digits for
* %g format
* caps = (int) flag indicating whether 'E' in exponent should be capatilized
* (for %E and %G formats only)
*
*Exit:
* returns : (void)
*
*Exceptions:
*******************************************************************************/
/*
* Function name: _cfltcvt
*
* Arguments: arg - double * pointer
* buf - char * pointer
* format - int
* ndec - int
* caps - int
*
* Description: _cfltcvt determines from the format, what routines to
* call to generate the correct floating point format
*
* Side Effects: none
*
* Author: Dave Weil, Jan 12, 1985
*/
void __cdecl _cfltcvt( double * arg, char * buffer, int format, int precision, int caps )
{
if (format == 'e' || format == 'E')
_cftoe(arg, buffer, precision, caps);
else if (format == 'f')
_cftof(arg, buffer, precision);
else
_cftog(arg, buffer, precision, caps);
}
/***
*_shift(s, dist) - shift a null-terminated string in memory (internal routine)
*Purpose:
* _shift is a helper routine that shifts a null-terminated string
* in memory, e.g., moves part of a buffer used for floating-point output
*
* modifies memory locations (s+dist) through (s+dist+strlen(s))
*
*Entry:
* s = (char *) pointer to string to move
* dist = (int) distance to move the string to the right (if negative, to left)
*
*Exit:
* returns : (void)
*
*Exceptions:
*******************************************************************************/
static void _CALLTYPE5 _shift( char *s, int dist )
{
if( dist )
memmove(s+dist, s, strlen(s)+1);
}