// Part of the Carbon Language project, under the Apache License v2.0 with LLVM // Exceptions. See /LICENSE for license information. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception #include "explorer/ast/declaration.h" #include "llvm/ADT/StringExtras.h" #include "llvm/Support/Casting.h" namespace Carbon { using llvm::cast; Declaration::~Declaration() = default; void Declaration::Print(llvm::raw_ostream& out) const { switch (kind()) { case DeclarationKind::InterfaceDeclaration: { const auto& iface_decl = cast(*this); PrintID(out); out << " {\n"; for (Nonnull m : iface_decl.members()) { out << *m; } out << "}\n"; break; } case DeclarationKind::ImplDeclaration: { const auto& impl_decl = cast(*this); PrintID(out); out << " {\n"; for (Nonnull m : impl_decl.members()) { out << *m; } out << "}\n"; break; } case DeclarationKind::FunctionDeclaration: cast(*this).PrintDepth(-1, out); break; case DeclarationKind::DestructorDeclaration: cast(*this).PrintDepth(-1, out); break; case DeclarationKind::ClassDeclaration: { const auto& class_decl = cast(*this); PrintID(out); if (class_decl.type_params().has_value()) { out << **class_decl.type_params(); } out << " {\n"; for (Nonnull m : class_decl.members()) { out << *m; } out << "}\n"; break; } case DeclarationKind::MixinDeclaration: { const auto& mixin_decl = cast(*this); PrintID(out); out << "{\n"; for (Nonnull m : mixin_decl.members()) { out << *m; } out << "}\n"; break; } case DeclarationKind::MixDeclaration: { const auto& mix_decl = cast(*this); PrintID(out); out << mix_decl.mixin() << ";"; break; } case DeclarationKind::ChoiceDeclaration: { const auto& choice = cast(*this); PrintID(out); out << " {\n"; for (Nonnull alt : choice.alternatives()) { out << *alt << ";\n"; } out << "}\n"; break; } case DeclarationKind::VariableDeclaration: { const auto& var = cast(*this); PrintID(out); if (var.has_initializer()) { out << " = " << var.initializer(); } out << ";\n"; break; } case DeclarationKind::AssociatedConstantDeclaration: PrintID(out); out << ";\n"; break; case DeclarationKind::SelfDeclaration: { out << "Self"; break; } case DeclarationKind::AliasDeclaration: { const auto& alias = cast(*this); PrintID(out); out << " = " << alias.target() << ";\n"; break; } } } void Declaration::PrintID(llvm::raw_ostream& out) const { switch (kind()) { case DeclarationKind::InterfaceDeclaration: { const auto& iface_decl = cast(*this); out << "interface " << iface_decl.name(); break; } case DeclarationKind::ImplDeclaration: { const auto& impl_decl = cast(*this); switch (impl_decl.kind()) { case ImplKind::InternalImpl: break; case ImplKind::ExternalImpl: out << "external "; break; } out << "impl " << *impl_decl.impl_type() << " as " << impl_decl.interface(); break; } case DeclarationKind::FunctionDeclaration: out << "fn " << cast(*this).name(); break; case DeclarationKind::DestructorDeclaration: out << cast(*this).name(); break; case DeclarationKind::ClassDeclaration: { const auto& class_decl = cast(*this); out << "class " << class_decl.name(); break; } case DeclarationKind::MixinDeclaration: { const auto& mixin_decl = cast(*this); out << "__mixin " << mixin_decl.name(); if (mixin_decl.self()->type().kind() != ExpressionKind::TypeTypeLiteral) { out << " for " << mixin_decl.self()->type(); } break; } case DeclarationKind::MixDeclaration: { out << "__mix "; break; } case DeclarationKind::ChoiceDeclaration: { const auto& choice = cast(*this); out << "choice " << choice.name(); break; } case DeclarationKind::VariableDeclaration: { const auto& var = cast(*this); out << "var " << var.binding(); break; } case DeclarationKind::AssociatedConstantDeclaration: { const auto& let = cast(*this); out << "let " << let.binding(); break; } case DeclarationKind::SelfDeclaration: { out << "Self"; break; } case DeclarationKind::AliasDeclaration: { const auto& alias = cast(*this); out << "alias " << alias.name(); break; } } } auto GetName(const Declaration& declaration) -> std::optional { switch (declaration.kind()) { case DeclarationKind::FunctionDeclaration: return cast(declaration).name(); case DeclarationKind::DestructorDeclaration: return cast(declaration).name(); case DeclarationKind::ClassDeclaration: return cast(declaration).name(); case DeclarationKind::MixinDeclaration: { return cast(declaration).name(); } case DeclarationKind::MixDeclaration: { return std::nullopt; } case DeclarationKind::ChoiceDeclaration: return cast(declaration).name(); case DeclarationKind::InterfaceDeclaration: return cast(declaration).name(); case DeclarationKind::VariableDeclaration: return cast(declaration).binding().name(); case DeclarationKind::AssociatedConstantDeclaration: return cast(declaration).binding().name(); case DeclarationKind::ImplDeclaration: return std::nullopt; case DeclarationKind::SelfDeclaration: return cast(declaration).name(); case DeclarationKind::AliasDeclaration: { return cast(declaration).name(); } } } void GenericBinding::Print(llvm::raw_ostream& out) const { out << name() << ":! " << type(); } void GenericBinding::PrintID(llvm::raw_ostream& out) const { out << name(); } void ReturnTerm::Print(llvm::raw_ostream& out) const { switch (kind_) { case ReturnKind::Omitted: return; case ReturnKind::Auto: out << "-> auto"; return; case ReturnKind::Expression: CARBON_CHECK(type_expression_.has_value()); out << "-> " << **type_expression_; return; } } namespace { // The deduced parameters of a function declaration. struct DeducedParameters { // The `me` parameter, if any. std::optional> me_pattern; // All other deduced parameters. std::vector> resolved_params; }; // Split the `me` pattern (if any) out of `deduced_params`. auto SplitDeducedParameters( SourceLocation source_loc, const std::vector>& deduced_params) -> ErrorOr { DeducedParameters result; for (Nonnull param : deduced_params) { switch (param->kind()) { case AstNodeKind::GenericBinding: result.resolved_params.push_back(&cast(*param)); break; case AstNodeKind::BindingPattern: { Nonnull binding = &cast(*param); if (binding->name() != "me") { return ProgramError(source_loc) << "illegal binding pattern in implicit parameter list"; } if (result.me_pattern.has_value()) { return ProgramError(source_loc) << "parameter list cannot contain more than one `me` " "parameter"; } result.me_pattern = binding; break; } case AstNodeKind::AddrPattern: { Nonnull addr_pattern = &cast(*param); Nonnull binding = &cast(addr_pattern->binding()); if (binding->name() != "me") { return ProgramError(source_loc) << "illegal binding pattern in implicit parameter list"; } if (result.me_pattern.has_value()) { return ProgramError(source_loc) << "parameter list cannot contain more than one `me` " "parameter"; } result.me_pattern = addr_pattern; break; } default: return ProgramError(source_loc) << "illegal AST node in implicit parameter list"; } } return result; } } // namespace auto DestructorDeclaration::CreateDestructor( Nonnull arena, SourceLocation source_loc, std::vector> deduced_params, Nonnull param_pattern, ReturnTerm return_term, std::optional> body) -> ErrorOr> { DeducedParameters split_params; CARBON_ASSIGN_OR_RETURN(split_params, SplitDeducedParameters(source_loc, deduced_params)); return arena->New( source_loc, std::move(split_params.resolved_params), split_params.me_pattern, param_pattern, return_term, body); } auto FunctionDeclaration::Create(Nonnull arena, SourceLocation source_loc, std::string name, std::vector> deduced_params, Nonnull param_pattern, ReturnTerm return_term, std::optional> body) -> ErrorOr> { DeducedParameters split_params; CARBON_ASSIGN_OR_RETURN(split_params, SplitDeducedParameters(source_loc, deduced_params)); return arena->New( source_loc, name, std::move(split_params.resolved_params), split_params.me_pattern, param_pattern, return_term, body); } void CallableDeclaration::PrintDepth(int depth, llvm::raw_ostream& out) const { out << "fn " << name_ << " "; if (!deduced_parameters_.empty()) { out << "["; llvm::ListSeparator sep; for (Nonnull deduced : deduced_parameters_) { out << sep << *deduced; } out << "]"; } out << *param_pattern_ << return_term_; if (body_) { out << " {\n"; (*body_)->PrintDepth(depth, out); out << "\n}\n"; } else { out << ";\n"; } } auto ImplDeclaration::Create(Nonnull arena, SourceLocation source_loc, ImplKind kind, Nonnull impl_type, Nonnull interface, std::vector> deduced_params, std::vector> members) -> ErrorOr> { std::vector> resolved_params; for (Nonnull param : deduced_params) { switch (param->kind()) { case AstNodeKind::GenericBinding: resolved_params.push_back(&cast(*param)); break; default: return ProgramError(source_loc) << "illegal AST node in implicit parameter list of impl"; } } Nonnull self_decl = arena->New(impl_type->source_loc()); return arena->New(source_loc, kind, impl_type, self_decl, interface, resolved_params, members); } void AlternativeSignature::Print(llvm::raw_ostream& out) const { out << "alt " << name() << " " << signature(); } void AlternativeSignature::PrintID(llvm::raw_ostream& out) const { out << name(); } } // namespace Carbon