mirror of
https://github.com/RGBCube/serenity
synced 2025-07-26 09:17:35 +00:00
LibCpp: Support parsing templatized function calls
This commit is contained in:
parent
fe4122bbae
commit
646aaa111b
4 changed files with 84 additions and 19 deletions
|
@ -483,4 +483,25 @@ String Name::full_name() const
|
||||||
return String::formatted("{}{}", builder.to_string(), m_name.is_null() ? "" : m_name->m_name);
|
return String::formatted("{}{}", builder.to_string(), m_name.is_null() ? "" : m_name->m_name);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void TemplatizedFunctionCall::dump(size_t indent) const
|
||||||
|
{
|
||||||
|
ASTNode::dump(indent);
|
||||||
|
|
||||||
|
print_indent(indent);
|
||||||
|
outln("{}", m_name->full_name());
|
||||||
|
|
||||||
|
print_indent(indent + 1);
|
||||||
|
outln("<");
|
||||||
|
for (auto& arg : m_template_arguments) {
|
||||||
|
arg.dump(indent + 1);
|
||||||
|
}
|
||||||
|
print_indent(indent + 1);
|
||||||
|
outln(">");
|
||||||
|
|
||||||
|
for (const auto& arg : m_arguments) {
|
||||||
|
arg.dump(indent + 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -460,22 +460,38 @@ public:
|
||||||
RefPtr<Expression> m_rhs;
|
RefPtr<Expression> m_rhs;
|
||||||
};
|
};
|
||||||
|
|
||||||
class FunctionCall final : public Expression {
|
class FunctionCall : public Expression {
|
||||||
public:
|
public:
|
||||||
FunctionCall(ASTNode* parent, Optional<Position> start, Optional<Position> end, const String& filename)
|
FunctionCall(ASTNode* parent, Optional<Position> start, Optional<Position> end, const String& filename)
|
||||||
: Expression(parent, start, end, filename)
|
: Expression(parent, start, end, filename)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
~FunctionCall() override = default;
|
virtual ~FunctionCall() override = default;
|
||||||
virtual const char* class_name() const override { return "FunctionCall"; }
|
virtual const char* class_name() const override { return "FunctionCall"; }
|
||||||
virtual void dump(size_t indent) const override;
|
virtual void dump(size_t indent) const override;
|
||||||
virtual bool is_function_call() const override { return true; }
|
virtual bool is_function_call() const override { return true; }
|
||||||
|
virtual bool is_templatized() const { return false; }
|
||||||
|
|
||||||
RefPtr<Name> m_name;
|
RefPtr<Name> m_name;
|
||||||
NonnullRefPtrVector<Expression> m_arguments;
|
NonnullRefPtrVector<Expression> m_arguments;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
class TemplatizedFunctionCall final : public FunctionCall {
|
||||||
|
public:
|
||||||
|
TemplatizedFunctionCall(ASTNode* parent, Optional<Position> start, Optional<Position> end, const String& filename)
|
||||||
|
: FunctionCall(parent, start, end, filename)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
~TemplatizedFunctionCall() override = default;
|
||||||
|
virtual const char* class_name() const override { return "TemplatizedFunctionCall"; }
|
||||||
|
virtual void dump(size_t indent) const override;
|
||||||
|
virtual bool is_templatized() const override { return true; }
|
||||||
|
|
||||||
|
NonnullRefPtrVector<Type> m_template_arguments;
|
||||||
|
};
|
||||||
|
|
||||||
class StringLiteral final : public Expression {
|
class StringLiteral final : public Expression {
|
||||||
public:
|
public:
|
||||||
StringLiteral(ASTNode* parent, Optional<Position> start, Optional<Position> end, const String& filename)
|
StringLiteral(ASTNode* parent, Optional<Position> start, Optional<Position> end, const String& filename)
|
||||||
|
|
|
@ -251,7 +251,7 @@ NonnullRefPtr<BlockStatement> Parser::parse_block_statement(ASTNode& parent)
|
||||||
return block_statement;
|
return block_statement;
|
||||||
}
|
}
|
||||||
|
|
||||||
Parser::MatchTypeResult Parser::match_type()
|
Parser::TemplatizedMatchResult Parser::match_type()
|
||||||
{
|
{
|
||||||
save_state();
|
save_state();
|
||||||
ScopeGuard state_guard = [this] { load_state(); };
|
ScopeGuard state_guard = [this] { load_state(); };
|
||||||
|
@ -259,17 +259,17 @@ Parser::MatchTypeResult Parser::match_type()
|
||||||
parse_type_qualifiers();
|
parse_type_qualifiers();
|
||||||
|
|
||||||
if (!match_name())
|
if (!match_name())
|
||||||
return MatchTypeResult::NoMatch;
|
return TemplatizedMatchResult::NoMatch;
|
||||||
parse_name(*m_root_node);
|
parse_name(*m_root_node);
|
||||||
|
|
||||||
if (peek(Token::Type::Less).has_value()) {
|
if (peek(Token::Type::Less).has_value()) {
|
||||||
if (match_template_arguments()) {
|
if (match_template_arguments()) {
|
||||||
return MatchTypeResult::Templatized;
|
return TemplatizedMatchResult::Templatized;
|
||||||
}
|
}
|
||||||
return MatchTypeResult::NoMatch;
|
return TemplatizedMatchResult::NoMatch;
|
||||||
}
|
}
|
||||||
|
|
||||||
return MatchTypeResult::Regular;
|
return TemplatizedMatchResult::Regular;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Parser::match_template_arguments()
|
bool Parser::match_template_arguments()
|
||||||
|
@ -282,7 +282,7 @@ bool Parser::match_template_arguments()
|
||||||
consume();
|
consume();
|
||||||
|
|
||||||
while (!eof() && peek().type() != Token::Type::Greater) {
|
while (!eof() && peek().type() != Token::Type::Greater) {
|
||||||
if (match_type() == MatchTypeResult::NoMatch)
|
if (match_type() == TemplatizedMatchResult::NoMatch)
|
||||||
return false;
|
return false;
|
||||||
parse_type(*m_root_node);
|
parse_type(*m_root_node);
|
||||||
}
|
}
|
||||||
|
@ -312,7 +312,7 @@ bool Parser::match_variable_declaration()
|
||||||
save_state();
|
save_state();
|
||||||
ScopeGuard state_guard = [this] { load_state(); };
|
ScopeGuard state_guard = [this] { load_state(); };
|
||||||
|
|
||||||
if (match_type() == MatchTypeResult::NoMatch) {
|
if (match_type() == TemplatizedMatchResult::NoMatch) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -441,7 +441,7 @@ NonnullRefPtr<Expression> Parser::parse_primary_expression(ASTNode& parent)
|
||||||
}
|
}
|
||||||
|
|
||||||
if (match_name()) {
|
if (match_name()) {
|
||||||
if (match_function_call())
|
if (match_function_call() != TemplatizedMatchResult::NoMatch)
|
||||||
return parse_function_call(parent);
|
return parse_function_call(parent);
|
||||||
return parse_name(parent);
|
return parse_name(parent);
|
||||||
}
|
}
|
||||||
|
@ -634,7 +634,7 @@ bool Parser::match_function_declaration()
|
||||||
|
|
||||||
parse_function_qualifiers();
|
parse_function_qualifiers();
|
||||||
|
|
||||||
if (match_type() == MatchTypeResult::NoMatch)
|
if (match_type() == TemplatizedMatchResult::NoMatch)
|
||||||
return false;
|
return false;
|
||||||
VERIFY(m_root_node);
|
VERIFY(m_root_node);
|
||||||
parse_type(*m_root_node);
|
parse_type(*m_root_node);
|
||||||
|
@ -904,22 +904,49 @@ void Parser::print_tokens() const
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Parser::match_function_call()
|
Parser::TemplatizedMatchResult Parser::match_function_call()
|
||||||
{
|
{
|
||||||
save_state();
|
save_state();
|
||||||
ScopeGuard state_guard = [this] { load_state(); };
|
ScopeGuard state_guard = [this] { load_state(); };
|
||||||
if (!match_name())
|
if (!match_name())
|
||||||
return false;
|
return TemplatizedMatchResult::NoMatch;
|
||||||
parse_name(*m_root_node);
|
parse_name(*m_root_node);
|
||||||
|
|
||||||
return match(Token::Type::LeftParen);
|
bool is_templatized = false;
|
||||||
|
if (match_template_arguments()) {
|
||||||
|
is_templatized = true;
|
||||||
|
parse_template_arguments(*m_root_node);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!match(Token::Type::LeftParen))
|
||||||
|
return TemplatizedMatchResult::NoMatch;
|
||||||
|
|
||||||
|
return is_templatized ? TemplatizedMatchResult::Templatized : TemplatizedMatchResult::Regular;
|
||||||
}
|
}
|
||||||
|
|
||||||
NonnullRefPtr<FunctionCall> Parser::parse_function_call(ASTNode& parent)
|
NonnullRefPtr<FunctionCall> Parser::parse_function_call(ASTNode& parent)
|
||||||
{
|
{
|
||||||
SCOPE_LOGGER();
|
SCOPE_LOGGER();
|
||||||
auto call = create_ast_node<FunctionCall>(parent, position(), {});
|
|
||||||
|
auto match_result = match_type();
|
||||||
|
if (match_result == TemplatizedMatchResult::NoMatch) {
|
||||||
|
error("expected type");
|
||||||
|
return create_ast_node<FunctionCall>(parent, position(), position());
|
||||||
|
}
|
||||||
|
|
||||||
|
bool is_templatized = match_result == TemplatizedMatchResult::Templatized;
|
||||||
|
|
||||||
|
RefPtr<FunctionCall> call;
|
||||||
|
if (is_templatized) {
|
||||||
|
call = create_ast_node<TemplatizedFunctionCall>(parent, position(), {});
|
||||||
|
} else {
|
||||||
|
call = create_ast_node<FunctionCall>(parent, position(), {});
|
||||||
|
}
|
||||||
|
|
||||||
call->m_name = parse_name(*call);
|
call->m_name = parse_name(*call);
|
||||||
|
if (is_templatized) {
|
||||||
|
static_cast<TemplatizedFunctionCall&>(*call).m_template_arguments = parse_template_arguments(*call);
|
||||||
|
}
|
||||||
|
|
||||||
NonnullRefPtrVector<Expression> args;
|
NonnullRefPtrVector<Expression> args;
|
||||||
consume(Token::Type::LeftParen);
|
consume(Token::Type::LeftParen);
|
||||||
|
@ -931,7 +958,8 @@ NonnullRefPtr<FunctionCall> Parser::parse_function_call(ASTNode& parent)
|
||||||
consume(Token::Type::RightParen);
|
consume(Token::Type::RightParen);
|
||||||
call->m_arguments = move(args);
|
call->m_arguments = move(args);
|
||||||
call->set_end(position());
|
call->set_end(position());
|
||||||
return call;
|
|
||||||
|
return call.release_nonnull();
|
||||||
}
|
}
|
||||||
|
|
||||||
NonnullRefPtr<StringLiteral> Parser::parse_string_literal(ASTNode& parent)
|
NonnullRefPtr<StringLiteral> Parser::parse_string_literal(ASTNode& parent)
|
||||||
|
|
|
@ -78,7 +78,6 @@ private:
|
||||||
bool match_whitespace();
|
bool match_whitespace();
|
||||||
bool match_variable_declaration();
|
bool match_variable_declaration();
|
||||||
bool match_expression();
|
bool match_expression();
|
||||||
bool match_function_call();
|
|
||||||
bool match_secondary_expression();
|
bool match_secondary_expression();
|
||||||
bool match_enum_declaration();
|
bool match_enum_declaration();
|
||||||
bool match_struct_declaration();
|
bool match_struct_declaration();
|
||||||
|
@ -91,12 +90,13 @@ private:
|
||||||
bool match_template_arguments();
|
bool match_template_arguments();
|
||||||
bool match_name();
|
bool match_name();
|
||||||
|
|
||||||
enum class MatchTypeResult {
|
enum class TemplatizedMatchResult {
|
||||||
NoMatch,
|
NoMatch,
|
||||||
Regular,
|
Regular,
|
||||||
Templatized,
|
Templatized,
|
||||||
};
|
};
|
||||||
MatchTypeResult match_type();
|
TemplatizedMatchResult match_type();
|
||||||
|
TemplatizedMatchResult match_function_call();
|
||||||
|
|
||||||
Optional<NonnullRefPtrVector<Parameter>> parse_parameter_list(ASTNode& parent);
|
Optional<NonnullRefPtrVector<Parameter>> parse_parameter_list(ASTNode& parent);
|
||||||
Optional<Token> consume_whitespace();
|
Optional<Token> consume_whitespace();
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue