| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141 |
- // 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/custom_witness.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);
- }
- 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 BuildCustomWitness(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, CoreInterface core_interface,
- SemIR::SpecificInterface specific_interface,
- const TypeStructure* best_impl_type_structure,
- SemIR::LocId best_impl_loc_id) -> SemIR::InstId {
- // TODO: Handle other interfaces.
- switch (core_interface) {
- case CoreInterface::Copy:
- return LookupCopyImpl(context, loc_id, self_type_id, specific_interface);
- case CoreInterface::Destroy:
- return LookupDestroyImpl(context, loc_id, self_type_id,
- specific_interface);
- case CoreInterface::Unknown:
- CARBON_FATAL("shouldn't be called with `Unknown`");
- }
- // 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
|