709 lines
17 KiB
C
709 lines
17 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);
|
||
|
}
|