Commit 568c4617 authored by Alan Donovan's avatar Alan Donovan

go/ast: add {L,R}paren token.Pos field to ast.TypeAssertExpr.

These are required to correctly determine the End() of the node.

Also set these fields in go/parser and use them in go/printer.

This is a backward-compatible API change.

R=gri, r
CC=golang-dev
https://golang.org/cl/10189043
parent b9ddb0d8
...@@ -304,8 +304,10 @@ type ( ...@@ -304,8 +304,10 @@ type (
// type assertion. // type assertion.
// //
TypeAssertExpr struct { TypeAssertExpr struct {
X Expr // expression X Expr // expression
Type Expr // asserted type; nil means type switch X.(type) Lparen token.Pos // position of "("
Type Expr // asserted type; nil means type switch X.(type)
Rparen token.Pos // position of ")"
} }
// A CallExpr node represents an expression followed by an argument list. // A CallExpr node represents an expression followed by an argument list.
...@@ -456,26 +458,21 @@ func (x *Ellipsis) End() token.Pos { ...@@ -456,26 +458,21 @@ func (x *Ellipsis) End() token.Pos {
} }
return x.Ellipsis + 3 // len("...") return x.Ellipsis + 3 // len("...")
} }
func (x *BasicLit) End() token.Pos { return token.Pos(int(x.ValuePos) + len(x.Value)) } func (x *BasicLit) End() token.Pos { return token.Pos(int(x.ValuePos) + len(x.Value)) }
func (x *FuncLit) End() token.Pos { return x.Body.End() } func (x *FuncLit) End() token.Pos { return x.Body.End() }
func (x *CompositeLit) End() token.Pos { return x.Rbrace + 1 } func (x *CompositeLit) End() token.Pos { return x.Rbrace + 1 }
func (x *ParenExpr) End() token.Pos { return x.Rparen + 1 } func (x *ParenExpr) End() token.Pos { return x.Rparen + 1 }
func (x *SelectorExpr) End() token.Pos { return x.Sel.End() } func (x *SelectorExpr) End() token.Pos { return x.Sel.End() }
func (x *IndexExpr) End() token.Pos { return x.Rbrack + 1 } func (x *IndexExpr) End() token.Pos { return x.Rbrack + 1 }
func (x *SliceExpr) End() token.Pos { return x.Rbrack + 1 } func (x *SliceExpr) End() token.Pos { return x.Rbrack + 1 }
func (x *TypeAssertExpr) End() token.Pos { func (x *TypeAssertExpr) End() token.Pos { return x.Rparen + 1 }
if x.Type != nil { func (x *CallExpr) End() token.Pos { return x.Rparen + 1 }
return x.Type.End() func (x *StarExpr) End() token.Pos { return x.X.End() }
} func (x *UnaryExpr) End() token.Pos { return x.X.End() }
return x.X.End() func (x *BinaryExpr) End() token.Pos { return x.Y.End() }
} func (x *KeyValueExpr) End() token.Pos { return x.Value.End() }
func (x *CallExpr) End() token.Pos { return x.Rparen + 1 } func (x *ArrayType) End() token.Pos { return x.Elt.End() }
func (x *StarExpr) End() token.Pos { return x.X.End() } func (x *StructType) End() token.Pos { return x.Fields.End() }
func (x *UnaryExpr) End() token.Pos { return x.X.End() }
func (x *BinaryExpr) End() token.Pos { return x.Y.End() }
func (x *KeyValueExpr) End() token.Pos { return x.Value.End() }
func (x *ArrayType) End() token.Pos { return x.Elt.End() }
func (x *StructType) End() token.Pos { return x.Fields.End() }
func (x *FuncType) End() token.Pos { func (x *FuncType) End() token.Pos {
if x.Results != nil { if x.Results != nil {
return x.Results.End() return x.Results.End()
......
...@@ -1150,7 +1150,7 @@ func (p *parser) parseTypeAssertion(x ast.Expr) ast.Expr { ...@@ -1150,7 +1150,7 @@ func (p *parser) parseTypeAssertion(x ast.Expr) ast.Expr {
defer un(trace(p, "TypeAssertion")) defer un(trace(p, "TypeAssertion"))
} }
p.expect(token.LPAREN) lparen := p.expect(token.LPAREN)
var typ ast.Expr var typ ast.Expr
if p.tok == token.TYPE { if p.tok == token.TYPE {
// type switch: typ == nil // type switch: typ == nil
...@@ -1158,9 +1158,9 @@ func (p *parser) parseTypeAssertion(x ast.Expr) ast.Expr { ...@@ -1158,9 +1158,9 @@ func (p *parser) parseTypeAssertion(x ast.Expr) ast.Expr {
} else { } else {
typ = p.parseType() typ = p.parseType()
} }
p.expect(token.RPAREN) rparen := p.expect(token.RPAREN)
return &ast.TypeAssertExpr{X: x, Type: typ} return &ast.TypeAssertExpr{X: x, Type: typ, Lparen: lparen, Rparen: rparen}
} }
func (p *parser) parseIndexOrSlice(x ast.Expr) ast.Expr { func (p *parser) parseIndexOrSlice(x ast.Expr) ast.Expr {
......
...@@ -754,13 +754,13 @@ func (p *printer) expr1(expr ast.Expr, prec1, depth int) { ...@@ -754,13 +754,13 @@ func (p *printer) expr1(expr ast.Expr, prec1, depth int) {
case *ast.TypeAssertExpr: case *ast.TypeAssertExpr:
p.expr1(x.X, token.HighestPrec, depth) p.expr1(x.X, token.HighestPrec, depth)
p.print(token.PERIOD, token.LPAREN) p.print(token.PERIOD, x.Lparen, token.LPAREN)
if x.Type != nil { if x.Type != nil {
p.expr(x.Type) p.expr(x.Type)
} else { } else {
p.print(token.TYPE) p.print(token.TYPE)
} }
p.print(token.RPAREN) p.print(x.Rparen, token.RPAREN)
case *ast.IndexExpr: case *ast.IndexExpr:
// TODO(gri): should treat[] like parentheses and undo one level of depth // TODO(gri): should treat[] like parentheses and undo one level of depth
......
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