...
Source file src/pkg/math/trig_reduce.go
1
2
3
4
5 package math
6
7 import (
8 "math/bits"
9 )
10
11
12
13
14 const reduceThreshold = (1 << 52) / (4 / Pi)
15
16
17
18
19
20
21
22
23 func trigReduce(x float64) (j uint64, z float64) {
24 const PI4 = Pi / 4
25 if x < PI4 {
26 return 0, x
27 }
28
29
30 ix := Float64bits(x)
31 exp := int(ix>>shift&mask) - bias - shift
32 ix &^= mask << shift
33 ix |= 1 << shift
34
35
36
37 digit, bitshift := uint(exp+61)/64, uint(exp+61)%64
38 z0 := (mPi4[digit] << bitshift) | (mPi4[digit+1] >> (64 - bitshift))
39 z1 := (mPi4[digit+1] << bitshift) | (mPi4[digit+2] >> (64 - bitshift))
40 z2 := (mPi4[digit+2] << bitshift) | (mPi4[digit+3] >> (64 - bitshift))
41
42 z2hi, _ := bits.Mul64(z2, ix)
43 z1hi, z1lo := bits.Mul64(z1, ix)
44 z0lo := z0 * ix
45 lo, c := bits.Add64(z1lo, z2hi, 0)
46 hi, _ := bits.Add64(z0lo, z1hi, c)
47
48 j = hi >> 61
49
50 hi = hi<<3 | lo>>61
51 lz := uint(bits.LeadingZeros64(hi))
52 e := uint64(bias - (lz + 1))
53
54 hi = (hi << (lz + 1)) | (lo >> (64 - (lz + 1)))
55 hi >>= 64 - shift
56
57 hi |= e << shift
58 z = Float64frombits(hi)
59
60 if j&1 == 1 {
61 j++
62 j &= 7
63 z--
64 }
65
66 return j, z * PI4
67 }
68
69
70
71
72
73 var mPi4 = [...]uint64{
74 0x0000000000000001,
75 0x45f306dc9c882a53,
76 0xf84eafa3ea69bb81,
77 0xb6c52b3278872083,
78 0xfca2c757bd778ac3,
79 0x6e48dc74849ba5c0,
80 0x0c925dd413a32439,
81 0xfc3bd63962534e7d,
82 0xd1046bea5d768909,
83 0xd338e04d68befc82,
84 0x7323ac7306a673e9,
85 0x3908bf177bf25076,
86 0x3ff12fffbc0b301f,
87 0xde5e2316b414da3e,
88 0xda6cfd9e4f96136e,
89 0x9e8c7ecd3cbfd45a,
90 0xea4f758fd7cbe2f6,
91 0x7a0e73ef14a525d4,
92 0xd7f6bf623f1aba10,
93 0xac06608df8f6d757,
94 }
95
View as plain text