fixed array subscript and member access parsing.
This commit is contained in:
parent
c8e88a6a98
commit
8b4b81e90b
1 changed files with 45 additions and 25 deletions
70
parser.c
70
parser.c
|
|
@ -213,6 +213,20 @@ static ast_node *parse_factor(parser *p)
|
||||||
node->expr.flt = parse_float(t->lexeme, t->lexeme_len);
|
node->expr.flt = parse_float(t->lexeme, t->lexeme_len);
|
||||||
return node;
|
return node;
|
||||||
}
|
}
|
||||||
|
else if (match(p, TOKEN_TRUE)) {
|
||||||
|
ast_node *node = arena_alloc(p->allocator, sizeof(ast_node));
|
||||||
|
node->type = NODE_BOOL;
|
||||||
|
node->position = p->previous->position;
|
||||||
|
node->expr.boolean = 1;
|
||||||
|
return node;
|
||||||
|
}
|
||||||
|
else if (match(p, TOKEN_FALSE)) {
|
||||||
|
ast_node *node = arena_alloc(p->allocator, sizeof(ast_node));
|
||||||
|
node->type = NODE_BOOL;
|
||||||
|
node->position = p->previous->position;
|
||||||
|
node->expr.boolean = 0;
|
||||||
|
return node;
|
||||||
|
}
|
||||||
else if (match_peek(p, TOKEN_IDENTIFIER))
|
else if (match_peek(p, TOKEN_IDENTIFIER))
|
||||||
{
|
{
|
||||||
/* If a `(` is found after an identifier, it should be a call. */
|
/* If a `(` is found after an identifier, it should be a call. */
|
||||||
|
|
@ -396,41 +410,47 @@ ast_node *parse_expression(parser *p)
|
||||||
* If after parsing an expression a `[` character
|
* If after parsing an expression a `[` character
|
||||||
* is found, it should be an array subscript expression.
|
* is found, it should be an array subscript expression.
|
||||||
*/
|
*/
|
||||||
if (match(p, TOKEN_LSQUARE))
|
if (match_peek(p, TOKEN_LSQUARE)) {
|
||||||
{
|
while (match(p, TOKEN_LSQUARE)) {
|
||||||
ast_node *index = parse_expression(p);
|
ast_node *index = parse_expression(p);
|
||||||
ast_node *node = arena_alloc(p->allocator, sizeof(ast_node));
|
ast_node *node = arena_alloc(p->allocator, sizeof(ast_node));
|
||||||
node->type = NODE_ARRAY_SUBSCRIPT;
|
node->type = NODE_ARRAY_SUBSCRIPT;
|
||||||
node->position = p->previous->position;
|
node->position = p->previous->position;
|
||||||
node->expr.subscript.expr = left;
|
node->expr.subscript.expr = left;
|
||||||
node->expr.subscript.index = index;
|
node->expr.subscript.index = index;
|
||||||
|
|
||||||
|
if (!match(p, TOKEN_RSQUARE))
|
||||||
|
{
|
||||||
|
error(p, "expected `]`.");
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
left = node;
|
||||||
|
|
||||||
if (!match(p, TOKEN_RSQUARE))
|
|
||||||
{
|
|
||||||
error(p, "expected `]`.");
|
|
||||||
return NULL;
|
|
||||||
}
|
}
|
||||||
return node;
|
return left;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* If after parsing an expression a `.` character
|
* If after parsing an expression a `.` character
|
||||||
* is found, it should be a member access expression.
|
* is found, it should be a member access expression.
|
||||||
*/
|
*/
|
||||||
if (match(p, TOKEN_DOT))
|
if (match_peek(p, TOKEN_DOT))
|
||||||
{
|
{
|
||||||
if (!match_peek(p, TOKEN_IDENTIFIER))
|
while (match(p, TOKEN_DOT)) {
|
||||||
{
|
if (!match_peek(p, TOKEN_IDENTIFIER)) {
|
||||||
error(p, "expected identifier after member access.");
|
error(p, "expected identifier after member access.");
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
ast_node *node = arena_alloc(p->allocator, sizeof(ast_node));
|
ast_node *node = arena_alloc(p->allocator, sizeof(ast_node));
|
||||||
node->type = NODE_ACCESS;
|
node->type = NODE_ACCESS;
|
||||||
node->position = p->previous->position;
|
node->position = p->previous->position;
|
||||||
node->expr.access.expr = left;
|
node->expr.access.expr = left;
|
||||||
node->expr.access.member = parse_expression(p);
|
node->expr.access.member = parse_factor(p);
|
||||||
|
|
||||||
return node;
|
left = node;
|
||||||
|
}
|
||||||
|
return left;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue