// 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/overload_resolution.h" #include "clang/Basic/DiagnosticSema.h" #include "clang/Sema/Overload.h" #include "clang/Sema/Sema.h" #include "toolchain/check/cpp/import.h" #include "toolchain/check/cpp/location.h" #include "toolchain/check/cpp/type_mapping.h" #include "toolchain/diagnostics/diagnostic_emitter.h" #include "toolchain/sem_ir/ids.h" #include "toolchain/sem_ir/typed_insts.h" namespace Carbon::Check { // Map a Carbon name into a C++ name. static auto GetCppName(Context& context, SemIR::NameId name_id) -> clang::DeclarationName { // TODO: Some special names should probably use different formatting. In // particular, NameId::CppOperator should probably map back to a // CXXOperatorName. auto name_str = context.names().GetFormatted(name_id); return clang::DeclarationName(&context.ast_context().Idents.get(name_str)); } // Adds the given overload candidates to the candidate set. static auto AddOverloadCandidataes(clang::Sema& sema, clang::OverloadCandidateSet& candidate_set, const clang::UnresolvedSetImpl& functions, clang::Expr* self_arg, llvm::ArrayRef args) -> void { constexpr bool SuppressUserConversions = false; constexpr bool PartialOverloading = false; constexpr clang::TemplateArgumentListInfo* ExplicitTemplateArgs = nullptr; for (auto found_decl : functions.pairs()) { auto* decl = found_decl.getDecl()->getUnderlyingDecl(); auto* template_decl = dyn_cast(decl); auto* fn_decl = template_decl ? template_decl->getTemplatedDecl() : cast(decl); auto* method_decl = dyn_cast(fn_decl); if (method_decl && !method_decl->isStatic() && !isa(fn_decl)) { clang::QualType self_type; clang::Expr::Classification self_classification; if (self_arg) { self_type = self_arg->getType(); self_classification = self_arg->Classify(sema.Context); } if (template_decl) { sema.AddMethodTemplateCandidate( template_decl, found_decl, cast(template_decl->getDeclContext()), ExplicitTemplateArgs, self_type, self_classification, args, candidate_set, SuppressUserConversions, PartialOverloading); } else { sema.AddMethodCandidate(method_decl, found_decl, method_decl->getParent(), self_type, self_classification, args, candidate_set, SuppressUserConversions, PartialOverloading); } } else { if (template_decl) { sema.AddTemplateOverloadCandidate( template_decl, found_decl, ExplicitTemplateArgs, args, candidate_set, SuppressUserConversions, PartialOverloading); } else { sema.AddOverloadCandidate(fn_decl, found_decl, args, candidate_set, SuppressUserConversions, PartialOverloading); } } } } auto PerformCppOverloadResolution(Context& context, SemIR::LocId loc_id, SemIR::CppOverloadSetId overload_set_id, SemIR::InstId self_id, llvm::ArrayRef arg_ids) -> SemIR::InstId { // Register an annotation scope to flush any Clang diagnostics when we return. // This is important to ensure that Clang diagnostics are properly interleaved // with Carbon diagnostics. Diagnostics::AnnotationScope annotate_diagnostics(&context.emitter(), [](auto& /*builder*/) {}); // Map Carbon call argument types to C++ types. clang::Expr* self_expr = nullptr; if (self_id.has_value()) { self_expr = InventClangArg(context, self_id); if (!self_expr) { return SemIR::ErrorInst::InstId; } } auto maybe_arg_exprs = InventClangArgs(context, arg_ids); if (!maybe_arg_exprs.has_value()) { return SemIR::ErrorInst::InstId; } auto& arg_exprs = *maybe_arg_exprs; const SemIR::CppOverloadSet& overload_set = context.cpp_overload_sets().Get(overload_set_id); clang::SourceLocation loc = GetCppLocation(context, loc_id); // Add candidate functions from the name lookup. clang::OverloadCandidateSet candidate_set( loc, clang::OverloadCandidateSet::CandidateSetKind::CSK_Normal); clang::Sema& sema = context.clang_sema(); AddOverloadCandidataes(sema, candidate_set, overload_set.candidate_functions, self_expr, arg_exprs); // Find best viable function among the candidates. clang::OverloadCandidateSet::iterator best_viable_fn; clang::OverloadingResult overloading_result = candidate_set.BestViableFunction(sema, loc, best_viable_fn); switch (overloading_result) { case clang::OverloadingResult::OR_Success: { // TODO: Handle the cases when Function is null. CARBON_CHECK(best_viable_fn->Function); sema.MarkFunctionReferenced(loc, best_viable_fn->Function); SemIR::InstId result = ImportCppFunctionDecl( context, loc_id, best_viable_fn->Function, arg_exprs.size()); return result; } case clang::OverloadingResult::OR_No_Viable_Function: { candidate_set.NoteCandidates( clang::PartialDiagnosticAt( loc, sema.PDiag(clang::diag::err_ovl_no_viable_function_in_call) << GetCppName(context, overload_set.name_id)), sema, clang::OCD_AllCandidates, arg_exprs); return SemIR::ErrorInst::InstId; } case clang::OverloadingResult::OR_Ambiguous: { candidate_set.NoteCandidates( clang::PartialDiagnosticAt( loc, sema.PDiag(clang::diag::err_ovl_ambiguous_call) << GetCppName(context, overload_set.name_id)), sema, clang::OCD_AmbiguousCandidates, arg_exprs); return SemIR::ErrorInst::InstId; } case clang::OverloadingResult::OR_Deleted: { sema.DiagnoseUseOfDeletedFunction( loc, clang::SourceRange(loc, loc), GetCppName(context, overload_set.name_id), candidate_set, best_viable_fn->Function, arg_exprs); return SemIR::ErrorInst::InstId; } } } } // namespace Carbon::Check