diff --git a/gnovm/pkg/gnolang/internal/softfloat/softfloat.go b/gnovm/pkg/gnolang/internal/softfloat/softfloat.go index 96ac86f6053..06f4bd8d7e9 100644 --- a/gnovm/pkg/gnolang/internal/softfloat/softfloat.go +++ b/gnovm/pkg/gnolang/internal/softfloat/softfloat.go @@ -61,3 +61,5 @@ func Fint32to32(x int32) uint32 { return fint32to32(x) } func Fint32to64(x int32) uint64 { return fint32to64(x) } func Fint64to32(x int64) uint32 { return fint64to32(x) } func Fint64to64(x int64) uint64 { return fint64to64(x) } +func Fuint64to32(x uint64) uint32 { return fuint64to32(x) } +func Fuint64to64(x uint64) uint64 { return fuint64to64(x) } diff --git a/gnovm/pkg/gnolang/op_binary.go b/gnovm/pkg/gnolang/op_binary.go index 24123d285ad..aee02f857d8 100644 --- a/gnovm/pkg/gnolang/op_binary.go +++ b/gnovm/pkg/gnolang/op_binary.go @@ -2,9 +2,11 @@ package gnolang import ( "fmt" + "math" "math/big" "github.com/cockroachdb/apd/v3" + "github.com/gnolang/gno/gnovm/pkg/gnolang/internal/softfloat" ) // ---------------------------------------- @@ -329,6 +331,18 @@ func (m *Machine) doOpBandn() { bandnAssign(lv, rv) } +func compFloat32(lv, rv *TypedValue) (int32, bool) { + lvf := softfloat.F32to64(math.Float32bits(lv.GetFloat32())) + rvf := softfloat.F32to64(math.Float32bits(rv.GetFloat32())) + return softfloat.Fcmp64(lvf, rvf) +} + +func compFloat64(lv, rv *TypedValue) (int32, bool) { + lvf := math.Float64bits(lv.GetFloat64()) + rvf := math.Float64bits(rv.GetFloat64()) + return softfloat.Fcmp64(lvf, rvf) +} + // ---------------------------------------- // logic functions @@ -384,9 +398,11 @@ func isEql(store Store, lv, rv *TypedValue) bool { case Uint64Kind: return (lv.GetUint64() == rv.GetUint64()) case Float32Kind: - return (lv.GetFloat32() == rv.GetFloat32()) // XXX determinism? + cmp, nan := compFloat32(lv, rv) + return cmp == 0 && !nan case Float64Kind: - return (lv.GetFloat64() == rv.GetFloat64()) // XXX determinism? + cmp, nan := compFloat64(lv, rv) + return cmp == 0 && !nan case BigintKind: lb := lv.V.(BigintValue).V rb := rv.V.(BigintValue).V @@ -525,9 +541,11 @@ func isLss(lv, rv *TypedValue) bool { case Uint64Kind: return (lv.GetUint64() < rv.GetUint64()) case Float32Kind: - return (lv.GetFloat32() < rv.GetFloat32()) // XXX determinism? + cmp, nan := compFloat32(lv, rv) + return (cmp < 0) && !nan case Float64Kind: - return (lv.GetFloat64() < rv.GetFloat64()) // XXX determinism? + cmp, nan := compFloat64(lv, rv) + return (cmp < 0) && !nan case BigintKind: lb := lv.V.(BigintValue).V rb := rv.V.(BigintValue).V @@ -569,9 +587,11 @@ func isLeq(lv, rv *TypedValue) bool { case Uint64Kind: return (lv.GetUint64() <= rv.GetUint64()) case Float32Kind: - return (lv.GetFloat32() <= rv.GetFloat32()) // XXX determinism? + cmp, nan := compFloat32(lv, rv) + return (cmp <= 0) && !nan case Float64Kind: - return (lv.GetFloat64() <= rv.GetFloat64()) // XXX determinism? + cmp, nan := compFloat64(lv, rv) + return (cmp <= 0) && !nan case BigintKind: lb := lv.V.(BigintValue).V rb := rv.V.(BigintValue).V @@ -613,9 +633,11 @@ func isGtr(lv, rv *TypedValue) bool { case Uint64Kind: return (lv.GetUint64() > rv.GetUint64()) case Float32Kind: - return (lv.GetFloat32() > rv.GetFloat32()) // XXX determinism? + cmp, nan := compFloat32(lv, rv) + return (cmp > 0) && !nan case Float64Kind: - return (lv.GetFloat64() > rv.GetFloat64()) // XXX determinism? + cmp, nan := compFloat64(lv, rv) + return (cmp > 0) && !nan case BigintKind: lb := lv.V.(BigintValue).V rb := rv.V.(BigintValue).V @@ -657,9 +679,11 @@ func isGeq(lv, rv *TypedValue) bool { case Uint64Kind: return (lv.GetUint64() >= rv.GetUint64()) case Float32Kind: - return (lv.GetFloat32() >= rv.GetFloat32()) // XXX determinism? + cmp, nan := compFloat32(lv, rv) + return (cmp >= 0) && !nan case Float64Kind: - return (lv.GetFloat64() >= rv.GetFloat64()) // XXX determinism? + cmp, nan := compFloat64(lv, rv) + return (cmp >= 0) && !nan case BigintKind: lb := lv.V.(BigintValue).V rb := rv.V.(BigintValue).V @@ -707,10 +731,14 @@ func addAssign(alloc *Allocator, lv, rv *TypedValue) { lv.SetUint64(lv.GetUint64() + rv.GetUint64()) case Float32Type: // NOTE: gno doesn't fuse *+. - lv.SetFloat32(lv.GetFloat32() + rv.GetFloat32()) // XXX determinism? + lvf := math.Float32bits(lv.GetFloat32()) + rvf := math.Float32bits(rv.GetFloat32()) + lv.SetFloat32(math.Float32frombits(softfloat.Fadd32(lvf, rvf))) case Float64Type: // NOTE: gno doesn't fuse *+. - lv.SetFloat64(lv.GetFloat64() + rv.GetFloat64()) // XXX determinism? + lvf := math.Float64bits(lv.GetFloat64()) + rvf := math.Float64bits(rv.GetFloat64()) + lv.SetFloat64(math.Float64frombits(softfloat.Fadd64(lvf, rvf))) case BigintType, UntypedBigintType: lb := lv.GetBigInt() lb = big.NewInt(0).Add(lb, rv.GetBigInt()) @@ -763,10 +791,14 @@ func subAssign(lv, rv *TypedValue) { lv.SetUint64(lv.GetUint64() - rv.GetUint64()) case Float32Type: // NOTE: gno doesn't fuse *+. - lv.SetFloat32(lv.GetFloat32() - rv.GetFloat32()) // XXX determinism? + lvf := math.Float32bits(lv.GetFloat32()) + rvf := math.Float32bits(rv.GetFloat32()) + lv.SetFloat32(math.Float32frombits(softfloat.Fsub32(lvf, rvf))) case Float64Type: // NOTE: gno doesn't fuse *+. - lv.SetFloat64(lv.GetFloat64() - rv.GetFloat64()) // XXX determinism? + lvf := math.Float64bits(lv.GetFloat64()) + rvf := math.Float64bits(rv.GetFloat64()) + lv.SetFloat64(math.Float64frombits(softfloat.Fsub64(lvf, rvf))) case BigintType, UntypedBigintType: lb := lv.GetBigInt() lb = big.NewInt(0).Sub(lb, rv.GetBigInt()) @@ -819,10 +851,14 @@ func mulAssign(lv, rv *TypedValue) { lv.SetUint64(lv.GetUint64() * rv.GetUint64()) case Float32Type: // NOTE: gno doesn't fuse *+. - lv.SetFloat32(lv.GetFloat32() * rv.GetFloat32()) // XXX determinism? + lvf := math.Float32bits(lv.GetFloat32()) + rvf := math.Float32bits(rv.GetFloat32()) + lv.SetFloat32(math.Float32frombits(softfloat.Fmul32(lvf, rvf))) case Float64Type: // NOTE: gno doesn't fuse *+. - lv.SetFloat64(lv.GetFloat64() * rv.GetFloat64()) // XXX determinism? + lvf := math.Float64bits(lv.GetFloat64()) + rvf := math.Float64bits(rv.GetFloat64()) + lv.SetFloat64(math.Float64frombits(softfloat.Fmul64(lvf, rvf))) case BigintType, UntypedBigintType: lb := lv.GetBigInt() lb = big.NewInt(0).Mul(lb, rv.GetBigInt()) diff --git a/gnovm/pkg/gnolang/values_conversions.go b/gnovm/pkg/gnolang/values_conversions.go index 9ec3427ed8f..67ce06f6f35 100644 --- a/gnovm/pkg/gnolang/values_conversions.go +++ b/gnovm/pkg/gnolang/values_conversions.go @@ -8,6 +8,7 @@ import ( "strconv" "github.com/cockroachdb/apd/v3" + "github.com/gnolang/gno/gnovm/pkg/gnolang/internal/softfloat" ) // t cannot be nil or untyped or DataByteType. @@ -136,11 +137,11 @@ GNO_CASE: tv.T = t tv.SetUint64(x) case Float32Kind: - x := float32(tv.GetInt()) // XXX determinism? + x := math.Float32frombits(softfloat.Fintto32(int64(tv.GetInt()))) tv.T = t tv.SetFloat32(x) case Float64Kind: - x := float64(tv.GetInt()) // XXX determinism? + x := math.Float64frombits(softfloat.Fintto64(int64(tv.GetInt()))) tv.T = t tv.SetFloat64(x) case StringKind: @@ -195,11 +196,11 @@ GNO_CASE: tv.T = t tv.SetUint64(x) case Float32Kind: - x := float32(tv.GetInt8()) // XXX determinism? + x := math.Float32frombits(softfloat.Fint32to32(int32(tv.GetInt8()))) tv.T = t tv.SetFloat32(x) case Float64Kind: - x := float64(tv.GetInt8()) // XXX determinism? + x := math.Float64frombits(softfloat.Fint32to64(int32(tv.GetInt8()))) tv.T = t tv.SetFloat64(x) case StringKind: @@ -254,11 +255,11 @@ GNO_CASE: tv.T = t tv.SetUint64(x) case Float32Kind: - x := float32(tv.GetInt16()) // XXX determinism? + x := math.Float32frombits(softfloat.Fint32to32(int32(tv.GetInt16()))) tv.T = t tv.SetFloat32(x) case Float64Kind: - x := float64(tv.GetInt16()) // XXX determinism? + x := math.Float64frombits(softfloat.Fint32to64(int32(tv.GetInt16()))) tv.T = t tv.SetFloat64(x) case StringKind: @@ -313,11 +314,11 @@ GNO_CASE: tv.T = t tv.SetUint64(x) case Float32Kind: - x := float32(tv.GetInt32()) // XXX determinism? + x := math.Float32frombits(softfloat.Fint32to32(tv.GetInt32())) tv.T = t tv.SetFloat32(x) case Float64Kind: - x := float64(tv.GetInt32()) // XXX determinism? + x := math.Float64frombits(softfloat.Fint32to64(tv.GetInt32())) tv.T = t tv.SetFloat64(x) case StringKind: @@ -372,11 +373,11 @@ GNO_CASE: tv.T = t tv.SetUint64(x) case Float32Kind: - x := float32(tv.GetInt64()) // XXX determinism? + x := math.Float32frombits(softfloat.Fint64to32(tv.GetInt64())) tv.T = t tv.SetFloat32(x) case Float64Kind: - x := float64(tv.GetInt64()) // XXX determinism? + x := math.Float64frombits(softfloat.Fint64to64(tv.GetInt64())) tv.T = t tv.SetFloat64(x) case StringKind: @@ -431,11 +432,11 @@ GNO_CASE: tv.T = t tv.SetUint64(x) case Float32Kind: - x := float32(tv.GetUint()) // XXX determinism? + x := math.Float32frombits(softfloat.Fuint64to32(uint64(tv.GetUint()))) tv.T = t tv.SetFloat32(x) case Float64Kind: - x := float64(tv.GetUint()) // XXX determinism? + x := math.Float64frombits(softfloat.Fuint64to64(uint64(tv.GetUint()))) tv.T = t tv.SetFloat64(x) case StringKind: @@ -490,11 +491,11 @@ GNO_CASE: tv.T = t tv.SetUint64(x) case Float32Kind: - x := float32(tv.GetUint8()) // XXX determinism? + x := math.Float32frombits(softfloat.Fuint64to32(uint64(tv.GetUint8()))) tv.T = t tv.SetFloat32(x) case Float64Kind: - x := float64(tv.GetUint8()) // XXX determinism? + x := math.Float64frombits(softfloat.Fuint64to64(uint64(tv.GetUint8()))) tv.T = t tv.SetFloat64(x) case StringKind: @@ -549,11 +550,11 @@ GNO_CASE: tv.T = t tv.SetUint64(x) case Float32Kind: - x := float32(tv.GetUint16()) // XXX determinism? + x := math.Float32frombits(softfloat.Fuint64to32(uint64(tv.GetUint16()))) tv.T = t tv.SetFloat32(x) case Float64Kind: - x := float64(tv.GetUint16()) // XXX determinism? + x := math.Float64frombits(softfloat.Fuint64to64(uint64(tv.GetUint16()))) tv.T = t tv.SetFloat64(x) case StringKind: @@ -608,11 +609,11 @@ GNO_CASE: tv.T = t tv.SetUint64(x) case Float32Kind: - x := float32(tv.GetUint32()) // XXX determinism? + x := math.Float32frombits(softfloat.Fuint64to32(uint64(tv.GetUint32()))) tv.T = t tv.SetFloat32(x) case Float64Kind: - x := float64(tv.GetUint32()) // XXX determinism? + x := math.Float64frombits(softfloat.Fuint64to64(uint64(tv.GetUint32()))) tv.T = t tv.SetFloat64(x) case StringKind: @@ -667,11 +668,11 @@ GNO_CASE: tv.T = t tv.SetUint64(x) case Float32Kind: - x := float32(tv.GetUint64()) // XXX determinism? + x := math.Float32frombits(softfloat.Fuint64to32(tv.GetUint64())) tv.T = t tv.SetFloat32(x) case Float64Kind: - x := float64(tv.GetUint64()) // XXX determinism? + x := math.Float64frombits(softfloat.Fuint64to64(tv.GetUint64())) tv.T = t tv.SetFloat64(x) case StringKind: @@ -686,51 +687,51 @@ GNO_CASE: case Float32Kind: switch k { case IntKind: - x := int(tv.GetFloat32()) // XXX determinism? + x := int(softfloat.F32toint64(math.Float32bits(tv.GetFloat32()))) tv.T = t tv.SetInt(x) case Int8Kind: - x := int8(tv.GetFloat32()) // XXX determinism? + x := int8(softfloat.F32toint32(math.Float32bits(tv.GetFloat32()))) tv.T = t tv.SetInt8(x) case Int16Kind: - x := int16(tv.GetFloat32()) // XXX determinism? + x := int16(softfloat.F32toint32(math.Float32bits(tv.GetFloat32()))) tv.T = t tv.SetInt16(x) case Int32Kind: - x := int32(tv.GetFloat32()) // XXX determinism? + x := int32(softfloat.F32toint32(math.Float32bits(tv.GetFloat32()))) tv.T = t tv.SetInt32(x) case Int64Kind: - x := int64(tv.GetFloat32()) // XXX determinism? + x := int64(softfloat.F32toint64(math.Float32bits(tv.GetFloat32()))) tv.T = t tv.SetInt64(x) case UintKind: - x := uint(tv.GetFloat32()) // XXX determinism? + x := uint(softfloat.F32touint64(math.Float32bits(tv.GetFloat32()))) tv.T = t tv.SetUint(x) case Uint8Kind: - x := uint8(tv.GetFloat32()) // XXX determinism? + x := uint8(softfloat.F32touint64(math.Float32bits(tv.GetFloat32()))) tv.T = t tv.SetUint8(x) case Uint16Kind: - x := uint16(tv.GetFloat32()) // XXX determinism? + x := uint16(softfloat.F32touint64(math.Float32bits(tv.GetFloat32()))) tv.T = t tv.SetUint16(x) case Uint32Kind: - x := uint32(tv.GetFloat32()) // XXX determinism? + x := uint32(softfloat.F32touint64(math.Float32bits(tv.GetFloat32()))) tv.T = t tv.SetUint32(x) case Uint64Kind: - x := uint64(tv.GetFloat32()) // XXX determinism? + x := uint64(softfloat.F32touint64(math.Float32bits(tv.GetFloat32()))) tv.T = t tv.SetUint64(x) case Float32Kind: - x := tv.GetFloat32() // XXX determinism? + x := math.Float32frombits(math.Float32bits(tv.GetFloat32())) // ??? tv.T = t tv.SetFloat32(x) case Float64Kind: - x := float64(tv.GetFloat32()) // XXX determinism? + x := math.Float64frombits(softfloat.F32to64(math.Float32bits(tv.GetFloat32()))) // ??? tv.T = t tv.SetFloat64(x) default: @@ -741,51 +742,51 @@ GNO_CASE: case Float64Kind: switch k { case IntKind: - x := int(tv.GetFloat64()) // XXX determinism? + x := int(softfloat.F64toint64(math.Float64bits(tv.GetFloat64()))) tv.T = t tv.SetInt(x) case Int8Kind: - x := int8(tv.GetFloat64()) // XXX determinism? + x := int8(softfloat.F64toint32(math.Float64bits(tv.GetFloat64()))) tv.T = t tv.SetInt8(x) case Int16Kind: - x := int16(tv.GetFloat64()) // XXX determinism? + x := int16(softfloat.F64toint32(math.Float64bits(tv.GetFloat64()))) tv.T = t tv.SetInt16(x) case Int32Kind: - x := int32(tv.GetFloat64()) // XXX determinism? + x := softfloat.F64toint32(math.Float64bits(tv.GetFloat64())) tv.T = t tv.SetInt32(x) case Int64Kind: - x := int64(tv.GetFloat64()) // XXX determinism? + x := softfloat.F64toint64(math.Float64bits(tv.GetFloat64())) tv.T = t tv.SetInt64(x) case UintKind: - x := uint(tv.GetFloat64()) // XXX determinism? + x := uint(softfloat.F64touint64(math.Float64bits(tv.GetFloat64()))) tv.T = t tv.SetUint(x) case Uint8Kind: - x := uint8(tv.GetFloat64()) // XXX determinism? + x := uint8(softfloat.F64touint64(math.Float64bits(tv.GetFloat64()))) tv.T = t tv.SetUint8(x) case Uint16Kind: - x := uint16(tv.GetFloat64()) // XXX determinism? + x := uint16(softfloat.F64touint64(math.Float64bits(tv.GetFloat64()))) tv.T = t tv.SetUint16(x) case Uint32Kind: - x := uint32(tv.GetFloat64()) // XXX determinism? + x := uint32(softfloat.F64touint64(math.Float64bits(tv.GetFloat64()))) tv.T = t tv.SetUint32(x) case Uint64Kind: - x := uint64(tv.GetFloat64()) // XXX determinism? + x := softfloat.F64touint64(math.Float64bits(tv.GetFloat64())) tv.T = t tv.SetUint64(x) case Float32Kind: - x := float32(tv.GetFloat64()) // XXX determinism? + x := math.Float32frombits(softfloat.F64to32(math.Float64bits(tv.GetFloat64()))) tv.T = t tv.SetFloat32(x) case Float64Kind: - x := tv.GetFloat64() // XXX determinism? + x := math.Float64frombits(math.Float64bits(tv.GetFloat64())) // ??? tv.T = t tv.SetFloat64(x) default: diff --git a/gnovm/tests/files/float5.gno b/gnovm/tests/files/float5.gno new file mode 100644 index 00000000000..410292b000f --- /dev/null +++ b/gnovm/tests/files/float5.gno @@ -0,0 +1,11 @@ +package main + +import "math" + +func main() { + var a, b float32 = 1.2, 3.4 + println(1.2 <= 3.4) +} + +// Output: +// true