// 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 "toolchain/check/context.h" #include "toolchain/check/convert.h" #include "toolchain/check/modifiers.h" #include "toolchain/sem_ir/typed_insts.h" namespace Carbon::Check { auto HandleClassIntroducer(Context& context, Parse::ClassIntroducerId node_id) -> bool { // Create an instruction block to hold the instructions created as part of the // class signature, such as generic parameters. context.inst_block_stack().Push(); // Push the bracketing node. context.node_stack().Push(node_id); // Optional modifiers and the name follow. context.decl_state_stack().Push(DeclState::Class); context.decl_name_stack().PushScopeAndStartName(); return true; } static auto BuildClassDecl(Context& context, Parse::AnyClassDeclId node_id) -> std::tuple { if (context.node_stack().PopIf()) { context.TODO(node_id, "generic class"); } if (context.node_stack().PopIf()) { context.TODO(node_id, "generic class"); } auto name_context = context.decl_name_stack().FinishName(); context.node_stack() .PopAndDiscardSoloNodeId(); // Process modifiers. CheckAccessModifiersOnDecl(context, Lex::TokenKind::Class, name_context.target_scope_id); LimitModifiersOnDecl(context, KeywordModifierSet::Class | KeywordModifierSet::Access, Lex::TokenKind::Class); auto modifiers = context.decl_state_stack().innermost().modifier_set; if (!!(modifiers & KeywordModifierSet::Access)) { context.TODO(context.decl_state_stack().innermost().modifier_node_id( ModifierOrder::Access), "access modifier"); } auto inheritance_kind = !!(modifiers & KeywordModifierSet::Abstract) ? SemIR::Class::Abstract : !!(modifiers & KeywordModifierSet::Base) ? SemIR::Class::Base : SemIR::Class::Final; context.decl_state_stack().Pop(DeclState::Class); auto decl_block_id = context.inst_block_stack().Pop(); // Add the class declaration. auto class_decl = SemIR::ClassDecl{SemIR::TypeId::TypeType, SemIR::ClassId::Invalid, decl_block_id}; auto class_decl_id = context.AddPlaceholderInst({node_id, class_decl}); // Check whether this is a redeclaration. auto existing_id = context.decl_name_stack().LookupOrAddName(name_context, class_decl_id); if (existing_id.is_valid()) { if (auto existing_class_decl = context.insts().Get(existing_id).TryAs()) { // This is a redeclaration of an existing class. class_decl.class_id = existing_class_decl->class_id; auto& class_info = context.classes().Get(class_decl.class_id); // The introducer kind must match the previous declaration. // TODO: The rule here is not yet decided. See #3384. if (class_info.inheritance_kind != inheritance_kind) { CARBON_DIAGNOSTIC(ClassRedeclarationDifferentIntroducer, Error, "Class redeclared with different inheritance kind."); CARBON_DIAGNOSTIC(ClassRedeclarationDifferentIntroducerPrevious, Note, "Previously declared here."); context.emitter() .Build(node_id, ClassRedeclarationDifferentIntroducer) .Note(existing_id, ClassRedeclarationDifferentIntroducerPrevious) .Emit(); } // TODO: Check that the generic parameter list agrees with the prior // declaration. } else { // This is a redeclaration of something other than a class. context.DiagnoseDuplicateName(class_decl_id, existing_id); } } // Create a new class if this isn't a valid redeclaration. bool is_new_class = !class_decl.class_id.is_valid(); if (is_new_class) { // TODO: If this is an invalid redeclaration of a non-class entity or there // was an error in the qualifier, we will have lost track of the class name // here. We should keep track of it even if the name is invalid. class_decl.class_id = context.classes().Add( {.name_id = name_context.name_id_for_new_inst(), .enclosing_scope_id = name_context.enclosing_scope_id_for_new_inst(), // `.self_type_id` depends on the ClassType, so is set below. .self_type_id = SemIR::TypeId::Invalid, .decl_id = class_decl_id, .inheritance_kind = inheritance_kind}); } // Write the class ID into the ClassDecl. context.ReplaceInstBeforeConstantUse(class_decl_id, {node_id, class_decl}); if (is_new_class) { // Build the `Self` type using the resulting type constant. auto& class_info = context.classes().Get(class_decl.class_id); class_info.self_type_id = context.GetTypeIdForTypeInst(class_decl_id); } return {class_decl.class_id, class_decl_id}; } auto HandleClassDecl(Context& context, Parse::ClassDeclId node_id) -> bool { BuildClassDecl(context, node_id); context.decl_name_stack().PopScope(); return true; } auto HandleClassDefinitionStart(Context& context, Parse::ClassDefinitionStartId node_id) -> bool { auto [class_id, class_decl_id] = BuildClassDecl(context, node_id); auto& class_info = context.classes().Get(class_id); // Track that this declaration is the definition. if (class_info.is_defined()) { CARBON_DIAGNOSTIC(ClassRedefinition, Error, "Redefinition of class {0}.", SemIR::NameId); CARBON_DIAGNOSTIC(ClassPreviousDefinition, Note, "Previous definition was here."); context.emitter() .Build(node_id, ClassRedefinition, class_info.name_id) .Note(class_info.definition_id, ClassPreviousDefinition) .Emit(); } else { class_info.definition_id = class_decl_id; class_info.scope_id = context.name_scopes().Add( class_decl_id, SemIR::NameId::Invalid, class_info.enclosing_scope_id); } // Enter the class scope. context.scope_stack().Push(class_decl_id, class_info.scope_id); // Introduce `Self`. context.name_scopes() .Get(class_info.scope_id) .names.insert({SemIR::NameId::SelfType, context.types().GetInstId(class_info.self_type_id)}); context.inst_block_stack().Push(); context.node_stack().Push(node_id, class_id); context.args_type_info_stack().Push(); // TODO: Handle the case where there's control flow in the class body. For // example: // // class C { // var v: if true then i32 else f64; // } // // We may need to track a list of instruction blocks here, as we do for a // function. class_info.body_block_id = context.inst_block_stack().PeekOrAdd(); return true; } auto HandleBaseIntroducer(Context& context, Parse::BaseIntroducerId /*node_id*/) -> bool { context.decl_state_stack().Push(DeclState::Base); return true; } auto HandleBaseColon(Context& /*context*/, Parse::BaseColonId /*node_id*/) -> bool { return true; } namespace { // Information gathered about a base type specified in a `base` declaration. struct BaseInfo { // A `BaseInfo` representing an erroneous base. static const BaseInfo Error; SemIR::TypeId type_id; SemIR::NameScopeId scope_id; }; constexpr BaseInfo BaseInfo::Error = {.type_id = SemIR::TypeId::Error, .scope_id = SemIR::NameScopeId::Invalid}; } // namespace // If `type_id` is a class type, get its corresponding `SemIR::Class` object. // Otherwise returns `nullptr`. static auto TryGetAsClass(Context& context, SemIR::TypeId type_id) -> SemIR::Class* { auto class_type = context.types().TryGetAs(type_id); if (!class_type) { return nullptr; } return &context.classes().Get(class_type->class_id); } // Diagnoses an attempt to derive from a final type. static auto DiagnoseBaseIsFinal(Context& context, Parse::NodeId node_id, SemIR::TypeId base_type_id) -> void { CARBON_DIAGNOSTIC(BaseIsFinal, Error, "Deriving from final type `{0}`. Base type must be an " "`abstract` or `base` class.", SemIR::TypeId); context.emitter().Emit(node_id, BaseIsFinal, base_type_id); } // Checks that the specified base type is valid. static auto CheckBaseType(Context& context, Parse::NodeId node_id, SemIR::InstId base_expr_id) -> BaseInfo { auto base_type_id = ExprAsType(context, node_id, base_expr_id); base_type_id = context.AsCompleteType(base_type_id, [&] { CARBON_DIAGNOSTIC(IncompleteTypeInBaseDecl, Error, "Base `{0}` is an incomplete type.", SemIR::TypeId); return context.emitter().Build(node_id, IncompleteTypeInBaseDecl, base_type_id); }); if (base_type_id == SemIR::TypeId::Error) { return BaseInfo::Error; } auto* base_class_info = TryGetAsClass(context, base_type_id); // The base must not be a final class. if (!base_class_info) { // For now, we treat all types that aren't introduced by a `class` // declaration as being final classes. // TODO: Once we have a better idea of which types are considered to be // classes, produce a better diagnostic for deriving from a non-class type. DiagnoseBaseIsFinal(context, node_id, base_type_id); return BaseInfo::Error; } if (base_class_info->inheritance_kind == SemIR::Class::Final) { DiagnoseBaseIsFinal(context, node_id, base_type_id); } CARBON_CHECK(base_class_info->scope_id.is_valid()) << "Complete class should have a scope"; return {.type_id = base_type_id, .scope_id = base_class_info->scope_id}; } auto HandleBaseDecl(Context& context, Parse::BaseDeclId node_id) -> bool { auto [base_type_node_id, base_type_expr_id] = context.node_stack().PopExprWithNodeId(); // Process modifiers. `extend` is required, none others are allowed. LimitModifiersOnDecl(context, KeywordModifierSet::Extend, Lex::TokenKind::Base); auto modifiers = context.decl_state_stack().innermost().modifier_set; if (!(modifiers & KeywordModifierSet::Extend)) { CARBON_DIAGNOSTIC(BaseMissingExtend, Error, "Missing `extend` before `base` declaration in class."); context.emitter().Emit(node_id, BaseMissingExtend); } context.decl_state_stack().Pop(DeclState::Base); auto enclosing_class_decl = context.GetCurrentScopeAs(); if (!enclosing_class_decl) { CARBON_DIAGNOSTIC(BaseOutsideClass, Error, "`base` declaration can only be used in a class."); context.emitter().Emit(node_id, BaseOutsideClass); return true; } auto& class_info = context.classes().Get(enclosing_class_decl->class_id); if (class_info.base_id.is_valid()) { CARBON_DIAGNOSTIC(BaseRepeated, Error, "Multiple `base` declarations in class. Multiple " "inheritance is not permitted."); CARBON_DIAGNOSTIC(BasePrevious, Note, "Previous `base` declaration is here."); context.emitter() .Build(node_id, BaseRepeated) .Note(class_info.base_id, BasePrevious) .Emit(); return true; } auto base_info = CheckBaseType(context, base_type_node_id, base_type_expr_id); // The `base` value in the class scope has an unbound element type. Instance // binding will be performed when it's found by name lookup into an instance. auto field_type_id = context.GetUnboundElementType(class_info.self_type_id, base_info.type_id); class_info.base_id = context.AddInst( {node_id, SemIR::BaseDecl{field_type_id, base_info.type_id, SemIR::ElementIndex(context.args_type_info_stack() .PeekCurrentBlockContents() .size())}}); // Add a corresponding field to the object representation of the class. // TODO: Consider whether we want to use `partial T` here. context.args_type_info_stack().AddInstId(context.AddInstInNoBlock( {node_id, SemIR::StructTypeField{SemIR::NameId::Base, base_info.type_id}})); // Bind the name `base` in the class to the base field. context.decl_name_stack().AddNameToLookup( context.decl_name_stack().MakeUnqualifiedName(node_id, SemIR::NameId::Base), class_info.base_id); // Extend the class scope with the base class. if (!!(modifiers & KeywordModifierSet::Extend)) { auto& class_scope = context.name_scopes().Get(class_info.scope_id); if (base_info.scope_id.is_valid()) { class_scope.extended_scopes.push_back(base_info.scope_id); } else { class_scope.has_error = true; } } return true; } auto HandleClassDefinition(Context& context, Parse::ClassDefinitionId /*node_id*/) -> bool { auto fields_id = context.args_type_info_stack().Pop(); auto class_id = context.node_stack().Pop(); context.inst_block_stack().Pop(); context.scope_stack().Pop(); context.decl_name_stack().PopScope(); // The class type is now fully defined. auto& class_info = context.classes().Get(class_id); class_info.object_repr_id = context.GetStructType(fields_id); return true; } } // namespace Carbon::Check