// 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/generic.h" #include "common/map.h" #include "toolchain/base/kind_switch.h" #include "toolchain/check/eval.h" #include "toolchain/check/generic_region_stack.h" #include "toolchain/check/subst.h" #include "toolchain/sem_ir/generic.h" #include "toolchain/sem_ir/ids.h" #include "toolchain/sem_ir/inst.h" #include "toolchain/sem_ir/typed_insts.h" namespace Carbon::Check { static auto MakeSelfSpecificId(Context& context, SemIR::GenericId generic_id) -> SemIR::SpecificId; static auto ResolveSpecificDeclaration(Context& context, SemIRLoc loc, SemIR::SpecificId specific_id) -> void; auto StartGenericDecl(Context& context) -> void { context.generic_region_stack().Push(); } auto StartGenericDefinition(Context& context) -> void { // Push a generic region even if we don't have a generic_id. We might still // have locally-introduced generic parameters to track: // // fn F() { // let T:! type = i32; // var x: T; // } context.generic_region_stack().Push(); } // Adds an instruction `generic_inst_id` to the eval block for a generic region, // which is the current instruction block. The instruction `generic_inst_id` is // expected to compute the value of the constant described by `const_inst_id` in // each specific. Forms and returns a corresponding symbolic constant ID that // refers to the substituted value of that instruction in each specific. static auto AddGenericConstantInstToEvalBlock( Context& context, SemIR::GenericId generic_id, SemIR::GenericInstIndex::Region region, SemIR::InstId const_inst_id, SemIR::InstId generic_inst_id) -> SemIR::ConstantId { auto index = SemIR::GenericInstIndex( region, context.inst_block_stack().PeekCurrentBlockContents().size()); context.inst_block_stack().AddInstId(generic_inst_id); return context.constant_values().AddSymbolicConstant( {.inst_id = const_inst_id, .generic_id = generic_id, .index = index}); } namespace { // A map from an instruction ID representing a canonical symbolic constant to an // instruction within an eval block of the generic that computes the specific // value for that constant. // // We arbitrarily use a small size of 256 bytes for the map. // TODO: Determine a better number based on measurements. using ConstantsInGenericMap = Map; // Substitution callbacks to rebuild a generic constant in the eval block for a // generic region. class RebuildGenericConstantInEvalBlockCallbacks final : public SubstInstCallbacks { public: RebuildGenericConstantInEvalBlockCallbacks( Context& context, SemIR::GenericId generic_id, SemIR::GenericInstIndex::Region region, SemIR::LocId loc_id, ConstantsInGenericMap& constants_in_generic) : context_(context), generic_id_(generic_id), region_(region), loc_id_(loc_id), constants_in_generic_(constants_in_generic) {} // Check for instructions for which we already have a mapping into the eval // block, and substitute them for the instructions in the eval block. auto Subst(SemIR::InstId& inst_id) const -> bool override { auto const_id = context_.constant_values().Get(inst_id); if (!const_id.has_value()) { // An unloaded import ref should never contain anything we need to // substitute into. Don't trigger loading it here. CARBON_CHECK( context_.insts().Is(inst_id), "Substituting into instruction with invalid constant ID: {0}", context_.insts().Get(inst_id)); return true; } if (!context_.constant_values().DependsOnGenericParameter(const_id)) { // This instruction doesn't have a symbolic constant value, so can't // contain any bindings that need to be substituted. return true; } // If this instruction is in the map, return the known result. if (auto result = constants_in_generic_.Lookup( context_.constant_values().GetInstId(const_id))) { // In order to reuse instructions from the generic as often as possible, // keep this instruction as-is if it already has the desired symbolic // constant value. if (const_id != context_.constant_values().Get(result.value())) { inst_id = result.value(); } CARBON_CHECK(inst_id.has_value()); return true; } // If the instruction is a symbolic binding, build a version in the eval // block. if (auto binding = context_.insts().TryGetAs(inst_id)) { if (context_.entity_names() .Get(binding->entity_name_id) .bind_index.has_value()) { inst_id = Rebuild(inst_id, *binding); return true; } } if (auto pattern = context_.insts().TryGetAs(inst_id)) { inst_id = Rebuild(inst_id, *pattern); return true; } return false; } // Build a new instruction in the eval block corresponding to the given // constant. auto Rebuild(SemIR::InstId orig_inst_id, SemIR::Inst new_inst) const -> SemIR::InstId override { auto const_inst_id = context_.constant_values().GetConstantInstId(orig_inst_id); // We might already have an instruction in the eval block if a transitive // operand of this instruction has the same constant value. auto result = constants_in_generic_.Insert(const_inst_id, [&] { // TODO: Add a function on `Context` to add the instruction without // inserting it into the dependent instructions list or computing a // constant value for it. // TODO: Is the location we pick here always appropriate for the new // instruction? auto inst_id = context_.sem_ir().insts().AddInNoBlock( SemIR::LocIdAndInst::UncheckedLoc(loc_id_, new_inst)); auto const_id = AddGenericConstantInstToEvalBlock( context_, generic_id_, region_, const_inst_id, inst_id); context_.constant_values().Set(inst_id, const_id); return inst_id; }); return result.value(); } private: Context& context_; SemIR::GenericId generic_id_; SemIR::GenericInstIndex::Region region_; SemIR::LocId loc_id_; ConstantsInGenericMap& constants_in_generic_; }; } // namespace // Adds instructions to compute the substituted version of `type_id` in each // specific into the eval block for the generic, which is the current // instruction block. Returns a symbolic type ID that refers to the substituted // type in each specific. static auto AddGenericTypeToEvalBlock( Context& context, SemIR::GenericId generic_id, SemIR::GenericInstIndex::Region region, SemIR::LocId loc_id, ConstantsInGenericMap& constants_in_generic, SemIR::TypeId type_id) -> SemIR::TypeId { // Substitute into the type's constant instruction and rebuild it in the eval // block. auto type_inst_id = SubstInst(context, context.types().GetInstId(type_id), RebuildGenericConstantInEvalBlockCallbacks( context, generic_id, region, loc_id, constants_in_generic)); return context.GetTypeIdForTypeInst(type_inst_id); } // Adds instructions to compute the substituted value of `inst_id` in each // specific into the eval block for the generic, which is the current // instruction block. Returns a symbolic constant instruction ID that refers to // the substituted constant value in each specific. static auto AddGenericConstantToEvalBlock( Context& context, SemIR::GenericId generic_id, SemIR::GenericInstIndex::Region region, ConstantsInGenericMap& constants_in_generic, SemIR::InstId inst_id) -> SemIR::ConstantId { // Substitute into the constant value and rebuild it in the eval block if // we've not encountered it before. auto const_inst_id = context.constant_values().GetConstantInstId(inst_id); auto new_inst_id = SubstInst(context, const_inst_id, RebuildGenericConstantInEvalBlockCallbacks( context, generic_id, region, context.insts().GetLocId(inst_id), constants_in_generic)); CARBON_CHECK(new_inst_id != const_inst_id, "Did not apply any substitutions to symbolic constant {0}", context.insts().Get(const_inst_id)); return context.constant_values().Get(new_inst_id); } // Populates a map of constants in a generic from the constants in the // declaration region, in preparation for building the definition region. static auto PopulateConstantsFromDeclaration( Context& context, SemIR::GenericId generic_id, ConstantsInGenericMap& constants_in_generic) { // For the definition region, populate constants from the declaration. auto decl_eval_block = context.inst_blocks().Get( context.generics().Get(generic_id).decl_block_id); constants_in_generic.GrowForInsertCount(decl_eval_block.size()); for (auto inst_id : decl_eval_block) { auto const_inst_id = context.constant_values().GetConstantInstId(inst_id); auto result = constants_in_generic.Insert(const_inst_id, inst_id); CARBON_CHECK(result.is_inserted(), "Duplicate constant in generic decl eval block: {0}", context.insts().Get(const_inst_id)); } } // Builds and returns a block of instructions whose constant values need to be // evaluated in order to resolve a generic to a specific. static auto MakeGenericEvalBlock(Context& context, SemIR::GenericId generic_id, SemIR::GenericInstIndex::Region region) -> SemIR::InstBlockId { context.inst_block_stack().Push(); ConstantsInGenericMap constants_in_generic; // For the definition region, populate constants from the declaration. if (region == SemIR::GenericInstIndex::Region::Definition) { PopulateConstantsFromDeclaration(context, generic_id, constants_in_generic); } // The work done in this loop might invalidate iterators into the generic // region stack, but shouldn't add new dependent instructions to the current // region. auto num_dependent_insts = context.generic_region_stack().PeekDependentInsts().size(); for (auto i : llvm::seq(num_dependent_insts)) { auto [inst_id, dep_kind] = context.generic_region_stack().PeekDependentInsts()[i]; // If the type is symbolic, replace it with a type specific to this generic. if ((dep_kind & GenericRegionStack::DependencyKind::SymbolicType) != GenericRegionStack::DependencyKind::None) { auto inst = context.insts().Get(inst_id); auto type_id = AddGenericTypeToEvalBlock( context, generic_id, region, context.insts().GetLocId(inst_id), constants_in_generic, inst.type_id()); // TODO: Eventually, completeness requirements should be modeled as // constraints on the generic rather than properties of the type. For now, // require the transformed type to be complete if the original was. if (context.types().IsComplete(inst.type_id())) { context.CompleteTypeOrCheckFail(type_id); } inst.SetType(type_id); context.sem_ir().insts().Set(inst_id, inst); } // If the instruction has a symbolic constant value, then make a note that // we'll need to evaluate this instruction when forming the specific. Update // the constant value of the instruction to refer to the result of that // eventual evaluation. if ((dep_kind & GenericRegionStack::DependencyKind::SymbolicConstant) != GenericRegionStack::DependencyKind::None) { // Update the constant value to refer to this generic. context.constant_values().Set( inst_id, AddGenericConstantToEvalBlock(context, generic_id, region, constants_in_generic, inst_id)); } } CARBON_CHECK( num_dependent_insts == context.generic_region_stack().PeekDependentInsts().size(), "Building eval block added new dependent insts, for example {0}", context.insts().Get(context.generic_region_stack() .PeekDependentInsts()[num_dependent_insts] .inst_id)); return context.inst_block_stack().Pop(); } // Builds and returns an eval block, given the list of canonical symbolic // constants that the instructions in the eval block should produce. This is // used when importing a generic. auto RebuildGenericEvalBlock(Context& context, SemIR::GenericId generic_id, SemIR::GenericInstIndex::Region region, llvm::ArrayRef const_ids) -> SemIR::InstBlockId { context.inst_block_stack().Push(); ConstantsInGenericMap constants_in_generic; // For the definition region, populate constants from the declaration. if (region == SemIR::GenericInstIndex::Region::Definition) { PopulateConstantsFromDeclaration(context, generic_id, constants_in_generic); } constants_in_generic.GrowForInsertCount(const_ids.size()); for (auto [i, inst_id] : llvm::enumerate(const_ids)) { // Build a constant in the inst block. AddGenericConstantToEvalBlock(context, generic_id, region, constants_in_generic, inst_id); CARBON_CHECK( context.inst_block_stack().PeekCurrentBlockContents().size() == i + 1, "Produced {0} instructions when importing {1}", (context.inst_block_stack().PeekCurrentBlockContents().size() - i), context.insts().Get(inst_id)); } return context.inst_block_stack().Pop(); } auto DiscardGenericDecl(Context& context) -> void { context.generic_region_stack().Pop(); } auto BuildGeneric(Context& context, SemIR::InstId decl_id) -> SemIR::GenericId { auto all_bindings = context.scope_stack().compile_time_bindings_stack().PeekAllValues(); if (all_bindings.empty()) { CARBON_CHECK(context.generic_region_stack().PeekDependentInsts().empty(), "Have dependent instruction {0} in declaration {1} but no " "compile time bindings are in scope.", context.insts().Get(context.generic_region_stack() .PeekDependentInsts() .front() .inst_id), context.insts().Get(decl_id)); context.generic_region_stack().Pop(); return SemIR::GenericId::None; } // Build the new Generic object. Note that we intentionally do not hold a // persistent reference to it throughout this function, because the `generics` // collection can have items added to it by import resolution while we are // building this generic. auto bindings_id = context.inst_blocks().Add(all_bindings); SemIR::GenericId generic_id = context.generics().Add( SemIR::Generic{.decl_id = decl_id, .bindings_id = bindings_id, .self_specific_id = SemIR::SpecificId::None}); // MakeSelfSpecificId could cause something to be imported, which would // invalidate the return value of `context.generics().Get(generic_id)`. auto self_specific_id = MakeSelfSpecificId(context, generic_id); context.generics().Get(generic_id).self_specific_id = self_specific_id; return generic_id; } auto FinishGenericDecl(Context& context, SemIRLoc loc, SemIR::GenericId generic_id) -> void { if (!generic_id.has_value()) { return; } auto decl_block_id = MakeGenericEvalBlock( context, generic_id, SemIR::GenericInstIndex::Region::Declaration); context.generic_region_stack().Pop(); context.generics().Get(generic_id).decl_block_id = decl_block_id; ResolveSpecificDeclaration(context, loc, context.generics().GetSelfSpecific(generic_id)); } auto BuildGenericDecl(Context& context, SemIR::InstId decl_id) -> SemIR::GenericId { SemIR::GenericId generic_id = BuildGeneric(context, decl_id); if (generic_id.has_value()) { FinishGenericDecl(context, decl_id, generic_id); } return generic_id; } auto FinishGenericRedecl(Context& context, SemIR::InstId /*decl_id*/, SemIR::GenericId /*generic_id*/) -> void { // TODO: Compare contents of this declaration with the existing one on the // generic. context.generic_region_stack().Pop(); } auto FinishGenericDefinition(Context& context, SemIR::GenericId generic_id) -> void { if (!generic_id.has_value()) { // TODO: We can have symbolic constants in a context that had a non-generic // declaration, for example if there's a local generic let binding in a // function definition. Handle this case somehow -- perhaps by forming // substituted constant values now. context.generic_region_stack().Pop(); return; } auto definition_block_id = MakeGenericEvalBlock( context, generic_id, SemIR::GenericInstIndex::Region::Definition); context.generics().Get(generic_id).definition_block_id = definition_block_id; context.generic_region_stack().Pop(); } static auto ResolveSpecificDeclaration(Context& context, SemIRLoc loc, SemIR::SpecificId specific_id) -> void { // If this is the first time we've formed this specific, evaluate its decl // block to form information about the specific. if (!context.specifics().Get(specific_id).decl_block_id.has_value()) { // Set a placeholder value as the decl block ID so we won't attempt to // recursively resolve the same specific. context.specifics().Get(specific_id).decl_block_id = SemIR::InstBlockId::Empty; auto decl_block_id = TryEvalBlockForSpecific(context, loc, specific_id, SemIR::GenericInstIndex::Region::Declaration); // Note that TryEvalBlockForSpecific may reallocate the list of specifics, // so re-lookup the specific here. context.specifics().Get(specific_id).decl_block_id = decl_block_id; } } auto MakeSpecific(Context& context, SemIRLoc loc, SemIR::GenericId generic_id, SemIR::InstBlockId args_id) -> SemIR::SpecificId { auto specific_id = context.specifics().GetOrAdd(generic_id, args_id); ResolveSpecificDeclaration(context, loc, specific_id); return specific_id; } auto MakeSpecific(Context& context, SemIRLoc loc, SemIR::GenericId generic_id, llvm::ArrayRef args) -> SemIR::SpecificId { auto args_id = context.inst_blocks().AddCanonical(args); return MakeSpecific(context, loc, generic_id, args_id); } static auto MakeSelfSpecificId(Context& context, SemIR::GenericId generic_id) -> SemIR::SpecificId { if (!generic_id.has_value()) { return SemIR::SpecificId::None; } auto& generic = context.generics().Get(generic_id); auto args = context.inst_blocks().Get(generic.bindings_id); // Form a canonical argument list for the generic. llvm::SmallVector arg_ids; arg_ids.reserve(args.size()); for (auto arg_id : args) { arg_ids.push_back(context.constant_values().GetConstantInstId(arg_id)); } auto args_id = context.inst_blocks().AddCanonical(arg_ids); return context.specifics().GetOrAdd(generic_id, args_id); } auto MakeSelfSpecific(Context& context, SemIRLoc loc, SemIR::GenericId generic_id) -> SemIR::SpecificId { // Build a corresponding specific. SemIR::SpecificId specific_id = MakeSelfSpecificId(context, generic_id); // TODO: This could be made more efficient. We don't need to perform // substitution here; we know we want identity mappings for all constants and // types. We could also consider not storing the mapping at all in this case. ResolveSpecificDeclaration(context, loc, specific_id); return specific_id; } auto ResolveSpecificDefinition(Context& context, SemIRLoc loc, SemIR::SpecificId specific_id) -> bool { // TODO: Handle recursive resolution of the same generic definition. auto& specific = context.specifics().Get(specific_id); auto generic_id = specific.generic_id; CARBON_CHECK(generic_id.has_value(), "Specific with no generic ID"); if (!specific.definition_block_id.has_value()) { // Evaluate the eval block for the definition of the generic. auto& generic = context.generics().Get(generic_id); if (!generic.definition_block_id.has_value()) { // The generic is not defined yet. return false; } auto definition_block_id = TryEvalBlockForSpecific( context, loc, specific_id, SemIR::GenericInstIndex::Region::Definition); // Note that TryEvalBlockForSpecific may reallocate the list of specifics, // so re-lookup the specific here. context.specifics().Get(specific_id).definition_block_id = definition_block_id; } return true; } auto GetInstForSpecific(Context& context, SemIR::SpecificId specific_id) -> SemIR::InstId { CARBON_CHECK(specific_id.has_value()); const auto& specific = context.specifics().Get(specific_id); const auto& generic = context.generics().Get(specific.generic_id); auto decl = context.insts().Get(generic.decl_id); CARBON_KIND_SWITCH(decl) { case CARBON_KIND(SemIR::ClassDecl class_decl): { return context.types().GetInstId( context.GetClassType(class_decl.class_id, specific_id)); } case CARBON_KIND(SemIR::InterfaceDecl interface_decl): { return context.types().GetInstId( context.GetInterfaceType(interface_decl.interface_id, specific_id)); } case SemIR::FunctionDecl::Kind: { return context.constant_values().GetInstId( TryEvalInst(context, SemIR::InstId::None, SemIR::SpecificFunction{ .type_id = context.GetSingletonType( SemIR::SpecificFunctionType::SingletonInstId), .callee_id = generic.decl_id, .specific_id = specific_id})); } case SemIR::AssociatedConstantDecl::Kind: { // TODO: We don't have a good instruction to use here. return generic.decl_id; } default: { CARBON_FATAL("Unknown kind for generic declaration {0}", decl); } } } } // namespace Carbon::Check