| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230 |
- // 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/cpp/impl_lookup.h"
- #include "clang/Sema/Sema.h"
- #include "toolchain/base/kind_switch.h"
- #include "toolchain/check/cpp/import.h"
- #include "toolchain/check/cpp/location.h"
- #include "toolchain/check/cpp/overload_resolution.h"
- #include "toolchain/check/impl.h"
- #include "toolchain/check/impl_lookup.h"
- #include "toolchain/check/import_ref.h"
- #include "toolchain/check/inst.h"
- #include "toolchain/check/type.h"
- #include "toolchain/sem_ir/ids.h"
- #include "toolchain/sem_ir/typed_insts.h"
- namespace Carbon::Check {
- // If the given type is a C++ class type, returns the corresponding class
- // declaration. Otherwise returns nullptr.
- // TODO: Handle qualified types.
- static auto TypeAsClassDecl(Context& context, SemIR::TypeId type_id)
- -> clang::CXXRecordDecl* {
- auto class_type = context.types().TryGetAs<SemIR::ClassType>(type_id);
- if (!class_type) {
- // Not a class.
- return nullptr;
- }
- SemIR::NameScopeId class_scope_id =
- context.classes().Get(class_type->class_id).scope_id;
- if (!class_scope_id.has_value()) {
- return nullptr;
- }
- const auto& scope = context.name_scopes().Get(class_scope_id);
- auto decl_id = scope.clang_decl_context_id();
- if (!decl_id.has_value()) {
- return nullptr;
- }
- return dyn_cast<clang::CXXRecordDecl>(
- context.clang_decls().Get(decl_id).key.decl);
- }
- // Builds a witness that the given type implements the given interface,
- // populating it with the specified set of values. Returns a corresponding
- // lookup result. Produces a diagnostic and returns `None` if the specified
- // values aren't suitable for the interface.
- static auto BuildWitness(Context& context, SemIR::LocId loc_id,
- SemIR::TypeId self_type_id,
- SemIR::SpecificInterface specific_interface,
- llvm::ArrayRef<SemIR::InstId> values)
- -> SemIR::InstId {
- const auto& interface =
- context.interfaces().Get(specific_interface.interface_id);
- auto assoc_entities =
- context.inst_blocks().GetOrEmpty(interface.associated_entities_id);
- if (assoc_entities.size() != values.size()) {
- context.TODO(loc_id, ("Unsupported definition of interface " +
- context.names().GetFormatted(interface.name_id))
- .str());
- return SemIR::ErrorInst::InstId;
- }
- llvm::SmallVector<SemIR::InstId> entries;
- // Build a witness with the current contents of the witness table. This will
- // grow as we progress through the impl. In theory this will build O(n^2)
- // table entries, but in practice n <= 2, so that's OK.
- //
- // This is necessary because later associated entities may refer to earlier
- // associated entities in their signatures. In particular, an associated
- // result type may be used as the return type of an associated function.
- //
- // TODO: Consider building one witness after all associated constants, and
- // then a second after all associated functions, rather than building one at
- // each step. For now this doesn't really matter since we don't have more than
- // one of each anyway.
- auto make_witness = [&] {
- return context.constant_values().GetInstId(EvalOrAddInst<SemIR::CppWitness>(
- context, loc_id,
- {.type_id = GetSingletonType(context, SemIR::WitnessType::TypeInstId),
- .elements_id = context.inst_blocks().Add(entries)}));
- };
- // Fill in the witness table.
- for (const auto& [assoc_entity_id, value_id] :
- llvm::zip_equal(assoc_entities, values)) {
- LoadImportRef(context, assoc_entity_id);
- auto decl_id =
- context.constant_values().GetInstId(SemIR::GetConstantValueInSpecific(
- context.sem_ir(), specific_interface.specific_id, assoc_entity_id));
- CARBON_CHECK(decl_id.has_value(), "Non-constant associated entity");
- auto decl = context.insts().Get(decl_id);
- CARBON_KIND_SWITCH(decl) {
- case CARBON_KIND(SemIR::StructValue struct_value): {
- if (struct_value.type_id == SemIR::ErrorInst::TypeId) {
- return SemIR::ErrorInst::InstId;
- }
- // TODO: If a thunk is needed, this will build a different value each
- // time it's called, so we won't properly deduplicate repeated
- // witnesses.
- // TODO: Skip calling make_witness if this function signature doesn't
- // involve `Self`.
- entries.push_back(CheckAssociatedFunctionImplementation(
- context,
- context.types().GetAs<SemIR::FunctionType>(struct_value.type_id),
- value_id, self_type_id, make_witness(),
- /*defer_thunk_definition=*/false));
- break;
- }
- case SemIR::AssociatedConstantDecl::Kind: {
- context.TODO(loc_id,
- "Associated constant in interface with synthesized impl");
- return SemIR::ErrorInst::InstId;
- }
- default:
- CARBON_CHECK(decl_id == SemIR::ErrorInst::InstId,
- "Unexpected kind of associated entity {0}", decl);
- return SemIR::ErrorInst::InstId;
- }
- }
- return make_witness();
- }
- static auto BuildSingleFunctionWitness(
- Context& context, SemIR::LocId loc_id, clang::FunctionDecl* cpp_fn,
- clang::DeclAccessPair found_decl, int num_params,
- SemIR::TypeId self_type_id, SemIR::SpecificInterface specific_interface)
- -> SemIR::InstId {
- auto fn_id = context.clang_sema().DiagnoseUseOfOverloadedDecl(
- cpp_fn, GetCppLocation(context, loc_id))
- ? SemIR::ErrorInst::InstId
- : ImportCppFunctionDecl(context, loc_id, cpp_fn, num_params);
- if (auto fn_decl =
- context.insts().TryGetAsWithId<SemIR::FunctionDecl>(fn_id)) {
- CheckCppOverloadAccess(context, loc_id, found_decl, fn_decl->inst_id);
- } else {
- CARBON_CHECK(fn_id == SemIR::ErrorInst::InstId);
- return SemIR::ErrorInst::InstId;
- }
- return BuildWitness(context, loc_id, self_type_id, specific_interface,
- {fn_id});
- }
- static auto LookupCopyImpl(Context& context, SemIR::LocId loc_id,
- SemIR::TypeId self_type_id,
- SemIR::SpecificInterface specific_interface)
- -> SemIR::InstId {
- auto* class_decl = TypeAsClassDecl(context, self_type_id);
- if (!class_decl) {
- // TODO: Should we also provide a `Copy` implementation for enumerations?
- return SemIR::InstId::None;
- }
- auto* ctor = context.clang_sema().LookupCopyingConstructor(
- class_decl, clang::Qualifiers::Const);
- if (!ctor) {
- // TODO: If the impl lookup failure is an error, we should produce a
- // diagnostic explaining why the class is not copyable.
- return SemIR::InstId::None;
- }
- return BuildSingleFunctionWitness(
- context, loc_id, ctor,
- clang::DeclAccessPair::make(ctor, ctor->getAccess()), /*num_params=*/1,
- self_type_id, specific_interface);
- }
- static auto LookupDestroyImpl(Context& context, SemIR::LocId loc_id,
- SemIR::TypeId self_type_id,
- SemIR::SpecificInterface specific_interface)
- -> SemIR::InstId {
- auto* class_decl = TypeAsClassDecl(context, self_type_id);
- if (!class_decl) {
- return SemIR::InstId::None;
- }
- auto* dtor = context.clang_sema().LookupDestructor(class_decl);
- if (!dtor) {
- // TODO: If the impl lookup failure is an error, we should produce a
- // diagnostic explaining why the class is not destructible.
- return SemIR::InstId::None;
- }
- return BuildSingleFunctionWitness(
- context, loc_id, dtor,
- clang::DeclAccessPair::make(dtor, dtor->getAccess()), /*num_params=*/0,
- self_type_id, specific_interface);
- }
- auto LookupCppImpl(Context& context, SemIR::LocId loc_id,
- SemIR::TypeId self_type_id,
- SemIR::SpecificInterface specific_interface,
- const TypeStructure* best_impl_type_structure,
- SemIR::LocId best_impl_loc_id) -> SemIR::InstId {
- // Determine whether this is an interface that we have special knowledge of.
- auto& interface = context.interfaces().Get(specific_interface.interface_id);
- if (!context.name_scopes().IsCorePackage(interface.parent_scope_id)) {
- return SemIR::InstId::None;
- }
- if (!interface.name_id.AsIdentifierId().has_value()) {
- return SemIR::InstId::None;
- }
- if (context.identifiers().Get(interface.name_id.AsIdentifierId()) == "Copy") {
- return LookupCopyImpl(context, loc_id, self_type_id, specific_interface);
- }
- if (context.identifiers().Get(interface.name_id.AsIdentifierId()) ==
- "Destroy") {
- return LookupDestroyImpl(context, loc_id, self_type_id, specific_interface);
- }
- // TODO: Handle other interfaces.
- // TODO: Infer a C++ type structure and check whether it's less strict than
- // the best Carbon type structure.
- static_cast<void>(best_impl_type_structure);
- static_cast<void>(best_impl_loc_id);
- return SemIR::InstId::None;
- }
- } // namespace Carbon::Check
|