/* * Copyright (c) 2021, Itamar S. * * SPDX-License-Identifier: BSD-2-Clause */ #include "AST.h" namespace Cpp { static void print_indent(FILE* output, int indent) { for (int i = 0; i < indent * 2; ++i) out(output, " "); } void ASTNode::dump(FILE* output, size_t indent) const { print_indent(output, indent); outln(output, "{}[{}:{}->{}:{}]", class_name(), start().line, start().column, end().line, end().column); } void TranslationUnit::dump(FILE* output, size_t indent) const { ASTNode::dump(output, indent); for (auto const& child : m_declarations) { child.dump(output, indent + 1); } } void FunctionDeclaration::dump(FILE* output, size_t indent) const { ASTNode::dump(output, indent); String qualifiers_string; if (!m_qualifiers.is_empty()) { print_indent(output, indent + 1); outln(output, "[{}]", String::join(' ', m_qualifiers)); } m_return_type->dump(output, indent + 1); if (!m_name.is_null()) { print_indent(output, indent + 1); outln(output, "{}", m_name->full_name()); } print_indent(output, indent + 1); outln(output, "("); for (auto const& arg : m_parameters) { arg.dump(output, indent + 1); } print_indent(output, indent + 1); outln(output, ")"); if (!m_definition.is_null()) { m_definition->dump(output, indent + 1); } } NonnullRefPtrVector FunctionDeclaration::declarations() const { NonnullRefPtrVector declarations; for (auto& arg : m_parameters) { declarations.append(arg); } if (m_definition) declarations.extend(m_definition->declarations()); return declarations; } void Type::dump(FILE* output, size_t indent) const { ASTNode::dump(output, indent); print_indent(output, indent + 1); outln(output, "{}", to_string()); } String NamedType::to_string() const { String qualifiers_string; if (!qualifiers().is_empty()) qualifiers_string = String::formatted("[{}] ", String::join(' ', qualifiers())); String name; if (is_auto()) name = "auto"; else name = m_name.is_null() ? ""sv : m_name->full_name(); return String::formatted("{}{}", qualifiers_string, name); } String Pointer::to_string() const { if (!m_pointee) return {}; StringBuilder builder; builder.append(m_pointee->to_string()); builder.append('*'); return builder.to_string(); } String Reference::to_string() const { if (!m_referenced_type) return {}; StringBuilder builder; builder.append(m_referenced_type->to_string()); if (m_kind == Kind::Lvalue) builder.append('&'); else builder.append("&&"sv); return builder.to_string(); } String FunctionType::to_string() const { StringBuilder builder; builder.append(m_return_type->to_string()); builder.append('('); bool first = true; for (auto& parameter : m_parameters) { if (first) first = false; else builder.append(", "sv); if (parameter.type()) builder.append(parameter.type()->to_string()); if (parameter.name() && !parameter.full_name().is_empty()) { builder.append(' '); builder.append(parameter.full_name()); } } builder.append(')'); return builder.to_string(); } void Parameter::dump(FILE* output, size_t indent) const { ASTNode::dump(output, indent); if (m_is_ellipsis) { print_indent(output, indent + 1); outln(output, "..."); } if (!m_name.is_null()) { print_indent(output, indent); outln(output, "{}", m_name->full_name()); } if (m_type) m_type->dump(output, indent + 1); } void FunctionDefinition::dump(FILE* output, size_t indent) const { ASTNode::dump(output, indent); print_indent(output, indent); outln(output, "{{"); for (auto const& statement : m_statements) { statement.dump(output, indent + 1); } print_indent(output, indent); outln(output, "}}"); } NonnullRefPtrVector FunctionDefinition::declarations() const { NonnullRefPtrVector declarations; for (auto& statement : m_statements) { declarations.extend(statement.declarations()); } return declarations; } void VariableDeclaration::dump(FILE* output, size_t indent) const { ASTNode::dump(output, indent); if (m_type) m_type->dump(output, indent + 1); print_indent(output, indent + 1); outln(output, "{}", full_name()); if (m_initial_value) m_initial_value->dump(output, indent + 1); } void Identifier::dump(FILE* output, size_t indent) const { ASTNode::dump(output, indent); print_indent(output, indent); outln(output, "{}", m_name); } void NumericLiteral::dump(FILE* output, size_t indent) const { ASTNode::dump(output, indent); print_indent(output, indent); outln(output, "{}", m_value); } void BinaryExpression::dump(FILE* output, size_t indent) const { ASTNode::dump(output, indent); char const* op_string = nullptr; switch (m_op) { case BinaryOp::Addition: op_string = "+"; break; case BinaryOp::Subtraction: op_string = "-"; break; case BinaryOp::Multiplication: op_string = "*"; break; case BinaryOp::Division: op_string = "/"; break; case BinaryOp::Modulo: op_string = "%"; break; case BinaryOp::GreaterThan: op_string = ">"; break; case BinaryOp::GreaterThanEquals: op_string = ">="; break; case BinaryOp::LessThan: op_string = "<"; break; case BinaryOp::LessThanEquals: op_string = "<="; break; case BinaryOp::BitwiseAnd: op_string = "&"; break; case BinaryOp::BitwiseOr: op_string = "|"; break; case BinaryOp::BitwiseXor: op_string = "^"; break; case BinaryOp::LeftShift: op_string = "<<"; break; case BinaryOp::RightShift: op_string = ">>"; break; case BinaryOp::EqualsEquals: op_string = "=="; break; case BinaryOp::NotEqual: op_string = "!="; break; case BinaryOp::LogicalOr: op_string = "||"; break; case BinaryOp::LogicalAnd: op_string = "&&"; break; case BinaryOp::Arrow: op_string = "->"; break; } m_lhs->dump(output, indent + 1); print_indent(output, indent + 1); VERIFY(op_string); outln(output, "{}", op_string); m_rhs->dump(output, indent + 1); } void AssignmentExpression::dump(FILE* output, size_t indent) const { ASTNode::dump(output, indent); char const* op_string = nullptr; switch (m_op) { case AssignmentOp::Assignment: op_string = "="; break; case AssignmentOp::AdditionAssignment: op_string = "+="; break; case AssignmentOp::SubtractionAssignment: op_string = "-="; break; } m_lhs->dump(output, indent + 1); print_indent(output, indent + 1); VERIFY(op_string); outln(output, "{}", op_string); m_rhs->dump(output, indent + 1); } void FunctionCall::dump(FILE* output, size_t indent) const { ASTNode::dump(output, indent); m_callee->dump(output, indent + 1); for (auto const& arg : m_arguments) { arg.dump(output, indent + 1); } } void StringLiteral::dump(FILE* output, size_t indent) const { ASTNode::dump(output, indent); print_indent(output, indent + 1); outln(output, "{}", m_value); } void ReturnStatement::dump(FILE* output, size_t indent) const { ASTNode::dump(output, indent); if (m_value) m_value->dump(output, indent + 1); } void EnumDeclaration::dump(FILE* output, size_t indent) const { ASTNode::dump(output, indent); print_indent(output, indent); outln(output, "{}", full_name()); for (auto& entry : m_entries) { print_indent(output, indent + 1); outln(output, "{}", entry.name); if (entry.value) entry.value->dump(output, indent + 2); } } void StructOrClassDeclaration::dump(FILE* output, size_t indent) const { ASTNode::dump(output, indent); print_indent(output, indent); outln(output, "{}", full_name()); if (!m_baseclasses.is_empty()) { print_indent(output, indent + 1); outln(output, ":"); for (size_t i = 0; i < m_baseclasses.size(); ++i) { auto& baseclass = m_baseclasses[i]; baseclass.dump(output, indent + 1); if (i < m_baseclasses.size() - 1) { print_indent(output, indent + 1); outln(output, ","); } } } outln(output, ""); for (auto& member : m_members) { member.dump(output, indent + 1); } } NonnullRefPtrVector StructOrClassDeclaration::declarations() const { NonnullRefPtrVector declarations; for (auto& member : m_members) declarations.append(member); return declarations; } void UnaryExpression::dump(FILE* output, size_t indent) const { ASTNode::dump(output, indent); char const* op_string = nullptr; switch (m_op) { case UnaryOp::BitwiseNot: op_string = "~"; break; case UnaryOp::Not: op_string = "!"; break; case UnaryOp::Plus: op_string = "+"; break; case UnaryOp::Minus: op_string = "-"; break; case UnaryOp::PlusPlus: op_string = "++"; break; case UnaryOp::Address: op_string = "&"; break; default: op_string = ""; } VERIFY(op_string); print_indent(output, indent + 1); outln(output, "{}", op_string); m_lhs->dump(output, indent + 1); } void BooleanLiteral::dump(FILE* output, size_t indent) const { ASTNode::dump(output, indent); print_indent(output, indent + 1); outln(output, "{}", m_value ? "true" : "false"); } void Pointer::dump(FILE* output, size_t indent) const { ASTNode::dump(output, indent); if (!m_pointee.is_null()) { m_pointee->dump(output, indent + 1); } } void Reference::dump(FILE* output, size_t indent) const { ASTNode::dump(output, indent); print_indent(output, indent + 1); outln(output, "{}", m_kind == Kind::Lvalue ? "&" : "&&"); if (!m_referenced_type.is_null()) { m_referenced_type->dump(output, indent + 1); } } void FunctionType::dump(FILE* output, size_t indent) const { ASTNode::dump(output, indent); if (m_return_type) m_return_type->dump(output, indent + 1); print_indent(output, indent + 1); outln("("); for (auto& parameter : m_parameters) parameter.dump(output, indent + 2); print_indent(output, indent + 1); outln(")"); } void MemberExpression::dump(FILE* output, size_t indent) const { ASTNode::dump(output, indent); m_object->dump(output, indent + 1); m_property->dump(output, indent + 1); } void BlockStatement::dump(FILE* output, size_t indent) const { ASTNode::dump(output, indent); for (auto& statement : m_statements) { statement.dump(output, indent + 1); } } void ForStatement::dump(FILE* output, size_t indent) const { ASTNode::dump(output, indent); if (m_init) m_init->dump(output, indent + 1); if (m_test) m_test->dump(output, indent + 1); if (m_update) m_update->dump(output, indent + 1); if (m_body) m_body->dump(output, indent + 1); } NonnullRefPtrVector Statement::declarations() const { if (is_declaration()) { NonnullRefPtrVector vec; auto const& decl = static_cast(*this); vec.empend(const_cast(decl)); return vec; } return {}; } NonnullRefPtrVector ForStatement::declarations() const { NonnullRefPtrVector declarations; if (m_init) declarations.extend(m_init->declarations()); if (m_body) declarations.extend(m_body->declarations()); return declarations; } NonnullRefPtrVector BlockStatement::declarations() const { NonnullRefPtrVector declarations; for (auto& statement : m_statements) { declarations.extend(statement.declarations()); } return declarations; } void IfStatement::dump(FILE* output, size_t indent) const { ASTNode::dump(output, indent); if (m_predicate) { print_indent(output, indent + 1); outln(output, "Predicate:"); m_predicate->dump(output, indent + 1); } if (m_then) { print_indent(output, indent + 1); outln(output, "Then:"); m_then->dump(output, indent + 1); } if (m_else) { print_indent(output, indent + 1); outln(output, "Else:"); m_else->dump(output, indent + 1); } } NonnullRefPtrVector IfStatement::declarations() const { NonnullRefPtrVector declarations; if (m_predicate) declarations.extend(m_predicate->declarations()); if (m_then) declarations.extend(m_then->declarations()); if (m_else) declarations.extend(m_else->declarations()); return declarations; } void NamespaceDeclaration::dump(FILE* output, size_t indent) const { ASTNode::dump(output, indent); print_indent(output, indent + 1); outln(output, "{}", full_name()); for (auto& decl : m_declarations) decl.dump(output, indent + 1); } void NullPointerLiteral::dump(FILE* output, size_t indent) const { ASTNode::dump(output, indent); } void Name::dump(FILE* output, size_t indent) const { ASTNode::dump(output, indent); print_indent(output, indent); outln(output, "{}", full_name()); } StringView Name::full_name() const { if (m_full_name.has_value()) return *m_full_name; StringBuilder builder; if (!m_scope.is_empty()) { for (auto& scope : m_scope) { builder.appendff("{}::", scope.name()); } } m_full_name = String::formatted("{}{}", builder.to_string(), m_name.is_null() ? ""sv : m_name->name()); return *m_full_name; } StringView TemplatizedName::full_name() const { if (m_full_name.has_value()) return *m_full_name; StringBuilder name; name.append(Name::full_name()); name.append('<'); for (auto& type : m_template_arguments) { name.append(type.to_string()); } name.append('>'); m_full_name = name.to_string(); return *m_full_name; } void CppCastExpression::dump(FILE* output, size_t indent) const { ASTNode::dump(output, indent); print_indent(output, indent); outln(output, "{}", m_cast_type); print_indent(output, indent + 1); outln(output, "<"); if (m_type) m_type->dump(output, indent + 1); print_indent(output, indent + 1); outln(output, ">"); if (m_expression) m_expression->dump(output, indent + 1); } void SizeofExpression::dump(FILE* output, size_t indent) const { ASTNode::dump(output, indent); if (m_type) m_type->dump(output, indent + 1); } void BracedInitList::dump(FILE* output, size_t indent) const { ASTNode::dump(output, indent); for (auto& exp : m_expressions) { exp.dump(output, indent + 1); } } void CStyleCastExpression::dump(FILE* output, size_t indent) const { ASTNode::dump(output, indent); if (m_type) m_type->dump(output, indent + 1); if (m_expression) m_expression->dump(output, indent + 1); } void Constructor::dump(FILE* output, size_t indent) const { print_indent(output, indent); outln(output, "C'tor"); print_indent(output, indent + 1); outln(output, "("); for (auto const& arg : parameters()) { arg.dump(output, indent + 1); } print_indent(output, indent + 1); outln(output, ")"); if (definition()) { definition()->dump(output, indent + 1); } } void Destructor::dump(FILE* output, size_t indent) const { print_indent(output, indent); outln(output, "D'tor"); print_indent(output, indent + 1); outln(output, "("); for (auto const& arg : parameters()) { arg.dump(output, indent + 1); } print_indent(output, indent + 1); outln(output, ")"); if (definition()) { definition()->dump(output, indent + 1); } } StringView Declaration::full_name() const { if (!m_full_name.has_value()) { if (m_name) m_full_name = m_name->full_name(); else m_full_name = String::empty(); } return *m_full_name; } }