1 | #include <stdio.h>
|
---|
2 | #include <string.h>
|
---|
3 |
|
---|
4 | #include "pyconfig.h"
|
---|
5 |
|
---|
6 | /* comp.sources.misc strtod(), as posted in comp.lang.tcl,
|
---|
7 | with bugfix for "123000.0" and acceptance of space after 'e' sign nuked.
|
---|
8 |
|
---|
9 | ************************************************************
|
---|
10 | * YOU MUST EDIT THE MACHINE-DEPENDENT DEFINITIONS BELOW!!! *
|
---|
11 | ************************************************************
|
---|
12 | */
|
---|
13 |
|
---|
14 | /* File : stdtod.c (Modified version of str2dbl.c)
|
---|
15 | Author : Richard A. O'Keefe @ Quintus Computer Systems, Inc.
|
---|
16 | Updated: Tuesday August 2nd, 1988
|
---|
17 | Defines: double strtod (char *str, char**ptr)
|
---|
18 | */
|
---|
19 |
|
---|
20 | /* This is an implementation of the strtod() function described in the
|
---|
21 | System V manuals, with a different name to avoid linker problems.
|
---|
22 | All that str2dbl() does itself is check that the argument is well-formed
|
---|
23 | and is in range. It leaves the work of conversion to atof(), which is
|
---|
24 | assumed to exist and deliver correct results (if they can be represented).
|
---|
25 |
|
---|
26 | There are two reasons why this should be provided to the net:
|
---|
27 | (a) some UNIX systems do not yet have strtod(), or do not have it
|
---|
28 | available in the BSD "universe" (but they do have atof()).
|
---|
29 | (b) some of the UNIX systems that *do* have it get it wrong.
|
---|
30 | (some crash with large arguments, some assign the wrong *ptr value).
|
---|
31 | There is a reason why *we* are providing it: we need a correct version
|
---|
32 | of strtod(), and if we give this one away maybe someone will look for
|
---|
33 | mistakes in it and fix them for us (:-).
|
---|
34 | */
|
---|
35 |
|
---|
36 | /* The following constants are machine-specific. MD{MIN,MAX}EXPT are
|
---|
37 | integers and MD{MIN,MAX}FRAC are strings such that
|
---|
38 | 0.${MDMAXFRAC}e${MDMAXEXPT} is the largest representable double,
|
---|
39 | 0.${MDMINFRAC}e${MDMINEXPT} is the smallest representable +ve double
|
---|
40 | MD{MIN,MAX}FRAC must not have any trailing zeros.
|
---|
41 | The values here are for IEEE-754 64-bit floats.
|
---|
42 | It is not perfectly clear to me whether an IEEE infinity should be
|
---|
43 | returned for overflow, nor what a portable way of writing one is,
|
---|
44 | so HUGE is just 0.MAXFRAC*10**MAXEXPT (this seems still to be the
|
---|
45 | UNIX convention).
|
---|
46 |
|
---|
47 | I do know about <values.h>, but the whole point of this file is that
|
---|
48 | we can't always trust that stuff to be there or to be correct.
|
---|
49 | */
|
---|
50 | static int MDMINEXPT = -323;
|
---|
51 | static char MDMINFRAC[] = "494065645841246544";
|
---|
52 | static double ZERO = 0.0;
|
---|
53 |
|
---|
54 | static int MDMAXEXPT = 309;
|
---|
55 | static char MDMAXFRAC[] = "17976931348623157";
|
---|
56 | static double HUGE = 1.7976931348623157e308;
|
---|
57 |
|
---|
58 | extern double atof(const char *); /* Only called when result known to be ok */
|
---|
59 |
|
---|
60 | #ifdef HAVE_ERRNO_H
|
---|
61 | #include <errno.h>
|
---|
62 | #endif
|
---|
63 | extern int errno;
|
---|
64 |
|
---|
65 | double strtod(char *str, char **ptr)
|
---|
66 | {
|
---|
67 | int sign, scale, dotseen;
|
---|
68 | int esign, expt;
|
---|
69 | char *save;
|
---|
70 | register char *sp, *dp;
|
---|
71 | register int c;
|
---|
72 | char *buforg, *buflim;
|
---|
73 | char buffer[64]; /* 45-digit significant + */
|
---|
74 | /* 13-digit exponent */
|
---|
75 | sp = str;
|
---|
76 | while (*sp == ' ') sp++;
|
---|
77 | sign = 1;
|
---|
78 | if (*sp == '-') sign -= 2, sp++;
|
---|
79 | dotseen = 0, scale = 0;
|
---|
80 | dp = buffer;
|
---|
81 | *dp++ = '0'; *dp++ = '.';
|
---|
82 | buforg = dp, buflim = buffer+48;
|
---|
83 | for (save = sp; (c = *sp); sp++)
|
---|
84 | if (c == '.') {
|
---|
85 | if (dotseen) break;
|
---|
86 | dotseen++;
|
---|
87 | } else
|
---|
88 | if ((unsigned)(c-'0') > (unsigned)('9'-'0')) {
|
---|
89 | break;
|
---|
90 | } else
|
---|
91 | if (c == '0') {
|
---|
92 | if (dp != buforg) {
|
---|
93 | /* This is not the first digit, so we want to keep it */
|
---|
94 | if (dp < buflim) *dp++ = c;
|
---|
95 | if (!dotseen) scale++;
|
---|
96 | } else {
|
---|
97 | /* No non-zero digits seen yet */
|
---|
98 | /* If a . has been seen, scale must be adjusted */
|
---|
99 | if (dotseen) scale--;
|
---|
100 | }
|
---|
101 | } else {
|
---|
102 | /* This is a nonzero digit, so we want to keep it */
|
---|
103 | if (dp < buflim) *dp++ = c;
|
---|
104 | /* If it precedes a ., scale must be adjusted */
|
---|
105 | if (!dotseen) scale++;
|
---|
106 | }
|
---|
107 | if (sp == save) {
|
---|
108 | if (ptr) *ptr = str;
|
---|
109 | errno = EDOM; /* what should this be? */
|
---|
110 | return ZERO;
|
---|
111 | }
|
---|
112 |
|
---|
113 | while (dp > buforg && dp[-1] == '0') --dp;
|
---|
114 | if (dp == buforg) *dp++ = '0';
|
---|
115 | *dp = '\0';
|
---|
116 | /* Now the contents of buffer are
|
---|
117 | +--+--------+-+--------+
|
---|
118 | |0.|fraction|\|leftover|
|
---|
119 | +--+--------+-+--------+
|
---|
120 | ^dp points here
|
---|
121 | where fraction begins with 0 iff it is "0", and has at most
|
---|
122 | 45 digits in it, and leftover is at least 16 characters.
|
---|
123 | */
|
---|
124 | save = sp, expt = 0, esign = 1;
|
---|
125 | do {
|
---|
126 | c = *sp++;
|
---|
127 | if (c != 'e' && c != 'E') break;
|
---|
128 | c = *sp++;
|
---|
129 | if (c == '-') esign -= 2, c = *sp++; else
|
---|
130 | if (c == '+' /* || c == ' ' */ ) c = *sp++;
|
---|
131 | if ((unsigned)(c-'0') > (unsigned)('9'-'0')) break;
|
---|
132 | while (c == '0') c = *sp++;
|
---|
133 | for (; (unsigned)(c-'0') <= (unsigned)('9'-'0'); c = *sp++)
|
---|
134 | expt = expt*10 + c-'0';
|
---|
135 | if (esign < 0) expt = -expt;
|
---|
136 | save = sp-1;
|
---|
137 | } while (0);
|
---|
138 | if (ptr) *ptr = save;
|
---|
139 | expt += scale;
|
---|
140 | /* Now the number is sign*0.fraction*10**expt */
|
---|
141 | errno = ERANGE;
|
---|
142 | if (expt > MDMAXEXPT) {
|
---|
143 | return HUGE*sign;
|
---|
144 | } else
|
---|
145 | if (expt == MDMAXEXPT) {
|
---|
146 | if (strcmp(buforg, MDMAXFRAC) > 0) return HUGE*sign;
|
---|
147 | } else
|
---|
148 | if (expt < MDMINEXPT) {
|
---|
149 | return ZERO*sign;
|
---|
150 | } else
|
---|
151 | if (expt == MDMINEXPT) {
|
---|
152 | if (strcmp(buforg, MDMINFRAC) < 0) return ZERO*sign;
|
---|
153 | }
|
---|
154 | /* We have now established that the number can be */
|
---|
155 | /* represented without overflow or underflow */
|
---|
156 | (void) sprintf(dp, "E%d", expt);
|
---|
157 | errno = 0;
|
---|
158 | return atof(buffer)*sign;
|
---|
159 | }
|
---|