strtod.c
上传用户:baixin
上传日期:2008-03-13
资源大小:4795k
文件大小:8k
- /* strtod.c function for stdlib */
- /* Copyright 1992-1995 Wind River Systems, Inc. */
- /*
- modification history
- --------------------
- 01e,11feb95,jdi doc fix.
- 01d,16aug93,dvs fixed value of endptr for strings that start with d, D, e, or E
- (SPR #2229)
- 01c,08feb93,jdi documentation cleanup for 5.1.
- 01b,20sep92,smb documentation additions.
- 01a,19jul92,smb written and documented
- */
- /*
- DESCRIPTION
- INCLUDE FILES: stdlib.h, math.h, assert.h, arrno.h
- SEE ALSO: American National Standard X3.159-1989
- NOMANUAL
- */
- #include "vxWorks.h"
- #include "stdlib.h"
- #include "math.h"
- #include "errno.h"
- #define Ise(c) ((c == 'e') || (c == 'E') || (c == 'd') || (c == 'D'))
- #define Isdigit(c) ((c <= '9') && (c >= '0'))
- #define Isspace(c) ((c == ' ') || (c == 't') || (c=='n') || (c=='v')
- || (c == 'r') || (c == 'f'))
- #define Issign(c) ((c == '-') || (c == '+'))
- #define Val(c) ((c - '0'))
- #define MAXE 308
- #define MINE (-308)
- static double powtab[] = {1.0,
- 10.0,
- 100.0,
- 1000.0,
- 10000.0};
- /* flags */
- #define SIGN 0x01
- #define ESIGN 0x02
- #define DECP 0x04
- /* locals */
- int __ten_mul (double *acc, int digit);
- double __adjust (double *acc, int dexp, int sign);
- double __exp10 (uint_t x);
- /*****************************************************************************
- *
- * strtod - convert the initial portion of a string to a double (ANSI)
- *
- * This routine converts the initial portion of a specified string <s> to a
- * double. First, it decomposes the input string into three parts: an initial,
- * possibly empty, sequence of white-space characters (as specified by the
- * isspace() function); a subject sequence resembling a floating-point
- * constant; and a final string of one or more unrecognized characters,
- * including the terminating null character of the input string. Then, it
- * attempts to convert the subject sequence to a floating-point number, and
- * returns the result.
- *
- * The expected form of the subject sequence is an optional plus or minus
- * decimal-point character, then an optional exponent part but no floating
- * suffix. The subject sequence is defined as the longest initial
- * subsequence of the input string, starting with the first non-white-space
- * character, that is of the expected form. The subject sequence contains
- * no characters if the input string is empty or consists entirely of
- * white space, or if the first non-white-space character is other than a
- * sign, a digit, or a decimal-point character.
- *
- * If the subject sequence has the expected form, the sequence of characters
- * starting with the first digit or the decimal-point character (whichever
- * occurs first) is interpreted as a floating constant, except that the
- * decimal-point character is used in place of a period, and that if neither
- * an exponent part nor a decimal-point character appears, a decimal point is
- * assumed to follow the last digit in the string. If the subject sequence
- * begins with a minus sign, the value resulting form the conversion is negated.
- * A pointer to the final string is stored in the object pointed to by <endptr>,
- * provided that <endptr> is not a null pointer.
- *
- * In other than the "C" locale, additional implementation-defined subject
- * sequence forms may be accepted. VxWorks supports only the "C" locale.
- *
- * If the subject sequence is empty or does not have the expected form, no
- * conversion is performed; the value of <s> is stored in the object pointed
- * to by <endptr>, provided that <endptr> is not a null pointer.
- *
- * INCLUDE FILES: stdlib.h
- *
- * RETURNS:
- * The converted value, if any. If no conversion could be performed, it
- * returns zero. If the correct value is outside the range of representable
- * values, it returns plus or minus HUGE_VAL (according to the sign of the
- * value), and stores the value of the macro ERANGE in `errno'. If the
- * correct value would cause underflow, it returns zero and stores the value
- * of the macro ERANGE in `errno'.
- *
- */
- double strtod
- (
- const char * s, /* string to convert */
- char ** endptr /* ptr to final string */
- )
- {
- /* Note that the point positioning is locale dependant */
- const char *start = s;
- double accum = 0.0;
- int flags = 0;
- int texp = 0;
- int e = 0;
- BOOL conv_done = FALSE;
- while (Isspace (*s)) s++;
- if (*s == EOS)
- { /* just leading spaces */
- if (endptr != NULL)
- *endptr = CHAR_FROM_CONST (start);
- return (0.0);
- }
- if (Issign (*s))
- {
- if (*s == '-') flags = SIGN;
- if (*++s == EOS)
- { /* "+|-" : should be an error ? */
- if (endptr != NULL)
- *endptr = CHAR_FROM_CONST (start);
- return (0.0);
- }
- }
- /* added code to fix problem with leading e, E, d, or D */
- if ( !Isdigit (*s) && (*s != '.'))
- {
- if (endptr != NULL)
- *endptr = CHAR_FROM_CONST (start);
- return (0.0);
- }
- for ( ; (Isdigit (*s) || (*s == '.')); s++)
- {
- conv_done = TRUE;
- if (*s == '.')
- flags |= DECP;
- else
- {
- if ( __ten_mul (&accum, Val(*s)) )
- texp++;
- if (flags & DECP)
- texp--;
- }
- }
- if (Ise (*s))
- {
- conv_done = TRUE;
- if (*++s != EOS) /* skip e|E|d|D */
- { /* ! ([s]xxx[.[yyy]]e) */
- while (Isspace (*s)) s++; /* Ansi allows spaces after e */
- if (*s != EOS)
- { /* ! ([s]xxx[.[yyy]]e[space]) */
- if (Issign (*s))
- if (*s++ == '-') flags |= ESIGN;
- if (*s != EOS)
- /* ! ([s]xxx[.[yyy]]e[s]) -- error?? */
- {
- for(; Isdigit (*s); s++)
- /* prevent from grossly overflowing */
- if (e < MAXE)
- e = e*10 + Val (*s);
- /* dont care what comes after this */
- if (flags & ESIGN)
- texp -= e;
- else
- texp += e;
- }
- }
- }
- }
- if (endptr != NULL)
- *endptr = CHAR_FROM_CONST ((conv_done) ? s : start);
- return (__adjust (&accum, (int) texp, (int) (flags & SIGN)));
- }
- /*******************************************************************************
- *
- * __ten_mul -
- *
- * multiply 64 bit accumulator by 10 and add digit.
- * The KA/CA way to do this should be to use
- * a 64-bit integer internally and use "adjust" to
- * convert it to float at the end of processing.
- *
- * AUTHOR:
- * Taken from cygnus.
- *
- * RETURNS:
- * NOMANUAL
- */
- LOCAL int __ten_mul
- (
- double *acc,
- int digit
- )
- {
- *acc *= 10;
- *acc += digit;
- return (0); /* no overflow */
- }
- /*******************************************************************************
- *
- * __adjust -
- *
- * return (*acc) scaled by 10**dexp.
- *
- * AUTHOR:
- * Taken from cygnus.
- *
- * RETURNS:
- * NOMANUAL
- */
- LOCAL double __adjust
- (
- double *acc, /* *acc the 64 bit accumulator */
- int dexp, /* dexp decimal exponent */
- int sign /* sign sign flag */
- )
- {
- double r;
- if (dexp > MAXE)
- {
- errno = ERANGE;
- return (sign) ? -HUGE_VAL : HUGE_VAL;
- }
- else if (dexp < MINE)
- {
- errno = ERANGE;
- return 0.0;
- }
- r = *acc;
- if (sign)
- r = -r;
- if (dexp==0)
- return r;
- if (dexp < 0)
- return r / __exp10 (abs (dexp));
- else
- return r * __exp10 (dexp);
- }
- /*******************************************************************************
- *
- * __exp10 -
- *
- * compute 10**x by successive squaring.
- *
- * AUTHOR:
- * Taken from cygnus.
- *
- * RETURNS:
- * NOMANUAL
- */
- double __exp10
- (
- uint_t x
- )
- {
- if (x < (sizeof (powtab) / sizeof (double)))
- return (powtab [x]);
- else if (x & 1)
- return (10.0 * __exp10 (x-1));
- else
- return (__exp10 (x/2) * __exp10 (x/2));
- }