| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023 |
- // 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
- #ifndef CARBON_TOOLCHAIN_SEM_IR_IDS_H_
- #define CARBON_TOOLCHAIN_SEM_IR_IDS_H_
- #include <limits>
- #include "common/check.h"
- #include "common/ostream.h"
- #include "toolchain/base/index_base.h"
- #include "toolchain/base/value_ids.h"
- #include "toolchain/diagnostics/diagnostic_emitter.h"
- #include "toolchain/parse/node_ids.h"
- // NOLINTNEXTLINE(readability-identifier-naming)
- namespace clang {
- // Forward declare indexed types, for integration with ValueStore.
- class SourceLocation;
- } // namespace clang
- namespace Carbon::SemIR {
- // Forward declare indexed types, for integration with ValueStore.
- class File;
- class ImportIRInst;
- class Inst;
- class NameScope;
- struct AssociatedConstant;
- struct Class;
- struct EntityName;
- struct ExprRegion;
- struct FacetTypeInfo;
- struct Function;
- struct Generic;
- struct IdentifiedFacetType;
- struct Specific;
- struct SpecificInterface;
- struct ImportCpp;
- struct ImportIR;
- struct Impl;
- struct Interface;
- struct StructTypeField;
- struct TypeInfo;
- // The ID of an instruction.
- struct InstId : public IdBase<InstId> {
- static constexpr llvm::StringLiteral Label = "inst";
- using ValueType = Inst;
- // The maximum ID, inclusive.
- static constexpr int Max = std::numeric_limits<int32_t>::max();
- // Represents the result of a name lookup that is temporarily disallowed
- // because the name is currently being initialized.
- static const InstId InitTombstone;
- using IdBase::IdBase;
- auto Print(llvm::raw_ostream& out) const -> void;
- };
- constexpr InstId InstId::InitTombstone = InstId(NoneIndex - 1);
- // And InstId whose value is a type. The fact it's a type is CHECKed on
- // construction, and this allows that check to be represented in the type
- // system.
- struct TypeInstId : public InstId {
- static const TypeInstId None;
- using InstId::InstId;
- static constexpr auto UnsafeMake(InstId id) -> TypeInstId {
- return TypeInstId(UnsafeCtor(), id);
- }
- private:
- struct UnsafeCtor {};
- explicit constexpr TypeInstId(UnsafeCtor /*unsafe*/, InstId id)
- : InstId(id) {}
- };
- constexpr TypeInstId TypeInstId::None = TypeInstId::UnsafeMake(InstId::None);
- // An ID of an instruction that is referenced absolutely by another instruction.
- // This should only be used as the type of a field within a typed instruction
- // class.
- //
- // When a typed instruction has a field of this type, that field represents an
- // absolute reference to another instruction that typically resides in a
- // different entity. This behaves in most respects like an InstId field, but
- // substitution into the typed instruction leaves the field unchanged rather
- // than substituting into it.
- class AbsoluteInstId : public InstId {
- public:
- // Support implicit conversion from InstId so that InstId and AbsoluteInstId
- // have the same interface.
- // NOLINTNEXTLINE(google-explicit-constructor)
- constexpr AbsoluteInstId(InstId inst_id) : InstId(inst_id) {}
- using InstId::InstId;
- };
- // An ID of an instruction that is used as the destination of an initializing
- // expression. This should only be used as the type of a field within a typed
- // instruction class.
- //
- // This behaves in most respects like an InstId field, but constant evaluation
- // of an instruction with a destination field will not evaluate this field, and
- // substitution will not substitute into it.
- //
- // TODO: Decide on how substitution should handle this. Multiple instructions
- // can refer to the same destination, so these don't have the tree structure
- // that substitution expects, but we might need to substitute into the result of
- // an instruction.
- class DestInstId : public InstId {
- public:
- // Support implicit conversion from InstId so that InstId and DestInstId
- // have the same interface.
- // NOLINTNEXTLINE(google-explicit-constructor)
- constexpr DestInstId(InstId inst_id) : InstId(inst_id) {}
- using InstId::InstId;
- };
- // An ID of an instruction that is referenced as a meta-operand of an action.
- // This should only be used as the type of a field within a typed instruction
- // class.
- //
- // This is used to model cases where an action's operand is not the value
- // produced by another instruction, but is the other instruction itself. This is
- // common for actions representing template instantiation.
- //
- // This behaves in most respects like an InstId field, but evaluation of the
- // instruction that has this field will not fail if the instruction does not
- // have a constant value. If the instruction has a constant value, it will still
- // be replaced by its constant value during evaluation like normal, but if it
- // has a non-constant value, the field is left unchanged by evaluation.
- class MetaInstId : public InstId {
- public:
- // Support implicit conversion from InstId so that InstId and MetaInstId
- // have the same interface.
- // NOLINTNEXTLINE(google-explicit-constructor)
- constexpr MetaInstId(InstId inst_id) : InstId(inst_id) {}
- using InstId::InstId;
- };
- // The ID of a constant value of an expression. An expression is either:
- //
- // - a concrete constant, whose value does not depend on any generic parameters,
- // such as `42` or `i32*` or `("hello", "world")`, or
- // - a symbolic constant, whose value includes a generic parameter, such as
- // `Vector(T*)`, or
- // - a runtime expression, such as `Print("hello")`.
- //
- // Concrete constants are a thin wrapper around the instruction ID of the
- // constant instruction that defines the constant. Symbolic constants are an
- // index into a separate table of `SymbolicConstant`s maintained by the constant
- // value store.
- struct ConstantId : public IdBase<ConstantId> {
- static constexpr llvm::StringLiteral Label = "constant";
- // An ID for an expression that is not constant.
- static const ConstantId NotConstant;
- // Returns the constant ID corresponding to a concrete constant, which should
- // either be in the `constants` block in the file or should be known to be
- // unique.
- static constexpr auto ForConcreteConstant(InstId const_id) -> ConstantId {
- return ConstantId(const_id.index);
- }
- // Returns the constant ID corresponding to a symbolic constant index.
- static constexpr auto ForSymbolicConstantIndex(int32_t symbolic_index)
- -> ConstantId {
- return ConstantId(FirstSymbolicIndex - symbolic_index);
- }
- using IdBase::IdBase;
- // Returns whether this represents a constant. Requires has_value.
- constexpr auto is_constant() const -> bool {
- CARBON_DCHECK(has_value());
- return *this != ConstantId::NotConstant;
- }
- // Returns whether this represents a symbolic constant. Requires has_value.
- constexpr auto is_symbolic() const -> bool {
- CARBON_DCHECK(has_value());
- return index <= FirstSymbolicIndex;
- }
- // Returns whether this represents a concrete constant. Requires has_value.
- constexpr auto is_concrete() const -> bool {
- CARBON_DCHECK(has_value());
- return index >= 0;
- }
- // Prints this ID to the given output stream. `disambiguate` indicates whether
- // concrete constants should be wrapped with "concrete_constant(...)" so that
- // they aren't printed the same as an InstId. This can be set to false if
- // there is no risk of ambiguity.
- auto Print(llvm::raw_ostream& out, bool disambiguate = true) const -> void;
- private:
- friend class ConstantValueStore;
- // TODO: C++23 makes std::abs constexpr, but until then we mirror std::abs
- // logic here. LLVM should still optimize this.
- static constexpr auto Abs(int32_t i) -> int32_t { return i > 0 ? i : -i; }
- // Returns the instruction that describes this concrete constant value.
- // Requires `is_concrete()`. Use `ConstantValueStore::GetInstId` to get the
- // instruction ID of a `ConstantId`.
- constexpr auto concrete_inst_id() const -> InstId {
- CARBON_DCHECK(is_concrete());
- return InstId(index);
- }
- // Returns the symbolic constant index that describes this symbolic constant
- // value. Requires `is_symbolic()`.
- constexpr auto symbolic_index() const -> int32_t {
- CARBON_DCHECK(is_symbolic());
- return FirstSymbolicIndex - index;
- }
- static constexpr int32_t NotConstantIndex = NoneIndex - 1;
- static constexpr int32_t FirstSymbolicIndex = NoneIndex - 2;
- };
- constexpr ConstantId ConstantId::NotConstant = ConstantId(NotConstantIndex);
- // The ID of a EntityName.
- struct EntityNameId : public IdBase<EntityNameId> {
- static constexpr llvm::StringLiteral Label = "entity_name";
- using ValueType = EntityName;
- using IdBase::IdBase;
- };
- // The index of a compile-time binding. This is the de Bruijn level for the
- // binding -- that is, this is the number of other compile time bindings whose
- // scope encloses this binding.
- struct CompileTimeBindIndex : public IndexBase<CompileTimeBindIndex> {
- static constexpr llvm::StringLiteral Label = "comp_time_bind";
- using IndexBase::IndexBase;
- };
- // The index of a `Call` parameter in a function. These are allocated
- // sequentially, left-to-right, to the function parameters that will have
- // arguments passed to them at runtime. In a `Call` instruction, a runtime
- // argument will have the position in the argument list corresponding to its
- // `Call` parameter index.
- struct CallParamIndex : public IndexBase<CallParamIndex> {
- static constexpr llvm::StringLiteral Label = "call_param";
- using IndexBase::IndexBase;
- };
- // The ID of a function.
- struct FunctionId : public IdBase<FunctionId> {
- static constexpr llvm::StringLiteral Label = "function";
- using ValueType = Function;
- using IdBase::IdBase;
- };
- // The ID of an IR within the set of all IRs being evaluated in the current
- // check execution.
- struct CheckIRId : public IdBase<CheckIRId> {
- static constexpr llvm::StringLiteral Label = "check_ir";
- using IdBase::IdBase;
- };
- // The ID of a class.
- struct ClassId : public IdBase<ClassId> {
- static constexpr llvm::StringLiteral Label = "class";
- using ValueType = Class;
- using IdBase::IdBase;
- };
- // The ID of an interface.
- struct InterfaceId : public IdBase<InterfaceId> {
- static constexpr llvm::StringLiteral Label = "interface";
- using ValueType = Interface;
- using IdBase::IdBase;
- };
- // The ID of an associated constant.
- struct AssociatedConstantId : public IdBase<AssociatedConstantId> {
- static constexpr llvm::StringLiteral Label = "assoc_const";
- using ValueType = AssociatedConstant;
- using IdBase::IdBase;
- };
- // The ID of an facet type value.
- struct FacetTypeId : public IdBase<FacetTypeId> {
- static constexpr llvm::StringLiteral Label = "facet_type";
- using ValueType = FacetTypeInfo;
- using IdBase::IdBase;
- };
- // The ID of an resolved facet type value.
- struct IdentifiedFacetTypeId : public IdBase<IdentifiedFacetTypeId> {
- static constexpr llvm::StringLiteral Label = "identified_facet_type";
- using ValueType = IdentifiedFacetType;
- using IdBase::IdBase;
- };
- // The ID of an impl.
- struct ImplId : public IdBase<ImplId> {
- static constexpr llvm::StringLiteral Label = "impl";
- using ValueType = Impl;
- using IdBase::IdBase;
- };
- // The ID of a generic.
- struct GenericId : public IdBase<GenericId> {
- static constexpr llvm::StringLiteral Label = "generic";
- using ValueType = Generic;
- using IdBase::IdBase;
- };
- // The ID of a specific, which is the result of specifying the generic arguments
- // for a generic.
- struct SpecificId : public IdBase<SpecificId> {
- using DiagnosticType = Diagnostics::TypeInfo<std::string>;
- static constexpr llvm::StringLiteral Label = "specific";
- using ValueType = Specific;
- using IdBase::IdBase;
- };
- // The ID of a SpecificInterface, which is an interface and a specific pair.
- struct SpecificInterfaceId : public IdBase<SpecificInterfaceId> {
- static constexpr llvm::StringLiteral Label = "specific_interface";
- using ValueType = SpecificInterface;
- using IdBase::IdBase;
- };
- // The index of an instruction that depends on generic parameters within a
- // region of a generic. A corresponding specific version of the instruction can
- // be found in each specific corresponding to that generic. This is a pair of a
- // region and an index, stored in 32 bits.
- struct GenericInstIndex : public IndexBase<GenericInstIndex> {
- // Where the value is first used within the generic.
- enum Region : uint8_t {
- // In the declaration.
- Declaration,
- // In the definition.
- Definition,
- };
- // An index with no value.
- static const GenericInstIndex None;
- explicit constexpr GenericInstIndex(Region region, int32_t index)
- : IndexBase(region == Declaration ? index
- : FirstDefinitionIndex - index) {
- CARBON_CHECK(index >= 0);
- }
- // Returns the index of the instruction within the region.
- auto index() const -> int32_t {
- CARBON_CHECK(has_value());
- return IndexBase::index >= 0 ? IndexBase::index
- : FirstDefinitionIndex - IndexBase::index;
- }
- // Returns the region within which this instruction was first used.
- auto region() const -> Region {
- CARBON_CHECK(has_value());
- return IndexBase::index >= 0 ? Declaration : Definition;
- }
- auto Print(llvm::raw_ostream& out) const -> void;
- private:
- static constexpr auto MakeNone() -> GenericInstIndex {
- GenericInstIndex result(Declaration, 0);
- result.IndexBase::index = NoneIndex;
- return result;
- }
- static constexpr int32_t FirstDefinitionIndex = NoneIndex - 1;
- };
- constexpr GenericInstIndex GenericInstIndex::None =
- GenericInstIndex::MakeNone();
- struct ImportCppId : public IdBase<ImportCppId> {
- static constexpr llvm::StringLiteral Label = "import_cpp";
- using ValueType = ImportCpp;
- using IdBase::IdBase;
- };
- // The ID of an IR within the set of imported IRs, both direct and indirect.
- struct ImportIRId : public IdBase<ImportIRId> {
- static constexpr llvm::StringLiteral Label = "ir";
- using ValueType = ImportIR;
- // The implicit `api` import, for an `impl` file. A null entry is added if
- // there is none, as in an `api`, in which case this ID should not show up in
- // instructions.
- static const ImportIRId ApiForImpl;
- // The `Cpp` import. A null entry is added if there is none, in which case
- // this ID should not show up in instructions.
- static const ImportIRId Cpp;
- using IdBase::IdBase;
- };
- constexpr ImportIRId ImportIRId::ApiForImpl = ImportIRId(0);
- constexpr ImportIRId ImportIRId::Cpp = ImportIRId(ApiForImpl.index + 1);
- // A boolean value.
- struct BoolValue : public IdBase<BoolValue> {
- // Not used by `Print`, but for `IdKind`.
- static constexpr llvm::StringLiteral Label = "bool";
- static const BoolValue False;
- static const BoolValue True;
- // Returns the `BoolValue` corresponding to `b`.
- static constexpr auto From(bool b) -> BoolValue { return b ? True : False; }
- // Returns the `bool` corresponding to this `BoolValue`.
- constexpr auto ToBool() -> bool {
- CARBON_CHECK(*this == False || *this == True, "Invalid bool value {0}",
- index);
- return *this != False;
- }
- using IdBase::IdBase;
- auto Print(llvm::raw_ostream& out) const -> void;
- };
- constexpr BoolValue BoolValue::False = BoolValue(0);
- constexpr BoolValue BoolValue::True = BoolValue(1);
- // An integer kind value -- either "signed" or "unsigned".
- //
- // This might eventually capture any other properties of an integer type that
- // affect its semantics, such as overflow behavior.
- struct IntKind : public IdBase<IntKind> {
- // Not used by `Print`, but for `IdKind`.
- static constexpr llvm::StringLiteral Label = "int_kind";
- static const IntKind Unsigned;
- static const IntKind Signed;
- using IdBase::IdBase;
- // Returns whether this type is signed.
- constexpr auto is_signed() -> bool { return *this == Signed; }
- auto Print(llvm::raw_ostream& out) const -> void;
- };
- constexpr IntKind IntKind::Unsigned = IntKind(0);
- constexpr IntKind IntKind::Signed = IntKind(1);
- // A float kind value.
- struct FloatKind : public IdBase<FloatKind> {
- // Not used by `Print`, but for `IdKind`.
- static constexpr llvm::StringLiteral Label = "float_kind";
- using IdBase::IdBase;
- auto Print(llvm::raw_ostream& out) const -> void { out << "float"; }
- };
- // An X-macro for special names. Uses should look like:
- //
- // #define CARBON_SPECIAL_NAME_ID_FOR_XYZ(Name) ...
- // CARBON_SPECIAL_NAME_ID(CARBON_SPECIAL_NAME_ID_FOR_XYZ)
- // #undef CARBON_SPECIAL_NAME_ID_FOR_XYZ
- #define CARBON_SPECIAL_NAME_ID(X) \
- /* The name of `base`. */ \
- X(Base) \
- /* The name of the discriminant field (if any) in a choice. */ \
- X(ChoiceDiscriminant) \
- /* The name of the package `Core`. */ \
- X(Core) \
- /* The name of `destroy`. */ \
- X(Destroy) \
- /* The name of `package`. */ \
- X(PackageNamespace) \
- /* The name of `.Self`. */ \
- X(PeriodSelf) \
- /* The name of the return slot in a function. */ \
- X(ReturnSlot) \
- /* The name of `Self`. */ \
- X(SelfType) \
- /* The name of `self`. */ \
- X(SelfValue) \
- /* The name of `_`. */ \
- X(Underscore) \
- /* The name of `vptr`. */ \
- X(Vptr)
- // The ID of a name. A name is either a string or a special name such as
- // `self`, `Self`, or `base`.
- struct NameId : public IdBase<NameId> {
- static constexpr llvm::StringLiteral Label = "name";
- // names().GetFormatted() is used for diagnostics.
- using DiagnosticType = Diagnostics::TypeInfo<std::string>;
- // An enum of special names.
- enum class SpecialNameId : uint8_t {
- #define CARBON_SPECIAL_NAME_ID_FOR_ENUM(Name) Name,
- CARBON_SPECIAL_NAME_ID(CARBON_SPECIAL_NAME_ID_FOR_ENUM)
- #undef CARBON_SPECIAL_NAME_ID_FOR_ENUM
- };
- // For each SpecialNameId, provide a matching `NameId` instance for
- // convenience.
- #define CARBON_SPECIAL_NAME_ID_FOR_DECL(Name) static const NameId Name;
- CARBON_SPECIAL_NAME_ID(CARBON_SPECIAL_NAME_ID_FOR_DECL)
- #undef CARBON_SPECIAL_NAME_ID_FOR_DECL
- // The number of non-index (<0) that exist, and will need storage in name
- // lookup.
- static const int NonIndexValueCount;
- // Returns the NameId corresponding to a particular IdentifierId.
- static auto ForIdentifier(IdentifierId id) -> NameId;
- // Returns the NameId corresponding to a particular PackageNameId. This is the
- // name that is declared when the package is imported.
- static auto ForPackageName(PackageNameId id) -> NameId;
- using IdBase::IdBase;
- // Returns the IdentifierId corresponding to this NameId, or `None` if this is
- // a special name.
- auto AsIdentifierId() const -> IdentifierId {
- return index >= 0 ? IdentifierId(index) : IdentifierId::None;
- }
- // Expose special names for `switch`.
- constexpr auto AsSpecialNameId() const -> std::optional<SpecialNameId> {
- if (index >= NoneIndex) {
- return std::nullopt;
- }
- return static_cast<SpecialNameId>(NoneIndex - 1 - index);
- }
- auto Print(llvm::raw_ostream& out) const -> void;
- };
- // Define the special `static const NameId` values.
- #define CARBON_SPECIAL_NAME_ID_FOR_DEF(Name) \
- constexpr NameId NameId::Name = \
- NameId(NoneIndex - 1 - static_cast<int>(NameId::SpecialNameId::Name));
- CARBON_SPECIAL_NAME_ID(CARBON_SPECIAL_NAME_ID_FOR_DEF)
- #undef CARBON_SPECIAL_NAME_ID_FOR_DEF
- // Count non-index values, including `None` and special names.
- #define CARBON_SPECIAL_NAME_ID_FOR_COUNT(...) +1
- constexpr int NameId::NonIndexValueCount =
- 1 CARBON_SPECIAL_NAME_ID(CARBON_SPECIAL_NAME_ID_FOR_COUNT);
- #undef CARBON_SPECIAL_NAME_ID_FOR_COUNT
- // The ID of a name scope.
- struct NameScopeId : public IdBase<NameScopeId> {
- static constexpr llvm::StringLiteral Label = "name_scope";
- using ValueType = NameScope;
- // The package (or file) name scope, guaranteed to be the first added.
- static const NameScopeId Package;
- using IdBase::IdBase;
- };
- constexpr NameScopeId NameScopeId::Package = NameScopeId(0);
- // The ID of an instruction block.
- struct InstBlockId : public IdBase<InstBlockId> {
- static constexpr llvm::StringLiteral Label = "inst_block";
- // Types for BlockValueStore<InstBlockId>.
- using ElementType = InstId;
- using ValueType = llvm::MutableArrayRef<ElementType>;
- // The canonical empty block, reused to avoid allocating empty vectors. Always
- // the 0-index block.
- static const InstBlockId Empty;
- // Exported instructions. Empty until the File is fully checked; intermediate
- // state is in the Check::Context.
- static const InstBlockId Exports;
- // ImportRef instructions. Empty until the File is fully checked; intermediate
- // state is in the Check::Context.
- static const InstBlockId ImportRefs;
- // Global declaration initialization instructions. Empty if none are present.
- // Otherwise, __global_init function will be generated and this block will
- // be inserted into it.
- static const InstBlockId GlobalInit;
- // An ID for unreachable code.
- static const InstBlockId Unreachable;
- using IdBase::IdBase;
- auto Print(llvm::raw_ostream& out) const -> void;
- };
- constexpr InstBlockId InstBlockId::Empty = InstBlockId(0);
- constexpr InstBlockId InstBlockId::Exports = InstBlockId(1);
- constexpr InstBlockId InstBlockId::ImportRefs = InstBlockId(2);
- constexpr InstBlockId InstBlockId::GlobalInit = InstBlockId(3);
- constexpr InstBlockId InstBlockId::Unreachable = InstBlockId(NoneIndex - 1);
- // Contains either an `InstBlockId` value, an error value, or
- // `InstBlockId::None`.
- //
- // Error values are treated as values, though they are not representable as an
- // `InstBlockId` (unlike for the singleton error `InstId`).
- class InstBlockIdOrError {
- public:
- // NOLINTNEXTLINE(google-explicit-constructor)
- InstBlockIdOrError(InstBlockId inst_block_id)
- : InstBlockIdOrError(inst_block_id, false) {}
- static auto MakeError() -> InstBlockIdOrError {
- return {InstBlockId::None, true};
- }
- // Returns whether this class contains either an InstBlockId (other than
- // `None`) or an error.
- //
- // An error is treated as a value (as same for the singleton error `InstId`),
- // but it can not actually be materialized as an error value outside of this
- // class.
- auto has_value() const -> bool {
- return has_error_value() || inst_block_id_.has_value();
- }
- // Returns whether this class contains an error value.
- auto has_error_value() const -> bool { return error_; }
- // Returns the id of a non-empty inst block, or `None` if `has_value()` is
- // false.
- //
- // Only valid to call if `has_error_value()` is false.
- auto inst_block_id() const -> InstBlockId {
- CARBON_CHECK(!has_error_value());
- return inst_block_id_;
- }
- private:
- InstBlockIdOrError(InstBlockId inst_block_id, bool error)
- : inst_block_id_(inst_block_id), error_(error) {}
- InstBlockId inst_block_id_;
- bool error_;
- };
- // An ID of an instruction block that is referenced absolutely by an
- // instruction. This should only be used as the type of a field within a typed
- // instruction class. See AbsoluteInstId.
- class AbsoluteInstBlockId : public InstBlockId {
- public:
- // Support implicit conversion from InstBlockId so that InstBlockId and
- // AbsoluteInstBlockId have the same interface.
- // NOLINTNEXTLINE(google-explicit-constructor)
- constexpr AbsoluteInstBlockId(InstBlockId inst_block_id)
- : InstBlockId(inst_block_id) {}
- using InstBlockId::InstBlockId;
- };
- // An ID of an instruction block that is used as the declaration block within a
- // declaration instruction. This is a block that is nested within the
- // instruction, but doesn't contribute to its value. Such blocks are not
- // included in the fingerprint of the declaration. This should only be used as
- // the type of a field within a typed instruction class.
- class DeclInstBlockId : public InstBlockId {
- public:
- // Support implicit conversion from InstBlockId so that InstBlockId and
- // DeclInstBlockId have the same interface.
- // NOLINTNEXTLINE(google-explicit-constructor)
- constexpr DeclInstBlockId(InstBlockId inst_block_id)
- : InstBlockId(inst_block_id) {}
- using InstBlockId::InstBlockId;
- };
- // An ID of an instruction block that is used as a label in a branch instruction
- // or similar. This is a block that is not nested within the instruction, but
- // instead exists elsewhere in the enclosing executable region. This should
- // only be used as the type of a field within a typed instruction class.
- class LabelId : public InstBlockId {
- public:
- // Support implicit conversion from InstBlockId so that InstBlockId and
- // LabelId have the same interface.
- // NOLINTNEXTLINE(google-explicit-constructor)
- constexpr LabelId(InstBlockId inst_block_id) : InstBlockId(inst_block_id) {}
- using InstBlockId::InstBlockId;
- };
- // TODO: Move this out of sem_ir and into check, if we don't wind up using it
- // in the SemIR for expression patterns.
- struct ExprRegionId : public IdBase<ExprRegionId> {
- static constexpr llvm::StringLiteral Label = "region";
- using ValueType = ExprRegion;
- using IdBase::IdBase;
- };
- // The ID of a struct type field block.
- struct StructTypeFieldsId : public IdBase<StructTypeFieldsId> {
- static constexpr llvm::StringLiteral Label = "struct_type_fields";
- // Types for BlockValueStore<StructTypeFieldsId>.
- using ElementType = StructTypeField;
- using ValueType = llvm::MutableArrayRef<StructTypeField>;
- // The canonical empty block, reused to avoid allocating empty vectors. Always
- // the 0-index block.
- static const StructTypeFieldsId Empty;
- using IdBase::IdBase;
- };
- constexpr StructTypeFieldsId StructTypeFieldsId::Empty = StructTypeFieldsId(0);
- // The ID of a type.
- struct TypeId : public IdBase<TypeId> {
- static constexpr llvm::StringLiteral Label = "type";
- // `StringifyConstantInst` is used for diagnostics. However, where possible,
- // an `InstId` describing how the type was written should be preferred, using
- // `InstIdAsType` or `TypeOfInstId` as the diagnostic argument type.
- using DiagnosticType = Diagnostics::TypeInfo<std::string>;
- using IdBase::IdBase;
- // Returns the ID of the type corresponding to the constant `const_id`, which
- // must be of type `type`. As an exception, the type `Error` is of type
- // `Error`.
- static constexpr auto ForTypeConstant(ConstantId const_id) -> TypeId {
- return TypeId(const_id.index);
- }
- // Returns the constant ID that defines the type.
- auto AsConstantId() const -> ConstantId { return ConstantId(index); }
- // Returns whether this represents a symbolic type. Requires has_value.
- auto is_symbolic() const -> bool { return AsConstantId().is_symbolic(); }
- // Returns whether this represents a concrete type. Requires has_value.
- auto is_concrete() const -> bool { return AsConstantId().is_concrete(); }
- auto Print(llvm::raw_ostream& out) const -> void;
- };
- // The ID of a Clang Source Location.
- struct ClangSourceLocId : public IdBase<ClangSourceLocId> {
- static constexpr llvm::StringLiteral Label = "clang_source_loc";
- using ValueType = clang::SourceLocation;
- using IdBase::IdBase;
- };
- // An index for element access, for structs, tuples, and classes.
- struct ElementIndex : public IndexBase<ElementIndex> {
- static constexpr llvm::StringLiteral Label = "element";
- using IndexBase::IndexBase;
- };
- // The ID of a library name. This is either a string literal or `default`.
- struct LibraryNameId : public IdBase<LibraryNameId> {
- static constexpr llvm::StringLiteral Label = "library_name";
- using DiagnosticType = Diagnostics::TypeInfo<std::string>;
- // The name of `default`.
- static const LibraryNameId Default;
- // Track cases where the library name was set, but has been diagnosed and
- // shouldn't be used anymore.
- static const LibraryNameId Error;
- // Returns the LibraryNameId for a library name as a string literal.
- static auto ForStringLiteralValueId(StringLiteralValueId id) -> LibraryNameId;
- using IdBase::IdBase;
- // Converts a LibraryNameId back to a string literal.
- auto AsStringLiteralValueId() const -> StringLiteralValueId {
- CARBON_CHECK(index >= NoneIndex, "{0} must be handled directly", *this);
- return StringLiteralValueId(index);
- }
- auto Print(llvm::raw_ostream& out) const -> void;
- };
- constexpr LibraryNameId LibraryNameId::Default = LibraryNameId(NoneIndex - 1);
- constexpr LibraryNameId LibraryNameId::Error = LibraryNameId(NoneIndex - 2);
- // The ID of an ImportIRInst.
- struct ImportIRInstId : public IdBase<ImportIRInstId> {
- static constexpr llvm::StringLiteral Label = "import_ir_inst";
- using ValueType = ImportIRInst;
- // ImportIRInstId is restricted so that it can fit into LocId.
- static constexpr int32_t BitsWithNodeId = 29;
- // The maximum ID, non-inclusive.
- static constexpr int Max = (1 << BitsWithNodeId) - Parse::NodeId::Max - 2;
- constexpr explicit ImportIRInstId(int32_t index) : IdBase(index) {
- CARBON_DCHECK(index < Max, "Index out of range: {0}", index);
- }
- };
- // A SemIR location used as the location of instructions. This contains either a
- // InstId, NodeId, ImportIRInstId, or None. The intent is that any of these can
- // indicate the source of an instruction, and also be used to associate a line
- // in diagnostics.
- //
- // The structure is:
- // - None: The standard NoneIndex for all Id types, -1.
- // - InstId: positive values including zero; a full 31 bits.
- // - [0, 1 << 31)
- // - NodeId: negative values starting after None; the 24 bit NodeId range.
- // - [-2, -2 - (1 << 24))
- // - ImportIRInstId: remaining negative values; after NodeId, fills out negative
- // values to 29 bits.
- // - [-2 - (1 << 24), -(1 << 29))
- //
- // In addition, two bits are used for flags: `ImplicitBit` and `TokenOnlyBit`.
- // Note that these can only be used with negative, non-`InstId` values.
- //
- // Use `InstStore::GetCanonicalLocId()` to get a canonical `LocId` which will
- // not be backed by an `InstId`. Note that the canonical `LocId` may be `None`
- // even when the original `LocId` was not, so this operation needs to be done
- // before checking `has_value()`. Only canonical locations can be converted with
- // `ToImplicit()` or `ToTokenOnly()`.
- struct LocId : public IdBase<LocId> {
- // The contained index kind.
- enum class Kind {
- None,
- ImportIRInstId,
- InstId,
- NodeId,
- };
- static constexpr llvm::StringLiteral Label = "loc";
- using IdBase::IdBase;
- // NOLINTNEXTLINE(google-explicit-constructor)
- constexpr LocId(ImportIRInstId import_ir_inst_id)
- : IdBase(import_ir_inst_id.has_value()
- ? FirstImportIRInstId - import_ir_inst_id.index
- : NoneIndex) {}
- explicit constexpr LocId(InstId inst_id) : IdBase(inst_id.index) {}
- // NOLINTNEXTLINE(google-explicit-constructor)
- constexpr LocId(Parse::NoneNodeId /*none*/) : IdBase(NoneIndex) {}
- // NOLINTNEXTLINE(google-explicit-constructor)
- constexpr LocId(Parse::NodeId node_id)
- : IdBase(FirstNodeId - node_id.index) {}
- // Forms an equivalent LocId for a desugared location. Requires a
- // canonical location. See `InstStore::GetCanonicalLocId()`.
- //
- // TODO: Rename to something like `ToDesugared`.
- auto ToImplicit() const -> LocId {
- // This should only be called for NodeId or ImportIRInstId (i.e. canonical
- // locations), but we only set the flag for NodeId.
- CARBON_CHECK(kind() != Kind::InstId);
- if (kind() == Kind::NodeId) {
- return LocId(index & ~ImplicitBit);
- }
- return *this;
- }
- // Forms an equivalent `LocId` for a token-only diagnostic location. Requires
- // a canonical location. See `InstStore::GetCanonicalLocId()`.
- //
- // TODO: Consider making this a part of check/ diagnostics instead, as a free
- // function operation on `LocIdForDiagnostics`?
- // https://github.com/carbon-language/carbon-lang/pull/5355#discussion_r2064113186
- auto ToTokenOnly() const -> LocId {
- CARBON_CHECK(kind() != Kind::InstId);
- if (has_value()) {
- return LocId(index & ~TokenOnlyBit);
- }
- return *this;
- }
- // Returns the kind of the `LocId`.
- auto kind() const -> Kind {
- if (!has_value()) {
- return Kind::None;
- }
- if (index >= 0) {
- return Kind::InstId;
- }
- if (index_without_flags() <= FirstImportIRInstId) {
- return Kind::ImportIRInstId;
- }
- return Kind::NodeId;
- }
- // Returns true if the location corresponds to desugared instructions.
- // Requires a non-`InstId` location.
- auto is_implicit() const -> bool {
- return (kind() == Kind::NodeId) && (index & ImplicitBit) == 0;
- }
- // Returns true if the location is token-only for diagnostics.
- //
- // This means the displayed location will include only the location's specific
- // parse node, instead of also including its descendants. As such, this can
- // only be true for locations backed by a `NodeId`.
- auto is_token_only() const -> bool {
- return kind() != Kind::InstId && (index & TokenOnlyBit) == 0;
- }
- // Returns the equivalent `ImportIRInstId` when `kind()` matches or is `None`.
- auto import_ir_inst_id() const -> ImportIRInstId {
- if (!has_value()) {
- return ImportIRInstId::None;
- }
- CARBON_CHECK(kind() == Kind::ImportIRInstId, "{0}", index);
- return ImportIRInstId(FirstImportIRInstId - index_without_flags());
- }
- // Returns the equivalent `InstId` when `kind()` matches or is `None`.
- auto inst_id() const -> InstId {
- CARBON_CHECK(kind() == Kind::None || kind() == Kind::InstId, "{0}", index);
- return InstId(index);
- }
- // Returns the equivalent `NodeId` when `kind()` matches or is `None`.
- auto node_id() const -> Parse::NodeId {
- if (!has_value()) {
- return Parse::NodeId::None;
- }
- CARBON_CHECK(kind() == Kind::NodeId, "{0}", index);
- return Parse::NodeId(FirstNodeId - index_without_flags());
- }
- auto Print(llvm::raw_ostream& out) const -> void;
- private:
- // Whether a location corresponds to desugared instructions. This only applies
- // for `NodeId`.
- static constexpr int32_t ImplicitBit = 1 << 30;
- // See `is_token_only` for the use. This only applies for canonical locations
- // (i.e. those containing `NodeId` or `ImportIRInstId`).
- static constexpr int32_t TokenOnlyBit = 1 << 29;
- // The value of the 0 index for each of `NodeId` and `ImportIRInstId`.
- static constexpr int32_t FirstNodeId = NoneIndex - 1;
- static constexpr int32_t FirstImportIRInstId =
- FirstNodeId - Parse::NodeId::Max;
- auto index_without_flags() const -> int32_t {
- CARBON_DCHECK(index < NoneIndex, "Only for NodeId and ImportIRInstId");
- return index | ImplicitBit | TokenOnlyBit;
- }
- };
- // Polymorphic id for fields in `Any[...]` typed instruction category. Used for
- // fields where the specific instruction structs have different field types in
- // that position or do not have a field in that position at all. Allows
- // conversion with `Inst::As<>` from the specific typed instruction to the
- // `Any[...]` instruction category.
- //
- // This type participates in `Inst::FromRaw` in order to convert from specific
- // instructions to an `Any[...]` instruction category:
- // - In the case the specific instruction has a field of some `IdKind` in the
- // same position, the `Any[...]` type will hold its raw value in the
- // `AnyRawId` field.
- // - In the case the specific instruction has no field in the same position, the
- // `Any[...]` type will hold a default constructed `AnyRawId` with a `None`
- // value.
- struct AnyRawId : public AnyIdBase {
- // For IdKind.
- static constexpr llvm::StringLiteral Label = "any_raw";
- constexpr explicit AnyRawId() : AnyIdBase(AnyIdBase::NoneIndex) {}
- constexpr explicit AnyRawId(int32_t id) : AnyIdBase(id) {}
- };
- // A pair of an interface and a specific for that interface.
- struct SpecificInterface {
- InterfaceId interface_id;
- SpecificId specific_id;
- static const SpecificInterface None;
- friend auto operator==(const SpecificInterface& lhs,
- const SpecificInterface& rhs) -> bool = default;
- };
- constexpr SpecificInterface SpecificInterface::None = {
- .interface_id = InterfaceId::None, .specific_id = SpecificId::None};
- } // namespace Carbon::SemIR
- #endif // CARBON_TOOLCHAIN_SEM_IR_IDS_H_
|