Source file
src/strconv/atof.go
1
2
3
4
5 package strconv
6
7
8
9
10
11
12
13 import "math"
14
15 var optimize = true
16
17
18
19
20 func commonPrefixLenIgnoreCase(s, prefix string) int {
21 n := min(len(prefix), len(s))
22 for i := 0; i < n; i++ {
23 c := s[i]
24 if 'A' <= c && c <= 'Z' {
25 c += 'a' - 'A'
26 }
27 if c != prefix[i] {
28 return i
29 }
30 }
31 return n
32 }
33
34
35
36
37
38
39 func special(s string) (f float64, n int, ok bool) {
40 if len(s) == 0 {
41 return 0, 0, false
42 }
43 sign := 1
44 nsign := 0
45 switch s[0] {
46 case '+', '-':
47 if s[0] == '-' {
48 sign = -1
49 }
50 nsign = 1
51 s = s[1:]
52 fallthrough
53 case 'i', 'I':
54 n := commonPrefixLenIgnoreCase(s, "infinity")
55
56
57 if 3 < n && n < 8 {
58 n = 3
59 }
60 if n == 3 || n == 8 {
61 return math.Inf(sign), nsign + n, true
62 }
63 case 'n', 'N':
64 if commonPrefixLenIgnoreCase(s, "nan") == 3 {
65 return math.NaN(), 3, true
66 }
67 }
68 return 0, 0, false
69 }
70
71 func (b *decimal) set(s string) (ok bool) {
72 i := 0
73 b.neg = false
74 b.trunc = false
75
76
77 if i >= len(s) {
78 return
79 }
80 switch {
81 case s[i] == '+':
82 i++
83 case s[i] == '-':
84 b.neg = true
85 i++
86 }
87
88
89 sawdot := false
90 sawdigits := false
91 for ; i < len(s); i++ {
92 switch {
93 case s[i] == '_':
94
95 continue
96 case s[i] == '.':
97 if sawdot {
98 return
99 }
100 sawdot = true
101 b.dp = b.nd
102 continue
103
104 case '0' <= s[i] && s[i] <= '9':
105 sawdigits = true
106 if s[i] == '0' && b.nd == 0 {
107 b.dp--
108 continue
109 }
110 if b.nd < len(b.d) {
111 b.d[b.nd] = s[i]
112 b.nd++
113 } else if s[i] != '0' {
114 b.trunc = true
115 }
116 continue
117 }
118 break
119 }
120 if !sawdigits {
121 return
122 }
123 if !sawdot {
124 b.dp = b.nd
125 }
126
127
128
129
130
131
132 if i < len(s) && lower(s[i]) == 'e' {
133 i++
134 if i >= len(s) {
135 return
136 }
137 esign := 1
138 if s[i] == '+' {
139 i++
140 } else if s[i] == '-' {
141 i++
142 esign = -1
143 }
144 if i >= len(s) || s[i] < '0' || s[i] > '9' {
145 return
146 }
147 e := 0
148 for ; i < len(s) && ('0' <= s[i] && s[i] <= '9' || s[i] == '_'); i++ {
149 if s[i] == '_' {
150
151 continue
152 }
153 if e < 10000 {
154 e = e*10 + int(s[i]) - '0'
155 }
156 }
157 b.dp += e * esign
158 }
159
160 if i != len(s) {
161 return
162 }
163
164 ok = true
165 return
166 }
167
168
169
170
171
172 func readFloat(s string) (mantissa uint64, exp int, neg, trunc, hex bool, i int, ok bool) {
173 underscores := false
174
175
176 if i >= len(s) {
177 return
178 }
179 switch {
180 case s[i] == '+':
181 i++
182 case s[i] == '-':
183 neg = true
184 i++
185 }
186
187
188 base := uint64(10)
189 maxMantDigits := 19
190 expChar := byte('e')
191 if i+2 < len(s) && s[i] == '0' && lower(s[i+1]) == 'x' {
192 base = 16
193 maxMantDigits = 16
194 i += 2
195 expChar = 'p'
196 hex = true
197 }
198 sawdot := false
199 sawdigits := false
200 nd := 0
201 ndMant := 0
202 dp := 0
203 loop:
204 for ; i < len(s); i++ {
205 switch c := s[i]; true {
206 case c == '_':
207 underscores = true
208 continue
209
210 case c == '.':
211 if sawdot {
212 break loop
213 }
214 sawdot = true
215 dp = nd
216 continue
217
218 case '0' <= c && c <= '9':
219 sawdigits = true
220 if c == '0' && nd == 0 {
221 dp--
222 continue
223 }
224 nd++
225 if ndMant < maxMantDigits {
226 mantissa *= base
227 mantissa += uint64(c - '0')
228 ndMant++
229 } else if c != '0' {
230 trunc = true
231 }
232 continue
233
234 case base == 16 && 'a' <= lower(c) && lower(c) <= 'f':
235 sawdigits = true
236 nd++
237 if ndMant < maxMantDigits {
238 mantissa *= 16
239 mantissa += uint64(lower(c) - 'a' + 10)
240 ndMant++
241 } else {
242 trunc = true
243 }
244 continue
245 }
246 break
247 }
248 if !sawdigits {
249 return
250 }
251 if !sawdot {
252 dp = nd
253 }
254
255 if base == 16 {
256 dp *= 4
257 ndMant *= 4
258 }
259
260
261
262
263
264
265 if i < len(s) && lower(s[i]) == expChar {
266 i++
267 if i >= len(s) {
268 return
269 }
270 esign := 1
271 if s[i] == '+' {
272 i++
273 } else if s[i] == '-' {
274 i++
275 esign = -1
276 }
277 if i >= len(s) || s[i] < '0' || s[i] > '9' {
278 return
279 }
280 e := 0
281 for ; i < len(s) && ('0' <= s[i] && s[i] <= '9' || s[i] == '_'); i++ {
282 if s[i] == '_' {
283 underscores = true
284 continue
285 }
286 if e < 10000 {
287 e = e*10 + int(s[i]) - '0'
288 }
289 }
290 dp += e * esign
291 } else if base == 16 {
292
293 return
294 }
295
296 if mantissa != 0 {
297 exp = dp - ndMant
298 }
299
300 if underscores && !underscoreOK(s[:i]) {
301 return
302 }
303
304 ok = true
305 return
306 }
307
308
309 var powtab = []int{1, 3, 6, 9, 13, 16, 19, 23, 26}
310
311 func (d *decimal) floatBits(flt *floatInfo) (b uint64, overflow bool) {
312 var exp int
313 var mant uint64
314
315
316 if d.nd == 0 {
317 mant = 0
318 exp = flt.bias
319 goto out
320 }
321
322
323
324
325 if d.dp > 310 {
326 goto overflow
327 }
328 if d.dp < -330 {
329
330 mant = 0
331 exp = flt.bias
332 goto out
333 }
334
335
336 exp = 0
337 for d.dp > 0 {
338 var n int
339 if d.dp >= len(powtab) {
340 n = 27
341 } else {
342 n = powtab[d.dp]
343 }
344 d.Shift(-n)
345 exp += n
346 }
347 for d.dp < 0 || d.dp == 0 && d.d[0] < '5' {
348 var n int
349 if -d.dp >= len(powtab) {
350 n = 27
351 } else {
352 n = powtab[-d.dp]
353 }
354 d.Shift(n)
355 exp -= n
356 }
357
358
359 exp--
360
361
362
363
364 if exp < flt.bias+1 {
365 n := flt.bias + 1 - exp
366 d.Shift(-n)
367 exp += n
368 }
369
370 if exp-flt.bias >= 1<<flt.expbits-1 {
371 goto overflow
372 }
373
374
375 d.Shift(int(1 + flt.mantbits))
376 mant = d.RoundedInteger()
377
378
379 if mant == 2<<flt.mantbits {
380 mant >>= 1
381 exp++
382 if exp-flt.bias >= 1<<flt.expbits-1 {
383 goto overflow
384 }
385 }
386
387
388 if mant&(1<<flt.mantbits) == 0 {
389 exp = flt.bias
390 }
391 goto out
392
393 overflow:
394
395 mant = 0
396 exp = 1<<flt.expbits - 1 + flt.bias
397 overflow = true
398
399 out:
400
401 bits := mant & (uint64(1)<<flt.mantbits - 1)
402 bits |= uint64((exp-flt.bias)&(1<<flt.expbits-1)) << flt.mantbits
403 if d.neg {
404 bits |= 1 << flt.mantbits << flt.expbits
405 }
406 return bits, overflow
407 }
408
409
410 var float64pow10 = []float64{
411 1e0, 1e1, 1e2, 1e3, 1e4, 1e5, 1e6, 1e7, 1e8, 1e9,
412 1e10, 1e11, 1e12, 1e13, 1e14, 1e15, 1e16, 1e17, 1e18, 1e19,
413 1e20, 1e21, 1e22,
414 }
415 var float32pow10 = []float32{1e0, 1e1, 1e2, 1e3, 1e4, 1e5, 1e6, 1e7, 1e8, 1e9, 1e10}
416
417
418
419
420
421
422
423
424
425
426 func atof64exact(mantissa uint64, exp int, neg bool) (f float64, ok bool) {
427 if mantissa>>float64info.mantbits != 0 {
428 return
429 }
430 f = float64(mantissa)
431 if neg {
432 f = -f
433 }
434 switch {
435 case exp == 0:
436
437 return f, true
438
439
440 case exp > 0 && exp <= 15+22:
441
442
443 if exp > 22 {
444 f *= float64pow10[exp-22]
445 exp = 22
446 }
447 if f > 1e15 || f < -1e15 {
448
449 return
450 }
451 return f * float64pow10[exp], true
452 case exp < 0 && exp >= -22:
453 return f / float64pow10[-exp], true
454 }
455 return
456 }
457
458
459
460 func atof32exact(mantissa uint64, exp int, neg bool) (f float32, ok bool) {
461 if mantissa>>float32info.mantbits != 0 {
462 return
463 }
464 f = float32(mantissa)
465 if neg {
466 f = -f
467 }
468 switch {
469 case exp == 0:
470 return f, true
471
472
473 case exp > 0 && exp <= 7+10:
474
475
476 if exp > 10 {
477 f *= float32pow10[exp-10]
478 exp = 10
479 }
480 if f > 1e7 || f < -1e7 {
481
482 return
483 }
484 return f * float32pow10[exp], true
485 case exp < 0 && exp >= -10:
486 return f / float32pow10[-exp], true
487 }
488 return
489 }
490
491
492
493
494
495
496 func atofHex(s string, flt *floatInfo, mantissa uint64, exp int, neg, trunc bool) (float64, error) {
497 maxExp := 1<<flt.expbits + flt.bias - 2
498 minExp := flt.bias + 1
499 exp += int(flt.mantbits)
500
501
502
503
504
505
506
507 for mantissa != 0 && mantissa>>(flt.mantbits+2) == 0 {
508 mantissa <<= 1
509 exp--
510 }
511 if trunc {
512 mantissa |= 1
513 }
514 for mantissa>>(1+flt.mantbits+2) != 0 {
515 mantissa = mantissa>>1 | mantissa&1
516 exp++
517 }
518
519
520
521
522 for mantissa > 1 && exp < minExp-2 {
523 mantissa = mantissa>>1 | mantissa&1
524 exp++
525 }
526
527
528 round := mantissa & 3
529 mantissa >>= 2
530 round |= mantissa & 1
531 exp += 2
532 if round == 3 {
533 mantissa++
534 if mantissa == 1<<(1+flt.mantbits) {
535 mantissa >>= 1
536 exp++
537 }
538 }
539
540 if mantissa>>flt.mantbits == 0 {
541 exp = flt.bias
542 }
543 var err error
544 if exp > maxExp {
545 mantissa = 1 << flt.mantbits
546 exp = maxExp + 1
547 err = rangeError(fnParseFloat, s)
548 }
549
550 bits := mantissa & (1<<flt.mantbits - 1)
551 bits |= uint64((exp-flt.bias)&(1<<flt.expbits-1)) << flt.mantbits
552 if neg {
553 bits |= 1 << flt.mantbits << flt.expbits
554 }
555 if flt == &float32info {
556 return float64(math.Float32frombits(uint32(bits))), err
557 }
558 return math.Float64frombits(bits), err
559 }
560
561 const fnParseFloat = "ParseFloat"
562
563 func atof32(s string) (f float32, n int, err error) {
564 if val, n, ok := special(s); ok {
565 return float32(val), n, nil
566 }
567
568 mantissa, exp, neg, trunc, hex, n, ok := readFloat(s)
569 if !ok {
570 return 0, n, syntaxError(fnParseFloat, s)
571 }
572
573 if hex {
574 f, err := atofHex(s[:n], &float32info, mantissa, exp, neg, trunc)
575 return float32(f), n, err
576 }
577
578 if optimize {
579
580
581 if !trunc {
582 if f, ok := atof32exact(mantissa, exp, neg); ok {
583 return f, n, nil
584 }
585 }
586 f, ok := eiselLemire32(mantissa, exp, neg)
587 if ok {
588 if !trunc {
589 return f, n, nil
590 }
591
592
593
594 fUp, ok := eiselLemire32(mantissa+1, exp, neg)
595 if ok && f == fUp {
596 return f, n, nil
597 }
598 }
599 }
600
601
602 var d decimal
603 if !d.set(s[:n]) {
604 return 0, n, syntaxError(fnParseFloat, s)
605 }
606 b, ovf := d.floatBits(&float32info)
607 f = math.Float32frombits(uint32(b))
608 if ovf {
609 err = rangeError(fnParseFloat, s)
610 }
611 return f, n, err
612 }
613
614 func atof64(s string) (f float64, n int, err error) {
615 if val, n, ok := special(s); ok {
616 return val, n, nil
617 }
618
619 mantissa, exp, neg, trunc, hex, n, ok := readFloat(s)
620 if !ok {
621 return 0, n, syntaxError(fnParseFloat, s)
622 }
623
624 if hex {
625 f, err := atofHex(s[:n], &float64info, mantissa, exp, neg, trunc)
626 return f, n, err
627 }
628
629 if optimize {
630
631
632 if !trunc {
633 if f, ok := atof64exact(mantissa, exp, neg); ok {
634 return f, n, nil
635 }
636 }
637 f, ok := eiselLemire64(mantissa, exp, neg)
638 if ok {
639 if !trunc {
640 return f, n, nil
641 }
642
643
644
645 fUp, ok := eiselLemire64(mantissa+1, exp, neg)
646 if ok && f == fUp {
647 return f, n, nil
648 }
649 }
650 }
651
652
653 var d decimal
654 if !d.set(s[:n]) {
655 return 0, n, syntaxError(fnParseFloat, s)
656 }
657 b, ovf := d.floatBits(&float64info)
658 f = math.Float64frombits(b)
659 if ovf {
660 err = rangeError(fnParseFloat, s)
661 }
662 return f, n, err
663 }
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692 func ParseFloat(s string, bitSize int) (float64, error) {
693 f, n, err := parseFloatPrefix(s, bitSize)
694 if n != len(s) && (err == nil || err.(*NumError).Err != ErrSyntax) {
695 return 0, syntaxError(fnParseFloat, s)
696 }
697 return f, err
698 }
699
700 func parseFloatPrefix(s string, bitSize int) (float64, int, error) {
701 if bitSize == 32 {
702 f, n, err := atof32(s)
703 return float64(f), n, err
704 }
705 return atof64(s)
706 }
707
View as plain text