diff --git a/mlir/docs/Canonicalization.md b/mlir/docs/Canonicalization.md --- a/mlir/docs/Canonicalization.md +++ b/mlir/docs/Canonicalization.md @@ -1,4 +1,4 @@ -# Operation Canonicalization in MLIR +# Operation Canonicalization Canonicalization is an important part of compiler IR design: it makes it easier to implement reliable compiler transformations and to reason about what is @@ -45,20 +45,161 @@ * Constant folding - e.g. "(addi 1, 2)" to "3". Constant folding hooks are specified by operations. -* Move constant operands to commutative binary operators to the right side - - e.g. "(addi 4, x)" to "(addi x, 4)". - -## Builtin Ops Canonicalizations - -These transformations are applied to builtin ops: - -* `constant` ops are uniqued and hoisted into the entry block of the first - parent region that is isolated from above, e.g. the entry block of a - function. -* (TODO) Merge `affine.apply` operations that directly feed each other. - -## Standard Ops Canonicalizations - -* Shape folding of `alloc` operations to turn dynamic dimensions into static - ones. -* Folding `memref_cast` operations into users where possible. +* Move constant operands to commutative operators to the right side - e.g. + "(addi 4, x)" to "(addi x, 4)". + +* `constant-like` operations are uniqued and hoisted into the entry block of + the first parent region that is isolated from above, e.g. the entry block of + a function. + +## Defining Canonicalizations + +Two mechanisms are available with which to define canonicalizations; +`getCanonicalizationPatterns` and `fold`. + +### Canonicalizing with `getCanonicalizationPatterns` + +This mechanism allows for providing canonicalizations as a set of +`RewritePattern`s, either imperatively defined in C++ or declaratively as +[Declarative Rewrite Rules](DeclarativeRewrites.md). The pattern rewrite +infrastructure allows for expressing many different types of canonicalizations. +These transformations may be as simple as replacing a multiplication with a +shift, or even replacing a conditional branch with an unconditional one. + +In [ODS](OpDefinitions.md), an operation can set the `hasCanonicalizer` bit to +generate a declaration for the `getCanonicalizationPatterns` method. + +```tablegen +def MyOp : ... { + let hasCanonicalizer = 1; +} +``` + +Canonicalization patterns can then be provided in the source file: + +```c++ +void MyOp::getCanonicalizationPatterns(OwningRewritePatternList &patterns, + MLIRContext *context) { + patterns.insert<...>(...); +} +``` + +See the [quickstart guide](QuickstartRewrites.md) for information on defining +operation rewrites. + +### Canonicalizing with `fold` + +The `fold` mechanism is an intentionally limited, but powerful mechanism that +allows for applying canonicalizations in many places throughout the compiler. +For example, outside of the canonicalizer pass, `fold` is used within the +[dialect conversion infrastructure](#DialectConversion.md) as a legalization +mechanism, and can be invoked directly anywhere with an `OpBuilder` via +`OpBuilder::createOrFold`. + +`fold` has the restriction that no new operations may be created, and only the +root operation may be replaced. It allows for updating an operation in-place, or +returning a set of pre-existing values (or attributes) to replace the operation +with. This ensures that the `fold` method is a truly "local" transformation, and +can be invoked without the need for a pattern rewriter. + +In [ODS](OpDefinitions.md), an operation can set the `hasFolder` bit to generate +a declaration for the `fold` method. This method takes on a different form, +depending on the structure of the operation. + +```tablegen +def MyOp : ... { + let hasFolder = 1; +} +``` + +If the operation has a single result the following will be generated: + +```c++ +/// Implementations of this hook can only perform the following changes to the +/// operation: +/// +/// 1. They can leave the operation alone and without changing the IR, and +/// return nullptr. +/// 2. They can mutate the operation in place, without changing anything else +/// in the IR. In this case, return the operation itself. +/// 3. They can return an existing value or attribute that can be used instead +/// of the operation. The caller will remove the operation and use that +/// result instead. +/// +OpFoldResult MyOp::fold(ArrayRef<Attribute> operands) { + ... +} +``` + +Otherwise, the following is generated: + +```c++ +/// Implementations of this hook can only perform the following changes to the +/// operation: +/// +/// 1. They can leave the operation alone and without changing the IR, and +/// return failure. +/// 2. They can mutate the operation in place, without changing anything else +/// in the IR. In this case, return success. +/// 3. They can return a list of existing values or attribute that can be used +/// instead of the operation. In this case, fill in the results list and +/// return success. The results list must correspond 1-1 with the results of +/// the operation, partial folding is not supported. The caller will remove +/// the operation and use those results instead. +/// +LogicalResult MyOp::fold(ArrayRef<Attribute> operands, + SmallVectorImpl<OpFoldResult> &results) { + ... +} +``` + +In the above, for each method an `ArrayRef<Attribute>` is provided that +corresponds to the constant attribute value of each of the operands. These +operands are those that implement the `ConstantLike` trait. If any of the +operands are non-constant, a null `Attribute` value is provided instead. For +example, if MyOp provides three operands [`a`, `b`, `c`], but only `b` is +constant then `operands` will be of the form [Attribute(), b-value, +Attribute()]. + +Also above, is the use of `OpFoldResult`. This class represents the possible +result of folding an operation result: either an SSA `Value`, or an +`Attribute`(for a constant result). If an SSA `Value` is provided, it *must* +correspond to an existing value. The `fold` methods are not permitted to +generate new `Value`s. There are no specific restrictions on the form of the +`Attribute` value returned, but it is important to ensure that the `Attribute` +representation of a specific `Type` is consistent. + +#### Generating Constants from Attributes + +When a `fold` method returns an `Attribute` as the result, it signifies that +this result is "constant". The `Attribute` is the constant representation of the +value. Users of the `fold` method, such as the canonicalizer pass, will take +these `Attribute`s and materialize constant operations in the IR to represent +them. To enable this materialization, the dialect of the operation must +implement the `materializeConstant` hook. This hook takes in an `Attribute` +value, generally returned by `fold`, and produces a "constant-like" operation +that materializes that value. + +In [ODS](OpDefinitions.md), a dialect can set the `hasConstantMaterializer` bit +to generate a declaration for the `materializeConstant` method. + +```tablegen +def MyDialect_Dialect : ... { + let hasConstantMaterializer = 1; +} +``` + +Constants can then be materialized in the source file: + +```c++ +/// Hook to materialize a single constant operation from a given attribute value +/// with the desired resultant type. This method should use the provided builder +/// to create the operation without changing the insertion position. The +/// generated operation is expected to be constant-like. On success, this hook +/// should return the value generated to represent the constant value. +/// Otherwise, it should return nullptr on failure. +Operation *MyDialect::materializeConstant(OpBuilder &builder, Attribute value, + Type type, Location loc) { + ... +} +```