cmd/compile: check labels and branches during parse time

Instead of a separate check control flow pass (checkcfg.go)
operating on nodes, perform this check at parse time on the
new syntax tree. Permits this check to be done concurrently,
and doesn't depend on the specifics of the symbol's dclstack
implementation anymore. The remaining dclstack uses will be
removed in a follow-up change.

- added CheckBranches Mode flag (so we can turn off the check
  if we only care about syntactic correctness, e.g. for tests)

- adjusted test/goto.go error messages: the new branches
  checker only reports if a goto jumps into a block, but not
  which block (we may want to improve this again, eventually)

- also, the new branches checker reports one variable that
  is being jumped over by a goto, but it may not be the first
  one declared (this is fine either way)

- the new branches checker reports additional errors for
  fixedbugs/issue14006.go (not crucial to avoid those errors)

- the new branches checker now correctly reports only
  variable declarations being jumped over, rather than
  all declarations (issue 8042). Added respective tests.

Fixes #8042.

Change-Id: I53b6e1bda189748e1e1fb5b765a8a64337c27d40
Reviewed-on: https://go-review.googlesource.com/39998
Reviewed-by: Matthew Dempsky <mdempsky@google.com>
This commit is contained in:
Robert Griesemer 2017-04-07 15:41:45 -07:00
parent 1e3570ac86
commit 912a638b0c
9 changed files with 420 additions and 351 deletions

View file

