Skip to content

Commit

Permalink
feat: use soft float
Browse files Browse the repository at this point in the history
  • Loading branch information
omarsy committed Nov 23, 2024
1 parent 5c6ce77 commit 0ab6c55
Show file tree
Hide file tree
Showing 4 changed files with 182 additions and 64 deletions.
2 changes: 2 additions & 0 deletions gnovm/pkg/gnolang/internal/softfloat/softfloat.go
Original file line number Diff line number Diff line change
Expand Up @@ -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) }

Check warning on line 65 in gnovm/pkg/gnolang/internal/softfloat/softfloat.go

View check run for this annotation

Codecov / codecov/patch

gnovm/pkg/gnolang/internal/softfloat/softfloat.go#L57-L65

Added lines #L57 - L65 were not covered by tests
78 changes: 58 additions & 20 deletions gnovm/pkg/gnolang/op_binary.go
Original file line number Diff line number Diff line change
Expand Up @@ -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"
)

// ----------------------------------------
Expand Down Expand Up @@ -335,6 +337,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)

Check warning on line 349 in gnovm/pkg/gnolang/op_binary.go

View check run for this annotation

Codecov / codecov/patch

gnovm/pkg/gnolang/op_binary.go#L346-L349

Added lines #L346 - L349 were not covered by tests
}

// ----------------------------------------
// logic functions

Expand Down Expand Up @@ -390,9 +404,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

Check warning on line 411 in gnovm/pkg/gnolang/op_binary.go

View check run for this annotation

Codecov / codecov/patch

gnovm/pkg/gnolang/op_binary.go#L410-L411

Added lines #L410 - L411 were not covered by tests
case BigintKind:
lb := lv.V.(BigintValue).V
rb := rv.V.(BigintValue).V
Expand Down Expand Up @@ -531,9 +547,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

Check warning on line 551 in gnovm/pkg/gnolang/op_binary.go

View check run for this annotation

Codecov / codecov/patch

gnovm/pkg/gnolang/op_binary.go#L550-L551

Added lines #L550 - L551 were not covered by tests
case Float64Kind:
return (lv.GetFloat64() < rv.GetFloat64()) // XXX determinism?
cmp, nan := compFloat64(lv, rv)
return (cmp < 0) && !nan

Check warning on line 554 in gnovm/pkg/gnolang/op_binary.go

View check run for this annotation

Codecov / codecov/patch

gnovm/pkg/gnolang/op_binary.go#L553-L554

Added lines #L553 - L554 were not covered by tests
case BigintKind:
lb := lv.V.(BigintValue).V
rb := rv.V.(BigintValue).V
Expand Down Expand Up @@ -575,9 +593,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

Check warning on line 597 in gnovm/pkg/gnolang/op_binary.go

View check run for this annotation

Codecov / codecov/patch

gnovm/pkg/gnolang/op_binary.go#L596-L597

Added lines #L596 - L597 were not covered by tests
case Float64Kind:
return (lv.GetFloat64() <= rv.GetFloat64()) // XXX determinism?
cmp, nan := compFloat64(lv, rv)
return (cmp <= 0) && !nan

Check warning on line 600 in gnovm/pkg/gnolang/op_binary.go

View check run for this annotation

Codecov / codecov/patch

gnovm/pkg/gnolang/op_binary.go#L599-L600

Added lines #L599 - L600 were not covered by tests
case BigintKind:
lb := lv.V.(BigintValue).V
rb := rv.V.(BigintValue).V
Expand Down Expand Up @@ -619,9 +639,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

Check warning on line 643 in gnovm/pkg/gnolang/op_binary.go

View check run for this annotation

Codecov / codecov/patch

gnovm/pkg/gnolang/op_binary.go#L642-L643

Added lines #L642 - L643 were not covered by tests
case Float64Kind:
return (lv.GetFloat64() > rv.GetFloat64()) // XXX determinism?
cmp, nan := compFloat64(lv, rv)
return (cmp > 0) && !nan

Check warning on line 646 in gnovm/pkg/gnolang/op_binary.go

View check run for this annotation

Codecov / codecov/patch

gnovm/pkg/gnolang/op_binary.go#L645-L646

Added lines #L645 - L646 were not covered by tests
case BigintKind:
lb := lv.V.(BigintValue).V
rb := rv.V.(BigintValue).V
Expand Down Expand Up @@ -663,9 +685,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

Check warning on line 689 in gnovm/pkg/gnolang/op_binary.go

View check run for this annotation

Codecov / codecov/patch

gnovm/pkg/gnolang/op_binary.go#L688-L689

Added lines #L688 - L689 were not covered by tests
case Float64Kind:
return (lv.GetFloat64() >= rv.GetFloat64()) // XXX determinism?
cmp, nan := compFloat64(lv, rv)
return (cmp >= 0) && !nan

Check warning on line 692 in gnovm/pkg/gnolang/op_binary.go

View check run for this annotation

Codecov / codecov/patch

gnovm/pkg/gnolang/op_binary.go#L691-L692

Added lines #L691 - L692 were not covered by tests
case BigintKind:
lb := lv.V.(BigintValue).V
rb := rv.V.(BigintValue).V
Expand Down Expand Up @@ -713,10 +737,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())
Expand Down Expand Up @@ -769,10 +797,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())
Expand Down Expand Up @@ -825,10 +857,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())
Expand Down Expand Up @@ -916,18 +952,20 @@ func quoAssign(lv, rv *TypedValue) *Exception {
lv.SetUint64(lv.GetUint64() / rv.GetUint64())
case Float32Type:
// NOTE: gno doesn't fuse *+.
lvf := math.Float32bits(lv.GetFloat32())
rvf := math.Float32bits(rv.GetFloat32())
if rv.GetFloat32() == 0 {
return expt
}
lv.SetFloat32(lv.GetFloat32() / rv.GetFloat32())
// XXX FOR DETERMINISM, PANIC IF NAN.
lv.SetFloat32(math.Float32frombits(softfloat.Fdiv32(lvf, rvf)))
case Float64Type:
// NOTE: gno doesn't fuse *+.
lvf := math.Float64bits(lv.GetFloat64())
rvf := math.Float64bits(rv.GetFloat64())
if rv.GetFloat64() == 0 {
return expt
}
lv.SetFloat64(lv.GetFloat64() / rv.GetFloat64())
// XXX FOR DETERMINISM, PANIC IF NAN.
lv.SetFloat64(math.Float64frombits(softfloat.Fdiv64(lvf, rvf)))
case BigintType, UntypedBigintType:
if rv.GetBigInt().Sign() == 0 {
return expt
Expand Down
Loading

0 comments on commit 0ab6c55

Please sign in to comment.