Parser can now parse itself. Also parses std.array

This commit is contained in:
Hackerpilot 2013-07-01 01:50:54 +00:00
parent 95c7c25fc6
commit 153e7c3b8e
2 changed files with 193 additions and 87 deletions

View File

@ -555,14 +555,11 @@ public:
{ {
if (attribute !is null) if (attribute !is null)
visitor.visit(attribute); visitor.visit(attribute);
foreach (dec; declarations) if (declaration !is null)
{ visitor.visit(declaration);
if (dec !is null)
visitor.visit(dec);
}
} }
/** */ Attribute attribute; /** */ Attribute attribute;
/** */ Declaration[] declarations; /** */ Declaration declaration;
} }
/// ///
@ -765,6 +762,9 @@ public:
mixin(DEFAULT_ACCEPT); mixin(DEFAULT_ACCEPT);
/** */ Parameters parameters; /** */ Parameters parameters;
/** */ FunctionBody functionBody; /** */ FunctionBody functionBody;
/** */ Constraint constraint;
/** */ MemberFunctionAttribute[] memberFunctionAttributes;
/** */ TemplateParameters templateParameters;
} }
/// ///
@ -850,6 +850,7 @@ public:
/** */ ConditionalDeclaration conditionalDeclaration; /** */ ConditionalDeclaration conditionalDeclaration;
/** */ PragmaDeclaration pragmaDeclaration; /** */ PragmaDeclaration pragmaDeclaration;
/** */ VersionSpecification versionSpecification; /** */ VersionSpecification versionSpecification;
/** */ Declaration[] declarations;
} }
/// ///
@ -1605,7 +1606,6 @@ class PrimaryExpression : ASTNode
public: public:
mixin(DEFAULT_ACCEPT); mixin(DEFAULT_ACCEPT);
/** */ Token primary; /** */ Token primary;
/** */ Token identifier;
/** */ IdentifierOrTemplateInstance identifierOrTemplateInstance; /** */ IdentifierOrTemplateInstance identifierOrTemplateInstance;
/** */ TokenType basicType; /** */ TokenType basicType;
/** */ TypeofExpression typeofExpression; /** */ TypeofExpression typeofExpression;

View File