@ -1,310 +0,0 @@
// Copyright 2017 The Go Authors. All rights reserved.
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file.
package gc
import (
"cmd/compile/internal/types"
"cmd/internal/src"
)
// checkcontrolflow checks fn's control flow structures for correctness.
// It catches:
// * misplaced breaks and continues
// * bad labeled break and continues
// * invalid, unused, duplicate, and missing labels
// * gotos jumping over declarations and into blocks
func checkcontrolflow(fn *Node) {
c := controlflow{
labels: make(map[string]*cfLabel),
labeledNodes: make(map[*Node]*cfLabel),
}
c.pushPos(fn.Pos)
c.stmtList(fn.Nbody)
// Check that we used all labels.
for name, lab := range c.labels {
if !lab.used() && !lab.reported && !lab.defNode.Used() {
yyerrorl(lab.defNode.Pos, "label %v defined and not used", name)
lab.reported = true
}
if lab.used() && !lab.defined() && !lab.reported {
yyerrorl(lab.useNode.Pos, "label %v not defined", name)
lab.reported = true
}
}
// Check any forward gotos. Non-forward gotos have already been checked.
for _, n := range c.fwdGotos {
lab := c.labels[n.Left.Sym.Name]
// If the label is undefined, we have already have printed an error.
if lab.defined() {
c.checkgoto(n, lab.defNode)
}
}
}
type controlflow struct {
// Labels and labeled control flow nodes (OFOR, OFORUNTIL, OSWITCH, OSELECT) in f.
labels map[string]*cfLabel
labeledNodes map[*Node]*cfLabel
// Gotos that jump forward; required for deferred checkgoto calls.
fwdGotos []*Node
// Breaks are allowed in loops, switches, and selects.
allowBreak bool
// Continues are allowed only in loops.
allowContinue bool
// Position stack. The current position is top of stack.
pos []src.XPos
}
// cfLabel is a label tracked by a controlflow.
type cfLabel struct {
ctlNode *Node // associated labeled control flow node
defNode *Node // label definition Node (OLABEL)
// Label use Node (OGOTO, OBREAK, OCONTINUE).
// There might be multiple uses, but we only need to track one.
useNode *Node
reported bool // reported indicates whether an error has already been reported for this label
}
// defined reports whether the label has a definition (OLABEL node).
func (l *cfLabel) defined() bool { return l.defNode != nil }
// used reports whether the label has a use (OGOTO, OBREAK, or OCONTINUE node).
func (l *cfLabel) used() bool { return l.useNode != nil }
// label returns the label associated with sym, creating it if necessary.
func (c *controlflow) label(sym *types.Sym) *cfLabel {
lab := c.labels[sym.Name]
if lab == nil {
lab = new(cfLabel)
c.labels[sym.Name] = lab
}
return lab
}
// stmtList checks l.
func (c *controlflow) stmtList(l Nodes) {
for _, n := range l.Slice() {
c.stmt(n)
}
}
// stmt checks n.
func (c *controlflow) stmt(n *Node) {
c.pushPos(n.Pos)
defer c.popPos()
c.stmtList(n.Ninit)
checkedNbody := false
switch n.Op {
case OLABEL:
sym := n.Left.Sym
lab := c.label(sym)
// Associate label with its control flow node, if any
if ctl := n.labeledControl(); ctl != nil {
c.labeledNodes[ctl] = lab
}
if !lab.defined() {
lab.defNode = n
} else {
c.err("label %v already defined at %v", sym, linestr(lab.defNode.Pos))
lab.reported = true
}
case OGOTO:
lab := c.label(n.Left.Sym)
if !lab.used() {
lab.useNode = n
}
if lab.defined() {
c.checkgoto(n, lab.defNode)
} else {
c.fwdGotos = append(c.fwdGotos, n)
}
case OCONTINUE, OBREAK:
if n.Left == nil {
// plain break/continue
if n.Op == OCONTINUE && !c.allowContinue {
c.err("%v is not in a loop", n.Op)
} else if !c.allowBreak {
c.err("%v is not in a loop, switch, or select", n.Op)
}
break
}
// labeled break/continue; look up the target
sym := n.Left.Sym
lab := c.label(sym)
if !lab.used() {
lab.useNode = n.Left
}
if !lab.defined() {
c.err("%v label not defined: %v", n.Op, sym)
lab.reported = true
break
}
ctl := lab.ctlNode
if n.Op == OCONTINUE && ctl != nil && (ctl.Op == OSWITCH || ctl.Op == OSELECT) {
// Cannot continue in a switch or select.
ctl = nil
}
if ctl == nil {
// Valid label but not usable with a break/continue here, e.g.:
// for {
// continue abc
// }
// abc:
// for {}
c.err("invalid %v label %v", n.Op, sym)
lab.reported = true
}
case OFOR, OFORUNTIL, OSWITCH, OSELECT:
// set up for continue/break in body
allowBreak := c.allowBreak
allowContinue := c.allowContinue
c.allowBreak = true
switch n.Op {
case OFOR, OFORUNTIL:
c.allowContinue = true
}
lab := c.labeledNodes[n]
if lab != nil {
// labeled for loop
lab.ctlNode = n
}
// check body
c.stmtList(n.Nbody)
checkedNbody = true
// tear down continue/break
c.allowBreak = allowBreak
c.allowContinue = allowContinue
if lab != nil {
lab.ctlNode = nil
}
}
if !checkedNbody {
c.stmtList(n.Nbody)
}
c.stmtList(n.List)
c.stmtList(n.Rlist)
}
// pushPos pushes a position onto the position stack.
func (c *controlflow) pushPos(pos src.XPos) {
if !pos.IsKnown() {
pos = c.peekPos()
if Debug['K'] != 0 {
Warn("controlflow: unknown position")
}
}
c.pos = append(c.pos, pos)
}
// popLine pops the top of the position stack.
func (c *controlflow) popPos() { c.pos = c.pos[:len(c.pos)-1] }
// peekPos peeks at the top of the position stack.
func (c *controlflow) peekPos() src.XPos { return c.pos[len(c.pos)-1] }
// err reports a control flow error at the current position.
func (c *controlflow) err(msg string, args ...interface{}) {
yyerrorl(c.peekPos(), msg, args...)
}
// checkgoto checks that a goto from from to to does not
// jump into a block or jump over variable declarations.
func (c *controlflow) checkgoto(from *Node, to *Node) {
if from.Op != OGOTO || to.Op != OLABEL {
Fatalf("bad from/to in checkgoto: %v -> %v", from, to)
}
// from and to's Sym fields record dclstack's value at their
// position, which implicitly encodes their block nesting
// level and variable declaration position within that block.
//
// For valid gotos, to.Sym will be a tail of from.Sym.
// Otherwise, any link in to.Sym not also in from.Sym
// indicates a block/declaration being jumped into/over.
//
// TODO(mdempsky): We should only complain about jumping over
// variable declarations, but currently we reject type and
// constant declarations too (#8042).
if from.Sym == to.Sym {
return
}
nf := dcldepth(from.Sym)
nt := dcldepth(to.Sym)
// Unwind from.Sym so it's no longer than to.Sym. It's okay to
// jump out of blocks or backwards past variable declarations.
fs := from.Sym
for ; nf > nt; nf-- {
fs = fs.Link
}
if fs == to.Sym {
return
}
// Decide what to complain about. Unwind to.Sym until where it
// forked from from.Sym, and keep track of the innermost block
// and declaration we jumped into/over.
var block *types.Sym
var dcl *types.Sym
// If to.Sym is longer, unwind until it's the same length.
ts := to.Sym
for ; nt > nf; nt-- {
if ts.Pkg == nil {
block = ts
} else {
dcl = ts
}
ts = ts.Link
}
// Same length; unwind until we find their common ancestor.
for ts != fs {
if ts.Pkg == nil {
block = ts
} else {
dcl = ts
}
ts = ts.Link
fs = fs.Link
}
// Prefer to complain about 'into block' over declarations.
pos := from.Left.Pos
if block != nil {
yyerrorl(pos, "goto %v jumps into block starting at %v", from.Left.Sym, linestr(block.Lastlineno))
} else {
yyerrorl(pos, "goto %v jumps over declaration of %v at %v", from.Left.Sym, dcl, linestr(dcl.Lastlineno))
}
}
// dcldepth returns the declaration depth for a dclstack Sym; that is,
// the sum of the block nesting level and the number of declarations
// in scope.
func dcldepth(s *types.Sym) int {
n := 0
for ; s != nil; s = s.Link {
n++
}
return n
}

