Merge pull request #2871 from nspcc-dev/fix-shadowed-infinite-loop
Fix shadowed infinite loop
This commit is contained in:
commit
babc44bfb3
2 changed files with 60 additions and 7 deletions
|
@ -1301,21 +1301,45 @@ func (c *codegen) Visit(node ast.Node) ast.Visitor {
|
||||||
emit.Opcodes(c.prog.BinWriter, opcode.OVER, opcode.OVER)
|
emit.Opcodes(c.prog.BinWriter, opcode.OVER, opcode.OVER)
|
||||||
emit.Jmp(c.prog.BinWriter, opcode.JMPLEL, end)
|
emit.Jmp(c.prog.BinWriter, opcode.JMPLEL, end)
|
||||||
|
|
||||||
var keyLoaded bool
|
var (
|
||||||
needValue := n.Value != nil && n.Value.(*ast.Ident).Name != "_"
|
haveKey bool
|
||||||
if n.Key != nil && n.Key.(*ast.Ident).Name != "_" {
|
haveVal bool
|
||||||
|
keyIdent *ast.Ident
|
||||||
|
keyLoaded bool
|
||||||
|
valIdent *ast.Ident
|
||||||
|
)
|
||||||
|
if n.Key != nil {
|
||||||
|
keyIdent, haveKey = n.Key.(*ast.Ident)
|
||||||
|
if !haveKey {
|
||||||
|
c.prog.Err = errors.New("only simple identifiers can be used for range loop keys (see #2870)")
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
haveKey = (keyIdent.Name != "_")
|
||||||
|
}
|
||||||
|
if n.Value != nil {
|
||||||
|
valIdent, haveVal = n.Value.(*ast.Ident)
|
||||||
|
if !haveVal {
|
||||||
|
c.prog.Err = errors.New("only simple identifiers can be used for range loop values (see #2870)")
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
haveVal = (valIdent.Name != "_")
|
||||||
|
}
|
||||||
|
if haveKey {
|
||||||
if isMap {
|
if isMap {
|
||||||
c.rangeLoadKey()
|
c.rangeLoadKey()
|
||||||
if needValue {
|
if haveVal {
|
||||||
emit.Opcodes(c.prog.BinWriter, opcode.DUP)
|
emit.Opcodes(c.prog.BinWriter, opcode.DUP)
|
||||||
keyLoaded = true
|
keyLoaded = true
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
emit.Opcodes(c.prog.BinWriter, opcode.DUP)
|
emit.Opcodes(c.prog.BinWriter, opcode.DUP)
|
||||||
}
|
}
|
||||||
c.emitStoreVar("", n.Key.(*ast.Ident).Name)
|
if n.Tok == token.DEFINE {
|
||||||
|
c.scope.newLocal(keyIdent.Name)
|
||||||
|
}
|
||||||
|
c.emitStoreVar("", keyIdent.Name)
|
||||||
}
|
}
|
||||||
if needValue {
|
if haveVal {
|
||||||
if !isMap || !keyLoaded {
|
if !isMap || !keyLoaded {
|
||||||
c.rangeLoadKey()
|
c.rangeLoadKey()
|
||||||
}
|
}
|
||||||
|
@ -1327,7 +1351,10 @@ 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)
|
if n.Tok == token.DEFINE {
|
||||||
|
c.scope.newLocal(valIdent.Name)
|
||||||
|
}
|
||||||
|
c.emitStoreVar("", valIdent.Name)
|
||||||
}
|
}
|
||||||
|
|
||||||
ast.Walk(c, n.Body)
|
ast.Walk(c, n.Body)
|
||||||
|
|
|
@ -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) {
|
||||||
|
|
Loading…
Reference in a new issue