compiler: create new locals for range loops when needed, fix #2855

This commit is contained in:
Roman Khimov 2023-01-11 15:49:10 +03:00
parent 5ad1fcd321
commit 92acc71c80
2 changed files with 36 additions and 2 deletions

View file

@ -1313,7 +1313,11 @@ func (c *codegen) Visit(node ast.Node) ast.Visitor {
} else { } else {
emit.Opcodes(c.prog.BinWriter, opcode.DUP) emit.Opcodes(c.prog.BinWriter, opcode.DUP)
} }
c.emitStoreVar("", n.Key.(*ast.Ident).Name) keyIdent := n.Key.(*ast.Ident)
if n.Tok == token.DEFINE {
c.scope.newLocal(keyIdent.Name)
}
c.emitStoreVar("", keyIdent.Name)
} }
if needValue { if needValue {
if !isMap || !keyLoaded { if !isMap || !keyLoaded {
@ -1327,7 +1331,11 @@ func (c *codegen) Visit(node ast.Node) ast.Visitor {
opcode.SWAP, // key should be on top opcode.SWAP, // key should be on top
opcode.PICKITEM) opcode.PICKITEM)
} }
c.emitStoreVar("", n.Value.(*ast.Ident).Name) valIdent := n.Value.(*ast.Ident)
if n.Tok == token.DEFINE {
c.scope.newLocal(valIdent.Name)
}
c.emitStoreVar("", valIdent.Name)
} }
ast.Walk(c, n.Body) ast.Walk(c, n.Body)

View file

@ -714,6 +714,32 @@ var forLoopTestCases = []testCase{
`, `,
big.NewInt(6), big.NewInt(6),
}, },
{
"shadow range key",
`func F%d() int {
i := 10
ints := []int{1, 2, 3, 4, 5}
for i := range ints {
_ = i
}
return i
}
`,
big.NewInt(10),
},
{
"shadow range value",
`func F%d() int {
i := 10
ints := []int{1, 2, 3, 4, 5}
for _, i := range ints {
_ = i
}
return i
}
`,
big.NewInt(10),
},
} }
func TestForLoop(t *testing.T) { func TestForLoop(t *testing.T) {