Commit fc31d4e2 authored by Robert Griesemer's avatar Robert Griesemer

go/parser: improved error recovery after missing type

R=go1.11

This CL also introduces a new TODO in parser.go. To be
addressed in a separate CL to make this easier to review.

Also: Make parser's test harness easier to use by ignoring
auto-inserted (invisible) semicolons when computing error
positions. Adjusted testdata/commas.src accordingly.

Fixes #23434.

Change-Id: I050592d11d5f984f71185548394c000eea509205
Reviewed-on: https://go-review.googlesource.com/87898Reviewed-by: default avatarMatthew Dempsky <mdempsky@google.com>
parent ea006a85
...@@ -91,6 +91,12 @@ func expectedErrors(fset *token.FileSet, filename string, src []byte) map[token. ...@@ -91,6 +91,12 @@ func expectedErrors(fset *token.FileSet, filename string, src []byte) map[token.
} }
errors[pos] = string(s[2]) errors[pos] = string(s[2])
} }
case token.SEMICOLON:
// don't use the position of auto-inserted (invisible) semicolons
if lit != ";" {
break
}
fallthrough
default: default:
prev = pos prev = pos
var l int // token length var l int // token length
......
...@@ -419,7 +419,7 @@ func (p *parser) expectSemi() { ...@@ -419,7 +419,7 @@ func (p *parser) expectSemi() {
p.next() p.next()
default: default:
p.errorExpected(p.pos, "';'") p.errorExpected(p.pos, "';'")
syncStmt(p) p.syncStmt()
} }
} }
} }
...@@ -445,10 +445,12 @@ func assert(cond bool, msg string) { ...@@ -445,10 +445,12 @@ func assert(cond bool, msg string) {
} }
} }
// TODO(gri) The syncX methods below all use the same pattern. Factor.
// syncStmt advances to the next statement. // syncStmt advances to the next statement.
// Used for synchronization after an error. // Used for synchronization after an error.
// //
func syncStmt(p *parser) { func (p *parser) syncStmt() {
for { for {
switch p.tok { switch p.tok {
case token.BREAK, token.CONST, token.CONTINUE, token.DEFER, case token.BREAK, token.CONST, token.CONTINUE, token.DEFER,
...@@ -486,7 +488,7 @@ func syncStmt(p *parser) { ...@@ -486,7 +488,7 @@ func syncStmt(p *parser) {
// syncDecl advances to the next declaration. // syncDecl advances to the next declaration.
// Used for synchronization after an error. // Used for synchronization after an error.
// //
func syncDecl(p *parser) { func (p *parser) syncDecl() {
for { for {
switch p.tok { switch p.tok {
case token.CONST, token.TYPE, token.VAR: case token.CONST, token.TYPE, token.VAR:
...@@ -507,6 +509,30 @@ func syncDecl(p *parser) { ...@@ -507,6 +509,30 @@ func syncDecl(p *parser) {
} }
} }
// syncExprEnd advances to the likely end of an expression.
// Used for synchronization after an error.
//
func (p *parser) syncExprEnd() {
for {
switch p.tok {
case token.COMMA, token.COLON, token.SEMICOLON, token.RPAREN, token.RBRACK, token.RBRACE:
// see comments in syncStmt
if p.pos == p.syncPos && p.syncCnt < 10 {
p.syncCnt++
return
}
if p.pos > p.syncPos {
p.syncPos = p.pos
p.syncCnt = 0
return
}
case token.EOF:
return
}
p.next()
}
}
// safePos returns a valid file position for a given position: If pos // safePos returns a valid file position for a given position: If pos
// is valid to begin with, safePos returns pos. If pos is out-of-range, // is valid to begin with, safePos returns pos. If pos is out-of-range,
// safePos returns the EOF position. // safePos returns the EOF position.
...@@ -623,7 +649,7 @@ func (p *parser) parseType() ast.Expr { ...@@ -623,7 +649,7 @@ func (p *parser) parseType() ast.Expr {
if typ == nil { if typ == nil {
pos := p.pos pos := p.pos
p.errorExpected(pos, "type") p.errorExpected(pos, "type")
p.next() // make progress p.syncExprEnd()
return &ast.BadExpr{From: pos, To: p.pos} return &ast.BadExpr{From: pos, To: p.pos}
} }
...@@ -1166,7 +1192,7 @@ func (p *parser) parseOperand(lhs bool) ast.Expr { ...@@ -1166,7 +1192,7 @@ func (p *parser) parseOperand(lhs bool) ast.Expr {
// we have an error // we have an error
pos := p.pos pos := p.pos
p.errorExpected(pos, "operand") p.errorExpected(pos, "operand")
syncStmt(p) p.syncStmt()
return &ast.BadExpr{From: pos, To: p.pos} return &ast.BadExpr{From: pos, To: p.pos}
} }
...@@ -2202,7 +2228,7 @@ func (p *parser) parseStmt() (s ast.Stmt) { ...@@ -2202,7 +2228,7 @@ func (p *parser) parseStmt() (s ast.Stmt) {
switch p.tok { switch p.tok {
case token.CONST, token.TYPE, token.VAR: case token.CONST, token.TYPE, token.VAR:
s = &ast.DeclStmt{Decl: p.parseDecl(syncStmt)} s = &ast.DeclStmt{Decl: p.parseDecl((*parser).syncStmt)}
case case
// tokens that may start an expression // tokens that may start an expression
token.IDENT, token.INT, token.FLOAT, token.IMAG, token.CHAR, token.STRING, token.FUNC, token.LPAREN, // operands token.IDENT, token.INT, token.FLOAT, token.IMAG, token.CHAR, token.STRING, token.FUNC, token.LPAREN, // operands
...@@ -2247,7 +2273,7 @@ func (p *parser) parseStmt() (s ast.Stmt) { ...@@ -2247,7 +2273,7 @@ func (p *parser) parseStmt() (s ast.Stmt) {
// no statement found // no statement found
pos := p.pos pos := p.pos
p.errorExpected(pos, "statement") p.errorExpected(pos, "statement")
syncStmt(p) p.syncStmt()
s = &ast.BadStmt{From: pos, To: p.pos} s = &ast.BadStmt{From: pos, To: p.pos}
} }
...@@ -2530,7 +2556,7 @@ func (p *parser) parseFile() *ast.File { ...@@ -2530,7 +2556,7 @@ func (p *parser) parseFile() *ast.File {
if p.mode&ImportsOnly == 0 { if p.mode&ImportsOnly == 0 {
// rest of package body // rest of package body
for p.tok != token.EOF { for p.tok != token.EOF {
decls = append(decls, p.parseDecl(syncDecl)) decls = append(decls, p.parseDecl((*parser).syncDecl))
} }
} }
} }
......
...@@ -8,12 +8,12 @@ ...@@ -8,12 +8,12 @@
package p package p
var _ = []int{ var _ = []int{
0 /* ERROR "missing ','" */ 0/* ERROR HERE "missing ','" */
} }
var _ = []int{ var _ = []int{
0, 0,
1, 1,
2, 2,
3 /* ERROR "missing ','" */ 3/* ERROR HERE "missing ','" */
} }
// Copyright 2018 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.
// Test case for issue 23434: Better synchronization of
// parser after missing type. There should be exactly
// one error each time, with now follow errors.
package p
func g() {
m := make(map[string]! /* ERROR "expected type, found '!'" */ )
for {
x := 1
print(x)
}
}
func f() {
m := make(map[string]) /* ERROR "expected type, found '\)'" */
for {
x := 1
print(x)
}
}
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment