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

View File

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