Commit 31e85240 authored by Robert Griesemer's avatar Robert Griesemer

math/big: fix aliasing error in Add, Sub

Also:
- make representation more flexible (no need to store trailing 0 digits to match precision)
- simplify rounding as a consequence
- minor related fixes

TBR adonovan

Change-Id: Ie91075990688b506d28371ec3b633b8267397ebb
Reviewed-on: https://go-review.googlesource.com/4841Reviewed-by: default avatarRob Pike <r@golang.org>
parent 7a77d8d1
...@@ -29,10 +29,10 @@ const debugFloat = true // enable for debugging ...@@ -29,10 +29,10 @@ const debugFloat = true // enable for debugging
// //
// Each Float value also has a precision, rounding mode, and accuracy. // Each Float value also has a precision, rounding mode, and accuracy.
// //
// The precision is the number of mantissa bits used to represent the // The precision is the (maximum) number of mantissa bits available to
// value. The rounding mode specifies how a result should be rounded // represent the value. The rounding mode specifies how a result should
// to fit into the mantissa bits, and accuracy describes the rounding // be rounded to fit into the mantissa bits, and accuracy describes the
// error with respect to the exact result. // rounding error with respect to the exact result.
// //
// All operations, including setters, that specify a *Float for the result, // All operations, including setters, that specify a *Float for the result,
// usually via the receiver, round their result to the result's precision // usually via the receiver, round their result to the result's precision
...@@ -48,8 +48,8 @@ const debugFloat = true // enable for debugging ...@@ -48,8 +48,8 @@ const debugFloat = true // enable for debugging
// or denormalized numbers). Additionally, positive and negative zeros and // or denormalized numbers). Additionally, positive and negative zeros and
// infinities are fully supported. // infinities are fully supported.
// //
// The zero (uninitialized) value for a Float is ready to use and // The zero (uninitialized) value for a Float is ready to use and represents
// represents the number +0.0 of 0 bit precision. // the number +0.0 exactly, with precision 0 and rounding mode ToNearestEven.
// //
type Float struct { type Float struct {
mode RoundingMode mode RoundingMode
...@@ -60,11 +60,13 @@ type Float struct { ...@@ -60,11 +60,13 @@ type Float struct {
prec uint // TODO(gri) make this a 32bit field prec uint // TODO(gri) make this a 32bit field
} }
// Internal representation details: The mantissa bits x.mant of a Float x // Internal representation: The mantissa bits x.mant of a Float x are stored
// are stored in the shortest nat slice long enough to hold x.prec bits. // in a nat slice long enough to hold up to x.prec bits; the slice may (but
// doesn't have to) be shorter if the mantissa contains trailing 0 bits.
// Unless x is a zero or an infinity, x.mant is normalized such that the // Unless x is a zero or an infinity, x.mant is normalized such that the
// msb of x.mant == 1. Thus, if the precision is not a multiple of the // msb of x.mant == 1 (i.e., the msb is shifted all the way "to the left").
// the Word size _W, x.mant[0] contains trailing zero bits. Zero and Inf // Thus, if the mantissa has trailing 0 bits or x.prec is not a multiple
// of the the Word size _W, x.mant[0] has trailing zero bits. Zero and Inf
// values have an empty mantissa and a 0 or infExp exponent, respectively. // values have an empty mantissa and a 0 or infExp exponent, respectively.
// NewFloat returns a new Float with value x rounded // NewFloat returns a new Float with value x rounded
...@@ -292,52 +294,35 @@ func validate(args ...*Float) { ...@@ -292,52 +294,35 @@ func validate(args ...*Float) {
// have before calling round. z's mantissa must be normalized (with the msb set) // have before calling round. z's mantissa must be normalized (with the msb set)
// or empty. // or empty.
func (z *Float) round(sbit uint) { func (z *Float) round(sbit uint) {
if debugFloat {
validate(z)
}
z.acc = Exact z.acc = Exact
// handle zero and Inf // handle zero and Inf
m := uint(len(z.mant)) // mantissa length in words for current precision m := uint(len(z.mant)) // present mantissa length in words
if m == 0 { if m == 0 {
if z.exp != infExp { if z.exp != infExp {
z.exp = 0 z.exp = 0
} }
return return
} }
// z.prec > 0 // m > 0 implies z.prec > 0 (checked by validate)
if debugFloat {
validate(z)
}
bits := m * _W // available mantissa bits bits := m * _W // present mantissa bits
if bits == z.prec { if bits <= z.prec {
// mantissa fits Exactly => nothing to do // mantissa fits => nothing to do
return return
} }
// bits > z.prec
n := (z.prec + (_W - 1)) / _W // mantissa length in words for desired precision n := (z.prec + (_W - 1)) / _W // mantissa length in words for desired precision
if bits < z.prec {
// mantissa too small => extend
if m < n {
// slice too short => extend slice
if int(n) <= cap(z.mant) {
// reuse existing slice
z.mant = z.mant[:n]
copy(z.mant[n-m:], z.mant[:m])
z.mant[:n-m].clear()
} else {
// n > cap(z.mant) => allocate new slice
const e = 4 // extra capacity (see nat.make)
new := make(nat, n, n+e)
copy(new[n-m:], z.mant)
}
}
return
}
// Rounding is based on two bits: the rounding bit (rbit) and the // Rounding is based on two bits: the rounding bit (rbit) and the
// sticky bit (sbit). The rbit is the bit immediately before the // sticky bit (sbit). The rbit is the bit immediately before the
// mantissa bits (the "0.5"). The sbit is set if any of the bits // z.prec leading mantissa bits (the "0.5"). The sbit is set if any
// before the rbit are set (the "0.25", "0.125", etc.): // of the bits before the rbit are set (the "0.25", "0.125", etc.):
// //
// rbit sbit => "fractional part" // rbit sbit => "fractional part"
// //
...@@ -875,14 +860,16 @@ func (z *Float) uadd(x, y *Float) { ...@@ -875,14 +860,16 @@ func (z *Float) uadd(x, y *Float) {
// could make this code significantly faster // could make this code significantly faster
switch { switch {
case ex < ey: case ex < ey:
t := z.mant.shl(y.mant, uint(ey-ex)) // cannot re-use z.mant w/o testing for aliasing
z.mant = t.add(x.mant, t) t := nat(nil).shl(y.mant, uint(ey-ex))
z.mant = z.mant.add(x.mant, t)
default: default:
// ex == ey, no shift needed // ex == ey, no shift needed
z.mant = z.mant.add(x.mant, y.mant) z.mant = z.mant.add(x.mant, y.mant)
case ex > ey: case ex > ey:
t := z.mant.shl(x.mant, uint(ex-ey)) // cannot re-use z.mant w/o testing for aliasing
z.mant = t.add(t, y.mant) t := nat(nil).shl(x.mant, uint(ex-ey))
z.mant = z.mant.add(t, y.mant)
ex = ey ex = ey
} }
// len(z.mant) > 0 // len(z.mant) > 0
...@@ -908,13 +895,15 @@ func (z *Float) usub(x, y *Float) { ...@@ -908,13 +895,15 @@ func (z *Float) usub(x, y *Float) {
switch { switch {
case ex < ey: case ex < ey:
t := z.mant.shl(y.mant, uint(ey-ex)) // cannot re-use z.mant w/o testing for aliasing
t := nat(nil).shl(y.mant, uint(ey-ex))
z.mant = t.sub(x.mant, t) z.mant = t.sub(x.mant, t)
default: default:
// ex == ey, no shift needed // ex == ey, no shift needed
z.mant = z.mant.sub(x.mant, y.mant) z.mant = z.mant.sub(x.mant, y.mant)
case ex > ey: case ex > ey:
t := z.mant.shl(x.mant, uint(ex-ey)) // cannot re-use z.mant w/o testing for aliasing
t := nat(nil).shl(x.mant, uint(ex-ey))
z.mant = t.sub(t, y.mant) z.mant = t.sub(t, y.mant)
ex = ey ex = ey
} }
...@@ -1072,11 +1061,11 @@ func (z *Float) Add(x, y *Float) *Float { ...@@ -1072,11 +1061,11 @@ func (z *Float) Add(x, y *Float) *Float {
// TODO(gri) what about -0? // TODO(gri) what about -0?
if len(y.mant) == 0 { if len(y.mant) == 0 {
// TODO(gri) handle Inf // TODO(gri) handle Inf
return z.Round(x, z.prec, z.mode) return z.Set(x)
} }
if len(x.mant) == 0 { if len(x.mant) == 0 {
// TODO(gri) handle Inf // TODO(gri) handle Inf
return z.Round(y, z.prec, z.mode) return z.Set(y)
} }
// x, y != 0 // x, y != 0
...@@ -1113,13 +1102,10 @@ func (z *Float) Sub(x, y *Float) *Float { ...@@ -1113,13 +1102,10 @@ func (z *Float) Sub(x, y *Float) *Float {
// TODO(gri) what about -0? // TODO(gri) what about -0?
if len(y.mant) == 0 { if len(y.mant) == 0 {
// TODO(gri) handle Inf // TODO(gri) handle Inf
return z.Round(x, z.prec, z.mode) return z.Set(x)
} }
if len(x.mant) == 0 { if len(x.mant) == 0 {
prec := z.prec return z.Neg(y)
mode := z.mode
z.Neg(y)
return z.Round(z, prec, mode)
} }
// x, y != 0 // x, y != 0
......
...@@ -729,14 +729,20 @@ func TestFloatNeg(t *testing.T) { ...@@ -729,14 +729,20 @@ func TestFloatNeg(t *testing.T) {
} }
func TestFloatInc(t *testing.T) { func TestFloatInc(t *testing.T) {
var x, one Float const n = 10
// x.prec = 256 TODO(gri) This doesn't work at the moment for _, prec := range precList {
one.SetInt64(1) if 1<<prec < n {
for i := 0; i < 10; i++ { continue // prec must be large enough to hold all numbers from 0 to n
x.Add(&x, &one) }
} var x, one Float
if s := x.Format('g', 10); s != "10" { x.prec = prec
t.Errorf("got %s; want 10", s) one.SetInt64(1)
for i := 0; i < n; i++ {
x.Add(&x, &one)
}
if x.Cmp(new(Float).SetInt64(n)) != 0 {
t.Errorf("prec = %d: got %s; want %d", prec, &x, n)
}
} }
} }
......
...@@ -215,9 +215,9 @@ func (x *Float) Append(buf []byte, format byte, prec int) []byte { ...@@ -215,9 +215,9 @@ func (x *Float) Append(buf []byte, format byte, prec int) []byte {
return x.bigFtoa(buf, format, prec) return x.bigFtoa(buf, format, prec)
} }
// BUG(gri): Currently, String uses the 'p' (rather than 'g') format. // BUG(gri): Currently, String uses x.Format('g', 10) rather than x.Format('g', -1).
func (x *Float) String() string { func (x *Float) String() string {
return x.Format('p', 0) return x.Format('g', 10)
} }
// bstring appends the string of x in the format ["-"] mantissa "p" exponent // bstring appends the string of x in the format ["-"] mantissa "p" exponent
......
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