blob: e5406546b1950734d4679850ac801aeeec7f1757 [file] [log] [blame]
//===-- EnumAttr.td - Enum attributes ----------------------*- tablegen -*-===//
//
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
// See https://llvm.org/LICENSE.txt for license information.
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
//
//===----------------------------------------------------------------------===//
#ifndef ENUMATTR_TD
#define ENUMATTR_TD
include "mlir/IR/AttrTypeBase.td"
//===----------------------------------------------------------------------===//
// Enum attribute kinds
// Additional information for an enum case.
class EnumCase<string sym, int intVal, string strVal, int widthVal> {
// The C++ enumerant symbol.
string symbol = sym;
// The C++ enumerant value.
// If less than zero, there will be no explicit discriminator values assigned
// to enumerators in the generated enum class.
int value = intVal;
// The string representation of the enumerant. May be the same as symbol.
string str = strVal;
// The bitwidth of the enum.
int width = widthVal;
}
// An enum attribute case stored with IntegerAttr, which has an integer value,
// its representation as a string and a C++ symbol name which may be different.
// Not needed when using the newer `EnumCase` form for defining enum cases.
class IntEnumAttrCaseBase<I intType, string sym, string strVal, int intVal> :
EnumCase<sym, intVal, strVal, intType.bitwidth>,
SignlessIntegerAttrBase<intType, "case " # strVal> {
let predicate =
CPred<"::llvm::cast<::mlir::IntegerAttr>($_self).getInt() == " # intVal>;
}
// Cases of integer enums with a specific type. By default, the string
// representation is the same as the C++ symbol name.
class I32EnumCase<string sym, int val, string str = sym>
: EnumCase<sym, val, str, 32>;
class I64EnumCase<string sym, int val, string str = sym>
: EnumCase<sym, val, str, 64>;
// Cases of integer enum attributes with a specific type. By default, the string
// representation is the same as the C++ symbol name. These forms
// are not needed when using the newer `EnumCase` form.
class I32EnumAttrCase<string sym, int val, string str = sym>
: IntEnumAttrCaseBase<I32, sym, str, val>;
class I64EnumAttrCase<string sym, int val, string str = sym>
: IntEnumAttrCaseBase<I64, sym, str, val>;
// A bit enum case. `val` here is *not* the ordinal number of a bit
// that is set. It is an integer value with bits set to match the case.
class BitEnumCaseBase<string sym, int val, string str, int width> :
EnumCase<sym, val, str, width>;
// Bit enum attr cases. The string representation is the same as the C++ symbol
// name unless otherwise specified.
class I8BitEnumCase<string sym, int val, string str = sym>
: BitEnumCaseBase<sym, val, str, 8>;
class I16BitEnumCase<string sym, int val, string str = sym>
: BitEnumCaseBase<sym, val, str, 16>;
class I32BitEnumCase<string sym, int val, string str = sym>
: BitEnumCaseBase<sym, val, str, 32>;
class I64BitEnumCase<string sym, int val, string str = sym>
: BitEnumCaseBase<sym, val, str, 64>;
// A form of `BitEnumCaseBase` that also inherits from `Attr` and encodes
// the width of the enum, which was defined when enums were always
// stored in attributes.
class BitEnumAttrCaseBase<I intType, string sym, int val, string str = sym> :
BitEnumCaseBase<sym, val, str, intType.bitwidth>,
SignlessIntegerAttrBase<intType, "case " #str>;
class I8BitEnumAttrCase<string sym, int val, string str = sym>
: BitEnumAttrCaseBase<I8, sym, val, str>;
class I16BitEnumAttrCase<string sym, int val, string str = sym>
: BitEnumAttrCaseBase<I16, sym, val, str>;
class I32BitEnumAttrCase<string sym, int val, string str = sym>
: BitEnumAttrCaseBase<I32, sym, val, str>;
class I64BitEnumAttrCase<string sym, int val, string str = sym>
: BitEnumAttrCaseBase<I64, sym, val, str>;
// The special bit enum case with no bits set (i.e. value = 0).
class BitEnumCaseNone<string sym, string str, int width>
: BitEnumCaseBase<sym, 0, str, width>;
class I8BitEnumCaseNone<string sym, string str = sym>
: BitEnumCaseNone<sym, str, 8>;
class I16BitEnumCaseNone<string sym, string str = sym>
: BitEnumCaseNone<sym, str, 16>;
class I32BitEnumCaseNone<string sym, string str = sym>
: BitEnumCaseNone<sym, str, 32>;
class I64BitEnumCaseNone<string sym, string str = sym>
: BitEnumCaseNone<sym, str, 64>;
// Older forms, used when enums were necessarily attributes.
class I8BitEnumAttrCaseNone<string sym, string str = sym>
: I8BitEnumAttrCase<sym, 0, str>;
class I16BitEnumAttrCaseNone<string sym, string str = sym>
: I16BitEnumAttrCase<sym, 0, str>;
class I32BitEnumAttrCaseNone<string sym, string str = sym>
: I32BitEnumAttrCase<sym, 0, str>;
class I64BitEnumAttrCaseNone<string sym, string str = sym>
: I64BitEnumAttrCase<sym, 0, str>;
// A bit enum case for a single bit, specified by a bit position `pos`.
// The `pos` argument refers to the index of the bit, and is limited
// to be in the range [0, width).
class BitEnumCaseBit<string sym, int pos, string str, int width>
: BitEnumCaseBase<sym, !shl(1, pos), str, width> {
assert !and(!ge(pos, 0), !lt(pos, width)),
"bit position larger than underlying storage";
}
class I8BitEnumCaseBit<string sym, int pos, string str = sym>
: BitEnumCaseBit<sym, pos, str, 8>;
class I16BitEnumCaseBit<string sym, int pos, string str = sym>
: BitEnumCaseBit<sym, pos, str, 16>;
class I32BitEnumCaseBit<string sym, int pos, string str = sym>
: BitEnumCaseBit<sym, pos, str, 32>;
class I64BitEnumCaseBit<string sym, int pos, string str = sym>
: BitEnumCaseBit<sym, pos, str, 64>;
// A bit enum case for a single bit, specified by a bit position.
// The pos argument refers to the index of the bit, and is limited
// to be in the range [0, bitwidth).
class BitEnumAttrCaseBit<I intType, string sym, int pos, string str = sym>
: BitEnumAttrCaseBase<intType, sym, !shl(1, pos), str> {
assert !and(!ge(pos, 0), !lt(pos, intType.bitwidth)),
"bit position larger than underlying storage";
}
class I8BitEnumAttrCaseBit<string sym, int pos, string str = sym>
: BitEnumAttrCaseBit<I8, sym, pos, str>;
class I16BitEnumAttrCaseBit<string sym, int pos, string str = sym>
: BitEnumAttrCaseBit<I16, sym, pos, str>;
class I32BitEnumAttrCaseBit<string sym, int pos, string str = sym>
: BitEnumAttrCaseBit<I32, sym, pos, str>;
class I64BitEnumAttrCaseBit<string sym, int pos, string str = sym>
: BitEnumAttrCaseBit<I64, sym, pos, str>;
// A bit enum case for a group/list of previously declared cases, providing
// a convenient alias for that group.
class BitEnumCaseGroup<string sym, list<BitEnumCaseBase> cases, string str = sym>
: BitEnumCaseBase<sym,
!foldl(0, cases, value, bitcase, !or(value, bitcase.value)),
str, !head(cases).width>;
// The attribute-only form of `BitEnumCaseGroup`.
class BitEnumAttrCaseGroup<I intType, string sym,
list<BitEnumCaseBase> cases, string str = sym>
: BitEnumAttrCaseBase<intType, sym,
!foldl(0, cases, value, bitcase, !or(value, bitcase.value)),
str>;
class I8BitEnumAttrCaseGroup<string sym, list<BitEnumAttrCaseBase> cases,
string str = sym>
: BitEnumAttrCaseGroup<I8, sym, cases, str>;
class I16BitEnumAttrCaseGroup<string sym, list<BitEnumAttrCaseBase> cases,
string str = sym>
: BitEnumAttrCaseGroup<I16, sym, cases, str>;
class I32BitEnumAttrCaseGroup<string sym, list<BitEnumAttrCaseBase> cases,
string str = sym>
: BitEnumAttrCaseGroup<I32, sym, cases, str>;
class I64BitEnumAttrCaseGroup<string sym, list<BitEnumAttrCaseBase> cases,
string str = sym>
: BitEnumAttrCaseGroup<I64, sym, cases, str>;
// Information describing an enum and the functions that should be generated for it.
class EnumInfo<string name, string summaryValue, list<EnumCase> cases, int width> {
string summary = summaryValue;
// Generate a description of this enums members for the MLIR docs.
string description =
"Enum cases:\n" # !interleave(
!foreach(case, cases,
"* " # case.str # " (`" # case.symbol # "`)"), "\n");
// The C++ namespace for this enum
string cppNamespace = "";
// The C++ enum class name
string className = name;
// C++ type wrapped by attribute
string cppType = cppNamespace # "::" # className;
// List of all accepted cases
list<EnumCase> enumerants = cases;
// The following fields are only used by the EnumsGen backend to generate
// an enum class definition and conversion utility functions.
// The bitwidth underlying the class
int bitwidth = width;
// The underlying type for the C++ enum class. An empty string mean the
// underlying type is not explicitly specified.
string underlyingType = "uint" # width # "_t";
// The name of the utility function that converts a value of the underlying
// type to the corresponding symbol. It will have the following signature:
//
// ```c++
// std::optional<<qualified-enum-class-name>> <fn-name>(<underlying-type>);
// ```
string underlyingToSymbolFnName = "symbolize" # name;
// The name of the utility function that converts a string to the
// corresponding symbol. It will have the following signature:
//
// ```c++
// std::optional<<qualified-enum-class-name>> <fn-name>(llvm::StringRef);
// ```
string stringToSymbolFnName = "symbolize" # name;
// The name of the utility function that converts a symbol to the
// corresponding string. It will have the following signature:
//
// ```c++
// <return-type> <fn-name>(<qualified-enum-class-name>);
// ```
string symbolToStringFnName = "stringify" # name;
string symbolToStringFnRetType = "::llvm::StringRef";
// The name of the utility function that returns the max enum value used
// within the enum class. It will have the following signature:
//
// ```c++
// static constexpr unsigned <fn-name>();
// ```
string maxEnumValFnName = "getMaxEnumValFor" # name;
}
// A wrapper around `EnumInfo` that also makes the Enum an attribute
// if `genSeecializedAttr` is 1 (though `EnumAttr` is the preferred means
// to accomplish this) or declares that the enum will be stored in an attribute.
class EnumAttrInfo<
string name, list<EnumCase> cases, SignlessIntegerAttrBase baseClass> :
EnumInfo<name, baseClass.summary, cases, !cast<I>(baseClass.valueType).bitwidth>,
Attr<baseClass.predicate, baseClass.summary> {
// Generate specialized Attribute class
bit genSpecializedAttr = 1;
// The underlying Attribute class, which holds the enum value
Attr baseAttrClass = baseClass;
// The name of specialized Enum Attribute class
string specializedAttrClassName = name # Attr;
// Override Attr class fields for specialized class
let predicate = !if(genSpecializedAttr,
CPred<"::llvm::isa<" # cppNamespace # "::" # specializedAttrClassName # ">($_self)">,
baseAttrClass.predicate);
let storageType = !if(genSpecializedAttr,
cppNamespace # "::" # specializedAttrClassName,
baseAttrClass.storageType);
let returnType = !if(genSpecializedAttr,
cppNamespace # "::" # className,
baseAttrClass.returnType);
let constBuilderCall = !if(genSpecializedAttr,
cppNamespace # "::" # specializedAttrClassName # "::get($_builder.getContext(), $0)",
baseAttrClass.constBuilderCall);
let valueType = baseAttrClass.valueType;
// Parser and printer code used by the EnumParameter class, to be provided by
// derived classes
string parameterParser = ?;
string parameterPrinter = ?;
}
// An attribute holding a single integer value.
class IntEnum<string name, string summary, list<EnumCase> cases, int width>
: EnumInfo<name,
!if(!empty(summary), "allowed i" # width # " cases: " #
!interleave(!foreach(case, cases, case.value), ", "),
summary),
cases, width>;
class I32Enum<string name, string summary, list<EnumCase> cases>
: IntEnum<name, summary, cases, 32>;
class I64Enum<string name, string summary, list<EnumCase> cases>
: IntEnum<name, summary, cases, 32>;
// An enum attribute backed by IntegerAttr.
//
// Op attributes of this kind are stored as IntegerAttr. Extra verification will
// be generated on the integer though: only the values of the allowed cases are
// permitted as the integer value.
class IntEnumAttrBase<I intType, list<IntEnumAttrCaseBase> cases, string summary> :
SignlessIntegerAttrBase<intType, summary> {
let predicate = And<[
SignlessIntegerAttrBase<intType, summary>.predicate,
Or<!foreach(case, cases, case.predicate)>]>;
}
class IntEnumAttr<I intType, string name, string summary,
list<IntEnumAttrCaseBase> cases> :
EnumAttrInfo<name, cases,
IntEnumAttrBase<intType, cases,
!if(!empty(summary), "allowed " # intType.summary # " cases: " #
!interleave(!foreach(case, cases, case.value), ", "),
summary)>> {
// Parse a keyword and pass it to `stringToSymbol`. Emit an error if a the
// symbol is not valid.
let parameterParser = [{[&]() -> ::mlir::FailureOr<}] # cppType # [{> {
auto loc = $_parser.getCurrentLocation();
::llvm::StringRef enumKeyword;
if (::mlir::failed($_parser.parseKeyword(&enumKeyword)))
return ::mlir::failure();
auto maybeEnum = }] # cppNamespace # "::" #
stringToSymbolFnName # [{(enumKeyword);
if (maybeEnum)
return *maybeEnum;
return {(::llvm::LogicalResult)($_parser.emitError(loc) << "expected " }] #
[{<< "}] # cppType # [{" << " to be one of: " << }] #
!interleave(!foreach(enum, enumerants, "\"" # enum.str # "\""),
[{ << ", " << }]) # [{)};
}()}];
// Print the enum by calling `symbolToString`.
let parameterPrinter = "$_printer << " # symbolToStringFnName # "($_self)";
}
class I32EnumAttr<string name, string summary, list<I32EnumAttrCase> cases> :
IntEnumAttr<I32, name, summary, cases> {
let underlyingType = "uint32_t";
}
class I64EnumAttr<string name, string summary, list<I64EnumAttrCase> cases> :
IntEnumAttr<I64, name, summary, cases> {
let underlyingType = "uint64_t";
}
// The base mixin for bit enums that are stored as an integer.
// This is used by both BitEnum and BitEnumAttr, which need to have a set of
// extra properties that bit enums have which normal enums don't. However,
// we can't just use BitEnum as a base class of BitEnumAttr, since BitEnumAttr
// also inherits from EnumAttrInfo, causing double inheritance of EnumInfo.
class BitEnumBase<list<BitEnumCaseBase> cases> {
// Determine "valid" bits from enum cases for error checking
int validBits = !foldl(0, cases, value, bitcase, !or(value, bitcase.value));
// The delimiter used to separate bit enum cases in strings. Only "|" and
// "," (along with optional spaces) are supported due to the use of the
// parseSeparatorFn in parameterParser below.
// Spaces in the separator string are used for printing, but will be optional
// for parsing.
string separator = "|";
assert !or(!ge(!find(separator, "|"), 0), !ge(!find(separator, ","), 0)),
"separator must contain '|' or ',' for parameter parsing";
// Print the "primary group" only for bits that are members of case groups
// that have all bits present. When the value is 0, printing will display both
// both individual bit case names AND the names for all groups that the bit is
// contained in. When the value is 1, for each bit that is set AND is a member
// of a group with all bits set, only the "primary group" (i.e. the first
// group with all bits set in reverse declaration order) will be printed (for
// conciseness).
bit printBitEnumPrimaryGroups = 0;
// 1 if the operator<< for this enum should put quotes around values with
// multiple entries. Off by default in the general case but on for BitEnumAttrs
// since that was the original behavior.
bit printBitEnumQuoted = 0;
}
// A bit enum stored as an integer.
//
// Enums of these kind are staored as an integer. Attributes or properties deriving
// from this enum will have additional verification generated on them to make sure
// only allowed bits are set. Helper methods are generated to parse a sring of enum
// values generated by the specified separator to a symbol and vice versa.
class BitEnum<string name, string summary, list<BitEnumCaseBase> cases, int width>
: EnumInfo<name, summary, cases, width>, BitEnumBase<cases> {
// We need to return a string because we may concatenate symbols for multiple
// bits together.
let symbolToStringFnRetType = "std::string";
}
class I8BitEnum<string name, string summary,
list<BitEnumCaseBase> cases>
: BitEnum<name, summary, cases, 8>;
class I16BitEnum<string name, string summary,
list<BitEnumCaseBase> cases>
: BitEnum<name, summary, cases, 16>;
class I32BitEnum<string name, string summary,
list<BitEnumCaseBase> cases>
: BitEnum<name, summary, cases, 32>;
class I64BitEnum<string name, string summary,
list<BitEnumCaseBase> cases>
: BitEnum<name, summary, cases, 64>;
// A bit enum stored with an IntegerAttr.
//
// Op attributes of this kind are stored as IntegerAttr. Extra verification will
// be generated on the integer to make sure only allowed bits are set. Besides,
// helper methods are generated to parse a string separated with a specified
// delimiter to a symbol and vice versa.
class BitEnumAttrBase<I intType, list<BitEnumCaseBase> cases,
string summary>
: SignlessIntegerAttrBase<intType, summary> {
let predicate = And<[
SignlessIntegerAttrBase<intType, summary>.predicate,
// Make sure we don't have unknown bit set.
CPred<"!(::llvm::cast<::mlir::IntegerAttr>($_self).getValue().getZExtValue() & (~("
# !interleave(!foreach(case, cases, case.value # "u"), "|") #
")))">
]>;
}
class BitEnumAttr<I intType, string name, string summary,
list<BitEnumCaseBase> cases>
: EnumAttrInfo<name, cases, BitEnumAttrBase<intType, cases, summary>>,
BitEnumBase<cases> {
// We need to return a string because we may concatenate symbols for multiple
// bits together.
let symbolToStringFnRetType = "std::string";
// Parsing function that corresponds to the enum separator. Only
// "," and "|" are supported by this definition.
string parseSeparatorFn = !if(!ge(!find(separator, "|"), 0),
"parseOptionalVerticalBar",
"parseOptionalComma");
// Parse a keyword and pass it to `stringToSymbol`. Emit an error if a the
// symbol is not valid.
let parameterParser = [{[&]() -> ::mlir::FailureOr<}] # cppType # [{> {
}] # cppType # [{ flags = {};
auto loc = $_parser.getCurrentLocation();
::llvm::StringRef enumKeyword;
do {
if (::mlir::failed($_parser.parseKeyword(&enumKeyword)))
return ::mlir::failure();
auto maybeEnum = }] # cppNamespace # "::" #
stringToSymbolFnName # [{(enumKeyword);
if (!maybeEnum) {
return {(::llvm::LogicalResult)($_parser.emitError(loc) << }] #
[{"expected " << "}] # cppType # [{" << " to be one of: " << }] #
!interleave(!foreach(enum, enumerants, "\"" # enum.str # "\""),
[{ << ", " << }]) # [{)};
}
flags = flags | *maybeEnum;
} while(::mlir::succeeded($_parser.}] # parseSeparatorFn # [{()));
return flags;
}()}];
// Print the enum by calling `symbolToString`.
let parameterPrinter = "$_printer << " # symbolToStringFnName # "($_self)";
// Use old-style operator<< and FieldParser for compatibility
let printBitEnumQuoted = 1;
}
class I8BitEnumAttr<string name, string summary,
list<BitEnumCaseBase> cases>
: BitEnumAttr<I8, name, summary, cases> {
let underlyingType = "uint8_t";
}
class I16BitEnumAttr<string name, string summary,
list<BitEnumCaseBase> cases>
: BitEnumAttr<I16, name, summary, cases> {
let underlyingType = "uint16_t";
}
class I32BitEnumAttr<string name, string summary,
list<BitEnumCaseBase> cases>
: BitEnumAttr<I32, name, summary, cases> {
let underlyingType = "uint32_t";
}
class I64BitEnumAttr<string name, string summary,
list<BitEnumCaseBase> cases>
: BitEnumAttr<I64, name, summary, cases> {
let underlyingType = "uint64_t";
}
// A C++ enum as an attribute parameter. The parameter implements a parser and
// printer for the enum by dispatching calls to `stringToSymbol` and
// `symbolToString`.
class EnumParameter<EnumInfo enumInfo>
: AttrParameter<enumInfo.cppNamespace # "::" # enumInfo.className,
"an enum of type " # enumInfo.className> {
let parser = !if(!isa<EnumAttrInfo>(enumInfo),
!cast<EnumAttrInfo>(enumInfo).parameterParser, ?);
let printer = !if(!isa<EnumAttrInfo>(enumInfo),
!cast<EnumAttrInfo>(enumInfo).parameterPrinter, ?);
}
// An attribute backed by a C++ enum. The attribute contains a single
// parameter `value` whose type is the C++ enum class.
//
// Example:
//
// ```
// def MyEnum : I32EnumAttr<"MyEnum", "a simple enum", [
// I32EnumAttrCase<"First", 0, "first">,
// I32EnumAttrCase<"Second", 1, "second>]> {
// let genSpecializedAttr = 0;
// }
//
// def MyEnumAttr : EnumAttr<MyDialect, MyEnum, "enum">;
// ```
//
// By default, the assembly format of the attribute works best with operation
// assembly formats. For example:
//
// ```
// def MyOp : Op<MyDialect, "my_op"> {
// let arguments = (ins MyEnumAttr:$enum);
// let assemblyFormat = "$enum attr-dict";
// }
// ```
//
// The op will appear in the IR as `my_dialect.my_op first`. However, the
// generic format of the attribute will be `#my_dialect<"enum first">`. Override
// the attribute's assembly format as required.
class EnumAttr<Dialect dialect, EnumInfo enumInfo, string name = "",
list <Trait> traits = []>
: AttrDef<dialect, enumInfo.className, traits> {
let summary = enumInfo.summary;
let description = enumInfo.description;
// The backing enumeration.
EnumInfo enum = enumInfo;
// Inherit the C++ namespace from the enum.
let cppNamespace = enumInfo.cppNamespace;
// Define a constant builder for the attribute to convert from C++ enums.
let constBuilderCall = cppNamespace # "::" # cppClassName #
"::get($_builder.getContext(), $0)";
// Op attribute getters should return the underlying C++ enum type.
let returnType = enumInfo.cppNamespace # "::" # enumInfo.className;
// Convert from attribute to the underlying C++ type in op getters.
let convertFromStorage = "$_self.getValue()";
// The enum attribute has one parameter: the C++ enum value.
let parameters = (ins EnumParameter<enumInfo>:$value);
// If a mnemonic was provided, use it to generate a custom assembly format.
let mnemonic = name;
// The default assembly format for enum attributes. Selected to best work with
// operation assembly formats.
let assemblyFormat = "$value";
}
class _symbolToValue<EnumInfo enumInfo, string case> {
defvar cases =
!filter(iter, enumInfo.enumerants, !eq(iter.str, case));
assert !not(!empty(cases)), "failed to find enum-case '" # case # "'";
// `!empty` check to not cause an error if the cases are empty.
// The assertion catches the issue later and emits a proper error message.
string value = enumInfo.cppType # "::"
# !if(!empty(cases), "", !head(cases).symbol);
}
class _bitSymbolsToValue<EnumInfo bitEnum, string case> {
assert !isa<BitEnumBase>(bitEnum), "_bitSymbolsToValue not given a bit enum";
defvar pos = !find(case, "|");
// Recursive instantiation looking up the symbol before the `|` in
// enum cases.
string value = !if(
!eq(pos, -1), /*baseCase=*/_symbolToValue<bitEnum, case>.value,
/*rec=*/_symbolToValue<bitEnum, !substr(case, 0, pos)>.value # "|"
# _bitSymbolsToValue<bitEnum, !substr(case, !add(pos, 1))>.value
);
}
class ConstantEnumCaseBase<Attr attribute,
EnumInfo enumInfo, string case>
: ConstantAttr<attribute,
!if(!isa<BitEnumBase>(enumInfo),
_bitSymbolsToValue<enumInfo, case>.value,
_symbolToValue<enumInfo, case>.value
)
>;
/// Attribute constraint matching a constant enum case. `attribute` should be
/// one of `EnumInfo` or `EnumAttr` and `symbol` the string representation
/// of an enum case. Multiple enum values of a bit-enum can be combined using
/// `|` as a separator. Note that there mustn't be any whitespace around the
/// separator.
/// This attribute constraint is additionally buildable, making it possible to
/// use it in result patterns.
///
/// Examples:
/// * ConstantEnumCase<Arith_IntegerOverflowAttr, "nsw|nuw">
/// * ConstantEnumCase<Arith_CmpIPredicateAttr, "slt">
class ConstantEnumCase<Attr attribute, string case>
: ConstantEnumCaseBase<attribute,
!if(!isa<EnumInfo>(attribute), !cast<EnumInfo>(attribute),
!cast<EnumAttr>(attribute).enum), case> {
assert !or(!isa<EnumAttr>(attribute), !isa<EnumInfo>(attribute)),
"attribute must be one of 'EnumAttr' or 'EnumInfo'";
}
#endif // ENUMATTR_TD