View file

@ -36,7 +36,7 @@ func parseFiles(filenames []string) uint {
}
defer f.Close()
p.file, _ = syntax.Parse(base, f, p.error, p.pragma, 0) // errors are tracked via p.error
p.file, _ = syntax.Parse(base, f, p.error, p.pragma, syntax.CheckBranches) // errors are tracked via p.error
}(filename)
}

View file

@ -197,10 +197,6 @@ func compile(fn *Node) {
if nerrors != 0 {
return
}
checkcontrolflow(fn)
if nerrors != 0 {
return
}
if instrumenting {
instrument(fn)
}

View file

@ -0,0 +1,304 @@
// Copyright 2017 The Go Authors. All rights reserved.
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file.
package syntax
import (
"cmd/internal/src"
"fmt"
)
// TODO(gri) do this while parsing instead of in a separate pass?
// checkBranches checks correct use of labels and branch
// statements (break, continue, goto) in a function body.
// It catches:
// - misplaced breaks and continues
// - bad labeled breaks and continues
// - invalid, unused, duplicate, and missing labels
// - gotos jumping over variable declarations and into blocks
func checkBranches(body *BlockStmt, errh ErrorHandler) {
if body == nil {
return
}
// scope of all labels in this body
ls := &labelScope{errh: errh}
fwdGo2s := ls.blockBranches(nil, 0, nil, body.List)
// If there are any forward gotos left, no matching label was
// found for them. Either those labels were never defined, or
// they are inside blocks and not reachable from the gotos.
for _, go2 := range fwdGo2s {
var msg string
name := go2.Label.Value
if alt, found := ls.labels[name]; found {
msg = "goto %s jumps into block"
alt.used = true // avoid "defined and not used" error
} else {
msg = "label %s not defined"
}
ls.err(go2.Label.Pos(), msg, name)
}
// spec: "It is illegal to define a label that is never used."
for _, l := range ls.labels {
if !l.used {
l := l.lstmt.Label
ls.err(l.Pos(), "label %s defined and not used", l.Value)
}
}
}
type labelScope struct {
errh ErrorHandler
labels map[string]*label // all label declarations inside the function; allocated lazily
}
type label struct {
parent *block // block containing this label declaration
lstmt *LabeledStmt // statement declaring the label
used bool // whether the label is used or not
}
type block struct {
parent *block // immediately enclosing block, or nil
lstmt *LabeledStmt // labeled statement to which this block belongs, or nil
}
func (ls *labelScope) err(pos src.Pos, format string, args ...interface{}) {
ls.errh(Error{pos, fmt.Sprintf(format, args...)})
}
// declare declares the label introduced by s in block b and returns
// the new label. If the label was already declared, declare reports
// and error and the existing label is returned instead.
func (ls *labelScope) declare(b *block, s *LabeledStmt) *label {
name := s.Label.Value
labels := ls.labels
if labels == nil {
labels = make(map[string]*label)
ls.labels = labels
} else if alt := labels[name]; alt != nil {
ls.err(s.Pos(), "label %s already defined at %s", name, alt.lstmt.Label.Pos().String())
return alt
}
l := &label{b, s, false}
labels[name] = l
return l
}
// gotoTarget returns the labeled statement matching the given name and
// declared in block b or any of its enclosing blocks. The result is nil
// if the label is not defined, or doesn't match a valid labeled statement.
func (ls *labelScope) gotoTarget(b *block, name string) *label {
if l := ls.labels[name]; l != nil {
l.used = true // even if it's not a valid target
for ; b != nil; b = b.parent {
if l.parent == b {
return l
}
}
}
return nil
}
var invalid = new(LabeledStmt) // singleton to signal invalid enclosing target
// enclosingTarget returns the innermost enclosing labeled statement matching
// the given name. The result is nil if the label is not defined, and invalid
// if the label is defined but doesn't label a valid labeled statement.
func (ls *labelScope) enclosingTarget(b *block, name string) *LabeledStmt {
if l := ls.labels[name]; l != nil {
l.used = true // even if it's not a valid target (see e.g., test/fixedbugs/bug136.go)
for ; b != nil; b = b.parent {
if l.lstmt == b.lstmt {
return l.lstmt
}
}
return invalid
}
return nil
}
// context flags
const (
breakOk = 1 << iota
continueOk
)
// blockBranches processes a block's body and returns the list of unresolved (forward) gotos.
// parent is the immediately enclosing block (or nil), context provides information about the
// enclosing statements, and lstmt is the labeled statement this body belongs to, or nil.
func (ls *labelScope) blockBranches(parent *block, context uint, lstmt *LabeledStmt, body []Stmt) []*BranchStmt {
b := &block{parent: parent, lstmt: lstmt}
var varPos src.Pos
var varName Expr
var fwdGo2s, badGo2s []*BranchStmt
recordVarDecl := func(pos src.Pos, name Expr) {
varPos = pos
varName = name
// Any existing forward goto jumping over the variable
// declaration is invalid. The goto may still jump out
// of the block and be ok, but we don't know that yet.
// Remember all forward gotos as potential bad gotos.
badGo2s = append(badGo2s[:0], fwdGo2s...)
}
jumpsOverVarDecl := func(go2 *BranchStmt) bool {
if varPos.IsKnown() {
for _, bad := range badGo2s {
if go2 == bad {
return true
}
}
}
return false
}
innerBlock := func(flags uint, body []Stmt) {
fwdGo2s = append(fwdGo2s, ls.blockBranches(b, context|flags, lstmt, body)...)
}
for _, stmt := range body {
lstmt = nil
L:
switch s := stmt.(type) {
case *DeclStmt:
for _, d := range s.DeclList {
if v, ok := d.(*VarDecl); ok {
recordVarDecl(v.Pos(), v.NameList[0])
break // the first VarDecl will do
}
}
case *LabeledStmt:
// declare non-blank label
if name := s.Label.Value; name != "_" {
l := ls.declare(b, s)
// resolve matching forward gotos
i := 0
for _, go2 := range fwdGo2s {
if go2.Label.Value == name {
l.used = true
if jumpsOverVarDecl(go2) {
ls.err(
go2.Label.Pos(),
"goto %s jumps over declaration of %s at %s",
name, String(varName), varPos,
)
}
} else {
// no match - keep forward goto
fwdGo2s[i] = go2
i++
}
}
fwdGo2s = fwdGo2s[:i]
lstmt = s
}
// process labeled statement
stmt = s.Stmt
goto L
case *BranchStmt:
// unlabeled branch statement
if s.Label == nil {
switch s.Tok {
case _Break:
if context&breakOk == 0 {
ls.err(s.Pos(), "break is not in a loop, switch, or select")
}
case _Continue:
if context&continueOk == 0 {
ls.err(s.Pos(), "continue is not in a loop")
}
case _Fallthrough:
// nothing to do
case _Goto:
fallthrough // should always have a label
default:
panic("invalid BranchStmt")
}
break
}
// labeled branch statement
name := s.Label.Value
switch s.Tok {
case _Break:
// spec: "If there is a label, it must be that of an enclosing
// "for", "switch", or "select" statement, and that is the one
// whose execution terminates."
if t := ls.enclosingTarget(b, name); t != nil {
valid := false
switch t.Stmt.(type) {
case *SwitchStmt, *SelectStmt, *ForStmt:
valid = true
}
if !valid {
ls.err(s.Label.Pos(), "invalid break label %s", name)
}
} else {
ls.err(s.Label.Pos(), "break label not defined: %s", name)
}
case _Continue:
// spec: "If there is a label, it must be that of an enclosing
// "for" statement, and that is the one whose execution advances."
if t := ls.enclosingTarget(b, name); t != nil {
if _, ok := t.Stmt.(*ForStmt); !ok {
ls.err(s.Label.Pos(), "invalid continue label %s", name)
}
} else {
ls.err(s.Label.Pos(), "continue label not defined: %s", name)
}
case _Goto:
if ls.gotoTarget(b, name) == nil {
// label may be declared later - add goto to forward gotos
fwdGo2s = append(fwdGo2s, s)
}
case _Fallthrough:
fallthrough // should never have a label
default:
panic("invalid BranchStmt")
}
case *AssignStmt:
if s.Op == Def {
recordVarDecl(s.Pos(), s.Lhs)
}
case *BlockStmt:
// Unresolved forward gotos from the nested block
// become forward gotos for the current block.
innerBlock(0, s.List)
case *IfStmt:
innerBlock(0, s.Then.List)
if s.Else != nil {
innerBlock(0, []Stmt{s.Else})
}
case *ForStmt:
innerBlock(breakOk|continueOk, s.Body.List)
case *SwitchStmt:
for _, cc := range s.Body {
innerBlock(breakOk, cc.Body)
}
case *SelectStmt:
for _, cc := range s.Body {
innerBlock(breakOk, cc.Body)
}
}
}
return fwdGo2s
}

View file

@ -18,6 +18,7 @@ const trace = false
type parser struct {
base *src.PosBase
errh ErrorHandler
mode Mode
scanner
first error // first error encountered
@ -28,9 +29,10 @@ type parser struct {
indent []byte // tracing support
}
func (p *parser) init(base *src.PosBase, r io.Reader, errh ErrorHandler, pragh PragmaHandler) {
func (p *parser) init(base *src.PosBase, r io.Reader, errh ErrorHandler, pragh PragmaHandler, mode Mode) {
p.base = base
p.errh = errh
p.mode = mode
p.scanner.init(
r,
// Error and pragma handlers for scanner.
@ -494,6 +496,9 @@ func (p *parser) funcDeclOrNil() *FuncDecl {
f.Type = p.funcType()
if p.tok == _Lbrace {
f.Body = p.blockStmt("")
if p.mode&CheckBranches != 0 {
checkBranches(f.Body, p.errh)
}
}
f.Pragma = p.pragma
@ -722,6 +727,9 @@ func (p *parser) operand(keep_parens bool) Expr {
f.pos = pos
f.Type = t
f.Body = p.blockStmt("")
if p.mode&CheckBranches != 0 {
checkBranches(f.Body, p.errh)
}
p.xnest--
return f

View file

@ -14,6 +14,11 @@ import (
// Mode describes the parser mode.
type Mode uint
// Modes supported by the parser.
const (
CheckBranches Mode = 1 << iota // check correct use of labels, break, continue, and goto statements
)
// Error describes a syntax error. Error implements the error interface.
type Error struct {
Pos src.Pos
@ -63,7 +68,7 @@ func Parse(base *src.PosBase, src io.Reader, errh ErrorHandler, pragh PragmaHand
}()
var p parser
p.init(base, src, errh, pragh)
p.init(base, src, errh, pragh, mode)
p.next()
return p.fileOrNil(), p.first
}

View file

@ -53,12 +53,12 @@ func f() {
switch {
case z:
labelname:
labelname: // ERROR "label labelname defined and not used"
}
switch {
case z:
labelname: ;
labelname: ; // ERROR "label labelname already defined at LINE-5"
case false:
}
}

View file

@ -0,0 +1,66 @@
// compile
// Copyright 2017 The Go Authors. All rights reserved.
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file.
// Verify that gotos across non-variable declarations
// are accepted.
package p
func _() {
goto L1
const x = 0
L1:
goto L2
type T int
L2:
}
func _() {
{
goto L1
}
const x = 0
L1:
{
goto L2
}
type T int
L2:
}
func _(d int) {
if d > 0 {
goto L1
} else {
goto L2
}
const x = 0
L1:
switch d {
case 1:
goto L3
case 2:
default:
goto L4
}
type T1 int
L2:
const y = 1
L3:
for d > 0 {
if d < 10 {
goto L4
}
}
type T2 int
L4:
select {
default:
goto L5
}
type T3 int
L5:
}

View file

@ -77,7 +77,7 @@ L:
// error shows first offending variable
func _() {
goto L // ERROR "goto L jumps over declaration of x at LINE+1|goto jumps over declaration"
goto L // ERROR "goto L jumps over declaration of x at LINE+1|goto L jumps over declaration of y at LINE+3|goto jumps over declaration"
x := 1 // GCCGO_ERROR "defined here"
_ = x
y := 1
@ -87,7 +87,7 @@ L:
// goto not okay even if code path is dead
func _() {
goto L // ERROR "goto L jumps over declaration of x at LINE+1|goto jumps over declaration"
goto L // ERROR "goto L jumps over declaration of x at LINE+1|goto L jumps over declaration of y at LINE+3|goto jumps over declaration"
x := 1 // GCCGO_ERROR "defined here"
_ = x
y := 1
@ -114,7 +114,7 @@ L:
// goto into inner block not okay
func _() {
goto L // ERROR "goto L jumps into block starting at LINE+1|goto jumps into block"
goto L // ERROR "goto L jumps into block starting at LINE+1|goto L jumps into block|goto jumps into block"
{ // GCCGO_ERROR "block starts here"
L:
}
@ -125,12 +125,12 @@ func _() {
{ // GCCGO_ERROR "block starts here"
L:
}
goto L // ERROR "goto L jumps into block starting at LINE-3|goto jumps into block"
goto L // ERROR "goto L jumps into block starting at LINE-3|goto L jumps into block|goto jumps into block"
}
// error shows first (outermost) offending block
func _() {
goto L // ERROR "goto L jumps into block starting at LINE+1|goto jumps into block"
goto L // ERROR "goto L jumps into block starting at LINE+1|goto L jumps into block|goto jumps into block"
{
{
{ // GCCGO_ERROR "block starts here"
@ -142,7 +142,7 @@ func _() {
// error prefers block diagnostic over declaration diagnostic
func _() {
goto L // ERROR "goto L jumps into block starting at LINE+3|goto jumps into block"
goto L // ERROR "goto L jumps into block starting at LINE+3|goto L jumps into block|goto jumps into block"
x := 1
_ = x
{ // GCCGO_ERROR "block starts here"
@ -179,14 +179,14 @@ L:
}
func _() {
goto L // ERROR "goto L jumps into block starting at LINE+1|goto jumps into block"
goto L // ERROR "goto L jumps into block starting at LINE+1|goto L jumps into block|goto jumps into block"
if true { // GCCGO_ERROR "block starts here"
L:
}
}
func _() {
goto L // ERROR "goto L jumps into block starting at LINE+1|goto jumps into block"
goto L // ERROR "goto L jumps into block starting at LINE+1|goto L jumps into block|goto jumps into block"
if true { // GCCGO_ERROR "block starts here"
L:
} else {
@ -194,7 +194,7 @@ func _() {
}
func _() {
goto L // ERROR "goto L jumps into block starting at LINE+1|goto jumps into block"
goto L // ERROR "goto L jumps into block starting at LINE+1|goto L jumps into block|goto jumps into block"
if true {
} else { // GCCGO_ERROR "block starts here"
L:
@ -205,13 +205,13 @@ func _() {
if false { // GCCGO_ERROR "block starts here"
L:
} else {
goto L // ERROR "goto L jumps into block starting at LINE-3|goto jumps into block"
goto L // ERROR "goto L jumps into block starting at LINE-3|goto L jumps into block|goto jumps into block"
}
}
func _() {
if true {
goto L // ERROR "goto L jumps into block starting at LINE+1|goto jumps into block"
goto L // ERROR "goto L jumps into block starting at LINE+1|goto L jumps into block|goto jumps into block"
} else { // GCCGO_ERROR "block starts here"
L:
}
@ -219,7 +219,7 @@ func _() {
func _() {
if true {
goto L // ERROR "goto L jumps into block starting at LINE+1|goto jumps into block"
goto L // ERROR "goto L jumps into block starting at LINE+1|goto L jumps into block|goto jumps into block"
} else if false { // GCCGO_ERROR "block starts here"
L:
}
@ -227,7 +227,7 @@ func _() {
func _() {
if true {
goto L // ERROR "goto L jumps into block starting at LINE+1|goto jumps into block"
goto L // ERROR "goto L jumps into block starting at LINE+1|goto L jumps into block|goto jumps into block"
} else if false { // GCCGO_ERROR "block starts here"
L:
} else {
@ -241,7 +241,7 @@ func _() {
// really is LINE+1 (like in the previous test),
// even though it looks like it might be LINE+3 instead.
if true {
goto L // ERROR "goto L jumps into block starting at LINE+1|goto jumps into block"
goto L // ERROR "goto L jumps into block starting at LINE+1|goto L jumps into block|goto jumps into block"
} else if false {
} else { // GCCGO_ERROR "block starts here"
L:
@ -290,7 +290,7 @@ func _() {
for { // GCCGO_ERROR "block starts here"
L:
}
goto L // ERROR "goto L jumps into block starting at LINE-3|goto jumps into block"
goto L // ERROR "goto L jumps into block starting at LINE-3|goto L jumps into block|goto jumps into block"
}
func _() {
@ -299,49 +299,49 @@ func _() {
L1:
}
L:
goto L1 // ERROR "goto L1 jumps into block starting at LINE-5|goto jumps into block"
goto L1 // ERROR "goto L1 jumps into block starting at LINE-5|goto L1 jumps into block|goto jumps into block"
}
func _() {
for i < n { // GCCGO_ERROR "block starts here"
L:
}
goto L // ERROR "goto L jumps into block starting at LINE-3|goto jumps into block"
goto L // ERROR "goto L jumps into block starting at LINE-3|goto L jumps into block|goto jumps into block"
}
func _() {
for i = 0; i < n; i++ { // GCCGO_ERROR "block starts here"
L:
}
goto L // ERROR "goto L jumps into block starting at LINE-3|goto jumps into block"
goto L // ERROR "goto L jumps into block starting at LINE-3|goto L jumps into block|goto jumps into block"
}
func _() {
for i = range x { // GCCGO_ERROR "block starts here"
L:
}
goto L // ERROR "goto L jumps into block starting at LINE-3|goto jumps into block"
goto L // ERROR "goto L jumps into block starting at LINE-3|goto L jumps into block|goto jumps into block"
}
func _() {
for i = range c { // GCCGO_ERROR "block starts here"
L:
}
goto L // ERROR "goto L jumps into block starting at LINE-3|goto jumps into block"
goto L // ERROR "goto L jumps into block starting at LINE-3|goto L jumps into block|goto jumps into block"
}
func _() {
for i = range m { // GCCGO_ERROR "block starts here"
L:
}
goto L // ERROR "goto L jumps into block starting at LINE-3|goto jumps into block"
goto L // ERROR "goto L jumps into block starting at LINE-3|goto L jumps into block|goto jumps into block"
}
func _() {
for i = range s { // GCCGO_ERROR "block starts here"
L:
}
goto L // ERROR "goto L jumps into block starting at LINE-3|goto jumps into block"
goto L // ERROR "goto L jumps into block starting at LINE-3|goto L jumps into block|goto jumps into block"
}
// switch
@ -395,7 +395,7 @@ func _() {
}
func _() {
goto L // ERROR "goto L jumps into block starting at LINE+1|goto jumps into block"
goto L // ERROR "goto L jumps into block starting at LINE+1|goto L jumps into block|goto jumps into block"
switch i {
case 0:
L: // GCCGO_ERROR "block starts here"
@ -403,7 +403,7 @@ func _() {
}
func _() {
goto L // ERROR "goto L jumps into block starting at LINE+1|goto jumps into block"
goto L // ERROR "goto L jumps into block starting at LINE+1|goto L jumps into block|goto jumps into block"
switch i {
case 0:
L: // GCCGO_ERROR "block starts here"
@ -413,7 +413,7 @@ func _() {
}
func _() {
goto L // ERROR "goto L jumps into block starting at LINE+1|goto jumps into block"
goto L // ERROR "goto L jumps into block starting at LINE+1|goto L jumps into block|goto jumps into block"
switch i {
case 0:
default:
@ -424,7 +424,7 @@ func _() {
func _() {
switch i {
default:
goto L // ERROR "goto L jumps into block starting at LINE+1|goto jumps into block"
goto L // ERROR "goto L jumps into block starting at LINE+1|goto L jumps into block|goto jumps into block"
case 0:
L: // GCCGO_ERROR "block starts here"
}
@ -436,7 +436,7 @@ func _() {
L: // GCCGO_ERROR "block starts here"
;
default:
goto L // ERROR "goto L jumps into block starting at LINE-4|goto jumps into block"
goto L // ERROR "goto L jumps into block starting at LINE-4|goto L jumps into block|goto jumps into block"
}
}
@ -492,7 +492,7 @@ func _() {
}
func _() {
goto L // ERROR "goto L jumps into block starting at LINE+2|goto jumps into block"
goto L // ERROR "goto L jumps into block starting at LINE+2|goto L jumps into block|goto jumps into block"
select {
case c <- 1:
L: // GCCGO_ERROR "block starts here"
@ -500,7 +500,7 @@ func _() {
}
func _() {
goto L // ERROR "goto L jumps into block starting at LINE+2|goto jumps into block"
goto L // ERROR "goto L jumps into block starting at LINE+2|goto L jumps into block|goto jumps into block"
select {
case c <- 1:
L: // GCCGO_ERROR "block starts here"
@ -510,7 +510,7 @@ func _() {
}
func _() {
goto L // ERROR "goto L jumps into block starting at LINE+3|goto jumps into block"
goto L // ERROR "goto L jumps into block starting at LINE+3|goto L jumps into block|goto jumps into block"
select {
case <-c:
default:
@ -521,7 +521,7 @@ func _() {
func _() {
select {
default:
goto L // ERROR "goto L jumps into block starting at LINE+1|goto jumps into block"
goto L // ERROR "goto L jumps into block starting at LINE+1|goto L jumps into block|goto jumps into block"
case <-c:
L: // GCCGO_ERROR "block starts here"
}
@ -533,6 +533,6 @@ func _() {
L: // GCCGO_ERROR "block starts here"
;
default:
goto L // ERROR "goto L jumps into block starting at LINE-4|goto jumps into block"
goto L // ERROR "goto L jumps into block starting at LINE-4|goto L jumps into block|goto jumps into block"
}
}