| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373 |
- // 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/eval_inst.h"
- #include "toolchain/check/facet_type.h"
- #include "toolchain/check/import_ref.h"
- #include "toolchain/check/type.h"
- #include "toolchain/check/type_completion.h"
- namespace Carbon::Check {
- // Performs an access into an aggregate, retrieving the specified element.
- static auto PerformAggregateAccess(Context& context, SemIR::Inst inst)
- -> ConstantEvalResult {
- auto access_inst = inst.As<SemIR::AnyAggregateAccess>();
- if (auto aggregate = context.insts().TryGetAs<SemIR::AnyAggregateValue>(
- access_inst.aggregate_id)) {
- auto elements = context.inst_blocks().Get(aggregate->elements_id);
- auto index = static_cast<size_t>(access_inst.index.index);
- CARBON_CHECK(index < elements.size(), "Access out of bounds.");
- // `Phase` is not used here. If this element is a concrete constant, then
- // so is the result of indexing, even if the aggregate also contains a
- // symbolic context.
- return ConstantEvalResult::Existing(
- context.constant_values().Get(elements[index]));
- }
- return ConstantEvalResult::NewSamePhase(inst);
- }
- auto EvalConstantInst(Context& /*context*/, SemIRLoc /*loc*/,
- SemIR::ArrayInit inst) -> ConstantEvalResult {
- // TODO: Add an `ArrayValue` to represent a constant array object
- // representation instead of using a `TupleValue`.
- return ConstantEvalResult::NewSamePhase(
- SemIR::TupleValue{.type_id = inst.type_id, .elements_id = inst.inits_id});
- }
- auto EvalConstantInst(Context& context, SemIRLoc loc, SemIR::ArrayType inst)
- -> ConstantEvalResult {
- auto bound_inst = context.insts().Get(inst.bound_id);
- auto int_bound = bound_inst.TryAs<SemIR::IntValue>();
- if (!int_bound) {
- CARBON_CHECK(context.constant_values().Get(inst.bound_id).is_symbolic(),
- "Unexpected inst {0} for template constant int", bound_inst);
- return ConstantEvalResult::NewSamePhase(inst);
- }
- // TODO: We should check that the size of the resulting array type
- // fits in 64 bits, not just that the bound does. Should we use a
- // 32-bit limit for 32-bit targets?
- const auto& bound_val = context.ints().Get(int_bound->int_id);
- if (context.types().IsSignedInt(int_bound->type_id) &&
- bound_val.isNegative()) {
- CARBON_DIAGNOSTIC(ArrayBoundNegative, Error,
- "array bound of {0} is negative", TypedInt);
- context.emitter().Emit(loc, ArrayBoundNegative,
- {.type = int_bound->type_id, .value = bound_val});
- return ConstantEvalResult::Error;
- }
- if (bound_val.getActiveBits() > 64) {
- CARBON_DIAGNOSTIC(ArrayBoundTooLarge, Error,
- "array bound of {0} is too large", TypedInt);
- context.emitter().Emit(loc, ArrayBoundTooLarge,
- {.type = int_bound->type_id, .value = bound_val});
- return ConstantEvalResult::Error;
- }
- return ConstantEvalResult::NewSamePhase(inst);
- }
- auto EvalConstantInst(Context& context, SemIRLoc /*loc*/,
- SemIR::AsCompatible inst) -> ConstantEvalResult {
- // AsCompatible changes the type of the source instruction; its constant
- // value, if there is one, needs to be modified to be of the same type.
- auto value_id = context.constant_values().Get(inst.source_id);
- CARBON_CHECK(value_id.is_constant());
- auto value_inst =
- context.insts().Get(context.constant_values().GetInstId(value_id));
- value_inst.SetType(inst.type_id);
- return ConstantEvalResult::NewAnyPhase(value_inst);
- }
- auto EvalConstantInst(Context& context, SemIRLoc /*loc*/, SemIR::BindAlias inst)
- -> ConstantEvalResult {
- // An alias evaluates to the value it's bound to.
- return ConstantEvalResult::Existing(
- context.constant_values().Get(inst.value_id));
- }
- auto EvalConstantInst(Context& /*context*/, SemIRLoc /*loc*/,
- SemIR::BindValue /*inst*/) -> ConstantEvalResult {
- // TODO: Handle this once we've decided how to represent constant values of
- // reference expressions.
- return ConstantEvalResult::TODO;
- }
- auto EvalConstantInst(Context& context, SemIRLoc /*loc*/,
- SemIR::ClassElementAccess inst) -> ConstantEvalResult {
- return PerformAggregateAccess(context, inst);
- }
- auto EvalConstantInst(Context& context, SemIRLoc /*loc*/, SemIR::ClassDecl inst)
- -> ConstantEvalResult {
- // If the class has generic parameters, we don't produce a class type, but a
- // callable whose return value is a class type.
- if (context.classes().Get(inst.class_id).has_parameters()) {
- return ConstantEvalResult::NewSamePhase(SemIR::StructValue{
- .type_id = inst.type_id, .elements_id = SemIR::InstBlockId::Empty});
- }
- // A non-generic class declaration evaluates to the class type.
- return ConstantEvalResult::NewSamePhase(
- SemIR::ClassType{.type_id = SemIR::TypeType::SingletonTypeId,
- .class_id = inst.class_id,
- .specific_id = SemIR::SpecificId::None});
- }
- auto EvalConstantInst(Context& /*context*/, SemIRLoc /*loc*/,
- SemIR::ClassInit inst) -> ConstantEvalResult {
- // TODO: Add a `ClassValue` to represent a constant class object
- // representation instead of using a `StructValue`.
- return ConstantEvalResult::NewSamePhase(SemIR::StructValue{
- .type_id = inst.type_id, .elements_id = inst.elements_id});
- }
- auto EvalConstantInst(Context& context, SemIRLoc /*loc*/, SemIR::ConstType inst)
- -> ConstantEvalResult {
- // `const (const T)` evaluates to `const T`.
- if (context.types().Is<SemIR::ConstType>(inst.inner_id)) {
- return ConstantEvalResult::Existing(
- context.types().GetConstantId(inst.inner_id));
- }
- // Otherwise, `const T` evaluates to itself.
- return ConstantEvalResult::NewSamePhase(inst);
- }
- auto EvalConstantInst(Context& context, SemIRLoc /*loc*/, SemIR::Converted inst)
- -> ConstantEvalResult {
- // A conversion evaluates to the result of the conversion.
- return ConstantEvalResult::Existing(
- context.constant_values().Get(inst.result_id));
- }
- auto EvalConstantInst(Context& /*context*/, SemIRLoc /*loc*/,
- SemIR::Deref /*inst*/) -> ConstantEvalResult {
- // TODO: Handle this.
- return ConstantEvalResult::TODO;
- }
- auto EvalConstantInst(Context& context, SemIRLoc /*loc*/,
- SemIR::ExportDecl inst) -> ConstantEvalResult {
- // An export instruction evaluates to the exported declaration.
- return ConstantEvalResult::Existing(
- context.constant_values().Get(inst.value_id));
- }
- auto EvalConstantInst(Context& context, SemIRLoc /*loc*/,
- SemIR::FacetAccessType inst) -> ConstantEvalResult {
- if (auto facet_value = context.insts().TryGetAs<SemIR::FacetValue>(
- inst.facet_value_inst_id)) {
- return ConstantEvalResult::Existing(
- context.constant_values().Get(facet_value->type_inst_id));
- }
- return ConstantEvalResult::NewSamePhase(inst);
- }
- auto EvalConstantInst(Context& context, SemIRLoc /*loc*/,
- SemIR::FacetAccessWitness inst) -> ConstantEvalResult {
- if (auto facet_value = context.insts().TryGetAs<SemIR::FacetValue>(
- inst.facet_value_inst_id)) {
- return ConstantEvalResult::Existing(
- context.constant_values().Get(facet_value->witness_inst_id));
- }
- return ConstantEvalResult::NewSamePhase(inst);
- }
- auto EvalConstantInst(Context& context, SemIRLoc loc, SemIR::FloatType inst)
- -> ConstantEvalResult {
- return ValidateFloatType(context, loc, inst)
- ? ConstantEvalResult::NewSamePhase(inst)
- : ConstantEvalResult::Error;
- }
- auto EvalConstantInst(Context& /*context*/, SemIRLoc /*loc*/,
- SemIR::FunctionDecl inst) -> ConstantEvalResult {
- // A function declaration evaluates to a function object, which is an empty
- // object of function type.
- // TODO: Eventually we may need to handle captures here.
- return ConstantEvalResult::NewSamePhase(SemIR::StructValue{
- .type_id = inst.type_id, .elements_id = SemIR::InstBlockId::Empty});
- }
- auto EvalConstantInst(Context& context, SemIRLoc loc,
- SemIR::ImplWitnessAccess inst) -> ConstantEvalResult {
- // This is PerformAggregateAccess followed by GetConstantInSpecific.
- if (auto witness =
- context.insts().TryGetAs<SemIR::ImplWitness>(inst.witness_id)) {
- auto elements = context.inst_blocks().Get(witness->elements_id);
- auto index = static_cast<size_t>(inst.index.index);
- CARBON_CHECK(index < elements.size(), "Access out of bounds.");
- auto element = elements[index];
- if (!element.has_value()) {
- // TODO: Perhaps this should be a `{}` value with incomplete type?
- CARBON_DIAGNOSTIC(ImplAccessMemberBeforeComplete, Error,
- "accessing member from impl before the end of "
- "its definition");
- // TODO: Add note pointing to the impl declaration.
- context.emitter().Emit(loc, ImplAccessMemberBeforeComplete);
- return ConstantEvalResult::Error;
- }
- LoadImportRef(context, element);
- return ConstantEvalResult::Existing(GetConstantValueInSpecific(
- context.sem_ir(), witness->specific_id, element));
- }
- return ConstantEvalResult::NewSamePhase(inst);
- }
- auto EvalConstantInst(Context& /*context*/, SemIRLoc /*loc*/,
- SemIR::ImportRefUnloaded inst) -> ConstantEvalResult {
- CARBON_FATAL("ImportRefUnloaded should be loaded before TryEvalInst: {0}",
- inst);
- }
- auto EvalConstantInst(Context& context, SemIRLoc /*loc*/,
- SemIR::InitializeFrom inst) -> ConstantEvalResult {
- // Initialization is not performed in-place during constant evaluation, so
- // just return the value of the initializer.
- return ConstantEvalResult::Existing(
- context.constant_values().Get(inst.src_id));
- }
- auto EvalConstantInst(Context& context, SemIRLoc loc, SemIR::IntType inst)
- -> ConstantEvalResult {
- return ValidateIntType(context, loc, inst)
- ? ConstantEvalResult::NewSamePhase(inst)
- : ConstantEvalResult::Error;
- }
- auto EvalConstantInst(Context& context, SemIRLoc /*loc*/,
- SemIR::InterfaceDecl inst) -> ConstantEvalResult {
- // If the interface has generic parameters, we don't produce an interface
- // type, but a callable whose return value is an interface type.
- if (context.interfaces().Get(inst.interface_id).has_parameters()) {
- return ConstantEvalResult::NewSamePhase(SemIR::StructValue{
- .type_id = inst.type_id, .elements_id = SemIR::InstBlockId::Empty});
- }
- // A non-generic interface declaration evaluates to a facet type.
- return ConstantEvalResult::NewSamePhase(FacetTypeFromInterface(
- context, inst.interface_id, SemIR::SpecificId::None));
- }
- auto EvalConstantInst(Context& context, SemIRLoc /*loc*/, SemIR::NameRef inst)
- -> ConstantEvalResult {
- // A name reference evaluates to the value the name resolves to.
- return ConstantEvalResult::Existing(
- context.constant_values().Get(inst.value_id));
- }
- auto EvalConstantInst(Context& context, SemIRLoc loc,
- SemIR::RequireCompleteType inst) -> ConstantEvalResult {
- auto witness_type_id =
- GetSingletonType(context, SemIR::WitnessType::SingletonInstId);
- // If the type is a concrete constant, require it to be complete now.
- auto complete_type_id = inst.complete_type_id;
- if (context.types().GetConstantId(complete_type_id).is_concrete()) {
- if (!TryToCompleteType(context, complete_type_id, loc, [&] {
- // TODO: It'd be nice to report the original type prior to
- // evaluation here.
- CARBON_DIAGNOSTIC(IncompleteTypeInMonomorphization, Error,
- "type {0} is incomplete", SemIR::TypeId);
- return context.emitter().Build(loc, IncompleteTypeInMonomorphization,
- complete_type_id);
- })) {
- return ConstantEvalResult::Error;
- }
- return ConstantEvalResult::NewSamePhase(SemIR::CompleteTypeWitness{
- .type_id = witness_type_id,
- .object_repr_id = context.types().GetObjectRepr(complete_type_id)});
- }
- // If it's not a concrete constant, require it to be complete once it
- // becomes one.
- return ConstantEvalResult::NewSamePhase(inst);
- }
- auto EvalConstantInst(Context& context, SemIRLoc /*loc*/,
- SemIR::SpecificConstant inst) -> ConstantEvalResult {
- // Pull the constant value out of the specific.
- return ConstantEvalResult::Existing(SemIR::GetConstantValueInSpecific(
- context.sem_ir(), inst.specific_id, inst.inst_id));
- }
- auto EvalConstantInst(Context& context, SemIRLoc /*loc*/,
- SemIR::SpliceBlock inst) -> ConstantEvalResult {
- // SpliceBlock evaluates to the result value that is (typically) within the
- // block. This can be constant even if the block contains other non-constant
- // instructions.
- return ConstantEvalResult::Existing(
- context.constant_values().Get(inst.result_id));
- }
- auto EvalConstantInst(Context& context, SemIRLoc /*loc*/,
- SemIR::StructAccess inst) -> ConstantEvalResult {
- return PerformAggregateAccess(context, inst);
- }
- auto EvalConstantInst(Context& /*context*/, SemIRLoc /*loc*/,
- SemIR::StructInit inst) -> ConstantEvalResult {
- return ConstantEvalResult::NewSamePhase(SemIR::StructValue{
- .type_id = inst.type_id, .elements_id = inst.elements_id});
- }
- auto EvalConstantInst(Context& /*context*/, SemIRLoc /*loc*/,
- SemIR::Temporary /*inst*/) -> ConstantEvalResult {
- // TODO: Handle this. Can we just return the value of `init_id`?
- return ConstantEvalResult::TODO;
- }
- auto EvalConstantInst(Context& context, SemIRLoc /*loc*/,
- SemIR::TupleAccess inst) -> ConstantEvalResult {
- return PerformAggregateAccess(context, inst);
- }
- auto EvalConstantInst(Context& /*context*/, SemIRLoc /*loc*/,
- SemIR::TupleInit inst) -> ConstantEvalResult {
- return ConstantEvalResult::NewSamePhase(SemIR::TupleValue{
- .type_id = inst.type_id, .elements_id = inst.elements_id});
- }
- auto EvalConstantInst(Context& context, SemIRLoc /*loc*/,
- SemIR::UnaryOperatorNot inst) -> ConstantEvalResult {
- // `not true` -> `false`, `not false` -> `true`.
- // All other uses of unary `not` are non-constant.
- auto const_id = context.constant_values().Get(inst.operand_id);
- if (const_id.is_concrete()) {
- auto value = context.insts().GetAs<SemIR::BoolLiteral>(
- context.constant_values().GetInstId(const_id));
- value.value = SemIR::BoolValue::From(!value.value.ToBool());
- return ConstantEvalResult::NewSamePhase(value);
- }
- return ConstantEvalResult::NotConstant;
- }
- auto EvalConstantInst(Context& context, SemIRLoc /*loc*/,
- SemIR::ValueOfInitializer inst) -> ConstantEvalResult {
- // Values of value expressions and initializing expressions are represented in
- // the same way during constant evaluation, so just return the value of the
- // operand.
- return ConstantEvalResult::Existing(
- context.constant_values().Get(inst.init_id));
- }
- auto EvalConstantInst(Context& context, SemIRLoc /*loc*/,
- SemIR::ValueParamPattern inst) -> ConstantEvalResult {
- // TODO: Treat this as a non-expression (here and in GetExprCategory)
- // once generic deduction doesn't need patterns to have constant values.
- return ConstantEvalResult::Existing(
- context.constant_values().Get(inst.subpattern_id));
- }
- auto EvalConstantInst(Context& /*context*/, SemIRLoc /*loc*/,
- SemIR::VtablePtr /*inst*/) -> ConstantEvalResult {
- // TODO: Handle this.
- return ConstantEvalResult::TODO;
- }
- } // namespace Carbon::Check
|