| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164 |
- // 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/operator.h"
- #include <optional>
- #include "toolchain/check/call.h"
- #include "toolchain/check/context.h"
- #include "toolchain/check/cpp/call.h"
- #include "toolchain/check/cpp/operators.h"
- #include "toolchain/check/generic.h"
- #include "toolchain/check/member_access.h"
- #include "toolchain/check/name_lookup.h"
- #include "toolchain/sem_ir/class.h"
- #include "toolchain/sem_ir/ids.h"
- #include "toolchain/sem_ir/name_scope.h"
- #include "toolchain/sem_ir/typed_insts.h"
- namespace Carbon::Check {
- // Returns the `Op` function for the specified operator.
- static auto GetOperatorOpFunction(Context& context, SemIR::LocId loc_id,
- Operator op) -> SemIR::InstId {
- auto implicit_loc_id = context.insts().GetLocIdForDesugaring(loc_id);
- // Look up the interface, and pass it any generic arguments.
- // TODO: Improve diagnostics when the found `interface_id` isn't callable.
- auto interface_id =
- LookupNameInCore(context, implicit_loc_id, op.interface_name);
- if (!op.interface_args_ref.empty()) {
- interface_id = PerformCall(context, implicit_loc_id, interface_id,
- op.interface_args_ref);
- }
- // Look up the interface member.
- auto op_name_id = context.core_identifiers().AddNameId(op.op_name);
- return PerformMemberAccess(context, implicit_loc_id, interface_id,
- op_name_id);
- }
- // Returns whether the instruction is a C++ class type. Assumes the argument is
- // in canonical form and does not look through the constant value.
- static auto IsCppClassType(Context& context, SemIR::InstId inst_id) -> bool {
- auto class_type = context.insts().TryGetAs<SemIR::ClassType>(inst_id);
- if (!class_type) {
- // Not a class.
- return false;
- }
- SemIR::NameScopeId class_scope_id =
- context.classes().Get(class_type->class_id).scope_id;
- return class_scope_id.has_value() &&
- context.name_scopes().Get(class_scope_id).is_cpp_scope();
- }
- // Returns whether the instruction is a value of C++ class type.
- static auto HasCppClassType(Context& context, SemIR::InstId inst_id) -> bool {
- return IsCppClassType(context, context.types().GetTypeInstId(
- context.insts().Get(inst_id).type_id()));
- }
- auto BuildUnaryOperator(Context& context, SemIR::LocId loc_id, Operator op,
- SemIR::InstId operand_id, bool diagnose,
- DiagnosticContextFn missing_impl_diagnostic_context)
- -> SemIR::InstId {
- if (operand_id == SemIR::ErrorInst::InstId) {
- // Exit early for errors, which prevent forming an `Op` function.
- return SemIR::ErrorInst::InstId;
- }
- SemIR::InstId op_fn_id = SemIR::InstId::None;
- // For unary operators with a C++ class as the operand, try to import and call
- // the C++ operator.
- // TODO: Change impl lookup instead. See
- // https://github.com/carbon-language/carbon-lang/blob/db0a00d713015436844c55e7ac190a0f95556499/toolchain/check/operator.cpp#L76
- if (HasCppClassType(context, operand_id) ||
- llvm::any_of(op.interface_args_ref, [&](SemIR::InstId arg_id) {
- return IsCppClassType(context, arg_id);
- })) {
- op_fn_id = LookupCppOperator(context, loc_id, op, {operand_id});
- // If C++ operator lookup found a non-method operator, call it with one call
- // argument. Otherwise fall through to call it with a self argument.
- if (op_fn_id.has_value() && !IsCppOperatorMethod(context, op_fn_id)) {
- return PerformCall(context, loc_id, op_fn_id, {operand_id},
- /*is_operator_syntax=*/true);
- }
- }
- if (!op_fn_id.has_value()) {
- // Look up the operator function.
- op_fn_id = GetOperatorOpFunction(context, loc_id, op);
- }
- // Form `operand.(Op)`.
- auto bound_op_id =
- PerformCompoundMemberAccess(context, loc_id, operand_id, op_fn_id,
- diagnose, missing_impl_diagnostic_context);
- if (bound_op_id == SemIR::ErrorInst::InstId) {
- return SemIR::ErrorInst::InstId;
- }
- // Form `bound_op()`.
- return PerformCall(context, loc_id, bound_op_id, {},
- /*is_operator_syntax=*/true);
- }
- auto BuildBinaryOperator(Context& context, SemIR::LocId loc_id, Operator op,
- SemIR::InstId lhs_id, SemIR::InstId rhs_id,
- bool diagnose,
- DiagnosticContextFn missing_impl_diagnostic_context)
- -> SemIR::InstId {
- if (lhs_id == SemIR::ErrorInst::InstId) {
- // Exit early for errors, which prevent forming an `Op` function.
- return SemIR::ErrorInst::InstId;
- }
- SemIR::InstId op_fn_id = SemIR::InstId::None;
- // For binary operators with a C++ class as at least one of the operands, try
- // to import and call the C++ operator.
- // TODO: Instead of hooking this here, change impl lookup, so that a generic
- // constraint such as `T:! Core.Add` is satisfied by C++ class types that are
- // addable. See
- // https://github.com/carbon-language/carbon-lang/pull/5996/files/5d01fa69511b76f87efbc0387f5e40abcf4c911a#r2308666348
- // and
- // https://github.com/carbon-language/carbon-lang/pull/5996/files/5d01fa69511b76f87efbc0387f5e40abcf4c911a#r2308664536
- if (HasCppClassType(context, lhs_id) || HasCppClassType(context, rhs_id) ||
- llvm::any_of(op.interface_args_ref, [&](SemIR::InstId arg_id) {
- return IsCppClassType(context, arg_id);
- })) {
- op_fn_id = LookupCppOperator(context, loc_id, op, {lhs_id, rhs_id});
- // If C++ operator lookup found a non-method operator, call it with two call
- // arguments. Otherwise fall through to call it with a self argument and one
- // call argument.
- if (op_fn_id.has_value() && !IsCppOperatorMethod(context, op_fn_id)) {
- return PerformCall(context, loc_id, op_fn_id, {lhs_id, rhs_id},
- /*is_operator_syntax=*/true);
- }
- }
- if (!op_fn_id.has_value()) {
- // Look up the operator function.
- op_fn_id = GetOperatorOpFunction(context, loc_id, op);
- }
- // Form `lhs.(Op)`.
- auto bound_op_id =
- PerformCompoundMemberAccess(context, loc_id, lhs_id, op_fn_id, diagnose,
- missing_impl_diagnostic_context);
- if (bound_op_id == SemIR::ErrorInst::InstId) {
- return SemIR::ErrorInst::InstId;
- }
- // Form `bound_op(rhs)`.
- return PerformCall(context, loc_id, bound_op_id, {rhs_id},
- /*is_operator_syntax=*/true);
- }
- } // namespace Carbon::Check
|