Merge pull request #1355 from nspcc-dev/compiler/cleanstack

compiler: drop unused call results
This commit is contained in:
Roman Khimov 2020-08-24 22:07:02 +03:00 committed by GitHub
commit 246d12a7d6
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
5 changed files with 103 additions and 19 deletions

View file

@ -578,9 +578,13 @@ func (c *codegen) Visit(node ast.Node) ast.Visitor {
return nil return nil
case *ast.SwitchStmt: case *ast.SwitchStmt:
ast.Walk(c, n.Tag) eqOpcode := opcode.EQUAL
if n.Tag != nil {
eqOpcode, _ := convertToken(token.EQL, c.typeOf(n.Tag)) ast.Walk(c, n.Tag)
eqOpcode, _ = convertToken(token.EQL, c.typeOf(n.Tag))
} else {
emit.Bool(c.prog.BinWriter, true)
}
switchEnd, label := c.generateLabel(labelEnd) switchEnd, label := c.generateLabel(labelEnd)
lastSwitch := c.currentSwitch lastSwitch := c.currentSwitch
@ -795,6 +799,20 @@ func (c *codegen) Visit(node ast.Node) ast.Visitor {
emit.Call(c.prog.BinWriter, opcode.CALLL, f.label) emit.Call(c.prog.BinWriter, opcode.CALLL, f.label)
} }
if c.scope != nil && c.scope.voidCalls[n] {
var sz int
if f != nil {
sz = f.decl.Type.Results.NumFields()
} else if !isBuiltin {
// lambda invocation
f := c.typeOf(n.Fun).Underlying().(*types.Signature)
sz = f.Results().Len()
}
for i := 0; i < sz; i++ {
emit.Opcode(c.prog.BinWriter, opcode.DROP)
}
}
return nil return nil
case *ast.SelectorExpr: case *ast.SelectorExpr:

View file

@ -90,8 +90,24 @@ func (c *funcScope) analyzeVoidCalls(node ast.Node) bool {
} }
case *ast.BinaryExpr: case *ast.BinaryExpr:
return false return false
case *ast.IfStmt:
// we can't just return `false`, because we still need to process body
ce, ok := n.Cond.(*ast.CallExpr)
if ok {
c.voidCalls[ce] = false
}
case *ast.CaseClause:
for _, e := range n.List {
ce, ok := e.(*ast.CallExpr)
if ok {
c.voidCalls[ce] = false
}
}
case *ast.CallExpr: case *ast.CallExpr:
c.voidCalls[n] = true _, ok := c.voidCalls[n]
if !ok {
c.voidCalls[n] = true
}
return false return false
} }
return true return true
@ -141,7 +157,7 @@ func (c *funcScope) countArgs() int {
func (c *funcScope) stackSize() int64 { func (c *funcScope) stackSize() int64 {
size := c.countLocals() size := c.countLocals()
numArgs := c.countArgs() numArgs := c.countArgs()
return int64(size + numArgs + len(c.voidCalls)) return int64(size + numArgs)
} }
// newVariable creates a new local variable or argument in the scope of the function. // newVariable creates a new local variable or argument in the scope of the function.

View file

@ -24,8 +24,8 @@ func TestSimpleFunctionCall(t *testing.T) {
} }
func TestNotAssignedFunctionCall(t *testing.T) { func TestNotAssignedFunctionCall(t *testing.T) {
src := ` t.Run("Simple", func(t *testing.T) {
package testcase src := `package testcase
func Main() int { func Main() int {
getSomeInteger() getSomeInteger()
getSomeInteger() getSomeInteger()
@ -34,12 +34,53 @@ func TestNotAssignedFunctionCall(t *testing.T) {
func getSomeInteger() int { func getSomeInteger() int {
return 0 return 0
} }`
` eval(t, src, big.NewInt(0))
// disable stack checks because it is hard right now })
// to distinguish between simple function call traversal t.Run("If", func(t *testing.T) {
// and the same traversal inside an assignment. src := `package testcase
evalWithoutStackChecks(t, src, big.NewInt(0)) func f() bool { return true }
func Main() int {
if f() {
return 42
}
return 0
}`
eval(t, src, big.NewInt(42))
})
t.Run("Switch", func(t *testing.T) {
src := `package testcase
func f() bool { return true }
func Main() int {
switch true {
case f():
return 42
default:
return 0
}
}`
eval(t, src, big.NewInt(42))
})
t.Run("Builtin", func(t *testing.T) {
src := `package foo
import "github.com/nspcc-dev/neo-go/pkg/interop/util"
func Main() int {
util.FromAddress("NPAsqZkx9WhNd4P72uhZxBhLinSuNkxfB8")
util.FromAddress("NPAsqZkx9WhNd4P72uhZxBhLinSuNkxfB8")
return 1
}`
eval(t, src, big.NewInt(1))
})
t.Run("Lambda", func(t *testing.T) {
src := `package foo
func Main() int {
f := func() (int, int) { return 1, 2 }
f()
f()
return 42
}`
eval(t, src, big.NewInt(42))
})
} }
func TestMultipleFunctionCalls(t *testing.T) { func TestMultipleFunctionCalls(t *testing.T) {

View file

@ -18,6 +18,21 @@ var switchTestCases = []testCase{
}`, }`,
big.NewInt(2), big.NewInt(2),
}, },
{
"switch with no tag",
`package main
func f() bool { return false }
func Main() int {
switch {
case f():
return 1
case true:
return 2
}
return 3
}`,
big.NewInt(2),
},
{ {
"simple switch fail", "simple switch fail",
`package main `package main

View file

@ -32,12 +32,6 @@ func runTestCases(t *testing.T, tcases []testCase) {
} }
} }
func evalWithoutStackChecks(t *testing.T, src string, result interface{}) {
v := vmAndCompile(t, src)
require.NoError(t, v.Run())
assertResult(t, v, result)
}
func eval(t *testing.T, src string, result interface{}) { func eval(t *testing.T, src string, result interface{}) {
vm := vmAndCompile(t, src) vm := vmAndCompile(t, src)
err := vm.Run() err := vm.Run()