@ -119,6 +119,7 @@ struct Parser
*/ */
AliasDeclaration parseAliasDeclaration() AliasDeclaration parseAliasDeclaration()
{ {
mixin(traceEnterAndExit!(__FUNCTION__));
auto node = new AliasDeclaration; auto node = new AliasDeclaration;
if (expect(TokenType.alias_) is null) return null; if (expect(TokenType.alias_) is null) return null;
if (startsWith(TokenType.identifier, TokenType.assign)) if (startsWith(TokenType.identifier, TokenType.assign))
@ -168,6 +169,7 @@ alias core.sys.posix.stdio.fileno fileno;
*/ */
AliasInitializer parseAliasInitializer() AliasInitializer parseAliasInitializer()
{ {
mixin(traceEnterAndExit!(__FUNCTION__));
auto node = new AliasInitializer; auto node = new AliasInitializer;
auto i = expect(TokenType.identifier); auto i = expect(TokenType.identifier);
if (i is null) return null; if (i is null) return null;
@ -185,6 +187,7 @@ alias core.sys.posix.stdio.fileno fileno;
*/ */
AliasThisDeclaration parseAliasThisDeclaration() AliasThisDeclaration parseAliasThisDeclaration()
{ {
mixin(traceEnterAndExit!(__FUNCTION__));
auto node = new AliasThisDeclaration; auto node = new AliasThisDeclaration;
if (expect(TokenType.alias_) is null) return null; if (expect(TokenType.alias_) is null) return null;
auto ident = expect(TokenType.identifier); auto ident = expect(TokenType.identifier);
@ -202,6 +205,7 @@ alias core.sys.posix.stdio.fileno fileno;
*/ */
AlignAttribute parseAlignAttribute() AlignAttribute parseAlignAttribute()
{ {
mixin(traceEnterAndExit!(__FUNCTION__));
auto node = new AlignAttribute; auto node = new AlignAttribute;
expect(TokenType.align_); expect(TokenType.align_);
if (currentIs(TokenType.lParen)) if (currentIs(TokenType.lParen))
@ -266,8 +270,8 @@ alias core.sys.posix.stdio.fileno fileno;
*/ */
Arguments parseArguments() Arguments parseArguments()
{ {
auto node = new Arguments;
mixin(traceEnterAndExit!(__FUNCTION__)); mixin(traceEnterAndExit!(__FUNCTION__));
auto node = new Arguments;
if (expect(TokenType.lParen) is null) return null; if (expect(TokenType.lParen) is null) return null;
if (!currentIs(TokenType.rParen)) if (!currentIs(TokenType.rParen))
node.argumentList = parseArgumentList(); node.argumentList = parseArgumentList();
@ -755,10 +759,12 @@ alias core.sys.posix.stdio.fileno fileno;
* | $(LITERAL 'static') * | $(LITERAL 'static')
* | $(LITERAL 'pure') * | $(LITERAL 'pure')
* | $(LITERAL 'nothrow') * | $(LITERAL 'nothrow')
* | $(LITERAL 'enum')
* ;) * ;)
*/ */
Attribute parseAttribute() Attribute parseAttribute()
{ {
mixin(traceEnterAndExit!(__FUNCTION__));
auto node = new Attribute; auto node = new Attribute;
switch (current.type) switch (current.type)
{ {
@ -796,8 +802,12 @@ alias core.sys.posix.stdio.fileno fileno;
case TokenType.static_: case TokenType.static_:
case TokenType.pure_: case TokenType.pure_:
case TokenType.nothrow_: case TokenType.nothrow_:
case TokenType.enum_:
node.attribute = advance().type; node.attribute = advance().type;
break; break;
case TokenType.at:
node.atAttribute = parseAtAttribute();
break;
default: default:
error("Attribute expected"); error("Attribute expected");
return null; return null;
@ -814,23 +824,16 @@ alias core.sys.posix.stdio.fileno fileno;
*/ */
AttributedDeclaration parseAttributedDeclaration() AttributedDeclaration parseAttributedDeclaration()
{ {
mixin(traceEnterAndExit!(__FUNCTION__));
auto node = new AttributedDeclaration; auto node = new AttributedDeclaration;
node.attribute = parseAttribute(); node.attribute = parseAttribute();
switch (current.type) switch (current.type)
{ {
case TokenType.colon: case TokenType.colon:
advance(); advance();
auto dec = parseDeclaration();
if (dec is null) return null;
node.declarations ~= dec;
break;
case TokenType.lBrace:
while (moreTokens() && !currentIs(TokenType.rBrace))
node.declarations ~= parseDeclaration();
if (expect(TokenType.rBrace) is null) return null;
break; break;
default: default:
node.declarations ~= parseDeclaration(); node.declaration = parseDeclaration();
break; break;
} }
return node; return node;
@ -1197,6 +1200,7 @@ incorrect;
*/ */
Catch parseCatch() Catch parseCatch()
{ {
mixin(traceEnterAndExit!(__FUNCTION__));
auto node = new Catch; auto node = new Catch;
expect(TokenType.catch_); expect(TokenType.catch_);
if (expect(TokenType.lParen) is null) return null; if (expect(TokenType.lParen) is null) return null;
@ -1218,6 +1222,7 @@ incorrect;
*/ */
Catches parseCatches() Catches parseCatches()
{ {
mixin(traceEnterAndExit!(__FUNCTION__));
auto node = new Catches; auto node = new Catches;
while (true) while (true)
{ {
@ -1248,6 +1253,7 @@ incorrect;
*/ */
ClassBody parseClassBody() ClassBody parseClassBody()
{ {
mixin(traceEnterAndExit!(__FUNCTION__));
auto node = new ClassBody; auto node = new ClassBody;
if (expect(TokenType.lBrace) is null) return null; if (expect(TokenType.lBrace) is null) return null;
while (moreTokens() && !currentIs(TokenType.rBrace)) while (moreTokens() && !currentIs(TokenType.rBrace))
@ -1269,6 +1275,7 @@ incorrect;
*/ */
ClassDeclaration parseClassDeclaration() ClassDeclaration parseClassDeclaration()
{ {
mixin(traceEnterAndExit!(__FUNCTION__));
auto node = new ClassDeclaration; auto node = new ClassDeclaration;
expect(TokenType.class_); expect(TokenType.class_);
auto ident = expect(TokenType.identifier); auto ident = expect(TokenType.identifier);
@ -1403,6 +1410,7 @@ class ClassFour(A, B) if (someTest()) : Super {}}c;
*/ */
CompileCondition parseCompileCondition() CompileCondition parseCompileCondition()
{ {
mixin(traceEnterAndExit!(__FUNCTION__));
auto node = new CompileCondition; auto node = new CompileCondition;
switch (current.type) switch (current.type)
{ {
@ -1431,6 +1439,7 @@ class ClassFour(A, B) if (someTest()) : Super {}}c;
*/ */
ConditionalDeclaration parseConditionalDeclaration() ConditionalDeclaration parseConditionalDeclaration()
{ {
mixin(traceEnterAndExit!(__FUNCTION__));
auto node = new ConditionalDeclaration; auto node = new ConditionalDeclaration;
node.compileCondition = parseCompileCondition(); node.compileCondition = parseCompileCondition();
if (currentIs(TokenType.lBrace)) if (currentIs(TokenType.lBrace))
@ -1491,6 +1500,7 @@ class ClassFour(A, B) if (someTest()) : Super {}}c;
*/ */
ConditionalStatement parseConditionalStatement() ConditionalStatement parseConditionalStatement()
{ {
mixin(traceEnterAndExit!(__FUNCTION__));
auto node = new ConditionalStatement; auto node = new ConditionalStatement;
node.compileCondition = parseCompileCondition(); node.compileCondition = parseCompileCondition();
node.trueStatement = parseStatementNoCaseNoDefault(); node.trueStatement = parseStatementNoCaseNoDefault();
@ -1511,6 +1521,7 @@ class ClassFour(A, B) if (someTest()) : Super {}}c;
*/ */
Constraint parseConstraint() Constraint parseConstraint()
{ {
mixin(traceEnterAndExit!(__FUNCTION__));
auto node = new Constraint; auto node = new Constraint;
if (expect(TokenType.if_) is null) return null; if (expect(TokenType.if_) is null) return null;
if (expect(TokenType.lParen) is null) return null; if (expect(TokenType.lParen) is null) return null;
@ -1523,15 +1534,37 @@ class ClassFour(A, B) if (someTest()) : Super {}}c;
* Parses a Constructor * Parses a Constructor
* *
* $(GRAMMAR $(RULEDEF constructor): * $(GRAMMAR $(RULEDEF constructor):
* $(LITERAL 'this') $(RULE parameters) $(RULE functionBody) * $(LITERAL 'this') $(RULE templateParameters) $(RULE parameters) $(RULE memberFunctionAttribute)* $(RULE constraint)? $(RULE functionBody)
* | $(LITERAL 'this') $(RULE parameters) $(RULE memberFunctionAttribute)* ($(RULE functionBody) | $(LITERAL ';'))
* ;) * ;)
*/ */
Constructor parseConstructor() Constructor parseConstructor()
{ {
mixin(traceEnterAndExit!(__FUNCTION__));
auto node = new Constructor; auto node = new Constructor;
if (expect(TokenType.this_) is null) return null; if (expect(TokenType.this_) is null) return null;
auto p = peekPastParens();
bool isTemplate = false;
if (p !is null && p.type == TokenType.lParen)
{
isTemplate = true;
node.templateParameters = parseTemplateParameters();
}
node.parameters = parseParameters(); node.parameters = parseParameters();
node.functionBody = parseFunctionBody();
while(moreTokens() && currentIsMemberFunctionAttribute())
node.memberFunctionAttributes ~= parseMemberFunctionAttribute();
if (isTemplate && currentIs(TokenType.if_))
node.constraint = parseConstraint();
if (isTemplate)
node.functionBody = parseFunctionBody();
else if (currentIs(TokenType.semicolon))
advance();
else
node.functionBody = parseFunctionBody();
return node; return node;
} }
@ -1544,6 +1577,7 @@ class ClassFour(A, B) if (someTest()) : Super {}}c;
*/ */
ContinueStatement parseContinueStatement() ContinueStatement parseContinueStatement()
{ {
mixin(traceEnterAndExit!(__FUNCTION__));
if (expect(TokenType.continue_) is null) return null; if (expect(TokenType.continue_) is null) return null;
auto node = new ContinueStatement; auto node = new ContinueStatement;
switch (current.type) switch (current.type)
@ -1571,6 +1605,7 @@ class ClassFour(A, B) if (someTest()) : Super {}}c;
*/ */
DebugCondition parseDebugCondition() DebugCondition parseDebugCondition()
{ {
mixin(traceEnterAndExit!(__FUNCTION__));
auto node = new DebugCondition; auto node = new DebugCondition;
if (expect(TokenType.debug_) is null) return null; if (expect(TokenType.debug_) is null) return null;
if (currentIs(TokenType.lParen)) if (currentIs(TokenType.lParen))
@ -1600,6 +1635,7 @@ class ClassFour(A, B) if (someTest()) : Super {}}c;
*/ */
DebugSpecification parseDebugSpecification() DebugSpecification parseDebugSpecification()
{ {
mixin(traceEnterAndExit!(__FUNCTION__));
auto node = new DebugSpecification; auto node = new DebugSpecification;
if (expect(TokenType.debug_) is null) return null; if (expect(TokenType.debug_) is null) return null;
if (expect(TokenType.assign) is null) return null; if (expect(TokenType.assign) is null) return null;
@ -1641,6 +1677,7 @@ class ClassFour(A, B) if (someTest()) : Super {}}c;
* | $(RULE unionDeclaration) * | $(RULE unionDeclaration)
* | $(RULE unittest) * | $(RULE unittest)
* | $(RULE variableDeclaration) * | $(RULE variableDeclaration)
* | $(LITERAL '{') $(RULE declaration)+ $(LITERAL '}')
* ;) * ;)
*/ */
Declaration parseDeclaration() Declaration parseDeclaration()
@ -1649,6 +1686,16 @@ class ClassFour(A, B) if (someTest()) : Super {}}c;
auto node = new Declaration; auto node = new Declaration;
with (TokenType) switch (current.type) with (TokenType) switch (current.type)
{ {
case lBrace:
advance();
while (moreTokens() && !currentIs(rBrace))
{
auto declaration = parseDeclaration();
if (declaration !is null)
node.declarations ~= declaration;
}
expect(TokenType.rBrace);
break;
case alias_: case alias_:
if (startsWith(alias_, identifier, this_)) if (startsWith(alias_, identifier, this_))
node.aliasThisDeclaration = parseAliasThisDeclaration(); node.aliasThisDeclaration = parseAliasThisDeclaration();
@ -1667,11 +1714,20 @@ class ClassFour(A, B) if (someTest()) : Super {}}c;
case enum_: case enum_:
if (peekIsOneOf(TokenType.lBrace, TokenType.colon)) if (peekIsOneOf(TokenType.lBrace, TokenType.colon))
node.enumDeclaration = parseEnumDeclaration(); node.enumDeclaration = parseEnumDeclaration();
if (peekIs(TokenType.identifier)) else if (!peekIs(TokenType.identifier))
goto storageClass;
else
{ {
trace("here");
auto b = setBookmark(); auto b = setBookmark();
advance(); advance();
if (peekIsOneOf(TokenType.lBrace, TokenType.colon, TokenType.semicolon)) if (peekIs(TokenType.assign))
{
trace("varDec");
goToBookmark(b);
node.variableDeclaration = parseVariableDeclaration();
}
else if (peekIsOneOf(TokenType.lBrace, TokenType.colon, TokenType.semicolon))
{ {
goToBookmark(b); goToBookmark(b);
node.enumDeclaration = parseEnumDeclaration(); node.enumDeclaration = parseEnumDeclaration();
@ -1679,7 +1735,7 @@ class ClassFour(A, B) if (someTest()) : Super {}}c;
else else
{ {
goToBookmark(b); goToBookmark(b);
node.variableDeclaration = parseVariableDeclaration(); goto storageClass;
} }
} }
break; break;
@ -1760,19 +1816,10 @@ class ClassFour(A, B) if (someTest()) : Super {}}c;
|| startsWith(auto_, identifier, lParen)) || startsWith(auto_, identifier, lParen))
node.functionDeclaration = parseFunctionDeclaration(); node.functionDeclaration = parseFunctionDeclaration();
else else
node.variableDeclaration = parseVariableDeclaration(); goto storageClass;
break; break;
case ref_: case ref_:
if (startsWith(ref_, auto_, identifier, lParen) node.functionDeclaration = parseFunctionDeclaration();
|| startsWith(ref_, identifier, lParen))
{
node.functionDeclaration = parseFunctionDeclaration();
}
else
{
error(`Declaration expected`);
return null;
}
break; break;
case const_: case const_:
case immutable_: case immutable_:
@ -1797,6 +1844,7 @@ class ClassFour(A, B) if (someTest()) : Super {}}c;
case gshared: case gshared:
case pure_: case pure_:
case nothrow_: case nothrow_:
storageClass:
node.attributedDeclaration = parseAttributedDeclaration(); node.attributedDeclaration = parseAttributedDeclaration();
break; break;
default: default:
@ -1820,6 +1868,7 @@ class ClassFour(A, B) if (someTest()) : Super {}}c;
*/ */
DeclarationsAndStatements parseDeclarationsAndStatements() DeclarationsAndStatements parseDeclarationsAndStatements()
{ {
mixin(traceEnterAndExit!(__FUNCTION__));
auto node = new DeclarationsAndStatements; auto node = new DeclarationsAndStatements;
while (!currentIsOneOf(TokenType.rBrace) && moreTokens()) while (!currentIsOneOf(TokenType.rBrace) && moreTokens())
{ {
@ -1842,6 +1891,7 @@ class ClassFour(A, B) if (someTest()) : Super {}}c;
*/ */
DeclarationOrInvariant parseDeclarationOrInvariant() DeclarationOrInvariant parseDeclarationOrInvariant()
{ {
mixin(traceEnterAndExit!(__FUNCTION__));
auto node = new DeclarationOrInvariant; auto node = new DeclarationOrInvariant;
if (currentIs(TokenType.invariant_)) if (currentIs(TokenType.invariant_))
node.invariant_ = parseInvariant(); node.invariant_ = parseInvariant();
@ -1860,6 +1910,7 @@ class ClassFour(A, B) if (someTest()) : Super {}}c;
*/ */
DeclarationOrStatement parseDeclarationOrStatement() DeclarationOrStatement parseDeclarationOrStatement()
{ {
mixin(traceEnterAndExit!(__FUNCTION__));
auto node = new DeclarationOrStatement; auto node = new DeclarationOrStatement;
// "Any ambiguities in the grammar between Statements and // "Any ambiguities in the grammar between Statements and
// Declarations are resolved by the declarations taking precedence." // Declarations are resolved by the declarations taking precedence."
@ -1879,6 +1930,7 @@ class ClassFour(A, B) if (someTest()) : Super {}}c;
*/ */
Declarator parseDeclarator() Declarator parseDeclarator()
{ {
mixin(traceEnterAndExit!(__FUNCTION__));
auto node = new Declarator; auto node = new Declarator;
auto id = expect(TokenType.identifier); auto id = expect(TokenType.identifier);
if (id is null) return null; if (id is null) return null;
@ -1905,6 +1957,7 @@ class ClassFour(A, B) if (someTest()) : Super {}}c;
*/ */
DefaultStatement parseDefaultStatement() DefaultStatement parseDefaultStatement()
{ {
mixin(traceEnterAndExit!(__FUNCTION__));
auto node = new DefaultStatement; auto node = new DefaultStatement;
if (expect(TokenType.default_) is null) return null; if (expect(TokenType.default_) is null) return null;
if (expect(TokenType.colon) is null) return null; if (expect(TokenType.colon) is null) return null;
@ -1921,6 +1974,7 @@ class ClassFour(A, B) if (someTest()) : Super {}}c;
*/ */
DeleteExpression parseDeleteExpression() DeleteExpression parseDeleteExpression()
{ {
mixin(traceEnterAndExit!(__FUNCTION__));
auto node = new DeleteExpression; auto node = new DeleteExpression;
if (expect(TokenType.delete_) is null) return null; if (expect(TokenType.delete_) is null) return null;
node.unaryExpression = parseUnaryExpression(); node.unaryExpression = parseUnaryExpression();
@ -1936,6 +1990,7 @@ class ClassFour(A, B) if (someTest()) : Super {}}c;
*/ */
Deprecated parseDeprecated() Deprecated parseDeprecated()
{ {
mixin(traceEnterAndExit!(__FUNCTION__));
auto node = new Deprecated; auto node = new Deprecated;
if (expect(TokenType.deprecated_) is null) return null; if (expect(TokenType.deprecated_) is null) return null;
if (currentIs(TokenType.lParen)) if (currentIs(TokenType.lParen))
@ -1956,6 +2011,7 @@ class ClassFour(A, B) if (someTest()) : Super {}}c;
*/ */
Destructor parseDestructor() Destructor parseDestructor()
{ {
mixin(traceEnterAndExit!(__FUNCTION__));
auto node = new Destructor; auto node = new Destructor;
if (expect(TokenType.tilde) is null) return null; if (expect(TokenType.tilde) is null) return null;
if (expect(TokenType.this_) is null) return null; if (expect(TokenType.this_) is null) return null;
@ -1985,6 +2041,7 @@ class ClassFour(A, B) if (someTest()) : Super {}}c;
*/ */
DoStatement parseDoStatement() DoStatement parseDoStatement()
{ {
mixin(traceEnterAndExit!(__FUNCTION__));
auto node = new DoStatement; auto node = new DoStatement;
if (expect(TokenType.do_) is null) return null; if (expect(TokenType.do_) is null) return null;
node.statementNoCaseNoDefault = parseStatementNoCaseNoDefault(); node.statementNoCaseNoDefault = parseStatementNoCaseNoDefault();
@ -2006,16 +2063,20 @@ class ClassFour(A, B) if (someTest()) : Super {}}c;
*/ */
EnumBody parseEnumBody() EnumBody parseEnumBody()
{ {
mixin(traceEnterAndExit!(__FUNCTION__));
auto node = new EnumBody; auto node = new EnumBody;
if (!currentIs(TokenType.semicolon)) if (!currentIs(TokenType.semicolon))
{ {
expect (TokenType.lBrace); expect (TokenType.lBrace);
while (moreTokens()) while (moreTokens())
{ {
if (!currentIs(TokenType.comma)) if (!currentIsOneOf(TokenType.comma, TokenType.rBrace))
node.enumMembers ~= parseEnumMember(); node.enumMembers ~= parseEnumMember();
if (currentIs(TokenType.comma)) else if (currentIs(TokenType.comma))
{
advance();
continue; continue;
}
else if (currentIs(TokenType.rBrace)) else if (currentIs(TokenType.rBrace))
break; break;
else else
@ -2039,6 +2100,7 @@ class ClassFour(A, B) if (someTest()) : Super {}}c;
*/ */
EnumDeclaration parseEnumDeclaration() EnumDeclaration parseEnumDeclaration()
{ {
mixin(traceEnterAndExit!(__FUNCTION__));
auto node = new EnumDeclaration; auto node = new EnumDeclaration;
if (expect(TokenType.enum_) is null) return null; if (expect(TokenType.enum_) is null) return null;
if (currentIs(TokenType.identifier)) if (currentIs(TokenType.identifier))
@ -2062,6 +2124,7 @@ class ClassFour(A, B) if (someTest()) : Super {}}c;
*/ */
EnumMember parseEnumMember() EnumMember parseEnumMember()
{ {
mixin(traceEnterAndExit!(__FUNCTION__));
auto node = new EnumMember; auto node = new EnumMember;
if (currentIs(TokenType.identifier)) if (currentIs(TokenType.identifier))
{ {
@ -2113,6 +2176,7 @@ class ClassFour(A, B) if (someTest()) : Super {}}c;
*/ */
Expression parseExpression() Expression parseExpression()
{ {
mixin(traceEnterAndExit!(__FUNCTION__));
return parseCommaSeparatedRule!(Expression, AssignExpression)(); return parseCommaSeparatedRule!(Expression, AssignExpression)();
} }
@ -2125,6 +2189,7 @@ class ClassFour(A, B) if (someTest()) : Super {}}c;
*/ */
ExpressionStatement parseExpressionStatement(Expression expression = null) ExpressionStatement parseExpressionStatement(Expression expression = null)
{ {
mixin(traceEnterAndExit!(__FUNCTION__));
auto node = new ExpressionStatement; auto node = new ExpressionStatement;
node.expression = expression is null ? parseExpression() : expression; node.expression = expression is null ? parseExpression() : expression;
if (expect(TokenType.semicolon) is null) return null; if (expect(TokenType.semicolon) is null) return null;
@ -2140,6 +2205,7 @@ class ClassFour(A, B) if (someTest()) : Super {}}c;
*/ */
FinalSwitchStatement parseFinalSwitchStatement() FinalSwitchStatement parseFinalSwitchStatement()
{ {
mixin(traceEnterAndExit!(__FUNCTION__));
auto node = new FinalSwitchStatement; auto node = new FinalSwitchStatement;
if (expect(TokenType.final_) is null) return null; if (expect(TokenType.final_) is null) return null;
node.switchStatement = parseSwitchStatement(); node.switchStatement = parseSwitchStatement();
@ -2156,6 +2222,7 @@ class ClassFour(A, B) if (someTest()) : Super {}}c;
*/ */
Finally parseFinally() Finally parseFinally()
{ {
mixin(traceEnterAndExit!(__FUNCTION__));
auto node = new Finally; auto node = new Finally;
if (expect(TokenType.finally_) is null) return null; if (expect(TokenType.finally_) is null) return null;
node.statementNoCaseNoDefault = parseStatementNoCaseNoDefault(); node.statementNoCaseNoDefault = parseStatementNoCaseNoDefault();
@ -2171,6 +2238,7 @@ class ClassFour(A, B) if (someTest()) : Super {}}c;
*/ */
ForStatement parseForStatement() ForStatement parseForStatement()
{ {
mixin(traceEnterAndExit!(__FUNCTION__));
auto node = new ForStatement; auto node = new ForStatement;
if (expect(TokenType.for_) is null) return null; if (expect(TokenType.for_) is null) return null;
if (expect(TokenType.lParen) is null) return null; if (expect(TokenType.lParen) is null) return null;
@ -2446,8 +2514,10 @@ body {} // six
{ {
node.hasAuto = true; node.hasAuto = true;
advance(); advance();
break;
} }
break; else
goto default;
default: default:
node.returnType = type is null ? parseType() : type; node.returnType = type is null ? parseType() : type;
break; break;
@ -2464,7 +2534,8 @@ body {} // six
return null; return null;
} }
bool isTemplate = peekPastParens().type == TokenType.lParen; auto p = peekPastParens();
bool isTemplate = p !is null && p.type == TokenType.lParen;
if (isTemplate) if (isTemplate)
node.templateParameters = parseTemplateParameters(); node.templateParameters = parseTemplateParameters();
@ -3073,35 +3144,29 @@ invariant() foo();
* Parses an IsExpression * Parses an IsExpression
* *
* $(GRAMMAR $(RULEDEF isExpression): * $(GRAMMAR $(RULEDEF isExpression):
* $(LITERAL'is') $(LITERAL '$(LPAREN)') ($(RULE assignExpression) | ($(RULE type) $(LITERAL Identifier)? (($(LITERAL ':') | '==') $(RULE typeSpecialization) ($(LITERAL ',') $(RULE templateParameterList))?)?)) $(LITERAL '$(RPAREN)') * $(LITERAL'is') $(LITERAL '$(LPAREN)') ($(RULE type) $(LITERAL Identifier)? (($(LITERAL ':') | $(LITERAL '==')) $(RULE typeSpecialization) ($(LITERAL ',') $(RULE templateParameterList))?)?)) $(LITERAL '$(RPAREN)')
* ;) * ;)
*/ */
IsExpression parseIsExpression() IsExpression parseIsExpression()
{ {
mixin(traceEnterAndExit!(__FUNCTION__));
auto node = new IsExpression; auto node = new IsExpression;
if (expect(TokenType.is_) is null) return null; if (expect(TokenType.is_) is null) return null;
if (expect(TokenType.lParen) is null) return null; if (expect(TokenType.lParen) is null) return null;
if (isType()) node.type = parseType();
if (node.type is null) return null;
if (currentIs(TokenType.identifier))
node.identifier = advance();
if (currentIsOneOf(TokenType.equal, TokenType.colon))
{ {
node.type = parseType(); node.equalsOrColon = advance().type;
if (node.type is null) return null; node.typeSpecialization = parseTypeSpecialization();
if (currentIs(TokenType.identifier)) if (currentIs(TokenType.comma))
{ {
node.identifier = advance(); advance();
if (currentIsOneOf(TokenType.assign, TokenType.colon)) node.templateParameterList = parseTemplateParameterList();
{
node.equalsOrColon = advance().type;
node.typeSpecialization = parseTypeSpecialization();
if (currentIs(TokenType.comma))
{
advance();
node.templateParameterList = parseTemplateParameterList();
}
}
} }
} }
else
node.assignExpression = parseAssignExpression();
if (expect(TokenType.rParen) is null) return null; if (expect(TokenType.rParen) is null) return null;
return node; return node;
} }
@ -3115,6 +3180,7 @@ invariant() foo();
*/ */
KeyValuePair parseKeyValuePair() KeyValuePair parseKeyValuePair()
{ {
mixin(traceEnterAndExit!(__FUNCTION__));
auto node = new KeyValuePair; auto node = new KeyValuePair;
node.key = parseAssignExpression(); node.key = parseAssignExpression();
if (expect(TokenType.colon) is null) return null; if (expect(TokenType.colon) is null) return null;
@ -3131,6 +3197,7 @@ invariant() foo();
*/ */
KeyValuePairs parseKeyValuePairs() KeyValuePairs parseKeyValuePairs()
{ {
mixin(traceEnterAndExit!(__FUNCTION__));
auto node = new KeyValuePairs; auto node = new KeyValuePairs;
while (true) while (true)
{ {
@ -3154,6 +3221,7 @@ invariant() foo();
*/ */
LabeledStatement parseLabeledStatement() LabeledStatement parseLabeledStatement()
{ {
mixin(traceEnterAndExit!(__FUNCTION__));
auto node = new LabeledStatement; auto node = new LabeledStatement;
auto ident = expect(TokenType.identifier); auto ident = expect(TokenType.identifier);
if (ident is null) return null; if (ident is null) return null;
@ -3172,6 +3240,7 @@ invariant() foo();
*/ */
LambdaExpression parseLambdaExpression() LambdaExpression parseLambdaExpression()
{ {
mixin(traceEnterAndExit!(__FUNCTION__));
auto node = new LambdaExpression; auto node = new LambdaExpression;
if (currentIs(TokenType.identifier)) if (currentIs(TokenType.identifier))
node.identifier = advance(); node.identifier = advance();
@ -3210,6 +3279,7 @@ invariant() foo();
*/ */
LastCatch parseLastCatch() LastCatch parseLastCatch()
{ {
mixin(traceEnterAndExit!(__FUNCTION__));
auto node = new LastCatch; auto node = new LastCatch;
if (expect(TokenType.catch_) is null) return null; if (expect(TokenType.catch_) is null) return null;
if ((node.statementNoCaseNoDefault = parseStatementNoCaseNoDefault()) is null) if ((node.statementNoCaseNoDefault = parseStatementNoCaseNoDefault()) is null)
@ -3226,6 +3296,7 @@ invariant() foo();
*/ */
LinkageAttribute parseLinkageAttribute() LinkageAttribute parseLinkageAttribute()
{ {
mixin(traceEnterAndExit!(__FUNCTION__));
auto node = new LinkageAttribute; auto node = new LinkageAttribute;
expect(TokenType.extern_); expect(TokenType.extern_);
expect(TokenType.lParen); expect(TokenType.lParen);
@ -3254,6 +3325,7 @@ invariant() foo();
*/ */
MemberFunctionAttribute parseMemberFunctionAttribute() MemberFunctionAttribute parseMemberFunctionAttribute()
{ {
mixin(traceEnterAndExit!(__FUNCTION__));
auto node = new MemberFunctionAttribute; auto node = new MemberFunctionAttribute;
with (TokenType) switch (current.type) with (TokenType) switch (current.type)
{ {
@ -3283,6 +3355,7 @@ invariant() foo();
*/ */
MixinDeclaration parseMixinDeclaration() MixinDeclaration parseMixinDeclaration()
{ {
mixin(traceEnterAndExit!(__FUNCTION__));
auto node = new MixinDeclaration; auto node = new MixinDeclaration;
node.mixinExpression = parseMixinExpression(); node.mixinExpression = parseMixinExpression();
expect(TokenType.semicolon); expect(TokenType.semicolon);
@ -3298,6 +3371,7 @@ invariant() foo();
*/ */
MixinExpression parseMixinExpression() MixinExpression parseMixinExpression()
{ {
mixin(traceEnterAndExit!(__FUNCTION__));
auto node = new MixinExpression; auto node = new MixinExpression;
expect(TokenType.mixin_); expect(TokenType.mixin_);
expect(TokenType.lParen); expect(TokenType.lParen);
@ -3315,6 +3389,7 @@ invariant() foo();
*/ */
MixinTemplateName parseMixinTemplateName() MixinTemplateName parseMixinTemplateName()
{ {
mixin(traceEnterAndExit!(__FUNCTION__));
auto node = new MixinTemplateName; auto node = new MixinTemplateName;
if (currentIs(TokenType.typeof_)) if (currentIs(TokenType.typeof_))
{ {
@ -3905,7 +3980,7 @@ q{(int a, ...)
* Parses a PrimaryExpression * Parses a PrimaryExpression
* *
* $(GRAMMAR $(RULEDEF primaryExpression): * $(GRAMMAR $(RULEDEF primaryExpression):
* $(LITERAL Identifier) * $(RULE identifierOrTemplateInstance)
* | $(RULE basicType) $(LITERAL '.') $(LITERAL Identifier) * | $(RULE basicType) $(LITERAL '.') $(LITERAL Identifier)
* | $(RULE typeofExpression) * | $(RULE typeofExpression)
* | $(RULE typeidExpression) * | $(RULE typeidExpression)
@ -3951,14 +4026,14 @@ q{(int a, ...)
if (peekIs(TokenType.goesTo)) if (peekIs(TokenType.goesTo))
node.lambdaExpression = parseLambdaExpression(); node.lambdaExpression = parseLambdaExpression();
else else
node.identifier = advance(); node.identifierOrTemplateInstance = parseIdentifierOrTemplateInstance();
break; break;
mixin (BASIC_TYPE_CASE_RANGE); mixin (BASIC_TYPE_CASE_RANGE);
node.basicType = advance().type; node.basicType = advance().type;
expect(dot); expect(dot);
auto t = expect(identifier); auto t = expect(identifier);
if (t !is null) if (t !is null)
node.identifier = *t; node.primary = *t;
break; break;
case function_: case function_:
case delegate_: case delegate_:
@ -3984,7 +4059,8 @@ q{(int a, ...)
node.arrayLiteral = parseArrayLiteral(); node.arrayLiteral = parseArrayLiteral();
break; break;
case lParen: case lParen:
if (peekPastParens().type == TokenType.goesTo) auto p = peekPastParens();
if (p !is null && p.type == TokenType.goesTo)
node.lambdaExpression = parseLambdaExpression(); node.lambdaExpression = parseLambdaExpression();
else else
{ {
@ -4201,11 +4277,11 @@ q{(int a, ...)
{ {
auto node = new SliceExpression; auto node = new SliceExpression;
node.unaryExpression = unary is null ? parseUnaryExpression() : unary; node.unaryExpression = unary is null ? parseUnaryExpression() : unary;
expect(TokenType.lBrace); expect(TokenType.lBracket);
node.lower = parseAssignExpression(); node.lower = parseAssignExpression();
expect(TokenType.slice); expect(TokenType.slice);
node.upper = parseAssignExpression(); node.upper = parseAssignExpression();
expect(TokenType.rBrace); expect(TokenType.rBracket);
return node; return node;
} }
@ -4260,7 +4336,7 @@ q{(int a, ...)
/** /**
* Parses a StaticAssertStatement * Parses a StaticAssertStatement
* *
* $(GRAMMAR $(RULEDEF) staticAssertStatement): * $(GRAMMAR $(RULEDEF staticAssertStatement):
* $(LITERAL 'static') $(RULE assertExpression) $(LITERAL ';') * $(LITERAL 'static') $(RULE assertExpression) $(LITERAL ';')
* ;) * ;)
*/ */
@ -4818,9 +4894,9 @@ q{(int a, ...)
{ {
mixin(traceEnterAndExit!(__FUNCTION__)); mixin(traceEnterAndExit!(__FUNCTION__));
auto node = new TemplateParameters; auto node = new TemplateParameters;
assert (expect(TokenType.lParen)); if (expect(TokenType.lParen) is null) return null;
node.templateParameterList = parseTemplateParameterList(); node.templateParameterList = parseTemplateParameterList();
assert (expect(TokenType.rParen)); if (expect(TokenType.rParen) is null) return null;
return node; return node;
} }
@ -4942,7 +5018,7 @@ q{(int a, ...)
TemplateValueParameter parseTemplateValueParameter() TemplateValueParameter parseTemplateValueParameter()
{ {
auto node = new TemplateValueParameter; auto node = new TemplateValueParameter;
if ((node.type = parseType) is null) return null; if ((node.type = parseType()) is null) return null;
auto ident = expect(TokenType.identifier); auto ident = expect(TokenType.identifier);
if (ident is null) return null; if (ident is null) return null;
node.identifier = *ident; node.identifier = *ident;
@ -5080,6 +5156,7 @@ q{(int a, ...)
case TokenType.immutable_: case TokenType.immutable_:
case TokenType.inout_: case TokenType.inout_:
case TokenType.shared_: case TokenType.shared_:
case TokenType.scope_:
if (!peekIs(TokenType.lParen)) if (!peekIs(TokenType.lParen))
node.typeConstructors = parseTypeConstructors(); node.typeConstructors = parseTypeConstructors();
break; break;
@ -5166,20 +5243,23 @@ q{(int a, ...)
* | $(LITERAL 'immutable') * | $(LITERAL 'immutable')
* | $(LITERAL 'inout') * | $(LITERAL 'inout')
* | $(LITERAL 'shared') * | $(LITERAL 'shared')
* | $(LITERAL 'scope')
* ;) * ;)
*/ */
TokenType parseTypeConstructor(bool validate = true) TokenType parseTypeConstructor(bool validate = true)
{ {
mixin(traceEnterAndExit!(__FUNCTION__));
with (TokenType) switch (current.type) with (TokenType) switch (current.type)
{ {
case const_: case const_:
case immutable_: case immutable_:
case inout_: case inout_:
case shared_: case shared_:
case scope_:
return advance().type; return advance().type;
default: default:
if (validate) if (validate)
error(`"const", "immutable", "inout", or "shared" expected`); error(`"const", "immutable", "inout", "shared", or "scope" expected`);
return invalid; return invalid;
} }
} }
@ -5228,6 +5308,7 @@ q{(int a, ...)
*/ */
TypeSpecialization parseTypeSpecialization() TypeSpecialization parseTypeSpecialization()
{ {
mixin(traceEnterAndExit!(__FUNCTION__));
auto node = new TypeSpecialization; auto node = new TypeSpecialization;
with (TokenType) switch (current.type) with (TokenType) switch (current.type)
{ {
@ -5238,13 +5319,21 @@ q{(int a, ...)
case function_: case function_:
case delegate_: case delegate_:
case super_: case super_:
case return_:
case parameters:
if (peekIsOneOf(rParen, comma))
node.token = advance();
else
goto default;
break;
case const_: case const_:
case immutable_: case immutable_:
case inout_: case inout_:
case shared_: case shared_:
case return_: if (peekIsOneOf(rParen, comma))
case parameters: node.token = advance();
node.token = advance(); else
goto default;
break; break;
default: default:
node.type = parseType(); node.type = parseType();
@ -5467,25 +5556,32 @@ q{doStuff(5)}c;
* Parses an UnionDeclaration * Parses an UnionDeclaration
* *
* $(GRAMMAR $(RULEDEF unionDeclaration): * $(GRAMMAR $(RULEDEF unionDeclaration):
* $(LITERAL 'union') $(LITERAL Identifier) (($(RULE templateParameters) $(RULE constraint)? $(RULE structBody))? | ($(RULE structBody) | $(LITERAL ';'))) * $(LITERAL 'union') $(LITERAL Identifier) (($(RULE templateParameters) $(RULE constraint)? $(RULE structBody))? | ($(RULE structBody) | $(LITERAL ';')))
* | $(LITERAL 'union') $(RULE structBody)
* ;) * ;)
*/ */
UnionDeclaration parseUnionDeclaration() UnionDeclaration parseUnionDeclaration()
{ {
auto node = new UnionDeclaration; auto node = new UnionDeclaration;
expect(TokenType.union_); expect(TokenType.union_);
auto ident = expect(TokenType.identifier); bool templated = false;
if (ident is null) return null; if (currentIs(TokenType.identifier))
node.identifier = *ident;
if (currentIs(TokenType.lParen))
{ {
node.templateParameters = parseTemplateParameters(); node.identifier = advance();
if (currentIs(TokenType.if_)) if (currentIs(TokenType.lParen))
node.constraint = parseConstraint(); {
node.structBody = parseStructBody(); templated = true;
node.templateParameters = parseTemplateParameters();
if (currentIs(TokenType.if_))
node.constraint = parseConstraint();
node.structBody = parseStructBody();
}
else
goto semiOrStructBody;
} }
else else
{ {
semiOrStructBody:
if (currentIs(TokenType.semicolon)) if (currentIs(TokenType.semicolon))
advance(); advance();
else else
@ -5513,7 +5609,7 @@ q{doStuff(5)}c;
* Parses a VariableDeclaration * Parses a VariableDeclaration
* *
* $(GRAMMAR $(RULEDEF variableDeclaration): * $(GRAMMAR $(RULEDEF variableDeclaration):
* $(RULE storageClass)? $(RULE _type) $(RULE declarator) ($(LITERAL ',') $(RULE declarator))* $(LITERAL ';') * $(RULE _type) $(RULE declarator) ($(LITERAL ',') $(RULE declarator))* $(LITERAL ';')
* | $(RULE autoDeclaration) * | $(RULE autoDeclaration)
* ;) * ;)
*/ */
@ -5521,14 +5617,15 @@ q{doStuff(5)}c;
{ {
mixin (traceEnterAndExit!(__FUNCTION__)); mixin (traceEnterAndExit!(__FUNCTION__));
auto node = new VariableDeclaration; auto node = new VariableDeclaration;
if (currentIs(TokenType.auto_)) with (TokenType) if (currentIsOneOf(const_, immutable_, inout_, shared_,
abstract_, auto_, deprecated_, enum_, extern_, final_, nothrow_,
override_, pure_, gshared, scope_, static_, synchronized_)
&& !(peekIs(lParen)))
{ {
node.autoDeclaration = parseAutoDeclaration(); node.autoDeclaration = parseAutoDeclaration();
return node; return node;
} }
// TODO: handle storage class here
node.type = type is null ? parseType() : type; node.type = type is null ? parseType() : type;
while(true) while(true)
@ -5832,12 +5929,12 @@ private:
if (suppressMessages <= 0) if (suppressMessages <= 0)
{ {
++errorCount; ++errorCount;
auto column = index < tokens.length ? tokens[index].column : 0;
auto line = index < tokens.length ? tokens[index].line : 0;
if (errorFunction is null) if (errorFunction is null)
stderr.writefln("!! %s(%d:%d): %s", fileName, tokens[index].line, stderr.writefln("!! %s(%d:%d): %s", fileName, line, column, message);
tokens[index].column, message);
else else
errorFunction(fileName, tokens[index].line, tokens[index].column, errorFunction(fileName, line, column, message);
message);
} }
while (moreTokens()) while (moreTokens())
{ {
@ -5891,6 +5988,11 @@ private:
skipContent!(TokenType.lBracket, TokenType.rBracket)(); skipContent!(TokenType.lBracket, TokenType.rBracket)();
} }
const(Token)* peek()
{
return index + 1 < tokens.length ? &tokens[index + 1] : null;
}
const(Token)* peekPast(alias O, alias C)() const(Token)* peekPast(alias O, alias C)()
in in
{ {
@ -5901,8 +6003,10 @@ private:
int depth = 1; int depth = 1;
auto i = index; auto i = index;
++i; ++i;
while (index < tokens.length) while (i < tokens.length)
{ {
if (i >= tokens.length)
return null;
if (tokens[i] == O) if (tokens[i] == O)
++depth; ++depth;
else if (tokens[i] == C) else if (tokens[i] == C)
@ -6048,6 +6152,8 @@ private:
version (verbose) void trace(string message) version (verbose) void trace(string message)
{ {
if (suppressMessages > 0)
return;
if (index < tokens.length) if (index < tokens.length)
stderr.writeln(message, "(", current.line, ":", current.column, ")"); stderr.writeln(message, "(", current.line, ":", current.column, ")");
else else