diff --git a/mlir/docs/OpDefinitions.md b/mlir/docs/OpDefinitions.md --- a/mlir/docs/OpDefinitions.md +++ b/mlir/docs/OpDefinitions.md @@ -567,10 +567,39 @@ let hasVerifier = 1; ``` -This will generate a `LogicalResult verify()` method declaration on the op class -that can be defined with any additional verification constraints. This method -will be invoked after the auto-generated verification code. The order of trait -verification excluding those of `hasVerifier` should not be relied upon. +or + +```tablegen +let hasRegionVerifier = 1; +``` + +This will generate either `LogicalResult verify()` or +`LogicalResult verifyRegions()` method declaration on the op class +that can be defined with any additional verification constraints. These method +will be invoked on its verification order. + +#### Verification Ordering + +The verification of an operation involves several steps, + +1. InternalOpTrait will be verified first, they can be run independently. +1. `verifyInvariants` which is constructed by ODS, it verifies the type, + attributes, .etc. +1. Other Traits/Interfaces that have marked their verifier as `verifyTrait` or + `verifyWithRegions=0`. +1. Custom verifier which is defined in the op and has marked `hasVerifier=1` + +If an operation has regions, then it may have the second phase, + +1. Traits/Interfaces that have marked their verifier as `verifyRegionTrait` or + `verifyWithRegions=1`. This implies the verifier needs to access the + operations in its regions. +1. Custom verifier which is defined in the op and has marked + `hasRegionVerifier=1` + +Note that the second phase will be run after the operations in the region are +verified. Verifiers further down the order can rely on certain invariants being +verified by a previous verifier and do not need to re-verify them. ### Declarative Assembly Format diff --git a/mlir/docs/Traits.md b/mlir/docs/Traits.md --- a/mlir/docs/Traits.md +++ b/mlir/docs/Traits.md @@ -36,9 +36,12 @@ }; ``` -Operation traits may also provide a `verifyTrait` hook, that is called when -verifying the concrete operation. The trait verifiers will currently always be -invoked before the main `Op::verify`. +Operation traits may also provide a `verifyTrait` or `verifyRegionTrait` hook +that is called when verifying the concrete operation. The difference between +these two is that whether the verifier needs to access the regions, if so, the +operations in the regions will be verified before the verification of this +trait. The [verification order](OpDefinitions.md/#verification-ordering) +determines when a verifier will be invoked. ```c++ template @@ -53,8 +56,9 @@ ``` Note: It is generally good practice to define the implementation of the -`verifyTrait` hook out-of-line as a free function when possible to avoid -instantiating the implementation for every concrete operation type. +`verifyTrait` or `verifyRegionTrait` hook out-of-line as a free function when +possible to avoid instantiating the implementation for every concrete operation +type. Operation traits may also provide a `foldTrait` hook that is called when folding the concrete operation. The trait folders will only be invoked if the concrete diff --git a/mlir/include/mlir/Dialect/Affine/IR/AffineOps.h b/mlir/include/mlir/Dialect/Affine/IR/AffineOps.h --- a/mlir/include/mlir/Dialect/Affine/IR/AffineOps.h +++ b/mlir/include/mlir/Dialect/Affine/IR/AffineOps.h @@ -76,7 +76,7 @@ class AffineDmaStartOp : public Op { + OpTrait::OpInvariants, AffineMapAccessInterface::Trait> { public: using Op::Op; static ArrayRef getAttributeNames() { return {}; } @@ -227,7 +227,8 @@ static StringRef getOperationName() { return "affine.dma_start"; } static ParseResult parse(OpAsmParser &parser, OperationState &result); void print(OpAsmPrinter &p); - LogicalResult verifyInvariants(); + LogicalResult verifyInvariantsImpl(); + LogicalResult verifyInvariants() { return verifyInvariantsImpl(); } LogicalResult fold(ArrayRef cstOperands, SmallVectorImpl &results); @@ -268,7 +269,7 @@ class AffineDmaWaitOp : public Op { + OpTrait::OpInvariants, AffineMapAccessInterface::Trait> { public: using Op::Op; static ArrayRef getAttributeNames() { return {}; } @@ -315,7 +316,8 @@ static StringRef getTagMapAttrName() { return "tag_map"; } static ParseResult parse(OpAsmParser &parser, OperationState &result); void print(OpAsmPrinter &p); - LogicalResult verifyInvariants(); + LogicalResult verifyInvariantsImpl(); + LogicalResult verifyInvariants() { return verifyInvariantsImpl(); } LogicalResult fold(ArrayRef cstOperands, SmallVectorImpl &results); }; diff --git a/mlir/include/mlir/IR/OpBase.td b/mlir/include/mlir/IR/OpBase.td --- a/mlir/include/mlir/IR/OpBase.td +++ b/mlir/include/mlir/IR/OpBase.td @@ -2023,6 +2023,10 @@ // OpTrait definitions //===----------------------------------------------------------------------===// +// This represents a trait has its C++ implementation defined in core MLIR +// framework. Note that this will affect the verification order. +class InternalOpTrait; + // These classes are used to define operation specific traits. class NativeOpTrait traits = []> : NativeTrait { @@ -2051,48 +2055,51 @@ } // Op defines an affine scope. -def AffineScope : NativeOpTrait<"AffineScope">; +def AffineScope : NativeOpTrait<"AffineScope">, InternalOpTrait; // Op defines an automatic allocation scope. -def AutomaticAllocationScope : NativeOpTrait<"AutomaticAllocationScope">; +def AutomaticAllocationScope : + NativeOpTrait<"AutomaticAllocationScope">, InternalOpTrait; // Op supports operand broadcast behavior. def ResultsBroadcastableShape : NativeOpTrait<"ResultsBroadcastableShape">; // X op Y == Y op X -def Commutative : NativeOpTrait<"IsCommutative">; +def Commutative : NativeOpTrait<"IsCommutative">, InternalOpTrait; // op op X == op X (unary) / X op X == X (binary) -def Idempotent : NativeOpTrait<"IsIdempotent">; +def Idempotent : NativeOpTrait<"IsIdempotent">, InternalOpTrait; // op op X == X -def Involution : NativeOpTrait<"IsInvolution">; +def Involution : NativeOpTrait<"IsInvolution">, InternalOpTrait; // Op behaves like a constant. -def ConstantLike : NativeOpTrait<"ConstantLike">; +def ConstantLike : NativeOpTrait<"ConstantLike">, InternalOpTrait; // Op is isolated from above. -def IsolatedFromAbove : NativeOpTrait<"IsIsolatedFromAbove">; +def IsolatedFromAbove : NativeOpTrait<"IsIsolatedFromAbove">, InternalOpTrait; // Op results are float or vectors/tensors thereof. -def ResultsAreFloatLike : NativeOpTrait<"ResultsAreFloatLike">; +def ResultsAreFloatLike : NativeOpTrait<"ResultsAreFloatLike">, InternalOpTrait; // Op has the same operand type. -def SameTypeOperands : NativeOpTrait<"SameTypeOperands">; +def SameTypeOperands : NativeOpTrait<"SameTypeOperands">, InternalOpTrait; // Op has same shape for all operands. -def SameOperandsShape : NativeOpTrait<"SameOperandsShape">; +def SameOperandsShape : NativeOpTrait<"SameOperandsShape">, InternalOpTrait; // Op has same operand and result shape. -def SameOperandsAndResultShape : NativeOpTrait<"SameOperandsAndResultShape">; +def SameOperandsAndResultShape : + NativeOpTrait<"SameOperandsAndResultShape">, InternalOpTrait; // Op has the same element type (or type itself, if scalar) for all operands. -def SameOperandsElementType : NativeOpTrait<"SameOperandsElementType">; +def SameOperandsElementType : + NativeOpTrait<"SameOperandsElementType">, InternalOpTrait; // Op has the same operand and result element type (or type itself, if scalar). def SameOperandsAndResultElementType : - NativeOpTrait<"SameOperandsAndResultElementType">; + NativeOpTrait<"SameOperandsAndResultElementType">, InternalOpTrait; // Op is a terminator. -def Terminator : NativeOpTrait<"IsTerminator">; +def Terminator : NativeOpTrait<"IsTerminator">, InternalOpTrait; // Op can be safely normalized in the presence of MemRefs with // non-identity maps. -def MemRefsNormalizable : NativeOpTrait<"MemRefsNormalizable">; +def MemRefsNormalizable : NativeOpTrait<"MemRefsNormalizable">, InternalOpTrait; // Op is elementwise on tensor/vector operands and results. -def Elementwise : NativeOpTrait<"Elementwise">; +def Elementwise : NativeOpTrait<"Elementwise">, InternalOpTrait; // Elementwise op can be applied to scalars instead tensor/vector operands. -def Scalarizable : NativeOpTrait<"Scalarizable">; +def Scalarizable : NativeOpTrait<"Scalarizable">, InternalOpTrait; // Elementwise op can be applied to all-vector operands. -def Vectorizable : NativeOpTrait<"Vectorizable">; +def Vectorizable : NativeOpTrait<"Vectorizable">, InternalOpTrait; // Elementwise op can be applied to all-tensor operands. -def Tensorizable : NativeOpTrait<"Tensorizable">; +def Tensorizable : NativeOpTrait<"Tensorizable">, InternalOpTrait; // Group together `Elementwise`, `Scalarizable`, `Vectorizable`, and // `Tensorizable` for convenience. @@ -2104,18 +2111,18 @@ ]>; // Op's regions have a single block. -def SingleBlock : NativeOpTrait<"SingleBlock">; +def SingleBlock : NativeOpTrait<"SingleBlock">, InternalOpTrait; // Op's regions have a single block with the specified terminator. class SingleBlockImplicitTerminator - : ParamNativeOpTrait<"SingleBlockImplicitTerminator", op>; + : ParamNativeOpTrait<"SingleBlockImplicitTerminator", op>, InternalOpTrait; // Op's regions don't have terminator. -def NoTerminator : NativeOpTrait<"NoTerminator">; +def NoTerminator : NativeOpTrait<"NoTerminator">, InternalOpTrait; // Op's parent operation is the provided one. class HasParent - : ParamNativeOpTrait<"HasParent", op>; + : ParamNativeOpTrait<"HasParent", op>, InternalOpTrait; class ParentOneOf ops> : ParamNativeOpTrait<"HasParent", !interleave(ops, ", ")>; @@ -2147,13 +2154,15 @@ // vector that has the same number of elements as the number of ODS declared // operands. That means even if some operands are non-variadic, the attribute // still need to have an element for its size, which is always 1. -def AttrSizedOperandSegments : NativeOpTrait<"AttrSizedOperandSegments">; +def AttrSizedOperandSegments : + NativeOpTrait<"AttrSizedOperandSegments">, InternalOpTrait; // Similar to AttrSizedOperandSegments, but used for results. The attribute // should be named as `result_segment_sizes`. -def AttrSizedResultSegments : NativeOpTrait<"AttrSizedResultSegments">; +def AttrSizedResultSegments : + NativeOpTrait<"AttrSizedResultSegments">, InternalOpTrait; // Op attached regions have no arguments -def NoRegionArguments : NativeOpTrait<"NoRegionArguments">; +def NoRegionArguments : NativeOpTrait<"NoRegionArguments">, InternalOpTrait; //===----------------------------------------------------------------------===// // OpInterface definitions @@ -2191,6 +2200,11 @@ // the operation being verified. code verify = verifyBody; + // A bit indicating if the verifier needs to access the ops in the regions. If + // it set to `1`, the region ops will be verified before invoking this + // verifier. + bit verifyWithRegions = 0; + // Specify the list of traits that need to be verified before the verification // of this OpInterfaceTrait. list dependentTraits = traits; @@ -2457,6 +2471,16 @@ // operation class. The operation should implement this method and verify the // additional necessary invariants. bit hasVerifier = 0; + + // A bit indicating if the operation has additional invariants that need to + // verified and which associate with regions (aside from those verified by the + // traits). If set to `1`, an additional `LogicalResult verifyRegions()` + // declaration will be generated on the operation class. The operation should + // implement this method and verify the additional necessary invariants + // associate with regions. Note that this method is invoked after all the + // region ops are verified. + bit hasRegionVerifier = 0; + // A custom code block corresponding to the extra verification code of the // operation. // NOTE: This field is deprecated in favor of `hasVerifier` and is slated for diff --git a/mlir/include/mlir/IR/OpDefinition.h b/mlir/include/mlir/IR/OpDefinition.h --- a/mlir/include/mlir/IR/OpDefinition.h +++ b/mlir/include/mlir/IR/OpDefinition.h @@ -201,7 +201,8 @@ protected: /// If the concrete type didn't implement a custom verifier hook, just fall /// back to this one which accepts everything. - LogicalResult verifyInvariants() { return success(); } + LogicalResult verify() { return success(); } + LogicalResult verifyRegions() { return success(); } /// Parse the custom form of an operation. Unless overridden, this method will /// first try to get an operation parser from the op's dialect. Otherwise the @@ -377,6 +378,18 @@ }; } // namespace detail +/// `verifyInvariantsImpl` verifies the invariants like the types, attrs, .etc. +/// It should be run after core traits and before any other user defined traits. +/// In order to run it in the correct order, wrap it with OpInvariants trait so +/// that tblgen will be able to put it in the right order. +template +class OpInvariants : public TraitBase { +public: + static LogicalResult verifyTrait(Operation *op) { + return cast(op).verifyInvariantsImpl(); + } +}; + /// This class provides the API for ops that are known to have no /// SSA operand. template @@ -1573,6 +1586,14 @@ template using detect_has_verify_trait = llvm::is_detected; +/// Trait to check if T provides a `verifyTrait` method. +template +using has_verify_region_trait = + decltype(T::verifyRegionTrait(std::declval())); +template +using detect_has_verify_region_trait = + llvm::is_detected; + /// The internal implementation of `verifyTraits` below that returns the result /// of verifying the current operation with all of the provided trait types /// `Ts`. @@ -1590,6 +1611,26 @@ static LogicalResult verifyTraits(Operation *op) { return verifyTraitsImpl(op, (TraitTupleT *)nullptr); } + +/// The internal implementation of `verifyRegionTraits` below that returns the +/// result of verifying the current operation with all of the provided trait +/// types `Ts`. +template +static LogicalResult verifyRegionTraitsImpl(Operation *op, + std::tuple *) { + LogicalResult result = success(); + (void)std::initializer_list{ + (result = succeeded(result) ? Ts::verifyRegionTrait(op) : failure(), + 0)...}; + return result; +} + +/// Given a tuple type containing a set of traits that contain a +/// `verifyTrait` method, return the result of verifying the given operation. +template +static LogicalResult verifyRegionTraits(Operation *op) { + return verifyRegionTraitsImpl(op, (TraitTupleT *)nullptr); +} } // namespace op_definition_impl //===----------------------------------------------------------------------===// @@ -1604,7 +1645,8 @@ public: /// Inherit getOperation from `OpState`. using OpState::getOperation; - using OpState::verifyInvariants; + using OpState::verify; + using OpState::verifyRegions; /// Return if this operation contains the provided trait. template