Commit f39b518b authored by Robert Griesemer's avatar Robert Griesemer

- fixed a couple of corner cases (empty statements, empty composites)

- more robust printing in presence of errors
- fixed incorrect printing of function literals

R=r
OCL=17378
CL=17378
parent ba46bd1e
...@@ -23,17 +23,7 @@ pretty: $(PRETTY_OBJS) ...@@ -23,17 +23,7 @@ pretty: $(PRETTY_OBJS)
$(GO) $(LDFLAGS) -o $@ $(PRETTY_OBJS) $(GO) $(LDFLAGS) -o $@ $(PRETTY_OBJS)
test: pretty test: pretty
pretty -s *.go test.sh
pretty -s ../gosrc/*.go
pretty -s $(GOROOT)/test/sieve.go
pretty -s $(GOROOT)/src/pkg/*.go
pretty -s $(GOROOT)/src/lib/flag.go
pretty -s $(GOROOT)/src/lib/fmt.go
pretty -s $(GOROOT)/src/lib/rand.go
pretty -s $(GOROOT)/src/lib/math/*.go
pretty -s $(GOROOT)/src/lib/container/*.go
pretty -s $(GOROOT)/src/syscall/*.go
echo "DONE"
install: pretty install: pretty
cp pretty $(HOME)/bin/pretty cp pretty $(HOME)/bin/pretty
......
...@@ -74,8 +74,10 @@ export func NewList() *List { ...@@ -74,8 +74,10 @@ export func NewList() *List {
export type Expr struct { export type Expr struct {
pos, tok int; pos, tok int;
x, y *Expr; // binary (x, y) and unary (y) expressions x, y *Expr; // binary (x, y) and unary (y) expressions
// TODO find a more space efficient way to hold these
s string; // identifiers and literals s string; // identifiers and literals
t *Type; // operands that are types t *Type; // type expressions, function literal types
block *List; // stats for function literals
} }
...@@ -108,6 +110,9 @@ export func NewLit(pos, tok int, s string) *Expr { ...@@ -108,6 +110,9 @@ export func NewLit(pos, tok int, s string) *Expr {
} }
export var BadExpr = NewExpr(0, Scanner.ILLEGAL, nil, nil);
// ---------------------------------------------------------------------------- // ----------------------------------------------------------------------------
// Types // Types
...@@ -159,6 +164,9 @@ export func NewTypeExpr(t *Type) *Expr { ...@@ -159,6 +164,9 @@ export func NewTypeExpr(t *Type) *Expr {
} }
export var BadType = NewType(0, Scanner.ILLEGAL);
// ---------------------------------------------------------------------------- // ----------------------------------------------------------------------------
// Statements // Statements
...@@ -178,6 +186,9 @@ export func NewStat(pos, tok int) *Stat { ...@@ -178,6 +186,9 @@ export func NewStat(pos, tok int) *Stat {
} }
export var BadStat = NewStat(0, Scanner.ILLEGAL);
// ---------------------------------------------------------------------------- // ----------------------------------------------------------------------------
// Declarations // Declarations
...@@ -200,6 +211,9 @@ export func NewDecl(pos, tok int, exported bool) *Decl { ...@@ -200,6 +211,9 @@ export func NewDecl(pos, tok int, exported bool) *Decl {
} }
export var BadDecl = NewDecl(0, Scanner.ILLEGAL, false);
// ---------------------------------------------------------------------------- // ----------------------------------------------------------------------------
// Program // Program
......
...@@ -65,6 +65,7 @@ func (P *Parser) Next0() { ...@@ -65,6 +65,7 @@ func (P *Parser) Next0() {
P.tok, P.pos, P.val = t.tok, t.pos, t.val; P.tok, P.pos, P.val = t.tok, t.pos, t.val;
} }
P.opt_semi = false; P.opt_semi = false;
if P.verbose { if P.verbose {
P.PrintIndent(); P.PrintIndent();
print("[", P.pos, "] ", Scanner.TokenString(P.tok), "\n"); print("[", P.pos, "] ", Scanner.TokenString(P.tok), "\n");
...@@ -75,6 +76,13 @@ func (P *Parser) Next0() { ...@@ -75,6 +76,13 @@ func (P *Parser) Next0() {
func (P *Parser) Next() { func (P *Parser) Next() {
for P.Next0(); P.tok == Scanner.COMMENT; P.Next0() { for P.Next0(); P.tok == Scanner.COMMENT; P.Next0() {
P.comments.Add(Node.NewComment(P.pos, P.val)); P.comments.Add(Node.NewComment(P.pos, P.val));
if P.val == "/*ERROR*/" {
// the position of the next token is the position of the next expected error
} else if P.val == "/*SYNC*/" {
// synchronized at the next token
}
} }
} }
...@@ -111,6 +119,26 @@ func (P *Parser) OptSemicolon() { ...@@ -111,6 +119,26 @@ func (P *Parser) OptSemicolon() {
} }
// ----------------------------------------------------------------------------
// AST support
func ExprType(x *Node.Expr) *Node.Type {
var t *Node.Type;
if x.tok == Scanner.TYPE {
t = x.t;
} else if x.tok == Scanner.IDENT {
// assume a type name
t = Node.NewType(x.pos, Scanner.IDENT);
t.expr = x;
} else if x.tok == Scanner.PERIOD && x.y != nil && ExprType(x.x) != nil {
// possibly a qualified (type) identifier
t = Node.NewType(x.pos, Scanner.IDENT);
t.expr = x;
}
return t;
}
func (P *Parser) NoType(x *Node.Expr) *Node.Expr { func (P *Parser) NoType(x *Node.Expr) *Node.Expr {
if x != nil && x.tok == Scanner.TYPE { if x != nil && x.tok == Scanner.TYPE {
P.Error(x.pos, "expected expression, found type"); P.Error(x.pos, "expected expression, found type");
...@@ -137,7 +165,7 @@ func (P *Parser) ParseDeclaration() *Node.Decl; ...@@ -137,7 +165,7 @@ func (P *Parser) ParseDeclaration() *Node.Decl;
func (P *Parser) ParseIdent() *Node.Expr { func (P *Parser) ParseIdent() *Node.Expr {
P.Trace("Ident"); P.Trace("Ident");
var x *Node.Expr; x := Node.BadExpr;
if P.tok == Scanner.IDENT { if P.tok == Scanner.IDENT {
x = Node.NewLit(P.pos, Scanner.IDENT, P.val); x = Node.NewLit(P.pos, Scanner.IDENT, P.val);
if P.verbose { if P.verbose {
...@@ -176,13 +204,14 @@ func (P *Parser) ParseIdentList() *Node.Expr { ...@@ -176,13 +204,14 @@ func (P *Parser) ParseIdentList() *Node.Expr {
func (P *Parser) ParseType() *Node.Type { func (P *Parser) ParseType() *Node.Type {
P.Trace("Type"); P.Trace("Type");
typ := P.TryType(); t := P.TryType();
if typ == nil { if t == nil {
P.Error(P.pos, "type expected"); P.Error(P.pos, "type expected");
t = Node.BadType;
} }
P.Ecart(); P.Ecart();
return typ; return t;
} }
...@@ -474,11 +503,10 @@ func (P *Parser) ParsePointerType() *Node.Type { ...@@ -474,11 +503,10 @@ func (P *Parser) ParsePointerType() *Node.Type {
} }
// Returns nil if no type was found.
func (P *Parser) TryType() *Node.Type { func (P *Parser) TryType() *Node.Type {
P.Trace("Type (try)"); P.Trace("Type (try)");
var t *Node.Type; t := Node.BadType;
switch P.tok { switch P.tok {
case Scanner.IDENT: t = P.ParseTypeName(); case Scanner.IDENT: t = P.ParseTypeName();
case Scanner.LBRACK: t = P.ParseArrayType(); case Scanner.LBRACK: t = P.ParseArrayType();
...@@ -488,6 +516,7 @@ func (P *Parser) TryType() *Node.Type { ...@@ -488,6 +516,7 @@ func (P *Parser) TryType() *Node.Type {
case Scanner.MAP: t = P.ParseMapType(); case Scanner.MAP: t = P.ParseMapType();
case Scanner.STRUCT: t = P.ParseStructType(); case Scanner.STRUCT: t = P.ParseStructType();
case Scanner.MUL: t = P.ParsePointerType(); case Scanner.MUL: t = P.ParsePointerType();
default: t = nil; // no type found
} }
P.Ecart(); P.Ecart();
...@@ -503,7 +532,11 @@ func (P *Parser) ParseStatementList() *Node.List { ...@@ -503,7 +532,11 @@ func (P *Parser) ParseStatementList() *Node.List {
list := Node.NewList(); list := Node.NewList();
for P.tok != Scanner.CASE && P.tok != Scanner.DEFAULT && P.tok != Scanner.RBRACE && P.tok != Scanner.EOF { for P.tok != Scanner.CASE && P.tok != Scanner.DEFAULT && P.tok != Scanner.RBRACE && P.tok != Scanner.EOF {
list.Add(P.ParseStatement()); s := P.ParseStatement();
if s != nil {
// not the empty statement
list.Add(s);
}
if P.tok == Scanner.SEMICOLON { if P.tok == Scanner.SEMICOLON {
P.Next(); P.Next();
} else if P.opt_semi { } else if P.opt_semi {
...@@ -542,7 +575,7 @@ func (P *Parser) ParseBlock() *Node.List { ...@@ -542,7 +575,7 @@ func (P *Parser) ParseBlock() *Node.List {
// ---------------------------------------------------------------------------- // ----------------------------------------------------------------------------
// Expressions // Expressions
// TODO: Make this non-recursive. // TODO make this non-recursive
func (P *Parser) ParseExpressionList() *Node.Expr { func (P *Parser) ParseExpressionList() *Node.Expr {
P.Trace("ExpressionList"); P.Trace("ExpressionList");
...@@ -562,26 +595,29 @@ func (P *Parser) ParseExpressionList() *Node.Expr { ...@@ -562,26 +595,29 @@ func (P *Parser) ParseExpressionList() *Node.Expr {
func (P *Parser) ParseFunctionLit() *Node.Expr { func (P *Parser) ParseFunctionLit() *Node.Expr {
P.Trace("FunctionLit"); P.Trace("FunctionLit");
x := Node.NewLit(P.pos, Scanner.FUNC, "");
P.Expect(Scanner.FUNC); P.Expect(Scanner.FUNC);
P.ParseFunctionType(); x.t = P.ParseFunctionType();
P.scope_lev++; P.scope_lev++;
P.ParseBlock(); x.block = P.ParseBlock();
P.scope_lev--; P.scope_lev--;
P.Ecart(); P.Ecart();
return Node.NewLit(P.pos, Scanner.INT, "0"); // "null" expr return x;
} }
func (P *Parser) ParseOperand() *Node.Expr { func (P *Parser) ParseOperand() *Node.Expr {
P.Trace("Operand"); P.Trace("Operand");
var x *Node.Expr; x := Node.BadExpr;
switch P.tok { switch P.tok {
case Scanner.IDENT: case Scanner.IDENT:
x = P.ParseIdent(); x = P.ParseIdent();
case Scanner.LPAREN: case Scanner.LPAREN:
// TODO we could have a function type here as in: new(*())
// (currently not working)
P.Next(); P.Next();
P.expr_lev++; P.expr_lev++;
x = P.ParseExpression(); x = P.ParseExpression();
...@@ -607,7 +643,6 @@ func (P *Parser) ParseOperand() *Node.Expr { ...@@ -607,7 +643,6 @@ func (P *Parser) ParseOperand() *Node.Expr {
} else { } else {
P.Error(P.pos, "operand expected"); P.Error(P.pos, "operand expected");
P.Next(); // make progress P.Next(); // make progress
x = Node.NewLit(P.pos, Scanner.INT, "0"); // "null" expr
} }
} }
...@@ -697,6 +732,7 @@ func (P *Parser) ParseCall(x *Node.Expr) *Node.Expr { ...@@ -697,6 +732,7 @@ func (P *Parser) ParseCall(x *Node.Expr) *Node.Expr {
} }
// TODO make this non-recursive
func (P *Parser) ParseExpressionPairList(mode int) *Node.Expr { func (P *Parser) ParseExpressionPairList(mode int) *Node.Expr {
P.Trace("ExpressionPairList"); P.Trace("ExpressionPairList");
...@@ -726,10 +762,12 @@ func (P *Parser) ParseExpressionPairList(mode int) *Node.Expr { ...@@ -726,10 +762,12 @@ func (P *Parser) ParseExpressionPairList(mode int) *Node.Expr {
func (P *Parser) ParseCompositeLit(t *Node.Type) *Node.Expr { func (P *Parser) ParseCompositeLit(t *Node.Type) *Node.Expr {
P.Trace("CompositeLit"); P.Trace("CompositeLit");
pos := P.pos; x := P.NewExpr(P.pos, Scanner.LBRACE, nil, nil);
P.Expect(Scanner.LBRACE);
x := P.NewExpr(pos, Scanner.LBRACE, nil, P.ParseExpressionPairList(0));
x.t = t; x.t = t;
P.Expect(Scanner.LBRACE);
if P.tok != Scanner.RBRACE {
x.y = P.ParseExpressionPairList(0);
}
P.Expect(Scanner.RBRACE); P.Expect(Scanner.RBRACE);
P.Ecart(); P.Ecart();
...@@ -752,13 +790,7 @@ func (P *Parser) ParsePrimaryExpr() *Node.Expr { ...@@ -752,13 +790,7 @@ func (P *Parser) ParsePrimaryExpr() *Node.Expr {
// (composites inside control clauses must be parenthesized) // (composites inside control clauses must be parenthesized)
var t *Node.Type; var t *Node.Type;
if P.expr_lev > 0 { if P.expr_lev > 0 {
if x.tok == Scanner.TYPE { t = ExprType(x);
t = x.t;
} else if x.tok == Scanner.IDENT {
// assume a type name
t = Node.NewType(x.pos, Scanner.IDENT);
t.expr = x;
}
} }
if t != nil { if t != nil {
x = P.ParseCompositeLit(t); x = P.ParseCompositeLit(t);
...@@ -768,8 +800,8 @@ func (P *Parser) ParsePrimaryExpr() *Node.Expr { ...@@ -768,8 +800,8 @@ func (P *Parser) ParsePrimaryExpr() *Node.Expr {
default: goto exit; default: goto exit;
} }
} }
exit:
exit:
P.Ecart(); P.Ecart();
return x; return x;
} }
...@@ -778,7 +810,7 @@ exit: ...@@ -778,7 +810,7 @@ exit:
func (P *Parser) ParseUnaryExpr() *Node.Expr { func (P *Parser) ParseUnaryExpr() *Node.Expr {
P.Trace("UnaryExpr"); P.Trace("UnaryExpr");
var x *Node.Expr; x := Node.BadExpr;
switch P.tok { switch P.tok {
case Scanner.ADD, Scanner.SUB, Scanner.MUL, Scanner.NOT, Scanner.XOR, Scanner.ARROW, Scanner.AND: case Scanner.ADD, Scanner.SUB, Scanner.MUL, Scanner.NOT, Scanner.XOR, Scanner.ARROW, Scanner.AND:
pos, tok := P.pos, P.tok; pos, tok := P.pos, P.tok;
...@@ -840,8 +872,7 @@ func (P *Parser) ParseExpression() *Node.Expr { ...@@ -840,8 +872,7 @@ func (P *Parser) ParseExpression() *Node.Expr {
func (P *Parser) ParseSimpleStat() *Node.Stat { func (P *Parser) ParseSimpleStat() *Node.Stat {
P.Trace("SimpleStat"); P.Trace("SimpleStat");
var s *Node.Stat; s := Node.BadStat;
x := P.ParseExpressionList(); x := P.ParseExpressionList();
switch P.tok { switch P.tok {
...@@ -874,7 +905,7 @@ func (P *Parser) ParseSimpleStat() *Node.Stat { ...@@ -874,7 +905,7 @@ func (P *Parser) ParseSimpleStat() *Node.Stat {
pos, tok = P.pos, P.tok; pos, tok = P.pos, P.tok;
P.Next(); P.Next();
} else { } else {
pos, tok = x.pos, 0; // TODO give this a token value pos, tok = x.pos, Scanner.EXPRSTAT;
} }
s = Node.NewStat(pos, tok); s = Node.NewStat(pos, tok);
s.expr = x; s.expr = x;
...@@ -974,15 +1005,18 @@ func (P *Parser) ParseIfStat() *Node.Stat { ...@@ -974,15 +1005,18 @@ func (P *Parser) ParseIfStat() *Node.Stat {
s.post = P.ParseIfStat(); s.post = P.ParseIfStat();
} else { } else {
// For 6g compliance - should really be P.ParseBlock() // For 6g compliance - should really be P.ParseBlock()
t := P.ParseStatement(); s1 := P.ParseStatement();
if t.tok != Scanner.LBRACE { if s1 != nil {
// not the empty statement
if s1.tok != Scanner.LBRACE {
// wrap in a block if we don't have one // wrap in a block if we don't have one
t1 := Node.NewStat(P.pos, Scanner.LBRACE); b := Node.NewStat(P.pos, Scanner.LBRACE);
t1.block = Node.NewList(); b.block = Node.NewList();
t1.block.Add(t); b.block.Add(s1);
t = t1; s1 = b;
}
s.post = s1;
} }
s.post = t;
} }
} }
...@@ -1117,17 +1151,11 @@ func (P *Parser) ParseRangeStat() *Node.Stat { ...@@ -1117,17 +1151,11 @@ func (P *Parser) ParseRangeStat() *Node.Stat {
} }
func (P *Parser) ParseEmptyStat() {
P.Trace("EmptyStat");
P.Ecart();
}
func (P *Parser) ParseStatement() *Node.Stat { func (P *Parser) ParseStatement() *Node.Stat {
P.Trace("Statement"); P.Trace("Statement");
indent := P.indent; indent := P.indent;
var s *Node.Stat; s := Node.BadStat;
switch P.tok { switch P.tok {
case Scanner.CONST, Scanner.TYPE, Scanner.VAR: case Scanner.CONST, Scanner.TYPE, Scanner.VAR:
s = Node.NewStat(P.pos, P.tok); s = Node.NewStat(P.pos, P.tok);
...@@ -1162,7 +1190,8 @@ func (P *Parser) ParseStatement() *Node.Stat { ...@@ -1162,7 +1190,8 @@ func (P *Parser) ParseStatement() *Node.Stat {
case Scanner.SELECT: case Scanner.SELECT:
s = P.ParseSelectStat(); s = P.ParseSelectStat();
default: default:
P.ParseEmptyStat(); // for complete tracing output only // empty statement
s = nil;
} }
if indent != P.indent { if indent != P.indent {
...@@ -1250,7 +1279,7 @@ func (P *Parser) ParseVarSpec(exported bool) *Node.Decl { ...@@ -1250,7 +1279,7 @@ func (P *Parser) ParseVarSpec(exported bool) *Node.Decl {
} }
// TODO Replace this by using function pointers derived from methods. // TODO replace this by using function pointers derived from methods
func (P *Parser) ParseSpec(exported bool, keyword int) *Node.Decl { func (P *Parser) ParseSpec(exported bool, keyword int) *Node.Decl {
switch keyword { switch keyword {
case Scanner.IMPORT: return P.ParseImportSpec(); case Scanner.IMPORT: return P.ParseImportSpec();
...@@ -1266,7 +1295,7 @@ func (P *Parser) ParseSpec(exported bool, keyword int) *Node.Decl { ...@@ -1266,7 +1295,7 @@ func (P *Parser) ParseSpec(exported bool, keyword int) *Node.Decl {
func (P *Parser) ParseDecl(exported bool, keyword int) *Node.Decl { func (P *Parser) ParseDecl(exported bool, keyword int) *Node.Decl {
P.Trace("Decl"); P.Trace("Decl");
var d *Node.Decl; d := Node.BadDecl;
P.Expect(keyword); P.Expect(keyword);
if P.tok == Scanner.LPAREN { if P.tok == Scanner.LPAREN {
P.Next(); P.Next();
...@@ -1334,29 +1363,8 @@ func (P *Parser) ParseFunctionDecl(exported bool) *Node.Decl { ...@@ -1334,29 +1363,8 @@ func (P *Parser) ParseFunctionDecl(exported bool) *Node.Decl {
func (P *Parser) ParseExportDecl() *Node.Decl { func (P *Parser) ParseExportDecl() *Node.Decl {
P.Trace("ExportDecl"); P.Trace("ExportDecl");
// TODO This is deprecated syntax and should go away eventually.
// (Also at the moment the syntax is everything goes...)
//P.Expect(Scanner.EXPORT);
d := Node.NewDecl(P.pos, Scanner.EXPORT, false); d := Node.NewDecl(P.pos, Scanner.EXPORT, false);
has_paren := false;
if P.tok == Scanner.LPAREN {
P.Next();
has_paren = true;
}
d.ident = P.ParseIdentList(); d.ident = P.ParseIdentList();
/*
for P.tok == Scanner.IDENT {
P.ParseIdent();
if P.tok == Scanner.COMMA {
P.Next(); // TODO this seems wrong
}
}
*/
if has_paren {
P.Expect(Scanner.RPAREN)
}
P.Ecart(); P.Ecart();
return d; return d;
...@@ -1367,8 +1375,7 @@ func (P *Parser) ParseDeclaration() *Node.Decl { ...@@ -1367,8 +1375,7 @@ func (P *Parser) ParseDeclaration() *Node.Decl {
P.Trace("Declaration"); P.Trace("Declaration");
indent := P.indent; indent := P.indent;
var d *Node.Decl; d := Node.BadDecl;
exported := false; exported := false;
if P.tok == Scanner.EXPORT { if P.tok == Scanner.EXPORT {
if P.scope_lev == 0 { if P.scope_lev == 0 {
......
...@@ -90,6 +90,12 @@ func (P *Printer) CloseScope(paren string) { ...@@ -90,6 +90,12 @@ func (P *Printer) CloseScope(paren string) {
P.semi, P.newl = false, 1; P.semi, P.newl = false, 1;
} }
func (P *Printer) Error(pos int, tok int, msg string) {
P.String(0, "<");
P.Token(pos, tok);
P.String(0, " " + msg + ">");
}
// ---------------------------------------------------------------------------- // ----------------------------------------------------------------------------
// Types // Types
...@@ -140,11 +146,6 @@ func (P *Printer) Fields(list *Node.List) { ...@@ -140,11 +146,6 @@ func (P *Printer) Fields(list *Node.List) {
func (P *Printer) Type(t *Node.Type) { func (P *Printer) Type(t *Node.Type) {
if t == nil { // TODO remove this check
P.String(0, "<nil type>");
return;
}
switch t.tok { switch t.tok {
case Scanner.IDENT: case Scanner.IDENT:
P.Expr(t.expr); P.Expr(t.expr);
...@@ -192,7 +193,7 @@ func (P *Printer) Type(t *Node.Type) { ...@@ -192,7 +193,7 @@ func (P *Printer) Type(t *Node.Type) {
} }
default: default:
panic("UNREACHABLE"); P.Error(t.pos, t.tok, "type");
} }
} }
...@@ -200,6 +201,8 @@ func (P *Printer) Type(t *Node.Type) { ...@@ -200,6 +201,8 @@ func (P *Printer) Type(t *Node.Type) {
// ---------------------------------------------------------------------------- // ----------------------------------------------------------------------------
// Expressions // Expressions
func (P *Printer) Block(list *Node.List, indent bool);
func (P *Printer) Expr1(x *Node.Expr, prec1 int) { func (P *Printer) Expr1(x *Node.Expr, prec1 int) {
if x == nil { if x == nil {
return; // empty expression list return; // empty expression list
...@@ -214,6 +217,13 @@ func (P *Printer) Expr1(x *Node.Expr, prec1 int) { ...@@ -214,6 +217,13 @@ func (P *Printer) Expr1(x *Node.Expr, prec1 int) {
// literal // literal
P.String(x.pos, x.s); P.String(x.pos, x.s);
case Scanner.FUNC:
// function literal
P.String(x.pos, "func");
P.Type(x.t);
P.Block(x.block, true);
P.newl = 0;
case Scanner.COMMA: case Scanner.COMMA:
// list // list
P.Expr1(x.x, 0); P.Expr1(x.x, 0);
...@@ -344,13 +354,8 @@ func (P *Printer) ControlClause(s *Node.Stat) { ...@@ -344,13 +354,8 @@ func (P *Printer) ControlClause(s *Node.Stat) {
func (P *Printer) Declaration(d *Node.Decl, parenthesized bool); func (P *Printer) Declaration(d *Node.Decl, parenthesized bool);
func (P *Printer) Stat(s *Node.Stat) { func (P *Printer) Stat(s *Node.Stat) {
if s == nil { // TODO remove this check
P.String(0, "<nil stat>");
return;
}
switch s.tok { switch s.tok {
case 0: // TODO use a real token const case Scanner.EXPRSTAT:
// expression statement // expression statement
P.Expr(s.expr); P.Expr(s.expr);
P.semi = true; P.semi = true;
...@@ -430,8 +435,7 @@ func (P *Printer) Stat(s *Node.Stat) { ...@@ -430,8 +435,7 @@ func (P *Printer) Stat(s *Node.Stat) {
P.semi = true; P.semi = true;
default: default:
P.String(s.pos, "<stat>"); P.Error(s.pos, s.tok, "stat");
P.semi = true;
} }
} }
...@@ -441,11 +445,6 @@ func (P *Printer) Stat(s *Node.Stat) { ...@@ -441,11 +445,6 @@ func (P *Printer) Stat(s *Node.Stat) {
func (P *Printer) Declaration(d *Node.Decl, parenthesized bool) { func (P *Printer) Declaration(d *Node.Decl, parenthesized bool) {
if d == nil { // TODO remove this check
P.String(0, "<nil decl>");
return;
}
if !parenthesized { if !parenthesized {
if d.exported { if d.exported {
P.String(0, "export "); P.String(0, "export ");
......
...@@ -104,6 +104,9 @@ export const ( ...@@ -104,6 +104,9 @@ export const (
TYPE; TYPE;
VAR; VAR;
KEYWORDS_END; KEYWORDS_END;
// AST use only
EXPRSTAT;
) )
...@@ -201,6 +204,8 @@ export func TokenString(tok int) string { ...@@ -201,6 +204,8 @@ export func TokenString(tok int) string {
case SWITCH: return "switch"; case SWITCH: return "switch";
case TYPE: return "type"; case TYPE: return "type";
case VAR: return "var"; case VAR: return "var";
case EXPRSTAT: return "EXPRSTAT";
} }
return "token(" + Utils.IntToString(tok, 10) + ")"; return "token(" + Utils.IntToString(tok, 10) + ")";
...@@ -268,10 +273,12 @@ func digit_val(ch int) int { ...@@ -268,10 +273,12 @@ func digit_val(ch int) int {
export type Scanner struct { export type Scanner struct {
// error handling
filename string; // error reporting only filename string; // error reporting only
nerrors int; // number of errors nerrors int; // number of errors
errpos int; // last error position errpos int; // last error position
// scanning
src string; // scanned source src string; // scanned source
pos int; // current reading position pos int; // current reading position
ch int; // one char look-ahead ch int; // one char look-ahead
...@@ -400,13 +407,7 @@ func (S *Scanner) LineCol(pos int) (line, col int) { ...@@ -400,13 +407,7 @@ func (S *Scanner) LineCol(pos int) (line, col int) {
} }
func (S *Scanner) Error(pos int, msg string) { func (S *Scanner) ErrorMsg(pos int, msg string) {
const errdist = 10;
delta := pos - S.errpos; // may be negative!
if delta < 0 {
delta = -delta;
}
if delta > errdist || S.nerrors == 0 /* always report first error */ {
print(S.filename); print(S.filename);
if pos >= 0 { if pos >= 0 {
// print position // print position
...@@ -418,6 +419,18 @@ func (S *Scanner) Error(pos int, msg string) { ...@@ -418,6 +419,18 @@ func (S *Scanner) Error(pos int, msg string) {
} }
} }
print(": ", msg, "\n"); print(": ", msg, "\n");
}
func (S *Scanner) Error(pos int, msg string) {
const errdist = 10;
delta := pos - S.errpos; // may be negative!
if delta < 0 {
delta = -delta;
}
if delta > errdist || S.nerrors == 0 /* always report first error */ {
S.ErrorMsg(pos, msg);
S.nerrors++; S.nerrors++;
S.errpos = pos; S.errpos = pos;
} }
......
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