Changeset 391 for python/trunk/Python/strtod.c
- Timestamp:
- Mar 19, 2014, 11:31:01 PM (11 years ago)
- Location:
- python/trunk
- Files:
-
- 2 edited
Legend:
- Unmodified
- Added
- Removed
-
python/trunk
-
Property svn:mergeinfo
set to
/python/vendor/Python-2.7.6 merged eligible /python/vendor/current merged eligible
-
Property svn:mergeinfo
set to
-
python/trunk/Python/strtod.c
r2 r391 1 #include <stdio.h> 2 #include <string.h> 3 1 4 #include "pyconfig.h" 2 5 … … 15 18 */ 16 19 17 /* This is an implementation of the strtod() function described in the 20 /* This is an implementation of the strtod() function described in the 18 21 System V manuals, with a different name to avoid linker problems. 19 22 All that str2dbl() does itself is check that the argument is well-formed … … 23 26 There are two reasons why this should be provided to the net: 24 27 (a) some UNIX systems do not yet have strtod(), or do not have it 25 28 available in the BSD "universe" (but they do have atof()). 26 29 (b) some of the UNIX systems that *do* have it get it wrong. 27 30 (some crash with large arguments, some assign the wrong *ptr value). 28 31 There is a reason why *we* are providing it: we need a correct version 29 32 of strtod(), and if we give this one away maybe someone will look for 30 33 mistakes in it and fix them for us (:-). 31 34 */ 32 35 33 36 /* The following constants are machine-specific. MD{MIN,MAX}EXPT are 34 37 integers and MD{MIN,MAX}FRAC are strings such that 35 36 38 0.${MDMAXFRAC}e${MDMAXEXPT} is the largest representable double, 39 0.${MDMINFRAC}e${MDMINEXPT} is the smallest representable +ve double 37 40 MD{MIN,MAX}FRAC must not have any trailing zeros. 38 41 The values here are for IEEE-754 64-bit floats. … … 45 48 we can't always trust that stuff to be there or to be correct. 46 49 */ 47 static int MDMINEXPT= -323;48 static char MDMINFRAC[]= "494065645841246544";49 static double ZERO= 0.0;50 static int MDMINEXPT = -323; 51 static char MDMINFRAC[] = "494065645841246544"; 52 static double ZERO = 0.0; 50 53 51 static int MDMAXEXPT= 309;52 static char MDMAXFRAC[]= "17976931348623157";53 static double HUGE= 1.7976931348623157e308;54 static int MDMAXEXPT = 309; 55 static char MDMAXFRAC[] = "17976931348623157"; 56 static double HUGE = 1.7976931348623157e308; 54 57 55 extern double atof(const char *);/* Only called when result known to be ok */58 extern double atof(const char *); /* Only called when result known to be ok */ 56 59 57 60 #ifdef HAVE_ERRNO_H 58 61 #include <errno.h> 59 62 #endif 60 extern interrno;63 extern int errno; 61 64 62 65 double strtod(char *str, char **ptr) 63 66 { 64 65 66 67 68 69 70 char buffer[64];/* 45-digit significant + */71 72 73 74 75 76 77 dp = buffer; 78 79 80 for (save = sp; c = *sp; sp++)81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 errno = EDOM;/* what should this be? */107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 expt = expt*10 + c-'0'; 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 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; 156 159 }
Note:
See TracChangeset
for help on using the changeset viewer.