Commit 5bb89eb0 authored by Robert Griesemer's avatar Robert Griesemer

cmd/internal/gc: use big.Float to represent Mpflt bits

All multi-precision arithmetic is now based on math/big.

- passes all.bash
- added test cases for fixed bugs

Fixes #7740.
Fixes #6866.

Change-Id: I67268b91766970ced3b928260053ccdce8753d58
Reviewed-on: https://go-review.googlesource.com/7912Reviewed-by: default avatarRuss Cox <rsc@golang.org>
parent 4da157a7
...@@ -485,8 +485,8 @@ func typeinit() { ...@@ -485,8 +485,8 @@ func typeinit() {
okforarith[i] = true okforarith[i] = true
okforconst[i] = true okforconst[i] = true
issimple[i] = true issimple[i] = true
minfltval[i] = new(Mpflt) minfltval[i] = newMpflt()
maxfltval[i] = new(Mpflt) maxfltval[i] = newMpflt()
} }
if Iscomplex[i] { if Iscomplex[i] {
......
...@@ -23,8 +23,10 @@ func truncfltlit(oldv *Mpflt, t *Type) *Mpflt { ...@@ -23,8 +23,10 @@ func truncfltlit(oldv *Mpflt, t *Type) *Mpflt {
v.U.Fval = oldv v.U.Fval = oldv
overflow(v, t) overflow(v, t)
fv := new(Mpflt) fv := newMpflt()
*fv = *oldv
// *fv = *oldv
mpmovefltflt(fv, oldv)
// convert large precision literal floating // convert large precision literal floating
// into limited precision (float64 or float32) // into limited precision (float64 or float32)
...@@ -276,7 +278,7 @@ func copyval(v Val) Val { ...@@ -276,7 +278,7 @@ func copyval(v Val) Val {
v.U.Xval = i v.U.Xval = i
case CTFLT: case CTFLT:
f := new(Mpflt) f := newMpflt()
mpmovefltflt(f, v.U.Fval) mpmovefltflt(f, v.U.Fval)
v.U.Fval = f v.U.Fval = f
...@@ -313,13 +315,13 @@ func tocplx(v Val) Val { ...@@ -313,13 +315,13 @@ func tocplx(v Val) Val {
func toflt(v Val) Val { func toflt(v Val) Val {
switch v.Ctype { switch v.Ctype {
case CTINT, CTRUNE: case CTINT, CTRUNE:
f := new(Mpflt) f := newMpflt()
Mpmovefixflt(f, v.U.Xval) Mpmovefixflt(f, v.U.Xval)
v.Ctype = CTFLT v.Ctype = CTFLT
v.U.Fval = f v.U.Fval = f
case CTCPLX: case CTCPLX:
f := new(Mpflt) f := newMpflt()
mpmovefltflt(f, &v.U.Cval.Real) mpmovefltflt(f, &v.U.Cval.Real)
if mpcmpfltc(&v.U.Cval.Imag, 0) != 0 { if mpcmpfltc(&v.U.Cval.Imag, 0) != 0 {
Yyerror("constant %v%vi truncated to real", Fconv(&v.U.Cval.Real, obj.FmtSharp), Fconv(&v.U.Cval.Imag, obj.FmtSharp|obj.FmtSign)) Yyerror("constant %v%vi truncated to real", Fconv(&v.U.Cval.Real, obj.FmtSharp), Fconv(&v.U.Cval.Imag, obj.FmtSharp|obj.FmtSign))
......
...@@ -57,12 +57,9 @@ const ( ...@@ -57,12 +57,9 @@ const (
) )
const ( const (
Mpscale = 29 // safely smaller than bits in a long // TODO(gri) replace these with a single precision constant.
Mpprec = 16 // Mpscale*Mpprec is max number of bits Mpscale = 29 // safely smaller than bits in a long
Mpnorm = Mpprec - 1 // significant words in a normalized float Mpprec = 16 // Mpscale*Mpprec is max number of bits
Mpbase = 1 << Mpscale
Mpsign = Mpbase >> 1
Mpmask = Mpbase - 1
Mpdebug = 0 Mpdebug = 0
) )
...@@ -72,19 +69,12 @@ type Mpint struct { ...@@ -72,19 +69,12 @@ type Mpint struct {
Ovf bool // set if Val overflowed compiler limit (sticky) Ovf bool // set if Val overflowed compiler limit (sticky)
} }
// Mpfix is the original (old) representation of an integer constant. // Mpflt represents a floating-point constant.
// Still needed for Mpflt.
type Mpfix struct {
A [Mpprec]int
Neg uint8
Ovf uint8
}
type Mpflt struct { type Mpflt struct {
Val Mpfix Val big.Float
Exp int16
} }
// Mpcplx represents a complex constant.
type Mpcplx struct { type Mpcplx struct {
Real Mpflt Real Mpflt
Imag Mpflt Imag Mpflt
......
...@@ -1444,7 +1444,7 @@ casei: ...@@ -1444,7 +1444,7 @@ casei:
yylval.val.U.Cval = new(Mpcplx) yylval.val.U.Cval = new(Mpcplx)
Mpmovecflt(&yylval.val.U.Cval.Real, 0.0) Mpmovecflt(&yylval.val.U.Cval.Real, 0.0)
mpatoflt(&yylval.val.U.Cval.Imag, str) mpatoflt(&yylval.val.U.Cval.Imag, str)
if yylval.val.U.Cval.Imag.Val.Ovf != 0 { if yylval.val.U.Cval.Imag.Val.IsInf() {
Yyerror("overflow in imaginary constant") Yyerror("overflow in imaginary constant")
Mpmovecflt(&yylval.val.U.Cval.Real, 0.0) Mpmovecflt(&yylval.val.U.Cval.Real, 0.0)
} }
...@@ -1461,9 +1461,9 @@ caseout: ...@@ -1461,9 +1461,9 @@ caseout:
ungetc(c) ungetc(c)
str = lexbuf.String() str = lexbuf.String()
yylval.val.U.Fval = new(Mpflt) yylval.val.U.Fval = newMpflt()
mpatoflt(yylval.val.U.Fval, str) mpatoflt(yylval.val.U.Fval, str)
if yylval.val.U.Fval.Val.Ovf != 0 { if yylval.val.U.Fval.Val.IsInf() {
Yyerror("overflow in float constant") Yyerror("overflow in float constant")
Mpmovecflt(yylval.val.U.Fval, 0.0) Mpmovecflt(yylval.val.U.Fval, 0.0)
} }
......
This diff is collapsed.
...@@ -4,151 +4,7 @@ ...@@ -4,151 +4,7 @@
package gc package gc
// // shift left by s (or right by -s)
// return the significant
// words of the argument
//
func mplen(a *Mpfix) int {
n := -1
for i := 0; i < Mpprec; i++ {
if a.A[i] != 0 {
n = i
}
}
return n + 1
}
//
// left shift mpint by one
// ignores sign
//
func mplsh(a *Mpfix, quiet int) {
var x int
c := 0
for i := 0; i < Mpprec; i++ {
x = (a.A[i] << 1) + c
c = 0
if x >= Mpbase {
x -= Mpbase
c = 1
}
a.A[i] = x
}
a.Ovf = uint8(c)
if a.Ovf != 0 && quiet == 0 {
Yyerror("constant shift overflow")
}
}
//
// left shift mpint by Mpscale
// ignores sign
//
func mplshw(a *Mpfix, quiet int) {
i := Mpprec - 1
if a.A[i] != 0 {
a.Ovf = 1
if quiet == 0 {
Yyerror("constant shift overflow")
}
}
for ; i > 0; i-- {
a.A[i] = a.A[i-1]
}
a.A[i] = 0
}
//
// right shift mpint by one
// ignores sign and overflow
//
func mprsh(a *Mpfix) {
var x int
c := 0
lo := a.A[0] & 1
for i := Mpprec - 1; i >= 0; i-- {
x = a.A[i]
a.A[i] = (x + c) >> 1
c = 0
if x&1 != 0 {
c = Mpbase
}
}
if a.Neg != 0 && lo != 0 {
mpaddcfix(a, -1)
}
}
//
// right shift mpint by Mpscale
// ignores sign and overflow
//
func mprshw(a *Mpfix) {
var i int
lo := a.A[0]
for i = 0; i < Mpprec-1; i++ {
a.A[i] = a.A[i+1]
}
a.A[i] = 0
if a.Neg != 0 && lo != 0 {
mpaddcfix(a, -1)
}
}
//
// return the sign of (abs(a)-abs(b))
//
func mpcmp(a *Mpfix, b *Mpfix) int {
if a.Ovf != 0 || b.Ovf != 0 {
if nsavederrors+nerrors == 0 {
Yyerror("ovf in cmp")
}
return 0
}
var x int
for i := Mpprec - 1; i >= 0; i-- {
x = a.A[i] - b.A[i]
if x > 0 {
return +1
}
if x < 0 {
return -1
}
}
return 0
}
//
// negate a
// ignore sign and ovf
//
func mpneg(a *Mpfix) {
var x int
c := 0
for i := 0; i < Mpprec; i++ {
x = -a.A[i] - c
c = 0
if x < 0 {
x += Mpbase
c = 1
}
a.A[i] = x
}
}
func Mpshiftfix(a *Mpint, s int) { func Mpshiftfix(a *Mpint, s int) {
switch { switch {
case s > 0: case s > 0:
...@@ -162,32 +18,6 @@ func Mpshiftfix(a *Mpint, s int) { ...@@ -162,32 +18,6 @@ func Mpshiftfix(a *Mpint, s int) {
} }
} }
// shift left by s (or right by -s)
func _Mpshiftfix(a *Mpfix, s int) {
if s >= 0 {
for s >= Mpscale {
mplshw(a, 0)
s -= Mpscale
}
for s > 0 {
mplsh(a, 0)
s--
}
} else {
s = -s
for s >= Mpscale {
mprshw(a)
s -= Mpscale
}
for s > 0 {
mprsh(a)
s--
}
}
}
/// implements fix arithmetic /// implements fix arithmetic
func mpsetovf(a *Mpint) { func mpsetovf(a *Mpint) {
...@@ -221,73 +51,6 @@ func mpaddfixfix(a, b *Mpint, quiet int) { ...@@ -221,73 +51,6 @@ func mpaddfixfix(a, b *Mpint, quiet int) {
} }
} }
func _mpaddfixfix(a *Mpfix, b *Mpfix, quiet int) {
if a.Ovf != 0 || b.Ovf != 0 {
if nsavederrors+nerrors == 0 {
Yyerror("ovf in mpaddxx")
}
a.Ovf = 1
return
}
c := 0
if a.Neg != b.Neg {
// perform a-b
switch mpcmp(a, b) {
case 0:
_Mpmovecfix(a, 0)
case 1:
var x int
for i := 0; i < Mpprec; i++ {
x = a.A[i] - b.A[i] - c
c = 0
if x < 0 {
x += Mpbase
c = 1
}
a.A[i] = x
}
case -1:
a.Neg ^= 1
var x int
for i := 0; i < Mpprec; i++ {
x = b.A[i] - a.A[i] - c
c = 0
if x < 0 {
x += Mpbase
c = 1
}
a.A[i] = x
}
}
return
}
// perform a+b
var x int
for i := 0; i < Mpprec; i++ {
x = a.A[i] + b.A[i] + c
c = 0
if x >= Mpbase {
x -= Mpbase
c = 1
}
a.A[i] = x
}
a.Ovf = uint8(c)
if a.Ovf != 0 && quiet == 0 {
Yyerror("constant addition overflow")
}
return
}
func mpmulfixfix(a, b *Mpint) { func mpmulfixfix(a, b *Mpint) {
if a.Ovf || b.Ovf { if a.Ovf || b.Ovf {
if nsavederrors+nerrors == 0 { if nsavederrors+nerrors == 0 {
...@@ -304,110 +67,6 @@ func mpmulfixfix(a, b *Mpint) { ...@@ -304,110 +67,6 @@ func mpmulfixfix(a, b *Mpint) {
} }
} }
func _mpmulfixfix(a *Mpfix, b *Mpfix) {
if a.Ovf != 0 || b.Ovf != 0 {
if nsavederrors+nerrors == 0 {
Yyerror("ovf in mpmulfixfix")
}
a.Ovf = 1
return
}
// pick the smaller
// to test for bits
na := mplen(a)
nb := mplen(b)
var s Mpfix
var c *Mpfix
if na > nb {
_mpmovefixfix(&s, a)
c = b
na = nb
} else {
_mpmovefixfix(&s, b)
c = a
}
s.Neg = 0
var q Mpfix
_Mpmovecfix(&q, 0)
var j int
var x int
for i := 0; i < na; i++ {
x = c.A[i]
for j = 0; j < Mpscale; j++ {
if x&1 != 0 {
if s.Ovf != 0 {
q.Ovf = 1
goto out
}
_mpaddfixfix(&q, &s, 1)
if q.Ovf != 0 {
goto out
}
}
mplsh(&s, 1)
x >>= 1
}
}
out:
q.Neg = a.Neg ^ b.Neg
_mpmovefixfix(a, &q)
if a.Ovf != 0 {
Yyerror("constant multiplication overflow")
}
}
func mpmulfract(a *Mpfix, b *Mpfix) {
if a.Ovf != 0 || b.Ovf != 0 {
if nsavederrors+nerrors == 0 {
Yyerror("ovf in mpmulflt")
}
a.Ovf = 1
return
}
var s Mpfix
_mpmovefixfix(&s, b)
s.Neg = 0
var q Mpfix
_Mpmovecfix(&q, 0)
i := Mpprec - 1
x := a.A[i]
if x != 0 {
Yyerror("mpmulfract not normal")
}
var j int
for i--; i >= 0; i-- {
x = a.A[i]
if x == 0 {
mprshw(&s)
continue
}
for j = 0; j < Mpscale; j++ {
x <<= 1
if x&Mpbase != 0 {
_mpaddfixfix(&q, &s, 1)
}
mprsh(&s)
}
}
q.Neg = a.Neg ^ b.Neg
_mpmovefixfix(a, &q)
if a.Ovf != 0 {
Yyerror("constant multiplication overflow")
}
}
func mporfixfix(a, b *Mpint) { func mporfixfix(a, b *Mpint) {
if a.Ovf || b.Ovf { if a.Ovf || b.Ovf {
if nsavederrors+nerrors == 0 { if nsavederrors+nerrors == 0 {
...@@ -502,10 +161,6 @@ func mpnegfix(a *Mpint) { ...@@ -502,10 +161,6 @@ func mpnegfix(a *Mpint) {
a.Val.Neg(&a.Val) a.Val.Neg(&a.Val)
} }
func _mpnegfix(a *Mpfix) {
a.Neg ^= 1
}
func Mpgetfix(a *Mpint) int64 { func Mpgetfix(a *Mpint) int64 {
if a.Ovf { if a.Ovf {
if nsavederrors+nerrors == 0 { if nsavederrors+nerrors == 0 {
...@@ -517,148 +172,6 @@ func Mpgetfix(a *Mpint) int64 { ...@@ -517,148 +172,6 @@ func Mpgetfix(a *Mpint) int64 {
return a.Val.Int64() return a.Val.Int64()
} }
func _Mpgetfix(a *Mpfix) int64 {
if a.Ovf != 0 {
if nsavederrors+nerrors == 0 {
Yyerror("constant overflow")
}
return 0
}
v := int64(uint64(a.A[0]))
v |= int64(uint64(a.A[1]) << Mpscale)
v |= int64(uint64(a.A[2]) << (Mpscale + Mpscale))
if a.Neg != 0 {
v = int64(-uint64(v))
}
return v
}
func Mpmovecfix(a *Mpint, c int64) { func Mpmovecfix(a *Mpint, c int64) {
a.Val.SetInt64(c) a.Val.SetInt64(c)
} }
func _Mpmovecfix(a *Mpfix, c int64) {
a.Neg = 0
a.Ovf = 0
x := c
if x < 0 {
a.Neg = 1
x = int64(-uint64(x))
}
for i := 0; i < Mpprec; i++ {
a.A[i] = int(x & Mpmask)
x >>= Mpscale
}
}
func mpdivmodfixfix(q *Mpfix, r *Mpfix, n *Mpfix, d *Mpfix) {
var i int
ns := int(n.Neg)
ds := int(d.Neg)
n.Neg = 0
d.Neg = 0
_mpmovefixfix(r, n)
_Mpmovecfix(q, 0)
// shift denominator until it
// is larger than numerator
for i = 0; i < Mpprec*Mpscale; i++ {
if mpcmp(d, r) > 0 {
break
}
mplsh(d, 1)
}
// if it never happens
// denominator is probably zero
if i >= Mpprec*Mpscale {
q.Ovf = 1
r.Ovf = 1
n.Neg = uint8(ns)
d.Neg = uint8(ds)
Yyerror("constant division overflow")
return
}
// shift denominator back creating
// quotient a bit at a time
// when done the remaining numerator
// will be the remainder
for ; i > 0; i-- {
mplsh(q, 1)
mprsh(d)
if mpcmp(d, r) <= 0 {
mpaddcfix(q, 1)
_mpsubfixfix(r, d)
}
}
n.Neg = uint8(ns)
d.Neg = uint8(ds)
r.Neg = uint8(ns)
q.Neg = uint8(ns ^ ds)
}
func mpiszero(a *Mpfix) bool {
for i := Mpprec - 1; i >= 0; i-- {
if a.A[i] != 0 {
return false
}
}
return true
}
func mpdivfract(a *Mpfix, b *Mpfix) {
var n Mpfix
var d Mpfix
var j int
var x int
_mpmovefixfix(&n, a) // numerator
_mpmovefixfix(&d, b) // denominator
neg := int(n.Neg) ^ int(d.Neg)
n.Neg = 0
d.Neg = 0
for i := Mpprec - 1; i >= 0; i-- {
x = 0
for j = 0; j < Mpscale; j++ {
x <<= 1
if mpcmp(&d, &n) <= 0 {
if !mpiszero(&d) {
x |= 1
}
_mpsubfixfix(&n, &d)
}
mprsh(&d)
}
a.A[i] = x
}
a.Neg = uint8(neg)
}
func mptestfix(a *Mpfix) int {
var b Mpfix
_Mpmovecfix(&b, 0)
r := mpcmp(a, &b)
if a.Neg != 0 {
if r > 0 {
return -1
}
if r < 0 {
return +1
}
}
return r
}
...@@ -9,281 +9,75 @@ import ( ...@@ -9,281 +9,75 @@ import (
"math" "math"
) )
/* func newMpflt() *Mpflt {
* returns the leading non-zero var a Mpflt
* word of the number a.Val.SetPrec(Mpscale * Mpprec)
*/ return &a
func sigfig(a *Mpflt) int {
var i int
for i = Mpprec - 1; i >= 0; i-- {
if a.Val.A[i] != 0 {
break
}
}
//print("sigfig %d %d\n", i-z+1, z);
return i + 1
}
/*
* sets the exponent.
* a too large exponent is an error.
* a too small exponent rounds the number to zero.
*/
func mpsetexp(a *Mpflt, exp int) {
if int(int16(exp)) != exp {
if exp > 0 {
Yyerror("float constant is too large")
a.Exp = 0x7fff
} else {
Mpmovecflt(a, 0)
}
} else {
a.Exp = int16(exp)
}
}
/*
* shifts the leading non-zero
* word of the number to Mpnorm
*/
func mpnorm(a *Mpflt) {
os := sigfig(a)
if os == 0 {
// zero
a.Exp = 0
a.Val.Neg = 0
return
}
// this will normalize to the nearest word
x := a.Val.A[os-1]
s := (Mpnorm - os) * Mpscale
// further normalize to the nearest bit
for {
x <<= 1
if x&Mpbase != 0 {
break
}
s++
if x == 0 {
// this error comes from trying to
// convert an Inf or something
// where the initial x=0x80000000
s = (Mpnorm - os) * Mpscale
break
}
}
_Mpshiftfix(&a.Val, s)
mpsetexp(a, int(a.Exp)-s)
} }
/// implements float arihmetic /// implements float arihmetic
func mpaddfltflt(a *Mpflt, b *Mpflt) { func mpaddfltflt(a *Mpflt, b *Mpflt) {
if Mpdebug != 0 /*TypeKind(100016)*/ { if Mpdebug != 0 {
fmt.Printf("\n%v + %v", Fconv(a, 0), Fconv(b, 0)) fmt.Printf("\n%v + %v", Fconv(a, 0), Fconv(b, 0))
} }
sa := sigfig(a) a.Val.Add(&a.Val, &b.Val)
var s int
var sb int
if sa == 0 {
mpmovefltflt(a, b)
goto out
}
sb = sigfig(b)
if sb == 0 {
goto out
}
s = int(a.Exp) - int(b.Exp)
if s > 0 {
// a is larger, shift b right
var c Mpflt
mpmovefltflt(&c, b)
_Mpshiftfix(&c.Val, -s)
_mpaddfixfix(&a.Val, &c.Val, 0)
goto out
}
if s < 0 {
// b is larger, shift a right
_Mpshiftfix(&a.Val, s)
mpsetexp(a, int(a.Exp)-s) if Mpdebug != 0 {
_mpaddfixfix(&a.Val, &b.Val, 0)
goto out
}
_mpaddfixfix(&a.Val, &b.Val, 0)
out:
mpnorm(a)
if Mpdebug != 0 /*TypeKind(100016)*/ {
fmt.Printf(" = %v\n\n", Fconv(a, 0)) fmt.Printf(" = %v\n\n", Fconv(a, 0))
} }
} }
func mpmulfltflt(a *Mpflt, b *Mpflt) { func mpmulfltflt(a *Mpflt, b *Mpflt) {
if Mpdebug != 0 /*TypeKind(100016)*/ { if Mpdebug != 0 {
fmt.Printf("%v\n * %v\n", Fconv(a, 0), Fconv(b, 0)) fmt.Printf("%v\n * %v\n", Fconv(a, 0), Fconv(b, 0))
} }
sa := sigfig(a) a.Val.Mul(&a.Val, &b.Val)
if sa == 0 {
// zero
a.Exp = 0
a.Val.Neg = 0
return
}
sb := sigfig(b)
if sb == 0 {
// zero
mpmovefltflt(a, b)
return
}
mpmulfract(&a.Val, &b.Val)
mpsetexp(a, (int(a.Exp)+int(b.Exp))+Mpscale*Mpprec-Mpscale-1)
mpnorm(a) if Mpdebug != 0 {
if Mpdebug != 0 /*TypeKind(100016)*/ {
fmt.Printf(" = %v\n\n", Fconv(a, 0)) fmt.Printf(" = %v\n\n", Fconv(a, 0))
} }
} }
func mpdivfltflt(a *Mpflt, b *Mpflt) { func mpdivfltflt(a *Mpflt, b *Mpflt) {
if Mpdebug != 0 /*TypeKind(100016)*/ { if Mpdebug != 0 {
fmt.Printf("%v\n / %v\n", Fconv(a, 0), Fconv(b, 0)) fmt.Printf("%v\n / %v\n", Fconv(a, 0), Fconv(b, 0))
} }
sb := sigfig(b) a.Val.Quo(&a.Val, &b.Val)
if sb == 0 {
// zero and ovfl
a.Exp = 0
a.Val.Neg = 0
a.Val.Ovf = 1
Yyerror("constant division by zero")
return
}
sa := sigfig(a)
if sa == 0 {
// zero
a.Exp = 0
a.Val.Neg = 0
return
}
// adjust b to top
var c Mpflt
mpmovefltflt(&c, b)
_Mpshiftfix(&c.Val, Mpscale)
// divide
mpdivfract(&a.Val, &c.Val)
mpsetexp(a, (int(a.Exp)-int(c.Exp))-Mpscale*(Mpprec-1)+1)
mpnorm(a) if Mpdebug != 0 {
if Mpdebug != 0 /*TypeKind(100016)*/ {
fmt.Printf(" = %v\n\n", Fconv(a, 0)) fmt.Printf(" = %v\n\n", Fconv(a, 0))
} }
} }
func mpgetfltN(a *Mpflt, prec int, bias int) float64 { func mpgetfltN(a *Mpflt, prec int, bias int) float64 {
if a.Val.Ovf != 0 && nsavederrors+nerrors == 0 { var x float64
switch prec {
case 53:
x, _ = a.Val.Float64()
case 24:
// We should be using a.Val.Float32() here but that seems incorrect
// for certain denormal values (all.bash fails). The current code
// appears to work for all existing test cases, though there ought
// to be issues with denormal numbers that are incorrectly rounded.
// TODO(gri) replace with a.Val.Float32() once correctly working
// See also: https://github.com/golang/go/issues/10321
var t Mpflt
t.Val.SetPrec(24).Set(&a.Val)
x, _ = t.Val.Float64()
default:
panic("unreachable")
}
// check for overflow
if math.IsInf(x, 0) && nsavederrors+nerrors == 0 {
Yyerror("mpgetflt ovf") Yyerror("mpgetflt ovf")
} }
s := sigfig(a) return x
if s == 0 {
return 0
}
if s != Mpnorm {
Yyerror("mpgetflt norm")
mpnorm(a)
}
for a.Val.A[Mpnorm-1]&Mpsign == 0 {
_Mpshiftfix(&a.Val, 1)
mpsetexp(a, int(a.Exp)-1) // can set 'a' to zero
s = sigfig(a)
if s == 0 {
return 0
}
}
// pick up the mantissa, a rounding bit, and a tie-breaking bit in a uvlong
s = prec + 2
v := uint64(0)
var i int
for i = Mpnorm - 1; s >= Mpscale; i-- {
v = v<<Mpscale | uint64(a.Val.A[i])
s -= Mpscale
}
if s > 0 {
v = v<<uint(s) | uint64(a.Val.A[i])>>uint(Mpscale-s)
if a.Val.A[i]&((1<<uint(Mpscale-s))-1) != 0 {
v |= 1
}
i--
}
for ; i >= 0; i-- {
if a.Val.A[i] != 0 {
v |= 1
}
}
// gradual underflow
e := Mpnorm*Mpscale + int(a.Exp) - prec
minexp := bias + 1 - prec + 1
if e < minexp {
s := minexp - e
if s > prec+1 {
s = prec + 1
}
if v&((1<<uint(s))-1) != 0 {
v |= 1 << uint(s)
}
v >>= uint(s)
e = minexp
}
// round to even
v |= (v & 4) >> 2
v += v & 1
v >>= 2
f := float64(v)
f = math.Ldexp(f, e)
if a.Val.Neg != 0 {
f = -f
}
return f
} }
func mpgetflt(a *Mpflt) float64 { func mpgetflt(a *Mpflt) float64 {
...@@ -295,62 +89,17 @@ func mpgetflt32(a *Mpflt) float64 { ...@@ -295,62 +89,17 @@ func mpgetflt32(a *Mpflt) float64 {
} }
func Mpmovecflt(a *Mpflt, c float64) { func Mpmovecflt(a *Mpflt, c float64) {
if Mpdebug != 0 /*TypeKind(100016)*/ { if Mpdebug != 0 {
fmt.Printf("\nconst %g", c) fmt.Printf("\nconst %g", c)
} }
_Mpmovecfix(&a.Val, 0)
a.Exp = 0
var f float64
var l int
var i int
if c == 0 {
goto out
}
if c < 0 {
a.Val.Neg = 1
c = -c
}
f, i = math.Frexp(c)
a.Exp = int16(i)
for i := 0; i < 10; i++ { a.Val.SetFloat64(c)
f = f * Mpbase
l = int(math.Floor(f))
f = f - float64(l)
a.Exp -= Mpscale
a.Val.A[0] = l
if f == 0 {
break
}
_Mpshiftfix(&a.Val, Mpscale)
}
out: if Mpdebug != 0 {
mpnorm(a)
if Mpdebug != 0 /*TypeKind(100016)*/ {
fmt.Printf(" = %v\n", Fconv(a, 0)) fmt.Printf(" = %v\n", Fconv(a, 0))
} }
} }
func mpnegflt(a *Mpflt) { func mpnegflt(a *Mpflt) {
a.Val.Neg ^= 1 a.Val.Neg(&a.Val)
}
func mptestflt(a *Mpflt) int {
if Mpdebug != 0 /*TypeKind(100016)*/ {
fmt.Printf("\n%v?", Fconv(a, 0))
}
s := sigfig(a)
if s != 0 {
s = +1
if a.Val.Neg != 0 {
s = -1
}
}
if Mpdebug != 0 /*TypeKind(100016)*/ {
fmt.Printf(" = %d\n", s)
}
return s
} }
...@@ -693,7 +693,7 @@ func Nodintconst(v int64) *Node { ...@@ -693,7 +693,7 @@ func Nodintconst(v int64) *Node {
func nodfltconst(v *Mpflt) *Node { func nodfltconst(v *Mpflt) *Node {
c := Nod(OLITERAL, nil, nil) c := Nod(OLITERAL, nil, nil)
c.Addable = 1 c.Addable = 1
c.Val.U.Fval = new(Mpflt) c.Val.U.Fval = newMpflt()
mpmovefltflt(c.Val.U.Fval, v) mpmovefltflt(c.Val.U.Fval, v)
c.Val.Ctype = CTFLT c.Val.Ctype = CTFLT
c.Type = Types[TIDEAL] c.Type = Types[TIDEAL]
......
// run
// Copyright 2015 The Go Authors. All rights reserved.
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file.
// WARNING: GENERATED FILE - DO NOT MODIFY MANUALLY!
// (To generate, in go/types directory: go test -run=Hilbert -H=2 -out="h2.src")
// This program tests arbitrary precision constant arithmetic
// by generating the constant elements of a Hilbert matrix H,
// its inverse I, and the product P = H*I. The product should
// be the identity matrix.
package main
func main() {
if !ok {
print()
return
}
}
// Hilbert matrix, n = 2
const (
h0_0, h0_1 = 1.0 / (iota + 1), 1.0 / (iota + 2)
h1_0, h1_1
)
// Inverse Hilbert matrix
const (
i0_0 = +1 * b2_1 * b2_1 * b0_0 * b0_0
i0_1 = -2 * b2_0 * b3_1 * b1_0 * b1_0
i1_0 = -2 * b3_1 * b2_0 * b1_1 * b1_1
i1_1 = +3 * b3_0 * b3_0 * b2_1 * b2_1
)
// Product matrix
const (
p0_0 = h0_0*i0_0 + h0_1*i1_0
p0_1 = h0_0*i0_1 + h0_1*i1_1
p1_0 = h1_0*i0_0 + h1_1*i1_0
p1_1 = h1_0*i0_1 + h1_1*i1_1
)
// Verify that product is identity matrix
const ok = p0_0 == 1 && p0_1 == 0 &&
p1_0 == 0 && p1_1 == 1 &&
true
func print() {
println(p0_0, p0_1)
println(p1_0, p1_1)
}
// Binomials
const (
b0_0 = f0 / (f0 * f0)
b1_0 = f1 / (f0 * f1)
b1_1 = f1 / (f1 * f0)
b2_0 = f2 / (f0 * f2)
b2_1 = f2 / (f1 * f1)
b2_2 = f2 / (f2 * f0)
b3_0 = f3 / (f0 * f3)
b3_1 = f3 / (f1 * f2)
b3_2 = f3 / (f2 * f1)
b3_3 = f3 / (f3 * f0)
)
// Factorials
const (
f0 = 1
f1 = 1
f2 = f1 * 2
f3 = f2 * 3
)
// run
// Copyright 2015 The Go Authors. All rights reserved.
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file.
// This test computes the precision of the compiler's internal multiprecision floats.
package main
import (
"fmt"
"math"
"runtime"
)
const ulp = (1.0 + (2.0 / 3.0)) - (5.0 / 3.0)
func main() {
// adjust precision depending on compiler
var prec float64
switch runtime.Compiler {
case "gc":
prec = 16 * 29
case "gccgo":
prec = 256
default:
// unknown compiler
return
}
p := 1 - math.Log(math.Abs(ulp))/math.Log(2)
if math.Abs(p-prec) > 1e-10 {
fmt.Printf("BUG: got %g; want %g\n", p, prec)
}
}
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment