diff --git a/mlir/docs/Bufferization.md b/mlir/docs/Bufferization.md --- a/mlir/docs/Bufferization.md +++ b/mlir/docs/Bufferization.md @@ -222,6 +222,312 @@ skips the analysis and inserts a copy on every buffer write, just like the dialect conversion-based bufferization. +## Buffer Deallocation + +One-Shot Bufferize does not deallocate any buffers that it allocates. This job +is delegated to the +[`-buffer-deallocation`](https://mlir.llvm.org/docs/Passes/#-buffer-deallocation-adds-all-required-dealloc-operations-for-all-allocations-in-the-input-program) +pass. This pass processes operations implementing `FunctionOpInterface` +one-by-one without analysing the call-graph. This means, that there have to be +[some rules](#function-boundary-api) on how MemRefs are handled when being +passed from one function to another. The rest of the pass revolves heavily +around the `bufferization.dealloc` operation which is inserted at the end of +each basic block with appropriate operands and should be optimized using the +Buffer Deallocation Simplification pass (`--buffer-deallocation-simplification`) +and the regular canonicalizer (`--canonicalize`). Lowering the +`bufferization.dealloc` operation directly using +`--convert-bufferization-to-memref` without beforehand optimization is not +recommended as it will lead to very inefficient code (the runtime-cost of +`bufferization.dealloc` is `O(|memrefs|^2+|memref|*|retained|)`). + +### Function boundary ABI + +The Buffer Deallocation pass operates on the level of operations implementing +the `FunctionOpInterface`. Such operations can take MemRefs as argument, but +also return them. To ensure compatibility among all functions (including +external ones), some rules have to be enforced: +* When a MemRef is passed as a function argument, ownership is never consumed. + It is always the caller's responsibility to deallocate such MemRefs. +* Returning a MemRef from a function always passes ownership to the caller, + i.e., it is also the caller's responsibility to deallocate memrefs returned + from a called function. +* A function must not return a MemRef that aliases one of its arguments (in + this case a copy has to be created). Note that in this context two subviews + of the same buffer that don't overlap are also considered to alias. + +### Inserting `bufferization.dealloc` operations + +`bufferization.dealloc` operations are unconditionally inserted at the end of +each basic block (just before the terminator). The majority of the pass is about +finding the correct operands for this operation. There are three variadic +operand lists to be populated, the first contains all MemRef values that may +need to be deallocated, the second list contains their associated ownership +values (of `i1` type), and the third list contains MemRef values that are still +needed at a later point and should thus not be deallocated. The genericity of +this operation allows us to deal with any kind of aliasing behavior since it +inserts runtime aliasing checks when not enough information can be collected +statically. Additionally, it allows us to only think about correctness and +split off optimizations reducing the number of operands and +`bufferization.dealloc` operations into simple canonicalization patterns instead +of having to do all of them as part of this pass. + +**Ownerships** + +To do so, we use a concept of +ownership of memrefs which allows us to materialize an `i1` value for any SSA +value of `memref` type on demand, indicating whether the basic block in which it +was materialized has ownership of this MemRef. Ideally, this is a constant +`true` or `false`, but might also be a non-constant SSA value. To keep track of +those ownership values without immediately materializing them (which might +require insertion of `bufferization.clone` operations or operations checking for +aliasing at runtime at positions where we don't actually need a materialized +value), we use the `Ownership` class. This class represents the ownership in +three states forming a lattice on a partial order: +``` +forall X in SSA values. uninitialized < unique(X) < unknown +forall X, Y in SSA values. + unique(X) == unique(Y) iff X and Y always evaluate to the same value + unique(X) != unique(Y) otherwise +``` +Intuitively, the states have the following meaning: +* Uninitialized: the ownership is not initialized yet, this is the default + state; once an operation is finished processing the ownership of all + operation results with MemRef type should not be uninitialized anymore. +* Unique: there is a specific SSA value that can be queried to check ownership + without materializing any additional IR +* Unknown: no specific SSA value is available without materializing additional + IR, typically this is because two ownerships in 'Unique' state would have to + be merged manually (e.g., the result of an `arith.select` either has the + ownership of the then or else case depending on the condition value, + inserting another `arith.select` for the ownership values can perform the + merge and provide a 'Unique' ownership for the result), however, in the + general case this 'Unknown' state has to be assigned. + +Implied by the above partial order, the pass combines two ownerships in the +following way: + +| Ownership 1 | Ownership 2 | Combined Ownership | +|:--------------|:--------------|:-------------------| +| uninitialized | uninitialized | uninitialized | +| unique(X) | uninitialized | unique(X) | +| unique(X) | unique(X) | unique(X) | +| unique(X) | unique(Y) | unknown | +| unknown | unique | unknown | +| unknown | uninitialized | unknown | +| + symmetric cases | + +**Collecting the list of MemRefs that potentially needs to be deallocated** + +For a given block, the list of MemRefs that potentially need to be deallocated +at the end of that block is computed by keeping track of all values for which +the block potentially takes over ownership. This includes MemRefs provided as +basic block arguments, interface handlers for operations like `memref.alloc` and +`func.call`, but also liveness information in regions with multiple basic +blocks. More concretely, it is computed by taking the MemRefs in the 'in' set +of the liveness analysis of the current basic block B, appended by the MemRef +block arguments and by the set of MemRefs allocated in B itself (determined by +the interface handlers), then subtracted (also determined by the interface +handlers) by the set of MemRefs deallocated in B. + +Note that we don't have to take the intersection of the liveness 'in' set with +the 'out' set of the predecessor block because a value that is in the 'in' set +must be defined in an ancestor block that dominates all direct predecessors and +thus the 'in' set of this block is a subset of the 'out' sets of each +predecessor. + +``` +memrefs = filter((liveIn(block) U + allocated(block) U arguments(block)) \ deallocated(block), isMemRef) +``` + +The list of conditions for the second variadic operands list of +`bufferization.dealloc` is computed by querying the stored ownership value for +each of the MemRefs collected as described above. The ownership state is updated +by the interface handlers while processing the basic block. + +**Collecting the list of MemRefs to retain** + +Given a basic block B, the list of MemRefs that have to be retained can be +different for each successor block S. For the two basic blocks B and S and the +values passed via block arguments to the destination block S, we compute the +list of MemRefs that have to be retained in B by taking the MemRefs in the +successor operand list of the terminator and the MemRefs in the 'out' set of the +liveness analysis for B intersected with the 'in' set of the destination block +S. + +This list of retained values makes sure that we cannot run into use-after-free +situations even if no aliasing information is present at compile-time. + +``` +toRetain = filter(successorOperands + (liveOut(fromBlock) insersect + liveIn(toBlock)), isMemRef) +``` + +### Supported interfaces + +The pass uses liveness analysis and a few interfaces: +* `FunctionOpInterface` +* `CallOpInterface` +* `MemoryEffectOpInterface` +* `RegionBranchOpInterface` +* `RegionBranchTerminatorOpInterface` + +Due to insufficient information provided by the interface, it also special-cases +on the `cf.cond_br` operation and makes some assumptions about operations +implementing the `RegionBranchOpInterface` at the moment, but improving the +interfaces would allow us to remove those dependencies in the future. + +### Limitations + +The Buffer Deallocation pass has some requirements and limitations on the input +IR. These are checked in the beginning of the pass and errors are emitted +accordingly: +* The set of interfaces the pass operates on must be implemented (correctly). + E.g., if there is an operation present with a nested region, but does not + implement the `RegionBranchOpInterface`, an error is emitted because the + pass cannot know the semantics of this nested region (and does not make any + default assumptions on it). +* Operations or interfaces that are not considered in the pass, but would have + an important effect (in terms of correctness) on deallocation should not be + present. +* No explicit control-flow loops are present. Currently, only loops using + structural-control-flow are supported. However, this limitation could be + lifted in the future. +* Deallocation operations should not be present already. The pass should + handle them correctly already (at least in most cases), but it's not + supported yet due to insufficient testing. +* Terminators must implement either `RegionBranchTerminatorOpInterface` or + `BranchOpInterface`, but not both. Terminators with more than one successor + are not supported (except `cf.cond_br`). This is not a fundamental + limitation, but there is no use-case justifying the more complex + implementation at the moment. + +### Example + +The following example contains a few interesting cases: +* Basic block arguments are modified to also pass along the ownership + indicator, but not for entry bocks for non-private functions (assuming the + `private-function-dynamic-ownership` pass option is disabled) where the + function boundary ABI is applied instead. +* The result of `arith.select` initially has 'Unknown' assigned as ownership, + but once the `bufferization.dealloc` operation is inserted it is put in the + 'retained' list (since it has uses in a later basic block) and thus the + 'Unknown' ownership can be replaced with a 'Unique' ownership using the + corresponding result of the dealloc operation. +* The `cf.cond_br` + +```mlir +func.func @example(%memref: memref, %select_cond: i1, %br_cond: i1) { + %alloc = memref.alloc() : memref + %alloca = memref.alloca() : memref + %select = arith.select %select_cond, %alloc, %alloca : memref + cf.cond_br %br_cond, ^bb1(%alloc : memref), ^bb1(%memref : memref) +^bb1(%bbarg: memref): + test.copy(%bbarg, %select) : (memref, memref) + return +} +``` + +After running `--buffer-deallocation`, it looks like the following: + +```mlir +// Since this is not a private function, the signature will not be modified even +// when private-function-dynamic-ownership is enabled. +func.func @example(%memref: memref, %select_cond: i1, %br_cond: i1) { + %false = arith.constant false + %true = arith.constant true + %alloc = memref.alloc() : memref + %alloca = memref.alloca() : memref + %select = arith.select %select_cond, %alloc, %alloca : memref + + // We use `memref.extract_strided_metadata` to get the base memref since it is + // not allowed to pass arbitrary memrefs to `memref.dealloc`. This property is + // already enforced for `bufferization.dealloc` + %base_buffer_memref, ... = memref.extract_strided_metadata %memref + : memref -> memref, index, index, index + %base_buffer_alloc, ... = memref.extract_strided_metadata %alloc + : memref -> memref, index, index, index + %base_buffer_alloca, ... = memref.extract_strided_metadata %alloca + : memref -> memref, index, index, index + + // The deallocation conditions need to be adjusted to incorporate the branch + // condition. In this example, this requires only a single negation, but might + // also require multiple arith.andi operations. + %not_br_cond = arith.xori %true, %br_cond : i1 + + // There are two dealloc operations inserted in this basic block, one per + // successor. Both have the same list of MemRefs to deallocate and and the + // conditions only differ by the branch condition conjunct. + // Note, however, that the retained list differs. Here, both contain the + // %select value because it is used in both successors (since it's the same + // block), but the value passed via block argument differs (%memref vs. + // %alloc). + %10:2 = bufferization.dealloc + (%base_buffer_memref, %base_buffer_alloc, %base_buffer_alloca + : memref, memref, memref) + if (%false, %br_cond, %false) + retain (%alloc, %select : memref, memref) + + %11:2 = bufferization.dealloc + (%base_buffer_memref, %base_buffer_alloc, %base_buffer_alloca + : memref, memref, memref) + if (%false, %not_br_cond, %false) + retain (%memref, %select : memref, memref) + + // Because %select is used in ^bb1 without passing it via block argument, we + // need to update it's ownership value here by merging the ownership values + // returned by the dealloc operations + %new_ownership = arith.select %br_cond, %10#1, %11#1 : i1 + + // The terminator is modified to pass along the ownership indicator values + // with each MemRef value. + cf.cond_br %br_cond, ^bb1(%alloc, %10#0 : memref, i1), + ^bb1(%memref, %11#0 : memref, i1) + +// All non-entry basic blocks are modified to have an additional i1 argument for +// each MemRef value in the argument list. +^bb1(%13: memref, %14: i1): // 2 preds: ^bb0, ^bb0 + test.copy(%13, %select) : (memref, memref) + + %base_buffer_13, ... = memref.extract_strided_metadata %13 + : memref -> memref, index, index, index + %base_buffer_select, ... = memref.extract_strided_metadata %select + : memref -> memref, index, index, index + + // Here, we don't have a retained list, because the block has no successors + // and the return has no operands. + bufferization.dealloc (%base_buffer_13, %base_buffer_select + : memref, memref) + if (%14, %new_ownership) + return +} +``` + +## Buffer Deallocation Simplification Pass + +The [semantics of the `bufferization.dealloc` operation](https://mlir.llvm.org/docs/Dialects/BufferizationOps/#bufferizationdealloc-bufferizationdeallocop) +provide a lot of opportunities for optimizations which can be conveniently split +into patterns using the greedy pattern rewriter. Some of those patterns need +access to additional analyses such as an analysis that can determine whether two +MemRef values must, may, or never originate from the same buffer allocation. +These patterns are collected in the Buffer Deallocation Simplification pass, +while patterns that don't need additional analyses are registered as part of the +regular canonicalizer pass. This pass is best run after `--buffer-deallocation` +followed by `--canonicalize`. + +The pass applies patterns for the following simplifications: +* Remove MemRefs from retain list when guaranteed to not alias with any value + in the 'memref' operand list. This avoids an additional aliasing check with + the removed value. +* Split off values in the 'memref' list to new `bufferization.dealloc` + operations only containing this value in the 'memref' list when it is + guaranteed to not alias with any other value in the 'memref' list. This + avoids at least one aliasing check at runtime and enables using a more + efficient lowering for this new `bufferization.dealloc` operation. +* Remove values from the 'memref' operand list when it is guaranteed to alias + with at least one value in the 'retained' list and may not alias any other + value in the 'retain' list. + ## Memory Layouts One-Shot Bufferize bufferizes ops from top to bottom. This works well when all diff --git a/mlir/include/mlir/Dialect/Bufferization/Transforms/BufferUtils.h b/mlir/include/mlir/Dialect/Bufferization/Transforms/BufferUtils.h --- a/mlir/include/mlir/Dialect/Bufferization/Transforms/BufferUtils.h +++ b/mlir/include/mlir/Dialect/Bufferization/Transforms/BufferUtils.h @@ -121,6 +121,14 @@ Liveness liveness; }; +/// Compare two SSA values in a deterministic manner. Two block arguments are +/// ordered by argument number, block arguments are always less than operation +/// results, and operation results are ordered by the `isBeforeInBlock` order of +/// their defining operation. +struct ValueComparator { + bool operator()(const Value &lhs, const Value &rhs) const; +}; + // Create a global op for the given tensor-valued constant in the program. // Globals are created lazily at the top of the enclosing ModuleOp with pretty // names. Duplicates are avoided. diff --git a/mlir/include/mlir/Dialect/Bufferization/Transforms/Passes.h b/mlir/include/mlir/Dialect/Bufferization/Transforms/Passes.h --- a/mlir/include/mlir/Dialect/Bufferization/Transforms/Passes.h +++ b/mlir/include/mlir/Dialect/Bufferization/Transforms/Passes.h @@ -5,6 +5,7 @@ namespace mlir { class ModuleOp; +class FunctionOpInterface; namespace func { class FuncOp; @@ -30,7 +31,8 @@ std::unique_ptr createBufferDeallocationSimplificationPass(); /// Run buffer deallocation. -LogicalResult deallocateBuffers(Operation *op); +LogicalResult deallocateBuffers(FunctionOpInterface op, + bool privateFuncDynamicOwnership); /// Creates a pass that moves allocations upwards to reduce the number of /// required copies that are inserted during the BufferDeallocation pass. @@ -106,9 +108,6 @@ // Registration //===----------------------------------------------------------------------===// -/// Register external models for AllocationOpInterface. -void registerAllocationOpInterfaceExternalModels(DialectRegistry ®istry); - /// Generate the code for registering passes. #define GEN_PASS_REGISTRATION #include "mlir/Dialect/Bufferization/Transforms/Passes.h.inc" diff --git a/mlir/include/mlir/Dialect/Bufferization/Transforms/Passes.td b/mlir/include/mlir/Dialect/Bufferization/Transforms/Passes.td --- a/mlir/include/mlir/Dialect/Bufferization/Transforms/Passes.td +++ b/mlir/include/mlir/Dialect/Bufferization/Transforms/Passes.td @@ -85,7 +85,19 @@ ``` }]; + let options = [ + Option<"privateFuncDynamicOwnership", "private-function-dynamic-ownership", + "bool", /*default=*/"false", + "Allows to add additional arguments to private functions to " + "dynamically pass ownership of memrefs to callees. This can enable " + "earlier deallocations.">, + ]; let constructor = "mlir::bufferization::createBufferDeallocationPass()"; + + let dependentDialects = [ + "mlir::bufferization::BufferizationDialect", "mlir::arith::ArithDialect", + "mlir::memref::MemRefDialect", "mlir::scf::SCFDialect" + ]; } def BufferDeallocationSimplification : diff --git a/mlir/lib/Dialect/Bufferization/IR/BufferizationOps.cpp b/mlir/lib/Dialect/Bufferization/IR/BufferizationOps.cpp --- a/mlir/lib/Dialect/Bufferization/IR/BufferizationOps.cpp +++ b/mlir/lib/Dialect/Bufferization/IR/BufferizationOps.cpp @@ -777,6 +777,10 @@ : public OpRewritePattern { using OpRewritePattern::OpRewritePattern; + void initialize() { + setDebugName("bufferization-dealloc-remove-duplicate-dealloc-memrefs"); + } + LogicalResult matchAndRewrite(DeallocOp deallocOp, PatternRewriter &rewriter) const override { // Unique memrefs to be deallocated. @@ -819,6 +823,10 @@ : public OpRewritePattern { using OpRewritePattern::OpRewritePattern; + void initialize() { + setDebugName("bufferization-dealloc-remove-duplicate-retained-memrefs"); + } + LogicalResult matchAndRewrite(DeallocOp deallocOp, PatternRewriter &rewriter) const override { // Unique retained values @@ -864,6 +872,8 @@ struct EraseEmptyDealloc : public OpRewritePattern { using OpRewritePattern::OpRewritePattern; + void initialize() { setDebugName("bufferization-erase-empty-dealloc"); } + LogicalResult matchAndRewrite(DeallocOp deallocOp, PatternRewriter &rewriter) const override { if (deallocOp.getMemrefs().empty()) { @@ -893,6 +903,10 @@ struct EraseAlwaysFalseDealloc : public OpRewritePattern { using OpRewritePattern::OpRewritePattern; + void initialize() { + setDebugName("bufferization-erase-always-false-dealloc"); + } + LogicalResult matchAndRewrite(DeallocOp deallocOp, PatternRewriter &rewriter) const override { SmallVector newMemrefs, newConditions; @@ -930,6 +944,10 @@ struct SkipExtractMetadataOfAlloc : public OpRewritePattern { using OpRewritePattern::OpRewritePattern; + void initialize() { + setDebugName("bufferization-skip-extract-metadata-of-alloc"); + } + LogicalResult matchAndRewrite(DeallocOp deallocOp, PatternRewriter &rewriter) const override { SmallVector newMemrefs( diff --git a/mlir/lib/Dialect/Bufferization/Transforms/BufferDeallocation.cpp b/mlir/lib/Dialect/Bufferization/Transforms/BufferDeallocation.cpp --- a/mlir/lib/Dialect/Bufferization/Transforms/BufferDeallocation.cpp +++ b/mlir/lib/Dialect/Bufferization/Transforms/BufferDeallocation.cpp @@ -6,57 +6,26 @@ // //===----------------------------------------------------------------------===// // -// This file implements logic for computing correct alloc and dealloc positions. -// Furthermore, buffer deallocation also adds required new clone operations to -// ensure that all buffers are deallocated. The main class is the -// BufferDeallocationPass class that implements the underlying algorithm. In -// order to put allocations and deallocations at safe positions, it is -// significantly important to put them into the correct blocks. However, the -// liveness analysis does not pay attention to aliases, which can occur due to -// branches (and their associated block arguments) in general. For this purpose, -// BufferDeallocation firstly finds all possible aliases for a single value -// (using the BufferViewFlowAnalysis class). Consider the following example: -// -// ^bb0(%arg0): -// cf.cond_br %cond, ^bb1, ^bb2 -// ^bb1: -// cf.br ^exit(%arg0) -// ^bb2: -// %new_value = ... -// cf.br ^exit(%new_value) -// ^exit(%arg1): -// return %arg1; -// -// We should place the dealloc for %new_value in exit. However, we have to free -// the buffer in the same block, because it cannot be freed in the post -// dominator. However, this requires a new clone buffer for %arg1 that will -// contain the actual contents. Using the class BufferViewFlowAnalysis, we -// will find out that %new_value has a potential alias %arg1. In order to find -// the dealloc position we have to find all potential aliases, iterate over -// their uses and find the common post-dominator block (note that additional -// clones and buffers remove potential aliases and will influence the placement -// of the deallocs). In all cases, the computed block can be safely used to free -// the %new_value buffer (may be exit or bb2) as it will die and we can use -// liveness information to determine the exact operation after which we have to -// insert the dealloc. However, the algorithm supports introducing clone buffers -// and placing deallocs in safe locations to ensure that all buffers will be -// freed in the end. +// This file implements logic for computing correct `bufferization.dealloc` +// positions. Furthermore, buffer deallocation also adds required new clone +// operations to ensure that memrefs returned by functions never alias an +// argument. // // TODO: // The current implementation does not support explicit-control-flow loops and // the resulting code will be invalid with respect to program semantics. -// However, structured control-flow loops are fully supported. Furthermore, it -// doesn't accept functions which return buffers already. +// However, structured control-flow loops are fully supported. // //===----------------------------------------------------------------------===// -#include "mlir/Dialect/Bufferization/Transforms/Passes.h" - -#include "mlir/Dialect/Bufferization/IR/AllocationOpInterface.h" #include "mlir/Dialect/Bufferization/IR/Bufferization.h" #include "mlir/Dialect/Bufferization/Transforms/BufferUtils.h" +#include "mlir/Dialect/Bufferization/Transforms/Passes.h" +#include "mlir/Dialect/ControlFlow/IR/ControlFlowOps.h" #include "mlir/Dialect/Func/IR/FuncOps.h" #include "mlir/Dialect/MemRef/IR/MemRef.h" +#include "mlir/Dialect/SCF/IR/SCF.h" +#include "mlir/Interfaces/ControlFlowInterfaces.h" #include "llvm/ADT/SetOperations.h" namespace mlir { @@ -69,56 +38,22 @@ using namespace mlir; using namespace mlir::bufferization; -/// Walks over all immediate return-like terminators in the given region. -static LogicalResult walkReturnOperations( - Region *region, - llvm::function_ref func) { - for (Block &block : *region) { - Operation *terminator = block.getTerminator(); - // Skip non region-return-like terminators. - if (auto regionTerminator = - dyn_cast(terminator)) { - if (failed(func(regionTerminator))) - return failure(); - } - } - return success(); -} - -/// Checks if all operations that have at least one attached region implement -/// the RegionBranchOpInterface. This is not required in edge cases, where we -/// have a single attached region and the parent operation has no results. -static bool validateSupportedControlFlow(Operation *op) { - WalkResult result = op->walk([&](Operation *operation) { - // Only check ops that are inside a function. - if (!operation->getParentOfType()) - return WalkResult::advance(); - - auto regions = operation->getRegions(); - // Walk over all operations in a region and check if the operation has at - // least one region and implements the RegionBranchOpInterface. If there - // is an operation that does not fulfill this condition, we cannot apply - // the deallocation steps. Furthermore, we accept cases, where we have a - // region that returns no results, since, in that case, the intra-region - // control flow does not affect the transformation. - size_t size = regions.size(); - if (((size == 1 && !operation->getResults().empty()) || size > 1) && - !dyn_cast(operation)) { - operation->emitError("All operations with attached regions need to " - "implement the RegionBranchOpInterface."); - } +//===----------------------------------------------------------------------===// +// Helpers +//===----------------------------------------------------------------------===// - return WalkResult::advance(); - }); - return !result.wasSkipped(); +static Value getBoolValue(OpBuilder &builder, Location loc, bool value) { + return builder.create(loc, builder.getBoolAttr(value)); } -namespace { +static bool isMemref(Value v) { return v.getType().isa(); } //===----------------------------------------------------------------------===// // Backedges analysis //===----------------------------------------------------------------------===// +namespace { + /// A straight-forward program analysis which detects loop backedges induced by /// explicit control flow. class Backedges { @@ -194,467 +129,1154 @@ BackedgeSetT edgeSet; }; +} // namespace + //===----------------------------------------------------------------------===// // BufferDeallocation //===----------------------------------------------------------------------===// +namespace { +/// This class is used to track the ownership of values. The ownership can +/// either be not initialized yet ('Uninitialized' state), set to a unique SSA +/// value which indicates the ownership at runtime (or statically if it is a +/// constant value) ('Unique' state), or it cannot be represented in a single +/// SSA value ('Unknown' state). An artificial example of a case where ownership +/// cannot be represented in a single i1 SSA value could be the following: +/// `%0 = test.non_deterministic_select %arg0, %arg1 : i32` +/// Since the operation does not provide us a separate boolean indicator on +/// which of the two operands was selected, we would need to either insert an +/// alias check at runtime to determine if `%0` aliases with `%arg0` or `%arg1`, +/// or insert a `bufferization.clone` operation to get a fresh buffer which we +/// could assign ownership to. +/// +/// The three states this class can represent form a lattice on a partial order: +/// forall X in SSA values. uninitialized < unique(X) < unknown +/// forall X, Y in SSA values. +/// unique(X) == unique(Y) iff X and Y always evaluate to the same value +/// unique(X) != unique(Y) otherwise +class Ownership { +public: + /// Constructor that creates an 'Uninitialized' ownership. This is needed for + /// default-construction when used in DenseMap. + Ownership() = default; + + /// Constructor that creates an 'Unique' ownership. This is a non-explicit + /// constructor to allow implicit conversion from 'Value'. + Ownership(Value indicator) : indicator(indicator), state(State::Unique) {} + + /// Get an ownership value in 'Unknown' state. + static Ownership getUnknown() { + Ownership unknown; + unknown.indicator = Value(); + unknown.state = State::Unknown; + return unknown; + } + /// Get an ownership value in 'Unique' state with 'indicator' as parameter. + static Ownership getUnique(Value indicator) { return Ownership(indicator); } + /// Get an ownership value in 'Uninitialized' state. + static Ownership getUninitialized() { return Ownership(); } + + /// Check if this ownership value is in the 'Uninitialized' state. + bool isUninitialized() const { return state == State::Uninitialized; } + /// Check if this ownership value is in the 'Unique' state. + bool isUnique() const { return state == State::Unique; } + /// Check if this ownership value is in the 'Unknown' state. + bool isUnknown() const { return state == State::Unknown; } + + /// If this ownership value is in 'Unique' state, this function can be used to + /// get the indicator parameter. Using this function in any other state is UB. + Value getIndicator() const { + assert(isUnique() && "must have unique ownership to get the indicator"); + return indicator; + } + + /// Get the join of the two-element subset {this,other}. Does not modify + /// 'this'. + Ownership getCombined(Ownership other) const { + if (other.isUninitialized()) + return *this; + if (isUninitialized()) + return other; + + if (!isUnique() || !other.isUnique()) + return getUnknown(); + + if (indicator == other.indicator) + return *this; + + // Since we create a new constant i1 value for (almost) each use-site, we + // should compare the actual value rather than just the SSA Value to avoid + // unnecessary invalidations. + if (matchPattern(indicator, m_Zero()) && + matchPattern(other.indicator, m_Zero())) + return *this; + if (matchPattern(indicator, m_One()) && + matchPattern(other.indicator, m_One())) + return *this; + + // Return the join of the lattice if the indicator of both ownerships cannot + // be merged. + return getUnknown(); + } + + /// Modify 'this' ownership to be the join of the current 'this' and 'other'. + void combine(Ownership other) { *this = getCombined(other); } + +private: + enum class State { + Uninitialized, + Unique, + Unknown, + }; + + // The indicator value is only relevant in the 'Unique' state. + Value indicator; + State state = State::Uninitialized; +}; + /// The buffer deallocation transformation which ensures that all allocs in the -/// program have a corresponding de-allocation. As a side-effect, it might also -/// introduce clones that in turn leads to additional deallocations. -class BufferDeallocation : public BufferPlacementTransformationBase { +/// program have a corresponding de-allocation. +class BufferDeallocation { public: - using AliasAllocationMapT = - llvm::DenseMap; - - BufferDeallocation(Operation *op) - : BufferPlacementTransformationBase(op), dominators(op), - postDominators(op) {} - - /// Checks if all allocation operations either provide an already existing - /// deallocation operation or implement the AllocationOpInterface. In - /// addition, this method initializes the internal alias to - /// AllocationOpInterface mapping in order to get compatible - /// AllocationOpInterface implementations for aliases. - LogicalResult prepare() { - for (const BufferPlacementAllocs::AllocEntry &entry : allocs) { - // Get the defining allocation operation. - Value alloc = std::get<0>(entry); - auto allocationInterface = - alloc.getDefiningOp(); - // If there is no existing deallocation operation and no implementation of - // the AllocationOpInterface, we cannot apply the BufferDeallocation pass. - if (!std::get<1>(entry) && !allocationInterface) { - return alloc.getDefiningOp()->emitError( - "Allocation is not deallocated explicitly nor does the operation " - "implement the AllocationOpInterface."); - } + BufferDeallocation(Operation *op, bool privateFuncDynamicOwnership) + : liveness(op), privateFuncDynamicOwnership(privateFuncDynamicOwnership) { + } - // Register the current allocation interface implementation. - aliasToAllocations[alloc] = allocationInterface; + /// Performs the actual placement/creation of all dealloc operations. + LogicalResult deallocate(FunctionOpInterface op); - // Get the alias information for the current allocation node. - for (Value alias : aliases.resolve(alloc)) { - // TODO: check for incompatible implementations of the - // AllocationOpInterface. This could be realized by promoting the - // AllocationOpInterface to a DialectInterface. - aliasToAllocations[alias] = allocationInterface; - } +private: + /// The base case for the recursive template below. + template + typename std::enable_if>::type + handleOp(Operation *op) { + return op; + } + + /// Applies all the handlers of the interfaces in the template list + /// implemented by 'op'. If a handler removes the operation, no further + /// handlers are called on the operation. + /// The return value of the 'handleInterface' functions should be a + /// FailureOr indicating whether there was a failure or otherwise + /// returning the operation itself or a replacement operation. + /// Note: The difference compared to `TypeSwitch` is that all + /// matching cases are applied instead of just the first match. + template + FailureOr handleOp(Operation *op) { + Operation *next = op; + if (auto concreteOp = dyn_cast(op)) { + FailureOr result = handleInterface(concreteOp); + if (failed(result)) + return failure(); + next = *result; } - return success(); + return handleOp(next); } - /// Performs the actual placement/creation of all temporary clone and dealloc - /// nodes. - LogicalResult deallocate() { - // Add additional clones that are required. - if (failed(introduceClones())) - return failure(); + /// Apply all supported interface handlers to the given op. This is useful + /// when interface handlers introduce new operations which they have to call + /// all handlers on themselves. + FailureOr handleAllInterfaces(Operation *op) { + return handleOp(op); + } - // Place deallocations for all allocation entries. - return placeDeallocs(); + /// While CondBranchOp also implement the BranchOpInterface, we add a + /// special-case implementation here because the BranchOpInterface does not + /// offer all of the functionallity we need to insert dealloc oeprations in an + /// efficient way. More precisely, there is no way to extract the branch + /// condition without casting to CondBranchOp specifically. It is still + /// possible to implement deallocation for cases where we don't know to which + /// successor the terminator branches before the actual branch happens by + /// inserting auxillary blocks and putting the dealloc op there, however, this + /// can lead to less efficient code. + /// This function inserts two dealloc operations (one for each successor) and + /// adjusts the dealloc conditions according to the branch condition, then the + /// ownerships of the retained MemRefs are updated by combining the result + /// values of the two dealloc operations. + /// + /// Example: + /// ``` + /// ^bb1: + /// + /// cf.cond_br cond, ^bb2(), ^bb3() + /// ``` + /// becomes + /// ``` + /// // let (m, c) = getMemrefsAndConditionsToDeallocate(bb1) + /// // let r0 = getMemrefsToRetain(bb1, bb2, ) + /// // let r1 = getMemrefsToRetain(bb1, bb3, ) + /// ^bb1: + /// + /// let thenCond = map(c, (c) -> arith.andi cond, c) + /// let elseCond = map(c, (c) -> arith.andi (arith.xori cond, true), c) + /// o0 = bufferization.dealloc m if thenCond retain r0 + /// o1 = bufferization.dealloc m if elseCond retain r1 + /// // replace ownership(r0) with o0 element-wise + /// // replace ownership(r1) with o1 element-wise + /// // let ownership0 := (r) -> o in o0 corresponding to r + /// // let ownership1 := (r) -> o in o1 corresponding to r + /// // let cmn := intersection(r0, r1) + /// foreach (a, b) in zip(map(cmn, ownership0), map(cmn, ownership1)): + /// forall r in r0: replace ownership0(r) with arith.select cond, a, b) + /// forall r in r1: replace ownership1(r) with arith.select cond, a, b) + /// cf.cond_br cond, ^bb2(, o0), ^bb3(, o1) + /// ``` + FailureOr handleInterface(cf::CondBranchOp op); + + /// Make sure that for each forwarded MemRef value, a ownership indicator `i1` + /// value is forwarded as well such that the successor block knows whether the + /// MemRef has to be deallocated. + /// + /// Example: + /// ``` + /// ^bb1: + /// + /// cf.br ^bb2() + /// ``` + /// becomes + /// ``` + /// // let (m, c) = getMemrefsAndConditionsToDeallocate(bb1) + /// // let r = getMemrefsToRetain(bb1, bb2, ) + /// ^bb1: + /// + /// o = bufferization.dealloc m if c retain r + /// // replace ownership(r) with o element-wise + /// cf.br ^bb2(, o) + /// ``` + FailureOr handleInterface(BranchOpInterface op); + + /// Add an ownership indicator for every forwarding MemRef operand and result. + /// Note that the block arguments in the nested region are currently handled + /// centrally in the 'dealloc' function, but better interface support could + /// allow us to do this here for the nested region specifically to reduce the + /// amount of assumptions we make on the structure of ops implementing this + /// interface. + /// + /// Example: + /// ``` + /// %ret = scf.for %i = %c0 to %c10 step %c1 iter_args(%m = %memref) { + /// + /// scf.yield %m : memref<2xi32>, i1 + /// } + /// ``` + /// becomes + /// ``` + /// %ret:2 = scf.for %i = %c0 to %c10 step %c1 + /// iter_args(%m = %memref, %own = %false) { + /// + /// // Note that the scf.yield is handled by the + /// // RegionBranchTerminatorOpInterface (not this handler) + /// // let o = getMemrefWithUniqueOwnership(%own) + /// scf.yield %m, o : memref<2xi32>, i1 + /// } + /// ``` + FailureOr handleInterface(RegionBranchOpInterface op); + + /// If the private-function-dynamic-ownership pass option is enabled and the + /// called function is private, additional arguments and results are added for + /// each MemRef argument/result to pass the dynamic ownership indicator along. + /// Otherwise, updates the ownership map and list of memrefs to be deallocated + /// according to the function boundary ABI, i.e., assume ownership of all + /// returned MemRefs. + /// + /// Example (assume `private-function-dynamic-ownership` is enabled): + /// ``` + /// func.func @f(%arg0: memref<2xi32>) -> memref<2xi32> {...} + /// func.func private @g(%arg0: memref<2xi32>) -> memref<2xi32> {...} + /// + /// %ret_f = func.call @f(%memref) : (memref<2xi32>) -> memref<2xi32> + /// %ret_g = func.call @g(%memref) : (memref<2xi32>) -> memref<2xi32> + /// ``` + /// becomes + /// ``` + /// func.func @f(%arg0: memref<2xi32>) -> memref<2xi32> {...} + /// func.func private @g(%arg0: memref<2xi32>) -> memref<2xi32> {...} + /// + /// %ret_f = func.call @f(%memref) : (memref<2xi32>) -> memref<2xi32> + /// // set ownership(%ret_f) := true + /// // remember to deallocate %ret_f + /// + /// // (new_memref, own) = getmemrefWithUniqueOwnership(%memref) + /// %ret_g:2 = func.call @g(new_memref, own) : + /// (memref<2xi32>, i1) -> (memref<2xi32>, i1) + /// // set ownership(%ret_g#0) := %ret_g#1 + /// // remember to deallocate %ret_g + /// ``` + FailureOr handleInterface(CallOpInterface op); + + /// Takes care of allocation and free side-effects. It collects allocated + /// MemRefs that we have to add to manually deallocate, but also removes + /// values again that are already deallocated before the end of the block. It + /// also updates the ownership map accordingly. + /// + /// Example: + /// ``` + /// %alloc = memref.alloc() + /// %alloca = memref.alloca() + /// ``` + /// becomes + /// ``` + /// %alloc = memref.alloc() + /// %alloca = memref.alloca() + /// // set ownership(alloc) := true + /// // set ownership(alloca) := false + /// // remember to deallocate %alloc + /// ``` + FailureOr handleInterface(MemoryEffectOpInterface op); + + /// Takes care that the function boundary ABI is adhered to if the parent + /// operation implements FunctionOpInterface, inserting a + /// `bufferization.clone` if necessary, and inserts the + /// `bufferization.dealloc` operation according to the ops operands. + /// + /// Example: + /// ``` + /// ^bb1: + /// + /// func.return + /// ``` + /// becomes + /// ``` + /// // let (m, c) = getMemrefsAndConditionsToDeallocate(bb1) + /// // let r = getMemrefsToRetain(bb1, nullptr, ) + /// ^bb1: + /// + /// o = bufferization.dealloc m if c retain r + /// func.return + /// (if !isFunctionWithoutDynamicOwnership: append o) + /// ``` + FailureOr handleInterface(RegionBranchTerminatorOpInterface op); + + /// Construct a new operation which is exactly the same as the passed 'op' + /// except that the OpResults list is appended by new results of the passed + /// 'types'. + /// TODO: ideally, this would be implemented using an OpInterface because it + /// is used to append function results, loop iter_args, etc. and thus makes + /// some assumptions that the variadic list of those is at the end of the + /// OpResults range. + Operation *appendOpResults(Operation *op, ArrayRef types); + + /// A convenience template for the generic 'appendOpResults' function above to + /// avoid manual casting of the result. + template + OpTy appendOpResults(OpTy op, ArrayRef types) { + return cast(appendOpResults(op.getOperation(), types)); } + /// Performs deallocation of a single basic block. This is a private function + /// because some internal data structures have to be set up beforehand and + /// this function has to be called on blocks in a region in dominance order. + LogicalResult deallocate(Block *block); + + /// Small helper function to update the ownership map by taking the current + /// ownership ('Uninitialized' state if not yet present), computing the join + /// with the passed ownership and storing this new value in the map. By + /// default, it will be performed for the block where 'owned' is defined. If + /// the ownership of the given value should be updated for another block, the + /// 'block' argument can be explicitly passed. + void setOwnership(Value owned, Ownership ownership, Block *block = nullptr); + + /// Removes ownerships associated with all values in the passed range for + /// 'block'. + void clearOwnershipOf(ValueRange values, Block *block); + + /// After all relevant interfaces of an operation have been processed by the + /// 'handleInterface' functions, this function sets the ownership of operation + /// results that have not been set yet by the 'handleInterface' functions. It + /// generally assumes that each result can alias with every operand of the + /// operation, if there are MemRef typed results but no MemRef operands it + /// assigns 'false' as ownership. This happens, e.g., for the + /// memref.get_global operation. It would also be possible to query some alias + /// analysis to get more precise ownerships, however, the analysis would have + /// to be updated according to the IR modifications this pass performs (e.g., + /// re-building operations to have more result values, inserting clone + /// operations, etc.). + void populateOwnerships(Operation *op); + + /// Given two basic blocks and the values passed via block arguments to the + /// destination block, compute the list of MemRefs that have to be retained in + /// the 'fromBlock' to not run into a use-after-free situation. + /// This list consists of the MemRefs in the successor operand list of the + /// terminator and the MemRefs in the 'out' set of the liveness analysis + /// intersected with the 'in' set of the destination block. + /// + /// toRetain = filter(successorOperands + (liveOut(fromBlock) insersect + /// liveIn(toBlock)), isMemRef) + void getMemrefsToRetain(Block *fromBlock, Block *toBlock, + ValueRange destOperands, + SmallVectorImpl &toRetain); + + /// For a given block, computes the list of MemRefs that potentially need to + /// be deallocated at the end of that block. This list also contains values + /// that have to be retained (and are thus part of the list returned by + /// `getMemrefsToRetain`) and is computed by taking the MemRefs in the 'in' + /// set of the liveness analysis of 'block' appended by the set of MemRefs + /// allocated in 'block' itself and subtracted by the set of MemRefs + /// deallocated in 'block'. + /// Note that we don't have to take the intersection of the liveness 'in' set + /// with the 'out' set of the predecessor block because a value that is in the + /// 'in' set must be defined in an ancestor block that dominates all direct + /// predecessors and thus the 'in' set of this block is a subset of the 'out' + /// sets of each predecessor. + /// + /// memrefs = filter((liveIn(block) U + /// allocated(block) U arguments(block)) \ deallocated(block), isMemRef) + /// + /// The list of conditions is then populated by querying the internal + /// datastructures for the ownership value of that MemRef. + LogicalResult getMemrefsAndConditionsToDeallocate( + OpBuilder &builder, Location loc, Block *block, + SmallVectorImpl &memrefs, SmallVectorImpl &conditions); + + /// Given an SSA value of MemRef type, this function queries the ownership and + /// if it is not already in the 'Unique' state, potentially inserts IR to get + /// a new SSA value, returned as the first element of the pair, which has + /// 'Unique' ownership and can be used instead of the passed Value with the + /// the ownership indicator returned as the second element of the pair. + std::pair getMemrefWithUniqueOwnership(OpBuilder &builder, + Value memref); + + /// Given an SSA value of MemRef type, returns the same of a new SSA value + /// which has 'Unique' ownership where the ownership indicator is guaranteed + /// to be always 'true'. + Value getMemrefWithGuaranteedOwnership(OpBuilder &builder, Value memref); + + /// Returns whether the given operation implements FunctionOpInterface, has + /// private visibility, and the private-function-dynamic-ownership pass option + /// is enabled. + bool isFunctionWithoutDynamicOwnership(Operation *op); + + /// Checks all the preconditions that have to hold for the deallocation to be + /// applicable. + /// (1) Checks that there are not explicit control flow loops. + /// (2) Checks if all operations that have at least one attached region + /// implement the RegionBranchOpInterface. This is not required in edge cases, + /// where we have a single attached region and the parent operation has no + /// results. + /// (3) Checks that no deallocations already exist. Especially deallocations + /// in nested regions are not properly supported yet since this requires + /// ownership of the memref to be transferred to the nested region, which does + /// not happen by default. This constrained can be lifted in the future. + // (4) Checks that terminators with more than one successor except + // `cf.cond_br` are not present and that either BranchOpInterface or + // RegionBranchTerminatorOpInterface is implemented. + static LogicalResult verifyPreconditions(FunctionOpInterface op); + + /// When the 'private-function-dynamic-ownership' pass option is enabled, + /// additional `i1` arguments and return values are added for each MemRef + /// value in the function signature. This function takes care of updating the + /// `function_type` attribute of the function according to the actually + /// returned values from the terminators. + static LogicalResult updateFunctionSignature(FunctionOpInterface op); + private: - /// Introduces required clone operations to avoid memory leaks. - LogicalResult introduceClones() { - // Initialize the set of values that require a dedicated memory free - // operation since their operands cannot be safely deallocated in a post - // dominator. - SetVector valuesToFree; - llvm::SmallDenseSet> visitedValues; - SmallVector, 8> toProcess; - - // Check dominance relation for proper dominance properties. If the given - // value node does not dominate an alias, we will have to create a clone in - // order to free all buffers that can potentially leak into a post - // dominator. - auto findUnsafeValues = [&](Value source, Block *definingBlock) { - auto it = aliases.find(source); - if (it == aliases.end()) - return; - for (Value value : it->second) { - if (valuesToFree.count(value) > 0) - continue; - Block *parentBlock = value.getParentBlock(); - // Check whether we have to free this particular block argument or - // generic value. We have to free the current alias if it is either - // defined in a non-dominated block or it is defined in the same block - // but the current value is not dominated by the source value. - if (!dominators.dominates(definingBlock, parentBlock) || - (definingBlock == parentBlock && isa(value))) { - toProcess.emplace_back(value, parentBlock); - valuesToFree.insert(value); - } else if (visitedValues.insert(std::make_tuple(value, definingBlock)) - .second) - toProcess.emplace_back(value, definingBlock); - } - }; + // Mapping from each SSA value with MemRef type to the associated ownership in + // each block. + DenseMap, Ownership> ownershipMap; + + // Collects the list of MemRef values that potentially need to be deallocated + // per block. It is also fine (albeit not efficient) to add MemRef values that + // don't have to be deallocated, but only when the ownership is not 'Unknown'. + DenseMap> memrefsToDeallocatePerBlock; + + // Symbol cache to lookup functions from call operations to check attributes + // on the function operation. + SymbolTableCollection symbolTable; + + // The underlying liveness analysis to compute fine grained information about + // alloc and dealloc positions. + Liveness liveness; + + // A pass option indicating whether private functions should be modified to + // pass the ownership of MemRef values instead of adhering to the function + // boundary ABI. + bool privateFuncDynamicOwnership; +}; - // Detect possibly unsafe aliases starting from all allocations. - for (BufferPlacementAllocs::AllocEntry &entry : allocs) { - Value allocValue = std::get<0>(entry); - findUnsafeValues(allocValue, allocValue.getDefiningOp()->getBlock()); - } - // Try to find block arguments that require an explicit free operation - // until we reach a fix point. - while (!toProcess.empty()) { - auto current = toProcess.pop_back_val(); - findUnsafeValues(std::get<0>(current), std::get<1>(current)); +} // namespace + +//===----------------------------------------------------------------------===// +// BufferDeallocation Implementation +//===----------------------------------------------------------------------===// + +void BufferDeallocation::setOwnership(Value owned, Ownership ownership, + Block *block) { + // In most cases we care about the block where the value is defined. + if (block == nullptr) + block = owned.getParentBlock(); + + // Update ownership of current memref itself. + ownershipMap[{owned, block}].combine(ownership); +} + +void BufferDeallocation::clearOwnershipOf(ValueRange values, Block *block) { + for (Value val : values) { + ownershipMap[{val, block}] = Ownership::getUninitialized(); + } +} + +LogicalResult BufferDeallocation::verifyPreconditions(FunctionOpInterface op) { + // (1) Ensure that there are supported loops only (no explicit control flow + // loops). + Backedges backedges(op); + if (backedges.size()) { + op->emitError("Only structured control-flow loops are supported."); + return failure(); + } + + // (2) Check that the control flow structures are supported. + WalkResult result = op.getFunctionBody().walk([&](Operation *operation) { + auto regions = operation->getRegions(); + // Walk over all operations in a region and check if the operation has at + // least one region and implements the RegionBranchOpInterface. If there + // is an operation that does not fulfill this condition, we cannot apply + // the deallocation steps. Furthermore, we accept cases, where we have a + // region that returns no results, since, in that case, the intra-region + // control flow does not affect the transformation. + size_t size = regions.size(); + if (((size == 1 && !operation->getResults().empty()) || size > 1) && + !dyn_cast(operation)) { + operation->emitError("All operations with attached regions need to " + "implement the RegionBranchOpInterface."); + return WalkResult::interrupt(); } - // Update buffer aliases to ensure that we free all buffers and block - // arguments at the correct locations. - aliases.remove(valuesToFree); + return WalkResult::advance(); + }); + if (result.wasInterrupted()) + return failure(); - // Add new allocs and additional clone operations. - for (Value value : valuesToFree) { - if (failed(isa(value) - ? introduceBlockArgCopy(cast(value)) - : introduceValueCopyForRegionResult(value))) - return failure(); + // (3) The pass does not work properly when deallocations are already present. + // Alternatively, we could also remove all deallocations as a pre-pass. + result = op->walk([](Operation *op) { + return isa(op) ? WalkResult::interrupt() + : WalkResult::advance(); + }); + if (result.wasInterrupted()) + return emitError( + op->getLoc(), + "No deallocation operations must be present when running this pass!"); + + // (4) Check that terminators with more than one successor except `cf.cond_br` + // are not present and that either BranchOpInterface or + // RegionBranchTerminatorOpInterface is implemented. + result = op->walk([](Block *block) { + if (block->getParentOp()->hasTrait()) + return WalkResult::interrupt(); + Operation *terminator = block->getTerminator(); + if (!terminator) + return WalkResult::interrupt(); + + // Either one of those interfaces has to be implemented on terminators, but + // not both. + if (!isa( + terminator) && + !(isa(terminator) && + isa(terminator))) + return WalkResult::interrupt(); + + // We only support terminators with 0 or 1 successors for now and + // special-case the conditional branch op. + if (terminator->getSuccessors().size() > 1 && + !isa(terminator)) + return WalkResult::interrupt(); + + return WalkResult::advance(); + }); + if (result.wasInterrupted()) + return emitError( + op->getLoc(), + "Terminators with more than one successor " + "are not supported (except cf.cond_br) and either BranchOpInterface or " + "RegionBranchTerminatorOpInterface has to be implemented (but not " + "both)!"); + + return success(); +} + +LogicalResult +BufferDeallocation::updateFunctionSignature(FunctionOpInterface op) { + SmallVector returnOperandTypes(llvm::map_range( + op.getFunctionBody().getOps(), + [](RegionBranchTerminatorOpInterface op) { + return op.getSuccessorOperands(std::nullopt).getTypes(); + })); + if (!llvm::all_equal(returnOperandTypes)) + return failure(); + + // TODO: it would be nice if the FunctionOpInterface had a method to not only + // get the function type but also set it. + op->setAttr( + "function_type", + TypeAttr::get(FunctionType::get( + op->getContext(), op.getFunctionBody().front().getArgumentTypes(), + returnOperandTypes[0]))); + + return success(); +} + +LogicalResult BufferDeallocation::deallocate(FunctionOpInterface op) { + // Stop and emit a proper error message if we don't support the input IR. + if (failed(verifyPreconditions(op))) + return failure(); + + // Process the function block by block. + auto result = op->walk([&](Block *block) { + if (failed(deallocate(block))) + return WalkResult::interrupt(); + return WalkResult::advance(); + }); + if (result.wasInterrupted()) + return failure(); + + // Update the function signature if the function is private, dynamic ownership + // is enabled, and the function has memrefs as arguments or results. + return updateFunctionSignature(op); +} + +void BufferDeallocation::getMemrefsToRetain(Block *fromBlock, Block *toBlock, + ValueRange destOperands, + SmallVectorImpl &toRetain) { + for (Value operand : destOperands) { + if (!isMemref(operand)) + continue; + toRetain.push_back(operand); + } + + SmallPtrSet liveOut; + for (auto val : liveness.getLiveOut(fromBlock)) + if (isMemref(val)) + liveOut.insert(val); + + if (toBlock) + llvm::set_intersect(liveOut, liveness.getLiveIn(toBlock)); + + // liveOut has non-deterministic order because it was constructed by iterating + // over a hash-set. + SmallVector retainedByLiveness(liveOut.begin(), liveOut.end()); + std::sort(retainedByLiveness.begin(), retainedByLiveness.end(), + ValueComparator()); + toRetain.append(retainedByLiveness); +} + +LogicalResult BufferDeallocation::getMemrefsAndConditionsToDeallocate( + OpBuilder &builder, Location loc, Block *block, + SmallVectorImpl &memrefs, SmallVectorImpl &conditions) { + + for (auto [i, memref] : llvm::enumerate(memrefsToDeallocatePerBlock[block])) { + Ownership ownership = ownershipMap[{memref, block}]; + if (!ownership.isUnique()) + return emitError(memref.getLoc(), + "MemRef value does not have valid ownership"); + + // Simply cast unranked MemRefs to ranked memrefs with 0 dimensions such + // that we can call extract_strided_metadata on it. + if (auto unrankedMemRefTy = dyn_cast(memref.getType())) + memref = builder.create( + loc, MemRefType::get({}, unrankedMemRefTy.getElementType()), memref, + 0, SmallVector{}, SmallVector{}); + + memrefs.push_back( + builder.create(loc, memref) + .getResult(0)); + conditions.push_back(ownership.getIndicator()); + } + + return success(); +} + +LogicalResult BufferDeallocation::deallocate(Block *block) { + OpBuilder builder = OpBuilder::atBlockBegin(block); + + // Compute liveness transfers of ownership to this block. + for (auto li : liveness.getLiveIn(block)) { + if (!isMemref(li)) + continue; - // Register the value to require a final dealloc. Note that we do not have - // to assign a block here since we do not want to move the allocation node - // to another location. - allocs.registerAlloc(std::make_tuple(value, nullptr)); + if (li.getParentRegion() == block->getParent()) { + setOwnership(li, ownershipMap[{li, li.getParentBlock()}], block); + memrefsToDeallocatePerBlock[block].push_back(li); + continue; + } + + if (li.getParentRegion()->isProperAncestor(block->getParent())) { + Value falseVal = getBoolValue(builder, li.getLoc(), false); + setOwnership(li, falseVal, block); } - return success(); } - /// Introduces temporary clones in all predecessors and copies the source - /// values into the newly allocated buffers. - LogicalResult introduceBlockArgCopy(BlockArgument blockArg) { - // Allocate a buffer for the current block argument in the block of - // the associated value (which will be a predecessor block by - // definition). - Block *block = blockArg.getOwner(); - for (auto it = block->pred_begin(), e = block->pred_end(); it != e; ++it) { - // Get the terminator and the value that will be passed to our - // argument. - Operation *terminator = (*it)->getTerminator(); - auto branchInterface = cast(terminator); - SuccessorOperands operands = - branchInterface.getSuccessorOperands(it.getSuccessorIndex()); - - // Query the associated source value. - Value sourceValue = operands[blockArg.getArgNumber()]; - if (!sourceValue) { - return failure(); - } - // Wire new clone and successor operand. - // Create a new clone at the current location of the terminator. - auto clone = introduceCloneBuffers(sourceValue, terminator); - if (failed(clone)) - return failure(); - operands.slice(blockArg.getArgNumber(), 1).assign(*clone); + for (unsigned i = 0, e = block->getNumArguments(); i < e; ++i) { + BlockArgument arg = block->getArgument(i); + if (!isMemref(arg)) + continue; + + // Adhere to function boundary ABI + if (isFunctionWithoutDynamicOwnership(block->getParentOp()) && + block->isEntryBlock()) { + Value newArg = getBoolValue(builder, arg.getLoc(), false); + setOwnership(arg, newArg); + continue; } - // Check whether the block argument has implicitly defined predecessors via - // the RegionBranchOpInterface. This can be the case if the current block - // argument belongs to the first block in a region and the parent operation - // implements the RegionBranchOpInterface. - Region *argRegion = block->getParent(); - Operation *parentOp = argRegion->getParentOp(); - RegionBranchOpInterface regionInterface; - if (&argRegion->front() != block || - !(regionInterface = dyn_cast(parentOp))) - return success(); - - if (failed(introduceClonesForRegionSuccessors( - regionInterface, argRegion->getParentOp()->getRegions(), blockArg, - [&](RegionSuccessor &successorRegion) { - // Find a predecessor of our argRegion. - return successorRegion.getSuccessor() == argRegion; - }))) - return failure(); + // Pass MemRef ownerships along via `i1` values. + Value newArg = block->addArgument(builder.getI1Type(), arg.getLoc()); + setOwnership(arg, newArg); + memrefsToDeallocatePerBlock[block].push_back(arg); + } - // Check whether the block argument belongs to an entry region of the - // parent operation. In this case, we have to introduce an additional clone - // for buffer that is passed to the argument. - SmallVector successorRegions; - regionInterface.getSuccessorRegions(/*index=*/std::nullopt, - successorRegions); - auto *it = - llvm::find_if(successorRegions, [&](RegionSuccessor &successorRegion) { - return successorRegion.getSuccessor() == argRegion; - }); - if (it == successorRegions.end()) - return success(); - - // Determine the actual operand to introduce a clone for and rewire the - // operand to point to the clone instead. - auto operands = - regionInterface.getEntrySuccessorOperands(argRegion->getRegionNumber()); - size_t operandIndex = - llvm::find(it->getSuccessorInputs(), blockArg).getIndex() + - operands.getBeginOperandIndex(); - Value operand = parentOp->getOperand(operandIndex); - assert(operand == - operands[operandIndex - operands.getBeginOperandIndex()] && - "region interface operands don't match parentOp operands"); - auto clone = introduceCloneBuffers(operand, parentOp); - if (failed(clone)) + // For each operation in the block, handle the interfaces that affect aliasing + // and ownership of memrefs. + for (Operation &op : llvm::make_early_inc_range(*block)) { + FailureOr result = handleAllInterfaces(&op); + if (failed(result)) return failure(); - parentOp->setOperand(operandIndex, *clone); - return success(); + populateOwnerships(*result); + } + + // TODO: if block has no terminator, handle dealloc insertion here. + return success(); +} + +Operation *BufferDeallocation::appendOpResults(Operation *op, + ArrayRef types) { + SmallVector newTypes(op->getResultTypes()); + newTypes.append(types.begin(), types.end()); + auto *newOp = Operation::create(op->getLoc(), op->getName(), newTypes, + op->getOperands(), op->getAttrDictionary(), + op->getPropertiesStorage(), + op->getSuccessors(), op->getNumRegions()); + for (auto [oldRegion, newRegion] : + llvm::zip(op->getRegions(), newOp->getRegions())) + newRegion.takeBody(oldRegion); + + OpBuilder(op).insert(newOp); + op->replaceAllUsesWith(newOp->getResults().take_front(op->getNumResults())); + op->erase(); + + return newOp; +} + +FailureOr +BufferDeallocation::handleInterface(cf::CondBranchOp op) { + OpBuilder builder(op); + + // The list of memrefs to deallocate in this block is independent of which + // branch is taken. + SmallVector memrefs, conditions; + if (failed(getMemrefsAndConditionsToDeallocate( + builder, op.getLoc(), op->getBlock(), memrefs, conditions))) + return failure(); + + // Helper lambda to factor out common logic for inserting the dealloc + // operations for each successor. + auto insertDeallocForBranch = + [&](Block *target, MutableOperandRange destOperands, + const std::function &conditionModifier, + DenseMap &mapping) -> DeallocOp { + SmallVector toRetain; + getMemrefsToRetain(op->getBlock(), target, OperandRange(destOperands), + toRetain); + SmallVector adaptedConditions( + llvm::map_range(conditions, conditionModifier)); + auto deallocOp = builder.create( + op.getLoc(), memrefs, adaptedConditions, toRetain); + clearOwnershipOf(deallocOp.getRetained(), op->getBlock()); + for (auto [retained, ownership] : + llvm::zip(deallocOp.getRetained(), deallocOp.getUpdatedConditions())) { + setOwnership(retained, ownership, op->getBlock()); + mapping[retained] = ownership; + } + SmallVector replacements, ownerships; + for (Value operand : destOperands) { + replacements.push_back(operand); + if (isMemref(operand)) { + assert(mapping.contains(operand) && + "Should be contained at this point"); + ownerships.push_back(mapping[operand]); + } + } + replacements.append(ownerships); + destOperands.assign(replacements); + return deallocOp; + }; + + // Call the helper lambda and make sure the dealloc conditions are properly + // modified to reflect the branch condition as well. + DenseMap thenMapping, elseMapping; + DeallocOp thenTakenDeallocOp = insertDeallocForBranch( + op.getTrueDest(), op.getTrueDestOperandsMutable(), + [&](Value cond) { + return builder.create(op.getLoc(), cond, + op.getCondition()); + }, + thenMapping); + DeallocOp elseTakenDeallocOp = insertDeallocForBranch( + op.getFalseDest(), op.getFalseDestOperandsMutable(), + [&](Value cond) { + Value trueVal = builder.create( + op.getLoc(), builder.getBoolAttr(true)); + Value negation = builder.create(op.getLoc(), trueVal, + op.getCondition()); + return builder.create(op.getLoc(), cond, negation); + }, + elseMapping); + + // We specifically need to update the ownerships of values that are retained + // in both dealloc operations again to get a combined 'Unique' ownership + // instead of an 'Unknown' ownership. + SmallPtrSet thenValues(thenTakenDeallocOp.getRetained().begin(), + thenTakenDeallocOp.getRetained().end()); + SetVector commonValues; + for (Value val : elseTakenDeallocOp.getRetained()) { + if (thenValues.contains(val)) + commonValues.insert(val); + } + + for (Value retained : commonValues) { + clearOwnershipOf(retained, op->getBlock()); + Value combinedOwnership = builder.create( + op.getLoc(), op.getCondition(), thenMapping[retained], + elseMapping[retained]); + setOwnership(retained, combinedOwnership, op->getBlock()); } - /// Introduces temporary clones in front of all associated nested-region - /// terminators and copies the source values into the newly allocated buffers. - LogicalResult introduceValueCopyForRegionResult(Value value) { - // Get the actual result index in the scope of the parent terminator. - Operation *operation = value.getDefiningOp(); - auto regionInterface = cast(operation); - // Filter successors that return to the parent operation. - auto regionPredicate = [&](RegionSuccessor &successorRegion) { - // If the RegionSuccessor has no associated successor, it will return to - // its parent operation. - return !successorRegion.getSuccessor(); - }; - // Introduce a clone for all region "results" that are returned to the - // parent operation. This is required since the parent's result value has - // been considered critical. Therefore, the algorithm assumes that a clone - // of a previously allocated buffer is returned by the operation (like in - // the case of a block argument). - return introduceClonesForRegionSuccessors( - regionInterface, operation->getRegions(), value, regionPredicate); + return op.getOperation(); +} + +FailureOr +BufferDeallocation::handleInterface(RegionBranchOpInterface op) { + OpBuilder builder = OpBuilder::atBlockBegin(op->getBlock()); + + // TODO: the RegionBranchOpInterface does not provide all the necessary + // methods to perform this transformation without additional assumptions on + // the structure. For example, getting a mutable operand range of init args is + // missing or getting the (subset of) results matching the forwarded operands. + // Additionally, we also make these additional assumptions when we just append + // the arguments of all basic blocks (except entry blocks of functions) to + // have an ownership argument for each memref at the end of the arg list. It + // might be worth adding an interface method for adding additional arguments + // such that each op can implement it for itself. Otherwise, we have to live + // with these additional assumptions or build out the interface with a lot of + // other methods to implement it generically. + + SmallVector entryOperands(op.getEntrySuccessorOperands(0)); + unsigned numMemrefOperands = llvm::count_if(entryOperands, isMemref); + Value falseVal = getBoolValue(builder, op.getLoc(), false); + op->insertOperands(op->getNumOperands(), + SmallVector(numMemrefOperands, falseVal)); + + int counter = op->getNumResults(); + unsigned numMemrefResults = llvm::count_if(op->getResults(), isMemref); + SmallVector ownershipResults(numMemrefResults, builder.getI1Type()); + RegionBranchOpInterface newOp = appendOpResults(op, ownershipResults); + + for (auto result : llvm::make_filter_range(newOp->getResults(), isMemref)) { + setOwnership(result, newOp->getResult(counter++)); + memrefsToDeallocatePerBlock[newOp->getBlock()].push_back(result); + } + + return newOp.getOperation(); +} + +std::pair +BufferDeallocation::getMemrefWithUniqueOwnership(OpBuilder &builder, + Value memref) { + assert(ownershipMap.count({memref, memref.getParentBlock()}) && + "Value must already have been registered in the ownership map"); + + Ownership ownership = ownershipMap[{memref, memref.getParentBlock()}]; + if (ownership.isUnique()) + return {memref, ownership.getIndicator()}; + + // Instead of inserting a clone operation we could also insert a dealloc + // operation earlier in the block and use the updated ownerships returned by + // the op for the retained values. Alternatively, we could insert code to + // check aliasing at runtime and use this information to combine two unique + // ownerships more intelligently to not end up with an 'Unknown' ownership in + // the first place. + auto cloneOp = + builder.create(memref.getLoc(), memref); + Value condition = getBoolValue(builder, memref.getLoc(), true); + Value newMemref = cloneOp.getResult(); + setOwnership(newMemref, condition); + memrefsToDeallocatePerBlock[newMemref.getParentBlock()].push_back(newMemref); + return {newMemref, condition}; +} + +Value BufferDeallocation::getMemrefWithGuaranteedOwnership(OpBuilder &builder, + Value memref) { + // First, make sure we at least have 'Unique' ownership already. + std::pair newMemrefAndOnwership = + getMemrefWithUniqueOwnership(builder, memref); + Value newMemref = newMemrefAndOnwership.first; + Value condition = newMemrefAndOnwership.second; + + // Avoid inserting additional IR if ownership is already guaranteed. In + // particular, this is already the case when we had 'Unknown' ownership + // initially and a clone was inserted to get to 'Unique' ownership. + if (matchPattern(condition, m_One())) + return newMemref; + + // Insert a runtime check and only clone if we still don't have ownership at + // runtime. + Value maybeClone = + builder + .create( + memref.getLoc(), condition, + [&](OpBuilder &builder, Location loc) { + builder.create(loc, newMemref); + }, + [&](OpBuilder &builder, Location loc) { + Value clone = + builder.create(loc, newMemref); + builder.create(loc, clone); + }) + .getResult(0); + Value trueVal = getBoolValue(builder, memref.getLoc(), true); + setOwnership(maybeClone, trueVal); + memrefsToDeallocatePerBlock[maybeClone.getParentBlock()].push_back( + maybeClone); + return maybeClone; +} + +FailureOr +BufferDeallocation::handleInterface(BranchOpInterface op) { + // Skip conditional branches since we special case them for now. + if (isa(op.getOperation())) + return op.getOperation(); + + if (op->getNumSuccessors() == 0) + return emitError(op.getLoc(), "BranchOpInterface operations with zero " + "successors are not supported yet"); + + // Collect the values to deallocate and retain and use them to create the + // dealloc operation. + Block *block = op->getBlock(); + OpBuilder builder(op); + SmallVector memrefs, conditions, toRetain; + if (failed(getMemrefsAndConditionsToDeallocate(builder, op.getLoc(), block, + memrefs, conditions))) + return failure(); + + OperandRange forwardedOperands = + op.getSuccessorOperands(0).getForwardedOperands(); + getMemrefsToRetain(block, op->getSuccessor(0), forwardedOperands, toRetain); + + auto deallocOp = builder.create( + op.getLoc(), memrefs, conditions, toRetain); + + // We want to replace the current ownership of the retained values with the + // result values of the dealloc operation as they are always unique. + clearOwnershipOf(deallocOp.getRetained(), block); + for (auto [retained, ownership] : + llvm::zip(deallocOp.getRetained(), deallocOp.getUpdatedConditions())) { + setOwnership(retained, ownership, block); } - /// Introduces buffer clones for all terminators in the given regions. The - /// regionPredicate is applied to every successor region in order to restrict - /// the clones to specific regions. - template - LogicalResult introduceClonesForRegionSuccessors( - RegionBranchOpInterface regionInterface, MutableArrayRef regions, - Value argValue, const TPredicate ®ionPredicate) { - for (Region ®ion : regions) { - // Query the regionInterface to get all successor regions of the current - // one. - SmallVector successorRegions; - regionInterface.getSuccessorRegions(region.getRegionNumber(), - successorRegions); - // Try to find a matching region successor. - RegionSuccessor *regionSuccessor = - llvm::find_if(successorRegions, regionPredicate); - if (regionSuccessor == successorRegions.end()) + unsigned numAdditionalReturns = llvm::count_if(forwardedOperands, isMemref); + SmallVector newOperands(forwardedOperands); + auto additionalConditions = + deallocOp.getUpdatedConditions().take_front(numAdditionalReturns); + newOperands.append(additionalConditions.begin(), additionalConditions.end()); + op.getSuccessorOperands(0).getMutableForwardedOperands().assign(newOperands); + + return op.getOperation(); +} + +FailureOr BufferDeallocation::handleInterface(CallOpInterface op) { + OpBuilder builder(op); + + // Lookup the function operation and check if it has private visibility. If + // the function is referenced by SSA value instead of a Symbol, it's assumed + // to be always private. + Operation *funcOp = op.resolveCallable(&symbolTable); + bool isPrivate = true; + if (auto symbol = dyn_cast(funcOp)) + isPrivate &= (symbol.getVisibility() == SymbolTable::Visibility::Private); + + // If the private-function-dynamic-ownership option is enabled and we are + // calling a private function, we need to add an additional `i1` + // argument/result for each MemRef argument/result to dynamically pass the + // current ownership indicator rather than adhering to the function boundary + // ABI. + if (privateFuncDynamicOwnership && isPrivate) { + SmallVector newOperands, ownershipIndicatorsToAdd; + for (Value operand : op.getArgOperands()) { + if (!isMemref(operand)) { + newOperands.push_back(operand); continue; - // Get the operand index in the context of the current successor input - // bindings. - size_t operandIndex = - llvm::find(regionSuccessor->getSuccessorInputs(), argValue) - .getIndex(); - - std::optional successorRegionNumber; - if (Region *successorRegion = regionSuccessor->getSuccessor()) - successorRegionNumber = successorRegion->getRegionNumber(); - - // Iterate over all immediate terminator operations to introduce - // new buffer allocations. Thereby, the appropriate terminator operand - // will be adjusted to point to the newly allocated buffer instead. - if (failed(walkReturnOperations( - ®ion, [&](RegionBranchTerminatorOpInterface terminator) { - // Get the actual mutable operands for this terminator op. - auto terminatorOperands = - terminator.getMutableSuccessorOperands( - successorRegionNumber); - // Extract the source value from the current terminator. - // This conversion needs to exist on a separate line due to a - // bug in GCC conversion analysis. - OperandRange immutableTerminatorOperands = terminatorOperands; - Value sourceValue = immutableTerminatorOperands[operandIndex]; - // Create a new clone at the current location of the terminator. - auto clone = introduceCloneBuffers(sourceValue, terminator); - if (failed(clone)) - return failure(); - // Wire clone and terminator operand. - terminatorOperands.slice(operandIndex, 1).assign(*clone); - return success(); - }))) - return failure(); + } + auto [memref, condition] = getMemrefWithUniqueOwnership(builder, operand); + newOperands.push_back(memref); + ownershipIndicatorsToAdd.push_back(condition); + } + newOperands.append(ownershipIndicatorsToAdd.begin(), + ownershipIndicatorsToAdd.end()); + op.getArgOperandsMutable().assign(newOperands); + + unsigned numMemrefs = llvm::count_if(op->getResults(), isMemref); + SmallVector ownershipTypesToAppend(numMemrefs, builder.getI1Type()); + unsigned ownershipCounter = op->getNumResults(); + op = appendOpResults(op, ownershipTypesToAppend); + + for (auto result : llvm::make_filter_range(op->getResults(), isMemref)) { + setOwnership(result, op->getResult(ownershipCounter++)); + memrefsToDeallocatePerBlock[result.getParentBlock()].push_back(result); } - return success(); + + return op.getOperation(); } - /// Creates a new memory allocation for the given source value and clones - /// its content into the newly allocated buffer. The terminator operation is - /// used to insert the clone operation at the right place. - FailureOr introduceCloneBuffers(Value sourceValue, - Operation *terminator) { - // Avoid multiple clones of the same source value. This can happen in the - // presence of loops when a branch acts as a backedge while also having - // another successor that returns to its parent operation. Note: that - // copying copied buffers can introduce memory leaks since the invariant of - // BufferDeallocation assumes that a buffer will be only cloned once into a - // temporary buffer. Hence, the construction of clone chains introduces - // additional allocations that are not tracked automatically by the - // algorithm. - if (clonedValues.contains(sourceValue)) - return sourceValue; - // Create a new clone operation that copies the contents of the old - // buffer to the new one. - auto clone = buildClone(terminator, sourceValue); - if (succeeded(clone)) { - // Remember the clone of original source value. - clonedValues.insert(*clone); - } - return clone; + // According to the function boundary ABI we are guaranteed to get ownership + // of all MemRefs returned by the function. Thus we set ownership to constant + // 'true' and remember to deallocate it. + Value trueVal = getBoolValue(builder, op.getLoc(), true); + for (auto result : llvm::make_filter_range(op->getResults(), isMemref)) { + setOwnership(result, trueVal); + memrefsToDeallocatePerBlock[result.getParentBlock()].push_back(result); } - /// Finds correct dealloc positions according to the algorithm described at - /// the top of the file for all alloc nodes and block arguments that can be - /// handled by this analysis. - LogicalResult placeDeallocs() { - // Move or insert deallocs using the previously computed information. - // These deallocations will be linked to their associated allocation nodes - // since they don't have any aliases that can (potentially) increase their - // liveness. - for (const BufferPlacementAllocs::AllocEntry &entry : allocs) { - Value alloc = std::get<0>(entry); - auto aliasesSet = aliases.resolve(alloc); - assert(!aliasesSet.empty() && "must contain at least one alias"); - - // Determine the actual block to place the dealloc and get liveness - // information. - Block *placementBlock = - findCommonDominator(alloc, aliasesSet, postDominators); - const LivenessBlockInfo *livenessInfo = - liveness.getLiveness(placementBlock); - - // We have to ensure that the dealloc will be after the last use of all - // aliases of the given value. We first assume that there are no uses in - // the placementBlock and that we can safely place the dealloc at the - // beginning. - Operation *endOperation = &placementBlock->front(); - - // Iterate over all aliases and ensure that the endOperation will point - // to the last operation of all potential aliases in the placementBlock. - for (Value alias : aliasesSet) { - // Ensure that the start operation is at least the defining operation of - // the current alias to avoid invalid placement of deallocs for aliases - // without any uses. - Operation *beforeOp = endOperation; - if (alias.getDefiningOp() && - !(beforeOp = placementBlock->findAncestorOpInBlock( - *alias.getDefiningOp()))) - continue; - - Operation *aliasEndOperation = - livenessInfo->getEndOperation(alias, beforeOp); - // Check whether the aliasEndOperation lies in the desired block and - // whether it is behind the current endOperation. If yes, this will be - // the new endOperation. - if (aliasEndOperation->getBlock() == placementBlock && - endOperation->isBeforeInBlock(aliasEndOperation)) - endOperation = aliasEndOperation; - } - // endOperation is the last operation behind which we can safely store - // the dealloc taking all potential aliases into account. - - // If there is an existing dealloc, move it to the right place. - Operation *deallocOperation = std::get<1>(entry); - if (deallocOperation) { - deallocOperation->moveAfter(endOperation); - } else { - // If the Dealloc position is at the terminator operation of the - // block, then the value should escape from a deallocation. - Operation *nextOp = endOperation->getNextNode(); - if (!nextOp) - continue; - // If there is no dealloc node, insert one in the right place. - if (failed(buildDealloc(nextOp, alloc))) - return failure(); + return op.getOperation(); +} + +FailureOr +BufferDeallocation::handleInterface(MemoryEffectOpInterface op) { + auto *block = op->getBlock(); + + for (auto operand : llvm::make_filter_range(op->getOperands(), isMemref)) { + // TODO: existing deallocation operations are not properly supported yet. In + // particular, this only works if the originally allocated value is passed + // to he deallocation op, if a deallocation op is inside a nested region the + // ownership is also not properly handled because it is always assumed that + // ownership is not passed to a nested region (this could be improved by + // doing a pre-pass to collect for which values we have to pass ownership to + // which region, alternatively, we could pass ownership whenever there is no + // later use). + if (op.getEffectOnValue(operand).has_value()) + llvm::erase_if(memrefsToDeallocatePerBlock[block], + [&](const auto &memref) { return memref == operand; }); + } + + OpBuilder builder = OpBuilder::atBlockBegin(block); + for (auto res : llvm::make_filter_range(op->getResults(), isMemref)) { + auto allocEffect = op.getEffectOnValue(res); + if (allocEffect.has_value()) { + if (isa( + allocEffect->getResource())) { + // Make sure that the ownership of auto-managed allocations is set to + // false. This is important for operations that have at least one memref + // typed operand, e.g., something like `clone` but for auto-managed + // allocations, such that the default ownership population does not + // assume aliasing with those MemRef operands. + clearOwnershipOf(res, block); + setOwnership(res, getBoolValue(builder, op.getLoc(), false)); + continue; } + + setOwnership(res, getBoolValue(builder, op.getLoc(), true)); + memrefsToDeallocatePerBlock[block].push_back(res); } - return success(); } - /// Builds a deallocation operation compatible with the given allocation - /// value. If there is no registered AllocationOpInterface implementation for - /// the given value (e.g. in the case of a function parameter), this method - /// builds a memref::DeallocOp. - LogicalResult buildDealloc(Operation *op, Value alloc) { - OpBuilder builder(op); - auto it = aliasToAllocations.find(alloc); - if (it != aliasToAllocations.end()) { - // Call the allocation op interface to build a supported and - // compatible deallocation operation. - auto dealloc = it->second.buildDealloc(builder, alloc); - if (!dealloc) - return op->emitError() - << "allocations without compatible deallocations are " - "not supported"; - } else { - // Build a "default" DeallocOp for unknown allocation sources. - builder.create(alloc.getLoc(), alloc); + return op.getOperation(); +} + +FailureOr +BufferDeallocation::handleInterface(RegionBranchTerminatorOpInterface op) { + OpBuilder builder(op); + + // If this is a return operation of a function that is not private or the + // dynamic function boundary ownership is disabled, we need to return memref + // values for which we have guaranteed ownership to pass on to adhere to the + // function boundary ABI. + if (isFunctionWithoutDynamicOwnership(op->getParentOp())) { + for (OpOperand &val : op->getOpOperands()) { + if (!isMemref(val.get())) + continue; + + val.set(getMemrefWithGuaranteedOwnership(builder, val.get())); } - return success(); } - /// Builds a clone operation compatible with the given allocation value. If - /// there is no registered AllocationOpInterface implementation for the given - /// value (e.g. in the case of a function parameter), this method builds a - /// bufferization::CloneOp. - FailureOr buildClone(Operation *op, Value alloc) { - OpBuilder builder(op); - auto it = aliasToAllocations.find(alloc); - if (it != aliasToAllocations.end()) { - // Call the allocation op interface to build a supported and - // compatible clone operation. - auto clone = it->second.buildClone(builder, alloc); - if (clone) - return *clone; - return (LogicalResult)(op->emitError() - << "allocations without compatible clone ops " - "are not supported"); - } - // Build a "default" CloneOp for unknown allocation sources. - return builder.create(alloc.getLoc(), alloc) - .getResult(); + // TODO: getSuccessorRegions is not implemented by all operations we care + // about, but we would need to check now many successors there are and under + // wich condition they are taken, etc. + + MutableOperandRange operands = op.getMutableSuccessorOperands(std::nullopt); + + // Collect the values to deallocate and retain and use them to create the + // dealloc operation. + Block *block = op->getBlock(); + SmallVector memrefs, conditions, toRetain; + if (failed(getMemrefsAndConditionsToDeallocate(builder, op.getLoc(), block, + memrefs, conditions))) + return failure(); + + getMemrefsToRetain(block, nullptr, OperandRange(operands), toRetain); + auto deallocOp = builder.create( + op.getLoc(), memrefs, conditions, toRetain); + + // We want to replace the current ownership of the retained values with the + // result values of the dealloc operation as they are always unique. + clearOwnershipOf(deallocOp.getRetained(), block); + for (auto [retained, ownership] : + llvm::zip(deallocOp.getRetained(), deallocOp.getUpdatedConditions())) + setOwnership(retained, ownership, block); + + // Add an additional operand for every MemRef for the ownership indicator. + if (!isFunctionWithoutDynamicOwnership(block->getParentOp())) { + unsigned numMemRefs = llvm::count_if(operands, isMemref); + SmallVector newOperands{OperandRange(operands)}; + auto ownershipValues = + deallocOp.getUpdatedConditions().take_front(numMemRefs); + newOperands.append(ownershipValues.begin(), ownershipValues.end()); + operands.assign(newOperands); } - /// The dominator info to find the appropriate start operation to move the - /// allocs. - DominanceInfo dominators; + return op.getOperation(); +} + +bool BufferDeallocation::isFunctionWithoutDynamicOwnership(Operation *op) { + auto funcOp = dyn_cast(op); + return funcOp && (!privateFuncDynamicOwnership || + funcOp.getVisibility() != SymbolTable::Visibility::Private); +} - /// The post dominator info to move the dependent allocs in the right - /// position. - PostDominanceInfo postDominators; +void BufferDeallocation::populateOwnerships(Operation *op) { + for (auto res : op->getResults()) { + if (!isMemref(res)) + continue; + if (ownershipMap.count({res, op->getBlock()})) + continue; + + // Don't take ownership of a returned memref if no allocate side-effect is + // present, relevant for memref.get_global, for example. + if (op->getNumOperands() == 0) { + OpBuilder builder(op); + setOwnership(res, getBoolValue(builder, op->getLoc(), false)); + continue; + } - /// Stores already cloned buffers to avoid additional clones of clones. - ValueSetT clonedValues; + // Assume the result may alias with any operand and thus combine all their + // ownerships. + for (auto operand : op->getOperands()) { + if (!isMemref(operand)) + continue; - /// Maps aliases to their source allocation interfaces (inverse mapping). - AliasAllocationMapT aliasToAllocations; -}; + ownershipMap[{res, op->getBlock()}].combine( + ownershipMap[{operand, operand.getParentBlock()}]); + } + } +} //===----------------------------------------------------------------------===// // BufferDeallocationPass //===----------------------------------------------------------------------===// -struct DefaultAllocationInterface - : public bufferization::AllocationOpInterface::ExternalModel< - DefaultAllocationInterface, memref::AllocOp> { - static std::optional buildDealloc(OpBuilder &builder, - Value alloc) { - return builder.create(alloc.getLoc(), alloc) - .getOperation(); - } - static std::optional buildClone(OpBuilder &builder, Value alloc) { - return builder.create(alloc.getLoc(), alloc) - .getResult(); - } -}; - -struct DefaultReallocationInterface - : public bufferization::AllocationOpInterface::ExternalModel< - DefaultAllocationInterface, memref::ReallocOp> { - static std::optional buildDealloc(OpBuilder &builder, - Value realloc) { - return builder.create(realloc.getLoc(), realloc) - .getOperation(); - } -}; +namespace { /// The actual buffer deallocation pass that inserts and moves dealloc nodes /// into the right positions. Furthermore, it inserts additional clones if @@ -662,66 +1284,30 @@ struct BufferDeallocationPass : public bufferization::impl::BufferDeallocationBase< BufferDeallocationPass> { - void getDependentDialects(DialectRegistry ®istry) const override { - registry.insert(); - registry.insert(); - registerAllocationOpInterfaceExternalModels(registry); - } - void runOnOperation() override { func::FuncOp func = getOperation(); if (func.isExternal()) return; - if (failed(deallocateBuffers(func))) + if (failed(deallocateBuffers(func, privateFuncDynamicOwnership))) signalPassFailure(); } }; } // namespace -LogicalResult bufferization::deallocateBuffers(Operation *op) { - if (isa(op)) { - WalkResult result = op->walk([&](func::FuncOp funcOp) { - if (failed(deallocateBuffers(funcOp))) - return WalkResult::interrupt(); - return WalkResult::advance(); - }); - return success(!result.wasInterrupted()); - } - - // Ensure that there are supported loops only. - Backedges backedges(op); - if (backedges.size()) { - op->emitError("Only structured control-flow loops are supported."); - return failure(); - } - - // Check that the control flow structures are supported. - if (!validateSupportedControlFlow(op)) - return failure(); +//===----------------------------------------------------------------------===// +// Implement bufferization API +//===----------------------------------------------------------------------===// +LogicalResult +bufferization::deallocateBuffers(FunctionOpInterface op, + bool privateFuncDynamicOwnership) { // Gather all required allocation nodes and prepare the deallocation phase. - BufferDeallocation deallocation(op); - - // Check for supported AllocationOpInterface implementations and prepare the - // internal deallocation pass. - if (failed(deallocation.prepare())) - return failure(); + BufferDeallocation deallocation(op, privateFuncDynamicOwnership); // Place all required temporary clone and dealloc nodes. - if (failed(deallocation.deallocate())) - return failure(); - - return success(); -} - -void bufferization::registerAllocationOpInterfaceExternalModels( - DialectRegistry ®istry) { - registry.addExtension(+[](MLIRContext *ctx, memref::MemRefDialect *dialect) { - memref::AllocOp::attachInterface(*ctx); - memref::ReallocOp::attachInterface(*ctx); - }); + return deallocation.deallocate(op); } //===----------------------------------------------------------------------===// diff --git a/mlir/lib/Dialect/Bufferization/Transforms/BufferUtils.cpp b/mlir/lib/Dialect/Bufferization/Transforms/BufferUtils.cpp --- a/mlir/lib/Dialect/Bufferization/Transforms/BufferUtils.cpp +++ b/mlir/lib/Dialect/Bufferization/Transforms/BufferUtils.cpp @@ -201,3 +201,62 @@ global->moveBefore(&moduleOp.front()); return global; } + +//===----------------------------------------------------------------------===// +// ValueComparator +//===----------------------------------------------------------------------===// + +bool ValueComparator::operator()(const Value &lhs, const Value &rhs) const { + if (lhs == rhs) + return false; + + // Block arguments are less than results. + bool lhsIsBBArg = lhs.isa(); + if (lhsIsBBArg != rhs.isa()) { + return lhsIsBBArg; + } + + Region *lhsRegion; + Region *rhsRegion; + if (lhsIsBBArg) { + auto lhsBBArg = llvm::cast(lhs); + auto rhsBBArg = llvm::cast(rhs); + if (lhsBBArg.getArgNumber() != rhsBBArg.getArgNumber()) { + return lhsBBArg.getArgNumber() < rhsBBArg.getArgNumber(); + } + lhsRegion = lhsBBArg.getParentRegion(); + rhsRegion = rhsBBArg.getParentRegion(); + assert(lhsRegion != rhsRegion && + "lhsRegion == rhsRegion implies lhs == rhs"); + } else if (lhs.getDefiningOp() == rhs.getDefiningOp()) { + return llvm::cast(lhs).getResultNumber() < + llvm::cast(rhs).getResultNumber(); + } else { + lhsRegion = lhs.getDefiningOp()->getParentRegion(); + rhsRegion = rhs.getDefiningOp()->getParentRegion(); + if (lhsRegion == rhsRegion) { + return lhs.getDefiningOp()->isBeforeInBlock(rhs.getDefiningOp()); + } + } + + // lhsRegion != rhsRegion, so if we look at their ancestor chain, they + // - have different heights + // - or there's a spot where their region numbers differ + // - or their parent regions are the same and their parent ops are + // different. + while (lhsRegion && rhsRegion) { + if (lhsRegion->getRegionNumber() != rhsRegion->getRegionNumber()) { + return lhsRegion->getRegionNumber() < rhsRegion->getRegionNumber(); + } + if (lhsRegion->getParentRegion() == rhsRegion->getParentRegion()) { + return lhsRegion->getParentOp()->isBeforeInBlock( + rhsRegion->getParentOp()); + } + lhsRegion = lhsRegion->getParentRegion(); + rhsRegion = rhsRegion->getParentRegion(); + } + if (rhsRegion) + return true; + assert(lhsRegion && "this should only happen if lhs == rhs"); + return false; +} diff --git a/mlir/lib/Dialect/Bufferization/Transforms/Bufferize.cpp b/mlir/lib/Dialect/Bufferization/Transforms/Bufferize.cpp --- a/mlir/lib/Dialect/Bufferization/Transforms/Bufferize.cpp +++ b/mlir/lib/Dialect/Bufferization/Transforms/Bufferize.cpp @@ -194,7 +194,6 @@ void getDependentDialects(DialectRegistry ®istry) const override { registry .insert(); - registerAllocationOpInterfaceExternalModels(registry); } void runOnOperation() override { diff --git a/mlir/lib/Dialect/Bufferization/Transforms/CMakeLists.txt b/mlir/lib/Dialect/Bufferization/Transforms/CMakeLists.txt --- a/mlir/lib/Dialect/Bufferization/Transforms/CMakeLists.txt +++ b/mlir/lib/Dialect/Bufferization/Transforms/CMakeLists.txt @@ -30,6 +30,8 @@ MLIRMemRefDialect MLIRPass MLIRTensorDialect + MLIRSCFDialect + MLIRControlFlowDialect MLIRSideEffectInterfaces MLIRTransforms MLIRViewLikeInterface diff --git a/mlir/test/Dialect/Bufferization/Transforms/BufferDeallocation/dealloc-branchop-interface.mlir b/mlir/test/Dialect/Bufferization/Transforms/BufferDeallocation/dealloc-branchop-interface.mlir new file mode 100644 --- /dev/null +++ b/mlir/test/Dialect/Bufferization/Transforms/BufferDeallocation/dealloc-branchop-interface.mlir @@ -0,0 +1,566 @@ +// DEFINE: %{canonicalize} = -canonicalize=enable-patterns="bufferization-skip-extract-metadata-of-alloc,bufferization-erase-always-false-dealloc,bufferization-erase-empty-dealloc,bufferization-dealloc-remove-duplicate-retained-memrefs,bufferization-dealloc-remove-duplicate-dealloc-memrefs",region-simplify=false + +// RUN: mlir-opt -verify-diagnostics -buffer-deallocation %{canonicalize} \ +// RUN: -buffer-deallocation-simplification %{canonicalize} -split-input-file %s | FileCheck %s +// RUN: mlir-opt -verify-diagnostics -buffer-deallocation=private-function-dynamic-ownership=true -split-input-file %s > /dev/null + +// Test Case: +// bb0 +// / \ +// bb1 bb2 <- Initial position of AllocOp +// \ / +// bb3 +// BufferDeallocation expected behavior: bb2 contains an AllocOp which is +// passed to bb3. In the latter block, there should be a deallocation. +// Since bb1 does not contain an adequate alloc, the deallocation has to be +// made conditional on the branch taken in bb0. + +func.func @condBranch(%arg0: i1, %arg1: memref<2xf32>, %arg2: memref<2xf32>) { + cf.cond_br %arg0, ^bb2(%arg1 : memref<2xf32>), ^bb1 +^bb1: + %0 = memref.alloc() : memref<2xf32> + test.buffer_based in(%arg1: memref<2xf32>) out(%0: memref<2xf32>) + cf.br ^bb2(%0 : memref<2xf32>) +^bb2(%1: memref<2xf32>): + test.copy(%1, %arg2) : (memref<2xf32>, memref<2xf32>) + return +} + +// CHECK-LABEL: func @condBranch +// CHECK-SAME: ([[ARG0:%.+]]: i1, +// CHECK-SAME: [[ARG1:%.+]]: memref<2xf32>, +// CHECK-SAME: [[ARG2:%.+]]: memref<2xf32>) +// CHECK-NOT: bufferization.dealloc +// CHECK: cf.cond_br{{.*}}, ^bb2([[ARG1]], %false{{[0-9_]*}} :{{.*}}), ^bb1 +// CHECK: ^bb1: +// CHECK: %[[ALLOC1:.*]] = memref.alloc +// CHECK-NEXT: test.buffer_based +// CHECK-NEXT: cf.br ^bb2(%[[ALLOC1]], %true +// CHECK-NEXT: ^bb2([[ALLOC2:%.+]]: memref<2xf32>, [[COND1:%.+]]: i1): +// CHECK: test.copy +// CHECK-NEXT: [[BASE:%[a-zA-Z0-9_]+]]{{.*}} = memref.extract_strided_metadata [[ALLOC2]] +// CHECK-NEXT: bufferization.dealloc ([[BASE]] : {{.*}}) if ([[COND1]]) +// CHECK-NEXT: return + +// ----- + +// Test Case: +// bb0 +// / \ +// bb1 bb2 <- Initial position of AllocOp +// \ / +// bb3 +// BufferDeallocation expected behavior: The existing AllocOp has a dynamic +// dependency to block argument %0 in bb2. Since the dynamic type is passed +// to bb3 via the block argument %2, it is currently required to allocate a +// temporary buffer for %2 that gets copies of %arg0 and %1 with their +// appropriate shape dimensions. The copy buffer deallocation will be applied +// to %2 in block bb3. + +func.func @condBranchDynamicType( + %arg0: i1, + %arg1: memref, + %arg2: memref, + %arg3: index) { + cf.cond_br %arg0, ^bb2(%arg1 : memref), ^bb1(%arg3: index) +^bb1(%0: index): + %1 = memref.alloc(%0) : memref + test.buffer_based in(%arg1: memref) out(%1: memref) + cf.br ^bb2(%1 : memref) +^bb2(%2: memref): + test.copy(%2, %arg2) : (memref, memref) + return +} + +// CHECK-LABEL: func @condBranchDynamicType +// CHECK-SAME: ([[ARG0:%.+]]: i1, [[ARG1:%.+]]: memref, [[ARG2:%.+]]: memref, [[ARG3:%.+]]: index) +// CHECK-NOT: bufferization.dealloc +// CHECK: cf.cond_br{{.*}}^bb2(%arg1, %false{{[0-9_]*}} :{{.*}}), ^bb1 +// CHECK: ^bb1([[IDX:%.*]]:{{.*}}) +// CHECK: [[ALLOC1:%.*]] = memref.alloc([[IDX]]) +// CHECK-NEXT: test.buffer_based +// CHECK-NEXT: cf.br ^bb2([[ALLOC1]], %true +// CHECK-NEXT: ^bb2([[ALLOC3:%.*]]:{{.*}}, [[COND:%.+]]:{{.*}}) +// CHECK: test.copy([[ALLOC3]], +// CHECK-NEXT: [[BASE:%[a-zA-Z0-9_]+]]{{.*}} = memref.extract_strided_metadata [[ALLOC3]] +// CHECK-NEXT: bufferization.dealloc ([[BASE]] : {{.*}}) if ([[COND]]) +// CHECK-NEXT: return + +// ----- + +// Test case: See above. + +func.func @condBranchUnrankedType( + %arg0: i1, + %arg1: memref<*xf32>, + %arg2: memref<*xf32>, + %arg3: index) { + cf.cond_br %arg0, ^bb2(%arg1 : memref<*xf32>), ^bb1(%arg3: index) +^bb1(%0: index): + %1 = memref.alloc(%0) : memref + %2 = memref.cast %1 : memref to memref<*xf32> + test.buffer_based in(%arg1: memref<*xf32>) out(%2: memref<*xf32>) + cf.br ^bb2(%2 : memref<*xf32>) +^bb2(%3: memref<*xf32>): + test.copy(%3, %arg2) : (memref<*xf32>, memref<*xf32>) + return +} + +// CHECK-LABEL: func @condBranchUnrankedType +// CHECK-SAME: ([[ARG0:%.+]]: i1, [[ARG1:%.+]]: memref<*xf32>, [[ARG2:%.+]]: memref<*xf32>, [[ARG3:%.+]]: index) +// CHECK-NOT: bufferization.dealloc +// CHECK: cf.cond_br{{.*}}^bb2([[ARG1]], %false{{[0-9_]*}} :{{.*}}), ^bb1 +// CHECK: ^bb1([[IDX:%.*]]:{{.*}}) +// CHECK: [[ALLOC1:%.*]] = memref.alloc([[IDX]]) +// CHECK-NEXT: [[CAST:%.+]] = memref.cast [[ALLOC1]] +// CHECK-NEXT: test.buffer_based +// CHECK-NEXT: cf.br ^bb2([[CAST]], %true +// CHECK-NEXT: ^bb2([[ALLOC3:%.*]]:{{.*}}, [[COND:%.+]]:{{.*}}) +// CHECK: test.copy([[ALLOC3]], +// CHECK-NEXT: [[CAST:%.+]] = memref.reinterpret_cast [[ALLOC3]] +// CHECK-NEXT: [[BASE:%[a-zA-Z0-9_]+]]{{.*}} = memref.extract_strided_metadata [[CAST]] +// CHECK-NEXT: bufferization.dealloc ([[BASE]] : {{.*}}) if ([[COND]]) +// CHECK-NEXT: return + +// TODO: we can get rid of first dealloc by doing some must-alias analysis + +// ----- + +// Test Case: +// bb0 +// / \ +// bb1 bb2 <- Initial position of AllocOp +// | / \ +// | bb3 bb4 +// | \ / +// \ bb5 +// \ / +// bb6 +// | +// bb7 +// BufferDeallocation expected behavior: The existing AllocOp has a dynamic +// dependency to block argument %0 in bb2. Since the dynamic type is passed to +// bb5 via the block argument %2 and to bb6 via block argument %3, it is +// currently required to pass along the condition under which the newly +// allocated buffer should be deallocated, since the path via bb1 does not +// allocate a buffer. + +func.func @condBranchDynamicTypeNested( + %arg0: i1, + %arg1: memref, + %arg2: memref, + %arg3: index) { + cf.cond_br %arg0, ^bb1, ^bb2(%arg3: index) +^bb1: + cf.br ^bb6(%arg1 : memref) +^bb2(%0: index): + %1 = memref.alloc(%0) : memref + test.buffer_based in(%arg1: memref) out(%1: memref) + cf.cond_br %arg0, ^bb3, ^bb4 +^bb3: + cf.br ^bb5(%1 : memref) +^bb4: + cf.br ^bb5(%1 : memref) +^bb5(%2: memref): + cf.br ^bb6(%2 : memref) +^bb6(%3: memref): + cf.br ^bb7(%3 : memref) +^bb7(%4: memref): + test.copy(%4, %arg2) : (memref, memref) + return +} + +// CHECK-LABEL: func @condBranchDynamicTypeNested +// CHECK-SAME: ([[ARG0:%.+]]: i1, [[ARG1:%.+]]: memref, [[ARG2:%.+]]: memref, [[ARG3:%.+]]: index) +// CHECK-NOT: bufferization.dealloc +// CHECK-NOT: bufferization.clone +// CHECK: cf.cond_br{{.*}} +// CHECK-NEXT: ^bb1 +// CHECK-NOT: bufferization.dealloc +// CHECK-NOT: bufferization.clone +// CHECK: cf.br ^bb5([[ARG1]], %false{{[0-9_]*}} : +// CHECK: ^bb2([[IDX:%.*]]:{{.*}}) +// CHECK: [[ALLOC1:%.*]] = memref.alloc([[IDX]]) +// CHECK-NEXT: test.buffer_based +// CHECK-NEXT: [[NOT_ARG0:%.+]] = arith.xori [[ARG0]], %true +// CHECK-NEXT: [[OWN:%.+]] = arith.select [[ARG0]], [[ARG0]], [[NOT_ARG0]] +// CHECK-NOT: bufferization.dealloc +// CHECK-NOT: bufferization.clone +// CHECK: cf.cond_br{{.*}}, ^bb3, ^bb3 +// CHECK-NEXT: ^bb3: +// CHECK-NOT: bufferization.dealloc +// CHECK-NOT: bufferization.clone +// CHECK: cf.br ^bb4([[ALLOC1]], [[OWN]] +// CHECK-NEXT: ^bb4([[ALLOC2:%.*]]:{{.*}}, [[COND1:%.+]]:{{.*}}) +// CHECK-NOT: bufferization.dealloc +// CHECK-NOT: bufferization.clone +// CHECK: cf.br ^bb5([[ALLOC2]], [[COND1]] +// CHECK-NEXT: ^bb5([[ALLOC4:%.*]]:{{.*}}, [[COND2:%.+]]:{{.*}}) +// CHECK-NEXT: [[BASE:%[a-zA-Z0-9_]+]]{{.*}} = memref.extract_strided_metadata [[ALLOC4]] +// CHECK-NEXT: [[OWN:%.+]]:2 = bufferization.dealloc ([[BASE]] :{{.*}}) if ([[COND2]]) retain ([[ALLOC4]], [[ARG2]] : +// CHECK: cf.br ^bb6([[ALLOC4]], [[OWN]]#0 +// CHECK-NEXT: ^bb6([[ALLOC5:%.*]]:{{.*}}, [[COND3:%.+]]:{{.*}}) +// CHECK: test.copy +// CHECK: [[BASE:%[a-zA-Z0-9_]+]]{{.*}} = memref.extract_strided_metadata [[ALLOC5]] +// CHECK-NEXT: bufferization.dealloc ([[BASE]] : {{.*}}) if ([[COND3]]) +// CHECK-NEXT: return + +// TODO: the dealloc in bb5 can be optimized away by adding another +// canonicalization pattern + +// ----- + +// Test Case: +// bb0 +// / \ +// | bb1 <- Initial position of AllocOp +// \ / +// bb2 +// BufferDeallocation expected behavior: It should insert a DeallocOp at the +// exit block after CopyOp since %1 is an alias for %0 and %arg1. + +func.func @criticalEdge(%arg0: i1, %arg1: memref<2xf32>, %arg2: memref<2xf32>) { + cf.cond_br %arg0, ^bb1, ^bb2(%arg1 : memref<2xf32>) +^bb1: + %0 = memref.alloc() : memref<2xf32> + test.buffer_based in(%arg1: memref<2xf32>) out(%0: memref<2xf32>) + cf.br ^bb2(%0 : memref<2xf32>) +^bb2(%1: memref<2xf32>): + test.copy(%1, %arg2) : (memref<2xf32>, memref<2xf32>) + return +} + +// CHECK-LABEL: func @criticalEdge +// CHECK-SAME: ([[ARG0:%.+]]: i1, [[ARG1:%.+]]: memref<2xf32>, [[ARG2:%.+]]: memref<2xf32>) +// CHECK-NOT: bufferization.dealloc +// CHECK-NOT: bufferization.clone +// CHECK: cf.cond_br{{.*}}, ^bb1, ^bb2([[ARG1]], %false +// CHECK: [[ALLOC1:%.*]] = memref.alloc() +// CHECK-NEXT: test.buffer_based +// CHECK-NEXT: cf.br ^bb2([[ALLOC1]], %true +// CHECK-NEXT: ^bb2([[ALLOC2:%.+]]:{{.*}}, [[COND:%.+]]: {{.*}}) +// CHECK: test.copy +// CHECK-NEXT: [[BASE:%[a-zA-Z0-9_]+]]{{.*}} = memref.extract_strided_metadata [[ALLOC2]] +// CHECK-NEXT: bufferization.dealloc ([[BASE]] : {{.*}}) if ([[COND]]) +// CHECK-NEXT: return + +// ----- + +// Test Case: +// bb0 <- Initial position of AllocOp +// / \ +// | bb1 +// \ / +// bb2 +// BufferDeallocation expected behavior: It only inserts a DeallocOp at the +// exit block after CopyOp since %1 is an alias for %0 and %arg1. + +func.func @invCriticalEdge(%arg0: i1, %arg1: memref<2xf32>, %arg2: memref<2xf32>) { + %0 = memref.alloc() : memref<2xf32> + test.buffer_based in(%arg1: memref<2xf32>) out(%0: memref<2xf32>) + cf.cond_br %arg0, ^bb1, ^bb2(%arg1 : memref<2xf32>) +^bb1: + cf.br ^bb2(%0 : memref<2xf32>) +^bb2(%1: memref<2xf32>): + test.copy(%1, %arg2) : (memref<2xf32>, memref<2xf32>) + return +} + +// CHECK-LABEL: func @invCriticalEdge +// CHECK-SAME: ([[ARG0:%.+]]: i1, [[ARG1:%.+]]: memref<2xf32>, [[ARG2:%.+]]: memref<2xf32>) +// CHECK: [[ALLOC:%.+]] = memref.alloc() +// CHECK-NEXT: test.buffer_based +// CHECK-NEXT: [[NOT_ARG0:%.+]] = arith.xori [[ARG0]], %true +// CHECK-NEXT: bufferization.dealloc ([[ALLOC]] : {{.*}}) if ([[NOT_ARG0]]) +// CHECK-NEXT: cf.cond_br{{.*}}^bb1, ^bb2([[ARG1]], %false +// CHECK-NEXT: ^bb1: +// CHECK-NOT: bufferization.dealloc +// CHECK-NOT: bufferization.clone +// CHECK: cf.br ^bb2([[ALLOC]], [[ARG0]] +// CHECK-NEXT: ^bb2([[ALLOC1:%.+]]:{{.*}}, [[COND:%.+]]:{{.*}}) +// CHECK: test.copy +// CHECK-NEXT: [[BASE:%[a-zA-Z0-9_]+]]{{.*}} = memref.extract_strided_metadata [[ALLOC1]] +// CHECK-NEXT: bufferization.dealloc ([[BASE]] : {{.*}}) if ([[COND]]) +// CHECK-NEXT: return + +// ----- + +// Test Case: +// bb0 <- Initial position of the first AllocOp +// / \ +// bb1 bb2 +// \ / +// bb3 <- Initial position of the second AllocOp +// BufferDeallocation expected behavior: It only inserts two missing +// DeallocOps in the exit block. %5 is an alias for %0. Therefore, the +// DeallocOp for %0 should occur after the last BufferBasedOp. The Dealloc for +// %7 should happen after CopyOp. + +func.func @ifElse(%arg0: i1, %arg1: memref<2xf32>, %arg2: memref<2xf32>) { + %0 = memref.alloc() : memref<2xf32> + test.buffer_based in(%arg1: memref<2xf32>) out(%0: memref<2xf32>) + cf.cond_br %arg0, + ^bb1(%arg1, %0 : memref<2xf32>, memref<2xf32>), + ^bb2(%0, %arg1 : memref<2xf32>, memref<2xf32>) +^bb1(%1: memref<2xf32>, %2: memref<2xf32>): + cf.br ^bb3(%1, %2 : memref<2xf32>, memref<2xf32>) +^bb2(%3: memref<2xf32>, %4: memref<2xf32>): + cf.br ^bb3(%3, %4 : memref<2xf32>, memref<2xf32>) +^bb3(%5: memref<2xf32>, %6: memref<2xf32>): + %7 = memref.alloc() : memref<2xf32> + test.buffer_based in(%5: memref<2xf32>) out(%7: memref<2xf32>) + test.copy(%7, %arg2) : (memref<2xf32>, memref<2xf32>) + return +} + +// CHECK-LABEL: func @ifElse +// CHECK-SAME: ([[ARG0:%.+]]: i1, [[ARG1:%.+]]: memref<2xf32>, [[ARG2:%.+]]: memref<2xf32>) +// CHECK: [[ALLOC0:%.+]] = memref.alloc() +// CHECK-NEXT: test.buffer_based +// CHECK-NOT: bufferization.dealloc +// CHECK-NOT: bufferization.clone +// CHECK-NEXT: [[NOT_ARG0:%.+]] = arith.xori [[ARG0]], %true +// CHECK-NEXT: cf.cond_br {{.*}}^bb1([[ARG1]], [[ALLOC0]], %false{{[0-9_]*}}, [[ARG0]] : {{.*}}), ^bb1([[ALLOC0]], [[ARG1]], [[NOT_ARG0]], %false{{[0-9_]*}} : {{.*}}) +// CHECK: ^bb2([[A0:%.+]]:{{.*}}, [[A1:%.+]]:{{.*}}, [[COND0:%.+]]: i1, [[COND1:%.+]]: i1): +// CHECK: [[ALLOC1:%.+]] = memref.alloc() +// CHECK-NEXT: test.buffer_based +// CHECK-NEXT: test.copy +// CHECK-NEXT: [[BASE0:%[a-zA-Z0-9_]+]]{{.*}} = memref.extract_strided_metadata [[A0]] +// CHECK-NEXT: [[BASE1:%[a-zA-Z0-9_]+]]{{.*}} = memref.extract_strided_metadata [[A1]] +// CHECK-NEXT: bufferization.dealloc ([[ALLOC1]] : {{.*}}) if (%true +// CHECK-NOT: retain +// CHECK-NEXT: bufferization.dealloc ([[BASE0]], [[BASE1]] : {{.*}}) if ([[COND0]], [[COND1]]) +// CHECK-NOT: retain +// CHECK-NEXT: return + +// TODO: Instead of deallocating the bbarg memrefs, a slightly better analysis +// could do an unconditional deallocation on ALLOC0 and move it before the +// test.copy (dealloc of ALLOC1 would remain after the copy) + +// ----- + +// Test Case: No users for buffer in if-else CFG +// bb0 <- Initial position of AllocOp +// / \ +// bb1 bb2 +// \ / +// bb3 +// BufferDeallocation expected behavior: It only inserts a missing DeallocOp +// in the exit block since %5 or %6 are the latest aliases of %0. + +func.func @ifElseNoUsers(%arg0: i1, %arg1: memref<2xf32>, %arg2: memref<2xf32>) { + %0 = memref.alloc() : memref<2xf32> + test.buffer_based in(%arg1: memref<2xf32>) out(%0: memref<2xf32>) + cf.cond_br %arg0, + ^bb1(%arg1, %0 : memref<2xf32>, memref<2xf32>), + ^bb2(%0, %arg1 : memref<2xf32>, memref<2xf32>) +^bb1(%1: memref<2xf32>, %2: memref<2xf32>): + cf.br ^bb3(%1, %2 : memref<2xf32>, memref<2xf32>) +^bb2(%3: memref<2xf32>, %4: memref<2xf32>): + cf.br ^bb3(%3, %4 : memref<2xf32>, memref<2xf32>) +^bb3(%5: memref<2xf32>, %6: memref<2xf32>): + test.copy(%arg1, %arg2) : (memref<2xf32>, memref<2xf32>) + return +} + +// CHECK-LABEL: func @ifElseNoUsers +// CHECK-SAME: ([[ARG0:%.+]]: i1, [[ARG1:%.+]]: memref<2xf32>, [[ARG2:%.+]]: memref<2xf32>) +// CHECK: [[ALLOC:%.+]] = memref.alloc() +// CHECK-NEXT: test.buffer_based +// CHECK-NEXT: [[NOT_ARG0:%.+]] = arith.xori [[ARG0]], %true +// CHECK-NEXT: cf.cond_br {{.*}}^bb1([[ARG1]], [[ALLOC]], %false{{[0-9_]*}}, [[ARG0]] : {{.*}}), ^bb1([[ALLOC]], [[ARG1]], [[NOT_ARG0]], %false{{[0-9_]*}} : {{.*}}) +// CHECK: ^bb2([[A0:%.+]]:{{.*}}, [[A1:%.+]]:{{.*}}, [[COND0:%.+]]: i1, [[COND1:%.+]]: i1): +// CHECK: test.copy +// CHECK-NEXT: [[BASE0:%[a-zA-Z0-9_]+]]{{.*}} = memref.extract_strided_metadata [[A0]] +// CHECK-NEXT: [[BASE1:%[a-zA-Z0-9_]+]]{{.*}} = memref.extract_strided_metadata [[A1]] +// CHECK-NEXT: bufferization.dealloc ([[BASE0]], [[BASE1]] : {{.*}}) if ([[COND0]], [[COND1]]) +// CHECK-NOT: retain +// CHECK-NEXT: return + +// TODO: slightly better analysis could just insert an unconditional dealloc on %0 + +// ----- + +// Test Case: +// bb0 <- Initial position of the first AllocOp +// / \ +// bb1 bb2 +// | / \ +// | bb3 bb4 +// \ \ / +// \ / +// bb5 <- Initial position of the second AllocOp +// BufferDeallocation expected behavior: Two missing DeallocOps should be +// inserted in the exit block. + +func.func @ifElseNested(%arg0: i1, %arg1: memref<2xf32>, %arg2: memref<2xf32>) { + %0 = memref.alloc() : memref<2xf32> + test.buffer_based in(%arg1: memref<2xf32>) out(%0: memref<2xf32>) + cf.cond_br %arg0, + ^bb1(%arg1, %0 : memref<2xf32>, memref<2xf32>), + ^bb2(%0, %arg1 : memref<2xf32>, memref<2xf32>) +^bb1(%1: memref<2xf32>, %2: memref<2xf32>): + cf.br ^bb5(%1, %2 : memref<2xf32>, memref<2xf32>) +^bb2(%3: memref<2xf32>, %4: memref<2xf32>): + cf.cond_br %arg0, ^bb3(%3 : memref<2xf32>), ^bb4(%4 : memref<2xf32>) +^bb3(%5: memref<2xf32>): + cf.br ^bb5(%5, %3 : memref<2xf32>, memref<2xf32>) +^bb4(%6: memref<2xf32>): + cf.br ^bb5(%3, %6 : memref<2xf32>, memref<2xf32>) +^bb5(%7: memref<2xf32>, %8: memref<2xf32>): + %9 = memref.alloc() : memref<2xf32> + test.buffer_based in(%7: memref<2xf32>) out(%9: memref<2xf32>) + test.copy(%9, %arg2) : (memref<2xf32>, memref<2xf32>) + return +} + +// CHECK-LABEL: func @ifElseNested +// CHECK-SAME: ([[ARG0:%.+]]: i1, [[ARG1:%.+]]: memref<2xf32>, [[ARG2:%.+]]: memref<2xf32>) +// CHECK: [[ALLOC0:%.+]] = memref.alloc() +// CHECK-NEXT: test.buffer_based +// CHECK-NEXT: [[NOT_ARG0:%.+]] = arith.xori [[ARG0]], %true +// CHECK-NEXT: cf.cond_br {{.*}}^bb1([[ARG1]], [[ALLOC0]], %false{{[0-9_]*}}, [[ARG0]] : {{.*}}), ^bb2([[ALLOC0]], [[ARG1]], [[NOT_ARG0]], %false{{[0-9_]*}} : +// CHECK: ^bb5([[A0:%.+]]: memref<2xf32>, [[A1:%.+]]: memref<2xf32>, [[COND0:%.+]]: i1, [[COND1:%.+]]: i1): +// CHECK: [[ALLOC1:%.+]] = memref.alloc() +// CHECK-NEXT: test.buffer_based +// CHECK-NEXT: test.copy +// CHECK-NEXT: [[BASE0:%[a-zA-Z0-9_]+]]{{.*}} = memref.extract_strided_metadata [[A0]] +// CHECK-NEXT: [[BASE1:%[a-zA-Z0-9_]+]]{{.*}} = memref.extract_strided_metadata [[A1]] +// CHECK-NEXT: bufferization.dealloc ([[ALLOC1]] : {{.*}}) if (%true +// CHECK-NOT: retain +// CHECK-NEXT: bufferization.dealloc ([[BASE0]], [[BASE1]] : {{.*}}) if ([[COND0]], [[COND1]]) +// CHECK-NOT: retain +// CHECK-NEXT: return + +// TODO: Instead of deallocating the bbarg memrefs, a slightly better analysis +// could do an unconditional deallocation on ALLOC0 and move it before the +// test.copy (dealloc of ALLOC1 would remain after the copy) + +// ----- + +// Test Case: +// bb0 +// / \ +// Initial pos of the 1st AllocOp -> bb1 bb2 <- Initial pos of the 2nd AllocOp +// \ / +// bb3 +// BufferDeallocation expected behavior: We need to introduce a copy for each +// buffer since the buffers are passed to bb3. The both missing DeallocOps are +// inserted in the respective block of the allocs. The copy is freed in the exit +// block. + +func.func @moving_alloc_and_inserting_missing_dealloc( + %cond: i1, + %arg0: memref<2xf32>, + %arg1: memref<2xf32>) { + cf.cond_br %cond, ^bb1, ^bb2 +^bb1: + %0 = memref.alloc() : memref<2xf32> + test.buffer_based in(%arg0: memref<2xf32>) out(%0: memref<2xf32>) + cf.br ^exit(%0 : memref<2xf32>) +^bb2: + %1 = memref.alloc() : memref<2xf32> + test.buffer_based in(%1: memref<2xf32>) out(%arg0: memref<2xf32>) + cf.br ^exit(%1 : memref<2xf32>) +^exit(%arg2: memref<2xf32>): + test.copy(%arg2, %arg1) : (memref<2xf32>, memref<2xf32>) + return +} + +// CHECK-LABEL: func @moving_alloc_and_inserting_missing_dealloc +// CHECK-SAME: ([[ARG0:%.+]]: i1, [[ARG0:%.+]]: memref<2xf32>, [[ARG0:%.+]]: memref<2xf32>) +// CHECK: ^bb1: +// CHECK: [[ALLOC0:%.+]] = memref.alloc() +// CHECK-NEXT: test.buffer_based +// CHECK-NEXT: cf.br ^bb3([[ALLOC0]], %true +// CHECK: ^bb2: +// CHECK: [[ALLOC1:%.+]] = memref.alloc() +// CHECK-NEXT: test.buffer_based +// CHECK-NEXT: cf.br ^bb3([[ALLOC1]], %true +// CHECK: ^bb3([[A0:%.+]]: memref<2xf32>, [[COND0:%.+]]: i1): +// CHECK: test.copy +// CHECK-NEXT: [[BASE:%[a-zA-Z0-9_]+]]{{.*}} = memref.extract_strided_metadata [[A0]] +// CHECK-NEXT: bufferization.dealloc ([[BASE]] : {{.*}}) if ([[COND0]]) +// CHECK-NEXT: return + +// ----- + +func.func @select_aliases(%arg0: index, %arg1: memref, %arg2: i1) { + %0 = memref.alloc(%arg0) : memref + %1 = memref.alloc(%arg0) : memref + %2 = arith.select %arg2, %0, %1 : memref + test.copy(%2, %arg1) : (memref, memref) + return +} + +// CHECK-LABEL: func @select_aliases +// CHECK: [[ALLOC0:%.+]] = memref.alloc( +// CHECK: [[ALLOC1:%.+]] = memref.alloc( +// CHECK: arith.select +// CHECK: test.copy +// CHECK: bufferization.dealloc ([[ALLOC0]] : {{.*}}) if (%true +// CHECK-NOT: retain +// CHECK: bufferization.dealloc ([[ALLOC1]] : {{.*}}) if (%true +// CHECK-NOT: retain + +// ----- + +func.func @select_aliases_not_same_ownership(%arg0: index, %arg1: memref, %arg2: i1) { + %0 = memref.alloc(%arg0) : memref + %1 = memref.alloca(%arg0) : memref + %2 = arith.select %arg2, %0, %1 : memref + cf.br ^bb1(%2 : memref) +^bb1(%arg3: memref): + test.copy(%arg3, %arg1) : (memref, memref) + return +} + +// CHECK-LABEL: func @select_aliases_not_same_ownership +// CHECK: ([[ARG0:%.+]]: index, [[ARG1:%.+]]: memref, [[ARG2:%.+]]: i1) +// CHECK: [[ALLOC0:%.+]] = memref.alloc( +// CHECK: [[ALLOC1:%.+]] = memref.alloca( +// CHECK: [[SELECT:%.+]] = arith.select +// CHECK: [[OWN:%.+]] = bufferization.dealloc ([[ALLOC0]] :{{.*}}) if (%true{{[0-9_]*}}) retain ([[SELECT]] : +// CHECK: cf.br ^bb1([[SELECT]], [[OWN]] : +// CHECK: ^bb1([[A0:%.+]]: memref, [[COND:%.+]]: i1) +// CHECK: test.copy +// CHECK: [[BASE0:%[a-zA-Z0-9_]+]]{{.*}} = memref.extract_strided_metadata [[A0]] +// CHECK: bufferization.dealloc ([[BASE0]] : {{.*}}) if ([[COND]]) +// CHECK-NOT: retain + +// ----- + +func.func @select_captured_in_next_block(%arg0: index, %arg1: memref, %arg2: i1, %arg3: i1) { + %0 = memref.alloc(%arg0) : memref + %1 = memref.alloca(%arg0) : memref + %2 = arith.select %arg2, %0, %1 : memref + cf.cond_br %arg3, ^bb1(%0 : memref), ^bb1(%arg1 : memref) +^bb1(%arg4: memref): + test.copy(%arg4, %2) : (memref, memref) + return +} + +// CHECK-LABEL: func @select_captured_in_next_block +// CHECK: ([[ARG0:%.+]]: index, [[ARG1:%.+]]: memref, [[ARG2:%.+]]: i1, [[ARG3:%.+]]: i1) +// CHECK: [[ALLOC0:%.+]] = memref.alloc( +// CHECK: [[ALLOC1:%.+]] = memref.alloca( +// CHECK: [[SELECT:%.+]] = arith.select +// CHECK: [[OWN0:%.+]]:2 = bufferization.dealloc ([[ALLOC0]] :{{.*}}) if ([[ARG3]]) retain ([[ALLOC0]], [[SELECT]] : +// CHECK: [[NOT_ARG3:%.+]] = arith.xori [[ARG3]], %true +// CHECK: [[OWN1:%.+]] = bufferization.dealloc ([[ALLOC0]] :{{.*}}) if ([[NOT_ARG3]]) retain ([[SELECT]] : +// CHECK: [[MERGED_OWN:%.+]] = arith.select [[ARG3]], [[OWN0]]#1, [[OWN1]] +// CHECK: cf.cond_br{{.*}}^bb1([[ALLOC0]], [[OWN0]]#0 :{{.*}}), ^bb1([[ARG1]], %false +// CHECK: ^bb1([[A0:%.+]]: memref, [[COND:%.+]]: i1) +// CHECK: test.copy +// CHECK: [[BASE0:%[a-zA-Z0-9_]+]]{{.*}} = memref.extract_strided_metadata [[SELECT]] +// CHECK: [[BASE1:%[a-zA-Z0-9_]+]]{{.*}} = memref.extract_strided_metadata [[A0]] +// CHECK: bufferization.dealloc ([[BASE0]], [[BASE1]] : {{.*}}) if ([[MERGED_OWN]], [[COND]]) + +// There are two interesting parts here: +// * The dealloc condition of %0 in the second block should be the corresponding +// result of the dealloc operation of the first block, because %0 has unknown +// ownership status and thus would other wise require a clone in the first +// block. +// * The dealloc of the first block must make sure that the branch condition and +// respective retained values are handled correctly, i.e., only the ones for the +// actual branch taken have to be retained. diff --git a/mlir/test/Dialect/Bufferization/Transforms/BufferDeallocation/dealloc-callop-interface.mlir b/mlir/test/Dialect/Bufferization/Transforms/BufferDeallocation/dealloc-callop-interface.mlir new file mode 100644 --- /dev/null +++ b/mlir/test/Dialect/Bufferization/Transforms/BufferDeallocation/dealloc-callop-interface.mlir @@ -0,0 +1,115 @@ +// DEFINE: %{canonicalize} = -canonicalize=enable-patterns="bufferization-skip-extract-metadata-of-alloc,bufferization-erase-always-false-dealloc,bufferization-erase-empty-dealloc,bufferization-dealloc-remove-duplicate-retained-memrefs,bufferization-dealloc-remove-duplicate-dealloc-memrefs",region-simplify=false + +// RUN: mlir-opt -verify-diagnostics -buffer-deallocation=private-function-dynamic-ownership=false \ +// RUN: %{canonicalize} -buffer-deallocation-simplification %{canonicalize} -split-input-file %s | FileCheck %s +// RUN: mlir-opt -verify-diagnostics -buffer-deallocation=private-function-dynamic-ownership=true \ +// RUN: %{canonicalize} --buffer-deallocation-simplification %{canonicalize} -split-input-file %s | FileCheck %s --check-prefix=CHECK-DYNAMIC + +func.func private @f(%arg0: memref) -> memref { + return %arg0 : memref +} + +func.func @function_call() { + %alloc = memref.alloc() : memref + %alloc2 = memref.alloc() : memref + %ret = call @f(%alloc) : (memref) -> memref + test.copy(%ret, %alloc2) : (memref, memref) + return +} + +// CHECK-LABEL: func @function_call() +// CHECK: [[ALLOC0:%.+]] = memref.alloc( +// CHECK-NEXT: [[ALLOC1:%.+]] = memref.alloc( +// CHECK-NEXT: [[RET:%.+]] = call @f([[ALLOC0]]) : (memref) -> memref +// CHECK-NEXT: test.copy +// CHECK-NEXT: [[BASE:%[a-zA-Z0-9_]+]]{{.*}} = memref.extract_strided_metadata [[RET]] +// COM: the following dealloc operation should be split into three since we can +// COM: be sure that the memrefs will never alias according to the buffer +// COM: deallocation ABI, however, the local alias analysis is not powerful +// COM: enough to detect this yet. +// CHECK-NEXT: bufferization.dealloc ([[ALLOC0]], [[ALLOC1]], [[BASE]] :{{.*}}) if (%true{{[0-9_]*}}, %true{{[0-9_]*}}, %true{{[0-9_]*}}) + +// CHECK-DYNAMIC-LABEL: func @function_call() +// CHECK-DYNAMIC: [[ALLOC0:%.+]] = memref.alloc( +// CHECK-DYNAMIC-NEXT: [[ALLOC1:%.+]] = memref.alloc( +// CHECK-DYNAMIC-NEXT: [[RET:%.+]]:2 = call @f([[ALLOC0]], %true{{[0-9_]*}}) : (memref, i1) -> (memref, i1) +// CHECK-DYNAMIC-NEXT: test.copy +// CHECK-DYNAMIC-NEXT: [[BASE:%[a-zA-Z0-9_]+]]{{.*}} = memref.extract_strided_metadata [[RET]]#0 +// CHECK-DYNAMIC-NEXT: bufferization.dealloc ([[ALLOC0]], [[ALLOC1]], [[BASE]] :{{.*}}) if (%true{{[0-9_]*}}, %true{{[0-9_]*}}, [[RET]]#1) + +// ----- + +func.func @f(%arg0: memref) -> memref { + return %arg0 : memref +} + +func.func @function_call_non_private() { + %alloc = memref.alloc() : memref + %alloc2 = memref.alloc() : memref + %ret = call @f(%alloc) : (memref) -> memref + test.copy(%ret, %alloc2) : (memref, memref) + return +} + +// CHECK-LABEL: func @function_call_non_private +// CHECK: [[ALLOC0:%.+]] = memref.alloc( +// CHECK: [[ALLOC1:%.+]] = memref.alloc( +// CHECK: [[RET:%.+]] = call @f([[ALLOC0]]) : (memref) -> memref +// CHECK-NEXT: test.copy +// CHECK-NEXT: [[BASE:%[a-zA-Z0-9_]+]]{{.*}} = memref.extract_strided_metadata [[RET]] +// CHECK-NEXT: bufferization.dealloc ([[ALLOC0]], [[ALLOC1]], [[BASE]] :{{.*}}) if (%true{{[0-9_]*}}, %true{{[0-9_]*}}, %true{{[0-9_]*}}) +// CHECK-NEXT: return + +// CHECK-DYNAMIC-LABEL: func @function_call_non_private +// CHECK-DYNAMIC: [[ALLOC0:%.+]] = memref.alloc( +// CHECK-DYNAMIC: [[ALLOC1:%.+]] = memref.alloc( +// CHECK-DYNAMIC: [[RET:%.+]] = call @f([[ALLOC0]]) : (memref) -> memref +// CHECK-DYNAMIC-NEXT: test.copy +// CHECK-DYNAMIC-NEXT: [[BASE:%[a-zA-Z0-9_]+]]{{.*}} = memref.extract_strided_metadata [[RET]] +// CHECK-DYNAMIC-NEXT: bufferization.dealloc ([[ALLOC0]], [[ALLOC1]], [[BASE]] :{{.*}}) if (%true{{[0-9_]*}}, %true{{[0-9_]*}}, %true{{[0-9_]*}}) +// CHECK-DYNAMIC-NEXT: return + +// ----- + +func.func private @f(%arg0: memref) -> memref { + return %arg0 : memref +} + +func.func @function_call_requries_merged_ownership_mid_block(%arg0: i1) { + %alloc = memref.alloc() : memref + %alloc2 = memref.alloca() : memref + %0 = arith.select %arg0, %alloc, %alloc2 : memref + %ret = call @f(%0) : (memref) -> memref + test.copy(%ret, %alloc) : (memref, memref) + return +} + +// CHECK-LABEL: func @function_call_requries_merged_ownership_mid_block +// CHECK: [[ALLOC0:%.+]] = memref.alloc( +// CHECK-NEXT: [[ALLOC1:%.+]] = memref.alloca( +// CHECK-NEXT: [[SELECT:%.+]] = arith.select{{.*}}[[ALLOC0]], [[ALLOC1]] +// CHECK-NEXT: [[RET:%.+]] = call @f([[SELECT]]) +// CHECK-NEXT: test.copy +// CHECK-NEXT: [[BASE:%[a-zA-Z0-9_]+]]{{.*}} = memref.extract_strided_metadata [[RET]] +// CHECK-NEXT: bufferization.dealloc ([[ALLOC0]], [[BASE]] : +// CHECK-SAME: if (%true{{[0-9_]*}}, %true{{[0-9_]*}}) +// CHECK-NOT: retain +// CHECK-NEXT: return + +// CHECK-DYNAMIC-LABEL: func @function_call_requries_merged_ownership_mid_block +// CHECK-DYNAMIC: [[ALLOC0:%.+]] = memref.alloc( +// CHECK-DYNAMIC-NEXT: [[ALLOC1:%.+]] = memref.alloca( +// CHECK-DYNAMIC-NEXT: [[SELECT:%.+]] = arith.select{{.*}}[[ALLOC0]], [[ALLOC1]] +// CHECK-DYNAMIC-NEXT: [[CLONE:%.+]] = bufferization.clone [[SELECT]] +// CHECK-DYNAMIC-NEXT: [[RET:%.+]]:2 = call @f([[CLONE]], %true{{[0-9_]*}}) +// CHECK-DYNAMIC-NEXT: test.copy +// CHECK-DYNAMIC-NEXT: [[BASE:%[a-zA-Z0-9_]+]]{{.*}} = memref.extract_strided_metadata [[RET]]#0 +// CHECK-DYNAMIC-NEXT: bufferization.dealloc ([[ALLOC0]], [[CLONE]], [[BASE]] : +// CHECK-DYNAMIC-SAME: if (%true{{[0-9_]*}}, %true{{[0-9_]*}}, [[RET]]#1) +// CHECK-DYNAMIC-NOT: retain +// CHECK-DYNAMIC-NEXT: return + +// TODO: the inserted clone is not necessary, we just have to know which of the +// two allocations was selected, either by checking aliasing of the result at +// runtime or by extracting the select condition using an OpInterface or by +// hardcoding the select op diff --git a/mlir/test/Dialect/Bufferization/Transforms/BufferDeallocation/dealloc-existing-deallocs.mlir b/mlir/test/Dialect/Bufferization/Transforms/BufferDeallocation/dealloc-existing-deallocs.mlir new file mode 100644 --- /dev/null +++ b/mlir/test/Dialect/Bufferization/Transforms/BufferDeallocation/dealloc-existing-deallocs.mlir @@ -0,0 +1,46 @@ +// DEFINE: %{canonicalize} = -canonicalize=enable-patterns="bufferization-skip-extract-metadata-of-alloc,bufferization-erase-always-false-dealloc,bufferization-erase-empty-dealloc,bufferization-dealloc-remove-duplicate-retained-memrefs,bufferization-dealloc-remove-duplicate-dealloc-memrefs",region-simplify=false + +// RUN: mlir-opt -verify-diagnostics -buffer-deallocation %{canonicalize} \ +// RUN: --buffer-deallocation-simplification %{canonicalize} -split-input-file %s | FileCheck %s +// RUN: mlir-opt -verify-diagnostics -buffer-deallocation=private-function-dynamic-ownership=true -split-input-file %s > /dev/null + +// Ensure we free the realloc, not the alloc. + +func.func @auto_dealloc() { + %c10 = arith.constant 10 : index + %c100 = arith.constant 100 : index + %alloc = memref.alloc(%c10) : memref + %realloc = memref.realloc %alloc(%c100) : memref to memref + return +} + +// CHECK-LABEL: func @auto_dealloc() +// CHECK-DAG: %[[C10:.*]] = arith.constant 10 : index +// CHECK-DAG: %[[C100:.*]] = arith.constant 100 : index +// CHECK: %[[A:.*]] = memref.alloc(%[[C10]]) +// iCHECK-NOT: bufferization.dealloc +// CHECK: %[[R:.*]] = memref.realloc %alloc(%[[C100]]) +// CHECK-NOT: bufferization.dealloc{{.*}}%[[A]] +// CHECK: bufferization.dealloc (%[[R]] :{{.*}}) if (%true{{[0-9_]*}}) +// CHECK-NOT: bufferization.dealloc + +// ----- + +func.func @auto_dealloc_inside_nested_region(%arg0: memref, %arg1: i1) -> memref { + %c100 = arith.constant 100 : index + %0 = scf.if %arg1 -> memref { + %realloc = memref.realloc %arg0(%c100) : memref to memref + scf.yield %realloc : memref + } else { + scf.yield %arg0 : memref + } + return %0 : memref +} + +// CHECK-LABEL: func @auto_dealloc_inside_nested_region +// CHECK-SAME: [[ARG0:%.+]]:{{.*}}, +// CHECK-SAME: [[ARG1:%.+]]:{{.*}}) +// CHECK: scf.if [[ARG1]] +// CHECK: [[REALLOC:%.+]] = memref.realloc [[ARG0]] +// CHECK: scf.yield [[REALLOC]], %true +// CHECK: scf.yield [[ARG0]], %false diff --git a/mlir/test/Dialect/Bufferization/Transforms/BufferDeallocation/dealloc-function-boundaries.mlir b/mlir/test/Dialect/Bufferization/Transforms/BufferDeallocation/dealloc-function-boundaries.mlir new file mode 100644 --- /dev/null +++ b/mlir/test/Dialect/Bufferization/Transforms/BufferDeallocation/dealloc-function-boundaries.mlir @@ -0,0 +1,131 @@ +// DEFINE: %{canonicalize} = -canonicalize=enable-patterns="bufferization-skip-extract-metadata-of-alloc,bufferization-erase-always-false-dealloc,bufferization-erase-empty-dealloc,bufferization-dealloc-remove-duplicate-retained-memrefs,bufferization-dealloc-remove-duplicate-dealloc-memrefs",region-simplify=false + +// RUN: mlir-opt -verify-diagnostics -buffer-deallocation=private-function-dynamic-ownership=false \ +// RUN: %{canonicalize} --buffer-deallocation-simplification %{canonicalize} -split-input-file %s | FileCheck %s +// RUN: mlir-opt -verify-diagnostics -buffer-deallocation=private-function-dynamic-ownership=true \ +// RUN: %{canonicalize} --buffer-deallocation-simplification %{canonicalize} -split-input-file %s | FileCheck %s --check-prefix=CHECK-DYNAMIC + +// Test Case: Existing AllocOp with no users. +// BufferDeallocation expected behavior: It should insert a DeallocOp right +// before ReturnOp. + +func.func private @emptyUsesValue(%arg0: memref<4xf32>) { + %0 = memref.alloc() : memref<4xf32> + return +} + +// CHECK-LABEL: func private @emptyUsesValue( +// CHECK: [[ALLOC:%.*]] = memref.alloc() +// CHECK-NEXT: bufferization.dealloc ([[ALLOC]] : +// CHECK-SAME: if (%true{{[0-9_]*}}) +// CHECK-NOT: retain +// CHECK-NEXT: return + +// CHECK-DYNAMIC-LABEL: func private @emptyUsesValue( +// CHECK-DYNAMIC-SAME: [[ARG0:%.+]]: memref<4xf32>, [[ARG1:%.+]]: i1) +// CHECK-DYNAMIC: [[ALLOC:%.*]] = memref.alloc() +// CHECK-DYNAMIC-NEXT: [[BASE:%[a-zA-Z0-9_]+]], {{.*}} = memref.extract_strided_metadata [[ARG0]] +// CHECK-DYNAMIC-NEXT: bufferization.dealloc ([[BASE]] :{{.*}}) if ([[ARG1]]) +// CHECK-DYNAMIC-NOT: retain +// CHECK-DYNAMIC-NEXT: bufferization.dealloc ([[ALLOC]] :{{.*}}) if (%true{{[0-9_]*}}) +// CHECK-DYNAMIC-NOT: retain +// CHECK-DYNAMIC-NEXT: return + +// ----- + +func.func @emptyUsesValue(%arg0: memref<4xf32>) { + %0 = memref.alloc() : memref<4xf32> + return +} + +// CHECK-LABEL: func @emptyUsesValue( + +// CHECK-DYNAMIC-LABEL: func @emptyUsesValue( +// CHECK-DYNAMIC: [[ALLOC:%.*]] = memref.alloc() +// CHECK-DYNAMIC-NEXT: bufferization.dealloc ([[ALLOC]] :{{.*}}) if (%true{{[0-9_]*}}) +// CHECK-DYNAMIC-NOT: retain +// CHECK-DYNAMIC-NEXT: return + +// ----- + +// Test Case: Dead operations in a single block. +// BufferDeallocation expected behavior: It only inserts the two missing +// DeallocOps after the last BufferBasedOp. + +func.func private @redundantOperations(%arg0: memref<2xf32>) { + %0 = memref.alloc() : memref<2xf32> + test.buffer_based in(%arg0: memref<2xf32>) out(%0: memref<2xf32>) + %1 = memref.alloc() : memref<2xf32> + test.buffer_based in(%0: memref<2xf32>) out(%1: memref<2xf32>) + return +} + +// CHECK-LABEL: func private @redundantOperations +// CHECK: (%[[ARG0:.*]]: {{.*}}) +// CHECK: %[[FIRST_ALLOC:.*]] = memref.alloc() +// CHECK-NEXT: test.buffer_based +// CHECK: %[[SECOND_ALLOC:.*]] = memref.alloc() +// CHECK-NEXT: test.buffer_based +// CHECK-NEXT: bufferization.dealloc (%[[FIRST_ALLOC]] : {{.*}}) if (%true{{[0-9_]*}}) +// CHECK-NEXT: bufferization.dealloc (%[[SECOND_ALLOC]] : {{.*}}) if (%true{{[0-9_]*}}) +// CHECK-NEXT: return + +// CHECK-DYNAMIC-LABEL: func private @redundantOperations +// CHECK-DYNAMIC: (%[[ARG0:.*]]: memref{{.*}}, %[[ARG1:.*]]: i1) +// CHECK-DYNAMIC: %[[FIRST_ALLOC:.*]] = memref.alloc() +// CHECK-DYNAMIC-NEXT: test.buffer_based +// CHECK-DYNAMIC: %[[SECOND_ALLOC:.*]] = memref.alloc() +// CHECK-DYNAMIC-NEXT: test.buffer_based +// CHECK-DYNAMIC-NEXT: %[[BASE:[a-zA-Z0-9_]+]], {{.*}} = memref.extract_strided_metadata %[[ARG0]] +// CHECK-DYNAMIC-NEXT: bufferization.dealloc (%[[BASE]] : {{.*}}) if (%[[ARG1]]) +// CHECK-DYNAMIC-NEXT: bufferization.dealloc (%[[FIRST_ALLOC]] : {{.*}}) if (%true{{[0-9_]*}}) +// CHECK-DYNAMIC-NEXT: bufferization.dealloc (%[[SECOND_ALLOC]] : {{.*}}) if (%true{{[0-9_]*}}) +// CHECK-DYNAMIC-NEXT: return + +// ----- + +// Test Case: buffer deallocation escaping +// BufferDeallocation expected behavior: It must not dealloc %arg1 and %x +// since they are operands of return operation and should escape from +// deallocating. It should dealloc %y after CopyOp. + +func.func private @memref_in_function_results( + %arg0: memref<5xf32>, + %arg1: memref<10xf32>, + %arg2: memref<5xf32>) -> (memref<10xf32>, memref<15xf32>) { + %x = memref.alloc() : memref<15xf32> + %y = memref.alloc() : memref<5xf32> + test.buffer_based in(%arg0: memref<5xf32>) out(%y: memref<5xf32>) + test.copy(%y, %arg2) : (memref<5xf32>, memref<5xf32>) + return %arg1, %x : memref<10xf32>, memref<15xf32> +} + +// CHECK-LABEL: func private @memref_in_function_results +// CHECK: (%[[ARG0:.*]]: memref<5xf32>, %[[ARG1:.*]]: memref<10xf32>, +// CHECK-SAME: %[[RESULT:.*]]: memref<5xf32>) +// CHECK: %[[X:.*]] = memref.alloc() +// CHECK: %[[Y:.*]] = memref.alloc() +// CHECK: test.copy +// CHECK-NEXT: %[[V0:.+]] = scf.if %false +// CHECK-NEXT: scf.yield %[[ARG1]] +// CHECK-NEXT: } else { +// CHECK-NEXT: %[[CLONE:.+]] = bufferization.clone %[[ARG1]] +// CHECK-NEXT: scf.yield %[[CLONE]] +// CHECK-NEXT: } +// CHECK: bufferization.dealloc (%[[Y]] : {{.*}}) if (%true{{[0-9_]*}}) +// CHECK-NOT: retain +// CHECK: return %[[V0]], %[[X]] + +// CHECK-DYNAMIC-LABEL: func private @memref_in_function_results +// CHECK-DYNAMIC: (%[[ARG0:.*]]: memref<5xf32>, %[[ARG1:.*]]: memref<10xf32>, +// CHECK-DYNAMIC-SAME: %[[RESULT:.*]]: memref<5xf32>, %[[ARG3:.*]]: i1, %[[ARG4:.*]]: i1, %[[ARG5:.*]]: i1) +// CHECK-DYNAMIC: %[[X:.*]] = memref.alloc() +// CHECK-DYNAMIC: %[[Y:.*]] = memref.alloc() +// CHECK-DYNAMIC: test.copy +// CHECK-DYNAMIC: %[[BASE0:[a-zA-Z0-9_]+]], {{.+}} = memref.extract_strided_metadata %[[ARG0]] +// CHECK-DYNAMIC: %[[BASE1:[a-zA-Z0-9_]+]], {{.+}} = memref.extract_strided_metadata %[[RESULT]] +// CHECK-DYNAMIC: bufferization.dealloc (%[[Y]] : {{.*}}) if (%true{{[0-9_]*}}) +// CHECK-DYNAMIC-NOT: retain +// CHECK-DYNAMIC: [[OWN:%.+]] = bufferization.dealloc (%[[BASE0]], %[[BASE1]] : {{.*}}) if (%[[ARG3]], %[[ARG5]]) retain (%[[ARG1]] : +// CHECK-DYNAMIC: [[OR:%.+]] = arith.ori [[OWN]], %[[ARG4]] +// CHECK-DYNAMIC: return %[[ARG1]], %[[X]], [[OR]], %true diff --git a/mlir/test/Dialect/Bufferization/Transforms/BufferDeallocation/dealloc-memoryeffect-interface.mlir b/mlir/test/Dialect/Bufferization/Transforms/BufferDeallocation/dealloc-memoryeffect-interface.mlir new file mode 100644 --- /dev/null +++ b/mlir/test/Dialect/Bufferization/Transforms/BufferDeallocation/dealloc-memoryeffect-interface.mlir @@ -0,0 +1,126 @@ +// DEFINE: %{canonicalize} = -canonicalize=enable-patterns="bufferization-skip-extract-metadata-of-alloc,bufferization-erase-always-false-dealloc,bufferization-erase-empty-dealloc,bufferization-dealloc-remove-duplicate-retained-memrefs,bufferization-dealloc-remove-duplicate-dealloc-memrefs",region-simplify=false + +// RUN: mlir-opt -verify-diagnostics -buffer-deallocation \ +// RUN: %{canonicalize} --buffer-deallocation-simplification %{canonicalize} -split-input-file %s | FileCheck %s +// RUN: mlir-opt -verify-diagnostics -buffer-deallocation=private-function-dynamic-ownership=true -split-input-file %s > /dev/null + +// Test Case: Dead operations in a single block. +// BufferDeallocation expected behavior: It only inserts the two missing +// DeallocOps after the last BufferBasedOp. + +// CHECK-LABEL: func @redundantOperations +func.func @redundantOperations(%arg0: memref<2xf32>) { + %0 = memref.alloc() : memref<2xf32> + test.buffer_based in(%arg0: memref<2xf32>) out(%0: memref<2xf32>) + %1 = memref.alloc() : memref<2xf32> + test.buffer_based in(%0: memref<2xf32>) out(%1: memref<2xf32>) + return +} + +// CHECK: (%[[ARG0:.*]]: {{.*}}) +// CHECK: %[[FIRST_ALLOC:.*]] = memref.alloc() +// CHECK-NOT: bufferization.dealloc +// CHECK: test.buffer_based in(%[[ARG0]]{{.*}}out(%[[FIRST_ALLOC]] +// CHECK-NOT: bufferization.dealloc +// CHECK: %[[SECOND_ALLOC:.*]] = memref.alloc() +// CHECK-NOT: bufferization.dealloc +// CHECK: test.buffer_based in(%[[FIRST_ALLOC]]{{.*}}out(%[[SECOND_ALLOC]] +// CHECK: bufferization.dealloc (%[[FIRST_ALLOC]] :{{.*}}) if (%true{{[0-9_]*}}) +// CHECK: bufferization.dealloc (%[[SECOND_ALLOC]] :{{.*}}) if (%true{{[0-9_]*}}) +// CHECK-NEXT: return + +// TODO: The dealloc could be split in two to avoid runtime aliasing checks +// since we can be sure at compile time that they will never alias. + +// ----- + +// CHECK-LABEL: func @allocaIsNotDeallocated +func.func @allocaIsNotDeallocated(%arg0: memref<2xf32>) { + %0 = memref.alloc() : memref<2xf32> + test.buffer_based in(%arg0: memref<2xf32>) out(%0: memref<2xf32>) + %1 = memref.alloca() : memref<2xf32> + test.buffer_based in(%0: memref<2xf32>) out(%1: memref<2xf32>) + return +} + +// CHECK: (%[[ARG0:.*]]: {{.*}}) +// CHECK: %[[FIRST_ALLOC:.*]] = memref.alloc() +// CHECK-NEXT: test.buffer_based in(%[[ARG0]]{{.*}}out(%[[FIRST_ALLOC]] +// CHECK-NEXT: %[[SECOND_ALLOC:.*]] = memref.alloca() +// CHECK-NEXT: test.buffer_based in(%[[FIRST_ALLOC]]{{.*}}out(%[[SECOND_ALLOC]] +// CHECK: bufferization.dealloc (%[[FIRST_ALLOC]] :{{.*}}) if (%true{{[0-9_]*}}) +// CHECK-NEXT: return + +// ----- + +// Test Case: Inserting missing DeallocOp in a single block. + +// CHECK-LABEL: func @inserting_missing_dealloc_simple +func.func @inserting_missing_dealloc_simple( + %arg0 : memref<2xf32>, + %arg1: memref<2xf32>) { + %0 = memref.alloc() : memref<2xf32> + test.buffer_based in(%arg0: memref<2xf32>) out(%0: memref<2xf32>) + test.copy(%0, %arg1) : (memref<2xf32>, memref<2xf32>) + return +} + +// CHECK: %[[ALLOC0:.*]] = memref.alloc() +// CHECK: test.copy +// CHECK: bufferization.dealloc (%[[ALLOC0]] :{{.*}}) if (%true{{[0-9_]*}}) + +// ----- + +// Test Case: The ownership indicator is set to false for alloca + +// CHECK-LABEL: func @alloca_ownership_indicator_is_false +func.func @alloca_ownership_indicator_is_false() { + %0 = memref.alloca() : memref<2xf32> + cf.br ^bb1(%0: memref<2xf32>) +^bb1(%arg0 : memref<2xf32>): + return +} + +// CHECK: %[[ALLOC0:.*]] = memref.alloca() +// CHECK-NEXT: cf.br ^bb1(%[[ALLOC0]], %false : +// CHECK-NEXT: ^bb1([[A0:%.+]]: memref<2xf32>, [[COND0:%.+]]: i1): +// CHECK: [[BASE:%[a-zA-Z0-9_]+]]{{.*}} = memref.extract_strided_metadata [[A0]] +// CHECK: bufferization.dealloc ([[BASE]] : {{.*}}) if ([[COND0]]) +// CHECK-NEXT: return + +// ----- + +func.func @dealloc_existing_clones(%arg0: memref, %arg1: memref) -> memref { + %0 = bufferization.clone %arg0 : memref to memref + %1 = bufferization.clone %arg1 : memref to memref + return %0 : memref +} + +// CHECK-LABEL: func @dealloc_existing_clones +// CHECK: (%[[ARG0:.*]]: memref, %[[ARG1:.*]]: memref) +// CHECK: %[[RES0:.*]] = bufferization.clone %[[ARG0]] +// CHECK: %[[RES1:.*]] = bufferization.clone %[[ARG1]] +// CHECK-NEXT: bufferization.dealloc (%[[RES1]] :{{.*}}) if (%true{{[0-9_]*}}) +// CHECK-NOT: retain +// CHECK-NEXT: return %[[RES0]] + +// TODO: The retain operand could be dropped to avoid runtime aliasing checks +// since We can guarantee at compile-time that it will never alias with the +// dealloc operand + +// ----- + +memref.global "private" constant @__constant_4xf32 : memref<4xf32> = dense<[1.000000e+00, 2.000000e+00, 3.000000e+00, 4.000000e+00]> + +func.func @op_without_aliasing_and_allocation() -> memref<4xf32> { + %0 = memref.get_global @__constant_4xf32 : memref<4xf32> + return %0 : memref<4xf32> +} + +// CHECK-LABEL: func @op_without_aliasing_and_allocation +// CHECK: [[GLOBAL:%.+]] = memref.get_global @__constant_4xf32 +// CHECK: [[RES:%.+]] = scf.if %false +// CHECK: scf.yield [[GLOBAL]] : +// CHECK: [[CLONE:%.+]] = bufferization.clone [[GLOBAL]] +// CHECK: scf.yield [[CLONE]] : +// CHECK: return [[RES]] : diff --git a/mlir/test/Dialect/Bufferization/Transforms/BufferDeallocation/dealloc-region-branchop-interface.mlir b/mlir/test/Dialect/Bufferization/Transforms/BufferDeallocation/dealloc-region-branchop-interface.mlir new file mode 100644 --- /dev/null +++ b/mlir/test/Dialect/Bufferization/Transforms/BufferDeallocation/dealloc-region-branchop-interface.mlir @@ -0,0 +1,675 @@ +// DEFINE: %{canonicalize} = -canonicalize=enable-patterns="bufferization-skip-extract-metadata-of-alloc,bufferization-erase-always-false-dealloc,bufferization-erase-empty-dealloc,bufferization-dealloc-remove-duplicate-retained-memrefs,bufferization-dealloc-remove-duplicate-dealloc-memrefs",region-simplify=false + +// RUN: mlir-opt -verify-diagnostics -buffer-deallocation \ +// RUN: %{canonicalize} --buffer-deallocation-simplification %{canonicalize} -split-input-file %s | FileCheck %s +// RUN: mlir-opt -verify-diagnostics -buffer-deallocation=private-function-dynamic-ownership=true -split-input-file %s > /dev/null + +// Test Case: Nested regions - This test defines a BufferBasedOp inside the +// region of a RegionBufferBasedOp. +// BufferDeallocation expected behavior: The AllocOp for the BufferBasedOp +// should remain inside the region of the RegionBufferBasedOp and it should insert +// the missing DeallocOp in the same region. The missing DeallocOp should be +// inserted after CopyOp. + +func.func @nested_regions_and_cond_branch( + %arg0: i1, + %arg1: memref<2xf32>, + %arg2: memref<2xf32>) { + cf.cond_br %arg0, ^bb1, ^bb2 +^bb1: + cf.br ^bb3(%arg1 : memref<2xf32>) +^bb2: + %0 = memref.alloc() : memref<2xf32> + test.region_buffer_based in(%arg1: memref<2xf32>) out(%0: memref<2xf32>) { + ^bb0(%gen1_arg0: f32, %gen1_arg1: f32): + %1 = memref.alloc() : memref<2xf32> + test.buffer_based in(%arg1: memref<2xf32>) out(%1: memref<2xf32>) + %tmp1 = math.exp %gen1_arg0 : f32 + test.region_yield %tmp1 : f32 + } + cf.br ^bb3(%0 : memref<2xf32>) +^bb3(%1: memref<2xf32>): + test.copy(%1, %arg2) : (memref<2xf32>, memref<2xf32>) + return +} + +// CHECK-LABEL: func @nested_regions_and_cond_branch +// CHECK-SAME: ([[ARG0:%.+]]: i1, [[ARG1:%.+]]: memref<2xf32>, [[ARG2:%.+]]: memref<2xf32>) +// CHECK: ^bb1: +// CHECK-NOT: bufferization.clone +// CHECK-NOT: bufferization.dealloc +// CHECK: cf.br ^bb3([[ARG1]], %false +// CHECK: ^bb2: +// CHECK: [[ALLOC0:%.+]] = memref.alloc() +// CHECK: test.region_buffer_based +// CHECK: [[ALLOC1:%.+]] = memref.alloc() +// CHECK: test.buffer_based +// CHECK: bufferization.dealloc ([[ALLOC1]] : memref<2xf32>) if (%true +// CHECK-NEXT: test.region_yield +// CHECK-NOT: bufferization.clone +// CHECK-NOT: bufferization.dealloc +// CHECK: cf.br ^bb3([[ALLOC0]], %true +// CHECK: ^bb3([[A0:%.+]]: memref<2xf32>, [[COND0:%.+]]: i1): +// CHECK: test.copy +// CHECK-NEXT: [[BASE:%[a-zA-Z0-9_]+]]{{.*}} = memref.extract_strided_metadata [[A0]] +// CHECK-NEXT: bufferization.dealloc ([[BASE]] : {{.*}}) if ([[COND0]]) +// CHECK: return + +// ----- + +// Test Case: nested region control flow +// The alloc %1 flows through both if branches until it is finally returned. +// Hence, it does not require a specific dealloc operation. However, %3 +// requires a dealloc. + +func.func @nested_region_control_flow( + %arg0 : index, + %arg1 : index) -> memref { + %0 = arith.cmpi eq, %arg0, %arg1 : index + %1 = memref.alloc(%arg0, %arg0) : memref + %2 = scf.if %0 -> (memref) { + scf.yield %1 : memref + } else { + %3 = memref.alloc(%arg0, %arg1) : memref + scf.yield %1 : memref + } + return %2 : memref +} + +// CHECK-LABEL: func @nested_region_control_flow +// CHECK: [[ALLOC:%.+]] = memref.alloc( +// CHECK: [[V0:%.+]]:2 = scf.if +// CHECK: scf.yield [[ALLOC]], %false +// CHECK: [[ALLOC1:%.+]] = memref.alloc( +// CHECK: bufferization.dealloc ([[ALLOC1]] :{{.*}}) if (%true{{[0-9_]*}}) +// CHECK-NOT: retain +// CHECK: scf.yield [[ALLOC]], %false +// CHECK: [[V1:%.+]] = scf.if [[V0]]#1 +// CHECK: scf.yield [[V0]]#0 +// CHECK: [[CLONE:%.+]] = bufferization.clone [[V0]]#0 +// CHECK: scf.yield [[CLONE]] +// CHECK: [[BASE:%[a-zA-Z0-9_]+]],{{.*}} = memref.extract_strided_metadata [[V0]]#0 +// CHECK: bufferization.dealloc ([[ALLOC]], [[BASE]] : {{.*}}) if (%true{{[0-9_]*}}, [[V0]]#1) retain ([[V1]] : +// CHECK: return [[V1]] + +// ----- + +// Test Case: nested region control flow with a nested buffer allocation in a +// divergent branch. +// Buffer deallocation places a copy for both %1 and %3, since they are +// returned in the end. + +func.func @nested_region_control_flow_div( + %arg0 : index, + %arg1 : index) -> memref { + %0 = arith.cmpi eq, %arg0, %arg1 : index + %1 = memref.alloc(%arg0, %arg0) : memref + %2 = scf.if %0 -> (memref) { + scf.yield %1 : memref + } else { + %3 = memref.alloc(%arg0, %arg1) : memref + scf.yield %3 : memref + } + return %2 : memref +} + +// CHECK-LABEL: func @nested_region_control_flow_div +// CHECK: [[ALLOC:%.+]] = memref.alloc( +// CHECK: [[V0:%.+]]:2 = scf.if +// CHECK: scf.yield [[ALLOC]], %false +// CHECK: [[ALLOC1:%.+]] = memref.alloc( +// CHECK: scf.yield [[ALLOC1]], %true +// CHECK: [[V1:%.+]] = scf.if [[V0]]#1 +// CHECK: scf.yield [[V0]]#0 +// CHECK: [[CLONE:%.+]] = bufferization.clone [[V0]]#0 +// CHECK: scf.yield [[CLONE]] +// CHECK: [[BASE:%[a-zA-Z0-9_]+]],{{.*}} = memref.extract_strided_metadata [[V0]]#0 +// CHECK: bufferization.dealloc ([[ALLOC]], [[BASE]] :{{.*}}) if (%true{{[0-9_]*}}, [[V0]]#1) retain ([[V1]] : +// CHECK: return [[V1]] + +// ----- + +// Test Case: nested region control flow within a region interface. +// No copies are required in this case since the allocation finally escapes +// the method. + +func.func @inner_region_control_flow(%arg0 : index) -> memref { + %0 = memref.alloc(%arg0, %arg0) : memref + %1 = test.region_if %0 : memref -> (memref) then { + ^bb0(%arg1 : memref): + test.region_if_yield %arg1 : memref + } else { + ^bb0(%arg1 : memref): + test.region_if_yield %arg1 : memref + } join { + ^bb0(%arg1 : memref): + test.region_if_yield %arg1 : memref + } + return %1 : memref +} + +// CHECK-LABEL: func.func @inner_region_control_flow +// CHECK: [[ALLOC:%.+]] = memref.alloc( +// CHECK: [[V0:%.+]]:2 = test.region_if [[ALLOC]], %false +// CHECK: ^bb0([[ARG1:%.+]]: memref, [[ARG2:%.+]]: i1): +// CHECK: test.region_if_yield [[ARG1]], [[ARG2]] +// CHECK: ^bb0([[ARG1:%.+]]: memref, [[ARG2:%.+]]: i1): +// CHECK: test.region_if_yield [[ARG1]], [[ARG2]] +// CHECK: ^bb0([[ARG1:%.+]]: memref, [[ARG2:%.+]]: i1): +// CHECK: test.region_if_yield [[ARG1]], [[ARG2]] +// CHECK: [[V1:%.+]] = scf.if [[V0]]#1 +// CHECK: scf.yield [[V0]]#0 +// CHECK: [[CLONE:%.+]] = bufferization.clone [[V0]]#0 +// CHECK: scf.yield [[CLONE]] +// CHECK: [[BASE:%[a-zA-Z0-9_]+]],{{.*}} = memref.extract_strided_metadata [[V0]]#0 +// CHECK: bufferization.dealloc ([[ALLOC]], [[BASE]] :{{.*}}) if (%true{{[0-9_]*}}, [[V0]]#1) retain ([[V1]] : +// CHECK: return [[V1]] + +// ----- + +func.func @nestedRegionsAndCondBranchAlloca( + %arg0: i1, + %arg1: memref<2xf32>, + %arg2: memref<2xf32>) { + cf.cond_br %arg0, ^bb1, ^bb2 +^bb1: + cf.br ^bb3(%arg1 : memref<2xf32>) +^bb2: + %0 = memref.alloc() : memref<2xf32> + test.region_buffer_based in(%arg1: memref<2xf32>) out(%0: memref<2xf32>) { + ^bb0(%gen1_arg0: f32, %gen1_arg1: f32): + %1 = memref.alloca() : memref<2xf32> + test.buffer_based in(%arg1: memref<2xf32>) out(%1: memref<2xf32>) + %tmp1 = math.exp %gen1_arg0 : f32 + test.region_yield %tmp1 : f32 + } + cf.br ^bb3(%0 : memref<2xf32>) +^bb3(%1: memref<2xf32>): + test.copy(%1, %arg2) : (memref<2xf32>, memref<2xf32>) + return +} + +// CHECK-LABEL: func @nestedRegionsAndCondBranchAlloca +// CHECK-SAME: ([[ARG0:%.+]]: i1, [[ARG1:%.+]]: memref<2xf32>, [[ARG2:%.+]]: memref<2xf32>) +// CHECK: ^bb1: +// CHECK: cf.br ^bb3([[ARG1]], %false +// CHECK: ^bb2: +// CHECK: [[ALLOC:%.+]] = memref.alloc() +// CHECK: test.region_buffer_based +// CHECK: memref.alloca() +// CHECK: test.buffer_based +// CHECK-NOT: bufferization.dealloc +// CHECK-NOT: bufferization.clone +// CHECK: test.region_yield +// CHECK: } +// CHECK: cf.br ^bb3([[ALLOC]], %true +// CHECK: ^bb3([[A0:%.+]]: memref<2xf32>, [[COND:%.+]]: i1): +// CHECK: test.copy +// CHECK: [[BASE:%[a-zA-Z0-9_]+]],{{.*}} = memref.extract_strided_metadata [[A0]] +// CHECK: bufferization.dealloc ([[BASE]] :{{.*}}) if ([[COND]]) + +// ----- + +func.func @nestedRegionControlFlowAlloca( + %arg0 : index, %arg1 : index, %arg2: f32) -> memref { + %0 = arith.cmpi eq, %arg0, %arg1 : index + %1 = memref.alloc(%arg0, %arg0) : memref + %2 = scf.if %0 -> (memref) { + scf.yield %1 : memref + } else { + %3 = memref.alloca(%arg0, %arg1) : memref + %c0 = arith.constant 0 : index + memref.store %arg2, %3[%c0, %c0] : memref + scf.yield %1 : memref + } + return %2 : memref +} + +// CHECK-LABEL: func @nestedRegionControlFlowAlloca +// CHECK: [[ALLOC:%.+]] = memref.alloc( +// CHECK: [[V0:%.+]]:2 = scf.if +// CHECK: scf.yield [[ALLOC]], %false +// CHECK: memref.alloca( +// CHECK: scf.yield [[ALLOC]], %false +// CHECK: [[V1:%.+]] = scf.if [[V0]]#1 +// CHECK: scf.yield [[V0]]#0 +// CHECK: [[CLONE:%.+]] = bufferization.clone [[V0]]#0 +// CHECK: scf.yield [[CLONE]] +// CHECK: [[BASE:%[a-zA-Z0-9_]+]],{{.*}} = memref.extract_strided_metadata [[V0]]#0 +// CHECK: bufferization.dealloc ([[ALLOC]], [[BASE]] :{{.*}}) if (%true{{[0-9_]*}}, [[V0]]#1) retain ([[V1]] : +// CHECK: return [[V1]] + +// ----- + +// Test Case: structured control-flow loop using a nested alloc. +// The iteration argument %iterBuf has to be freed before yielding %3 to avoid +// memory leaks. + +func.func @loop_alloc( + %lb: index, + %ub: index, + %step: index, + %buf: memref<2xf32>, + %res: memref<2xf32>) { + %0 = memref.alloc() : memref<2xf32> + %1 = scf.for %i = %lb to %ub step %step + iter_args(%iterBuf = %buf) -> memref<2xf32> { + %2 = arith.cmpi eq, %i, %ub : index + %3 = memref.alloc() : memref<2xf32> + scf.yield %3 : memref<2xf32> + } + test.copy(%1, %res) : (memref<2xf32>, memref<2xf32>) + return +} + +// CHECK-LABEL: func @loop_alloc +// CHECK-SAME: ([[ARG0:%.+]]: index, [[ARG1:%.+]]: index, [[ARG2:%.+]]: index, [[ARG3:%.+]]: memref<2xf32>, [[ARG4:%.+]]: memref<2xf32>) +// CHECK: [[ALLOC:%.+]] = memref.alloc() +// CHECK: [[V0:%.+]]:2 = scf.for {{.*}} iter_args([[ARG6:%.+]] = [[ARG3]], [[ARG7:%.+]] = %false +// CHECK: [[ALLOC1:%.+]] = memref.alloc() +// CHECK: [[BASE:%[a-zA-Z0-9_]+]],{{.*}} = memref.extract_strided_metadata [[ARG6]] +// CHECK: bufferization.dealloc ([[BASE]] :{{.*}}) if ([[ARG7]]) retain ([[ALLOC1]] : +// CHECK: scf.yield [[ALLOC1]], %true +// CHECK: test.copy +// CHECK: [[BASE:%[a-zA-Z0-9_]+]],{{.*}} = memref.extract_strided_metadata [[V0]]#0 +// CHECK: bufferization.dealloc ([[ALLOC]] :{{.*}}) if (%true +// CHECK-NOT: retain +// CHECK: bufferization.dealloc ([[BASE]] :{{.*}}) if ([[V0]]#1) +// CHECK-NOT: retain + +// ----- + +// Test Case: structured control-flow loop with a nested if operation. +// The loop yields buffers that have been defined outside of the loop and the +// backedges only use the iteration arguments (or one of its aliases). +// Therefore, we do not have to (and are not allowed to) free any buffers +// that are passed via the backedges. + +func.func @loop_nested_if_no_alloc( + %lb: index, + %ub: index, + %step: index, + %buf: memref<2xf32>, + %res: memref<2xf32>) { + %0 = memref.alloc() : memref<2xf32> + %1 = scf.for %i = %lb to %ub step %step + iter_args(%iterBuf = %buf) -> memref<2xf32> { + %2 = arith.cmpi eq, %i, %ub : index + %3 = scf.if %2 -> (memref<2xf32>) { + scf.yield %0 : memref<2xf32> + } else { + scf.yield %iterBuf : memref<2xf32> + } + scf.yield %3 : memref<2xf32> + } + test.copy(%1, %res) : (memref<2xf32>, memref<2xf32>) + return +} + +// CHECK-LABEL: func @loop_nested_if_no_alloc +// CHECK-SAME: ({{.*}}, [[ARG3:%.+]]: memref<2xf32>, [[ARG4:%.+]]: memref<2xf32>) +// CHECK: [[ALLOC:%.+]] = memref.alloc() +// CHECK: [[V0:%.+]]:2 = scf.for {{.*}} iter_args([[ARG6:%.+]] = [[ARG3]], [[ARG7:%.+]] = %false +// CHECK: [[V1:%.+]]:2 = scf.if +// CHECK: scf.yield [[ALLOC]], %false +// CHECK: scf.yield [[ARG6]], %false +// CHECK: [[BASE:%[a-zA-Z0-9_]+]],{{.*}} = memref.extract_strided_metadata [[ARG6]] +// CHECK: [[OWN:%.+]] = bufferization.dealloc ([[BASE]] :{{.*}}) if ([[ARG7]]) retain ([[V1]]#0 : +// CHECK: [[OWN_AGG:%.+]] = arith.ori [[OWN]], [[V1]]#1 +// CHECK: scf.yield [[V1]]#0, [[OWN_AGG]] +// CHECK: test.copy +// CHECK: [[BASE:%[a-zA-Z0-9_]+]],{{.*}} = memref.extract_strided_metadata [[V0]]#0 +// CHECK: bufferization.dealloc ([[ALLOC]], [[BASE]] :{{.*}}) if (%true{{[0-9_]*}}, [[V0]]#1) + +// TODO: we know statically that the inner dealloc will never deallocate +// anything, i.e., we can optimize it away + +// ----- + +// Test Case: structured control-flow loop with a nested if operation using +// a deeply nested buffer allocation. + +func.func @loop_nested_if_alloc( + %lb: index, + %ub: index, + %step: index, + %buf: memref<2xf32>) -> memref<2xf32> { + %0 = memref.alloc() : memref<2xf32> + %1 = scf.for %i = %lb to %ub step %step + iter_args(%iterBuf = %buf) -> memref<2xf32> { + %2 = arith.cmpi eq, %i, %ub : index + %3 = scf.if %2 -> (memref<2xf32>) { + %4 = memref.alloc() : memref<2xf32> + scf.yield %4 : memref<2xf32> + } else { + scf.yield %0 : memref<2xf32> + } + scf.yield %3 : memref<2xf32> + } + return %1 : memref<2xf32> +} + +// CHECK-LABEL: func @loop_nested_if_alloc +// CHECK-SAME: ({{.*}}, [[ARG3:%.+]]: memref<2xf32>) +// CHECK: [[ALLOC:%.+]] = memref.alloc() +// CHECK: [[V0:%.+]]:2 = scf.for {{.*}} iter_args([[ARG5:%.+]] = [[ARG3]], [[ARG6:%.+]] = %false +// CHECK: [[V1:%.+]]:2 = scf.if +// CHECK: [[ALLOC1:%.+]] = memref.alloc() +// CHECK: scf.yield [[ALLOC1]], %true +// CHECK: scf.yield [[ALLOC]], %false +// CHECK: [[BASE:%[a-zA-Z0-9_]+]],{{.*}} = memref.extract_strided_metadata [[ARG5]] +// CHECK: [[OWN:%.+]] = bufferization.dealloc ([[BASE]] :{{.*}}) if ([[ARG6]]) retain ([[V1]]#0 : +// CHECK: [[OWN_AGG:%.+]] = arith.ori [[OWN]], [[V1]]#1 +// CHECK: scf.yield [[V1]]#0, [[OWN_AGG]] +// CHECK: } +// CHECK: [[V2:%.+]] = scf.if [[V0]]#1 +// CHECK: scf.yield [[V0]]#0 +// CHECK: [[CLONE:%.+]] = bufferization.clone [[V0]]#0 +// CHECK: scf.yield [[CLONE]] +// CHECK: [[BASE:%[a-zA-Z0-9_]+]],{{.*}} = memref.extract_strided_metadata [[V0]]#0 +// CHECK: bufferization.dealloc ([[ALLOC]], [[BASE]] :{{.*}}) if (%true{{[0-9_]*}}, [[V0]]#1) retain ([[V2]] : +// CHECK: return [[V2]] + +// ----- + +// Test Case: several nested structured control-flow loops with a deeply nested +// buffer allocation inside an if operation. + +func.func @loop_nested_alloc( + %lb: index, + %ub: index, + %step: index, + %buf: memref<2xf32>, + %res: memref<2xf32>) { + %0 = memref.alloc() : memref<2xf32> + %1 = scf.for %i = %lb to %ub step %step + iter_args(%iterBuf = %buf) -> memref<2xf32> { + %2 = scf.for %i2 = %lb to %ub step %step + iter_args(%iterBuf2 = %iterBuf) -> memref<2xf32> { + %3 = scf.for %i3 = %lb to %ub step %step + iter_args(%iterBuf3 = %iterBuf2) -> memref<2xf32> { + %4 = memref.alloc() : memref<2xf32> + %5 = arith.cmpi eq, %i, %ub : index + %6 = scf.if %5 -> (memref<2xf32>) { + %7 = memref.alloc() : memref<2xf32> + scf.yield %7 : memref<2xf32> + } else { + scf.yield %iterBuf3 : memref<2xf32> + } + scf.yield %6 : memref<2xf32> + } + scf.yield %3 : memref<2xf32> + } + scf.yield %2 : memref<2xf32> + } + test.copy(%1, %res) : (memref<2xf32>, memref<2xf32>) + return +} + +// CHECK-LABEL: func @loop_nested_alloc +// CHECK: ({{.*}}, [[ARG3:%.+]]: memref<2xf32>, {{.*}}: memref<2xf32>) +// CHECK: [[ALLOC:%.+]] = memref.alloc() +// CHECK: [[V0:%.+]]:2 = scf.for {{.*}} iter_args([[ARG6:%.+]] = [[ARG3]], [[ARG7:%.+]] = %false +// CHECK: [[V1:%.+]]:2 = scf.for {{.*}} iter_args([[ARG9:%.+]] = [[ARG6]], [[ARG10:%.+]] = %false +// CHECK: [[V2:%.+]]:2 = scf.for {{.*}} iter_args([[ARG12:%.+]] = [[ARG9]], [[ARG13:%.+]] = %false +// CHECK: [[ALLOC1:%.+]] = memref.alloc() +// CHECK: [[V3:%.+]]:2 = scf.if +// CHECK: [[ALLOC2:%.+]] = memref.alloc() +// CHECK: scf.yield [[ALLOC2]], %true +// CHECK: } else { +// CHECK: scf.yield [[ARG12]], %false +// CHECK: } +// CHECK: [[BASE:%[a-zA-Z0-9_]+]],{{.*}} = memref.extract_strided_metadata [[ARG12]] +// CHECK: [[OWN:%.+]] = bufferization.dealloc ([[BASE]] :{{.*}}) if ([[ARG13]]) retain ([[V3]]#0 : +// CHECK: bufferization.dealloc ([[ALLOC1]] :{{.*}}) if (%true{{[0-9_]*}}) +// CHECK-NOT: retain +// CHECK: [[OWN_AGG:%.+]] = arith.ori [[OWN]], [[V3]]#1 +// CHECK: scf.yield [[V3]]#0, [[OWN_AGG]] +// CHECK: } +// CHECK: [[BASE:%[a-zA-Z0-9_]+]],{{.*}} = memref.extract_strided_metadata [[ARG9]] +// CHECK: [[OWN:%.+]] = bufferization.dealloc ([[BASE]] :{{.*}}) if ([[ARG10]]) retain ([[V2]]#0 : +// CHECK: [[OWN_AGG:%.+]] = arith.ori [[OWN]], [[V2]]#1 +// CHECK: scf.yield [[V2]]#0, [[OWN_AGG]] +// CHECK: } +// CHECK: [[BASE:%[a-zA-Z0-9_]+]],{{.*}} = memref.extract_strided_metadata [[ARG6]] +// CHECK: [[OWN:%.+]] = bufferization.dealloc ([[BASE]] :{{.*}}) if ([[ARG7]]) retain ([[V1]]#0 : +// CHECK: [[OWN_AGG:%.+]] = arith.ori [[OWN]], [[V1]]#1 +// CHECK: scf.yield [[V1]]#0, [[OWN_AGG]] +// CHECK: } +// CHECK: test.copy +// CHECK: [[BASE:%[a-zA-Z0-9_]+]],{{.*}} = memref.extract_strided_metadata [[V0]]#0 +// CHECK: bufferization.dealloc ([[ALLOC]] :{{.*}}) if (%true +// CHECK: bufferization.dealloc ([[BASE]] :{{.*}}) if ([[V0]]#1) + +// TODO: all the retain operands could be removed by doing some more thorough analysis + +// ----- + +func.func @affine_loop() -> f32 { + %buffer = memref.alloc() : memref<1024xf32> + %sum_init_0 = arith.constant 0.0 : f32 + %res = affine.for %i = 0 to 10 step 2 iter_args(%sum_iter = %sum_init_0) -> f32 { + %t = affine.load %buffer[%i] : memref<1024xf32> + %sum_next = arith.addf %sum_iter, %t : f32 + affine.yield %sum_next : f32 + } + return %res : f32 +} + +// CHECK-LABEL: func @affine_loop +// CHECK: [[ALLOC:%.+]] = memref.alloc() +// CHECK: affine.for {{.*}} iter_args(%arg1 = %cst) +// CHECK: affine.yield +// CHECK: bufferization.dealloc ([[ALLOC]] :{{.*}}) if (%true + +// ----- + +func.func @assumingOp( + %arg0: !shape.witness, + %arg2: memref<2xf32>, + %arg3: memref<2xf32>) { + // Confirm the alloc will be dealloc'ed in the block. + %1 = shape.assuming %arg0 -> memref<2xf32> { + %0 = memref.alloc() : memref<2xf32> + shape.assuming_yield %arg2 : memref<2xf32> + } + // Confirm the alloc will be returned and dealloc'ed after its use. + %3 = shape.assuming %arg0 -> memref<2xf32> { + %2 = memref.alloc() : memref<2xf32> + shape.assuming_yield %2 : memref<2xf32> + } + test.copy(%3, %arg3) : (memref<2xf32>, memref<2xf32>) + return +} + +// CHECK-LABEL: func @assumingOp +// CHECK: ({{.*}}, [[ARG1:%.+]]: memref<2xf32>, {{.*}}: memref<2xf32>) +// CHECK: [[V0:%.+]]:2 = shape.assuming +// CHECK: [[ALLOC:%.+]] = memref.alloc() +// CHECK: bufferization.dealloc ([[ALLOC]] :{{.*}}) if (%true{{[0-9_]*}}) +// CHECK-NOT: retain +// CHECK: shape.assuming_yield [[ARG1]], %false +// CHECK: } +// CHECK: [[V1:%.+]]:2 = shape.assuming +// CHECK: [[ALLOC:%.+]] = memref.alloc() +// CHECK: shape.assuming_yield [[ALLOC]], %true +// CHECK: } +// CHECK: test.copy +// CHECK: [[BASE0:%[a-zA-Z0-9_]+]],{{.*}} = memref.extract_strided_metadata [[V0]]#0 +// CHECK: [[BASE1:%[a-zA-Z0-9_]+]],{{.*}} = memref.extract_strided_metadata [[V1]]#0 +// CHECK: bufferization.dealloc ([[BASE0]] :{{.*}}) if ([[V0]]#1) +// CHECK-NOT: retain +// CHECK: bufferization.dealloc ([[BASE1]] :{{.*}}) if ([[V1]]#1) +// CHECK-NOT: retain +// CHECK: return + +// ----- + +// Test Case: The op "test.bar" does not implement the RegionBranchOpInterface. +// This is not allowed in buffer deallocation. + +func.func @noRegionBranchOpInterface() { + %0 = "test.bar"() ({ + // expected-error@+1 {{All operations with attached regions need to implement the RegionBranchOpInterface.}} + %1 = "test.bar"() ({ + "test.yield"() : () -> () + }) : () -> (i32) + "test.yield"() : () -> () + }) : () -> (i32) + "test.terminator"() : () -> () +} + +// ----- + +func.func @while_two_arg(%arg0: index) { + %a = memref.alloc(%arg0) : memref + scf.while (%arg1 = %a, %arg2 = %a) : (memref, memref) -> (memref, memref) { + %0 = "test.make_condition"() : () -> i1 + scf.condition(%0) %arg1, %arg2 : memref, memref + } do { + ^bb0(%arg1: memref, %arg2: memref): + %b = memref.alloc(%arg0) : memref + scf.yield %arg1, %b : memref, memref + } + return +} + +// CHECK-LABEL: func @while_two_arg +// CHECK: [[ALLOC:%.+]] = memref.alloc( +// CHECK: [[V0:%.+]]:4 = scf.while ({{.*}} = [[ALLOC]], {{.*}} = [[ALLOC]], {{.*}} = %false{{[0-9_]*}}, {{.*}} = %false{{[0-9_]*}}) +// CHECK: scf.condition +// CHECK: ^bb0([[ARG1:%.+]]: memref, [[ARG2:%.+]]: memref, [[ARG3:%.+]]: i1, [[ARG4:%.+]]: i1): +// CHECK: [[ALLOC1:%.+]] = memref.alloc( +// CHECK: [[BASE:%[a-zA-Z0-9_]+]],{{.*}} = memref.extract_strided_metadata [[ARG2]] +// CHECK: [[OWN:%.+]]:2 = bufferization.dealloc ([[BASE]] :{{.*}}) if ([[ARG4]]) retain ([[ARG1]], [[ALLOC1]] : +// CHECK: [[OWN_AGG:%.+]] = arith.ori [[OWN]]#0, [[ARG3]] +// CHECK: scf.yield [[ARG1]], [[ALLOC1]], [[OWN_AGG]], %true +// CHECK: [[BASE0:%[a-zA-Z0-9_]+]],{{.*}} = memref.extract_strided_metadata [[V0]]#0 +// CHECK: [[BASE1:%[a-zA-Z0-9_]+]],{{.*}} = memref.extract_strided_metadata [[V0]]#1 +// CHECK: bufferization.dealloc ([[ALLOC]], [[BASE0]], [[BASE1]] :{{.*}}) if (%true{{[0-9_]*}}, [[V0]]#2, [[V0]]#3) + +// ----- + +func.func @while_three_arg(%arg0: index) { + %a = memref.alloc(%arg0) : memref + scf.while (%arg1 = %a, %arg2 = %a, %arg3 = %a) : (memref, memref, memref) -> (memref, memref, memref) { + %0 = "test.make_condition"() : () -> i1 + scf.condition(%0) %arg1, %arg2, %arg3 : memref, memref, memref + } do { + ^bb0(%arg1: memref, %arg2: memref, %arg3: memref): + %b = memref.alloc(%arg0) : memref + %q = memref.alloc(%arg0) : memref + scf.yield %q, %b, %arg2: memref, memref, memref + } + return +} + +// CHECK-LABEL: func @while_three_arg +// CHECK: [[ALLOC:%.+]] = memref.alloc( +// CHECK: [[V0:%.+]]:6 = scf.while ({{.*}} = [[ALLOC]], {{.*}} = [[ALLOC]], {{.*}} = [[ALLOC]], {{.*}} = %false{{[0-9_]*}}, {{.*}} = %false{{[0-9_]*}}, {{.*}} = %false +// CHECK: scf.condition +// CHECK: ^bb0([[ARG1:%.+]]: memref, [[ARG2:%.+]]: memref, [[ARG3:%.+]]: memref, [[ARG4:%.+]]: i1, [[ARG5:%.+]]: i1, [[ARG6:%.+]]: i1): +// CHECK: [[ALLOC1:%.+]] = memref.alloc( +// CHECK: [[ALLOC2:%.+]] = memref.alloc( +// CHECK: [[BASE0:%[a-zA-Z0-9_]+]],{{.*}} = memref.extract_strided_metadata [[ARG1]] +// CHECK: [[BASE1:%[a-zA-Z0-9_]+]],{{.*}} = memref.extract_strided_metadata [[ARG2]] +// CHECK: [[BASE2:%[a-zA-Z0-9_]+]],{{.*}} = memref.extract_strided_metadata [[ARG3]] +// CHECK: [[OWN:%.+]]:3 = bufferization.dealloc ([[BASE0]], [[BASE1]], [[BASE2]], [[ALLOC1]] :{{.*}}) if ([[ARG4]], [[ARG5]], [[ARG6]], %true{{[0-9_]*}}) retain ([[ALLOC2]], [[ALLOC1]], [[ARG2]] : +// CHECK: scf.yield [[ALLOC2]], [[ALLOC1]], [[ARG2]], %true{{[0-9_]*}}, [[OWN]]#1, [[OWN]]#2 : +// CHECK: } +// CHECK: [[BASE0:%[a-zA-Z0-9_]+]],{{.*}} = memref.extract_strided_metadata [[V0]]#0 +// CHECK: [[BASE1:%[a-zA-Z0-9_]+]],{{.*}} = memref.extract_strided_metadata [[V0]]#1 +// CHECK: [[BASE2:%[a-zA-Z0-9_]+]],{{.*}} = memref.extract_strided_metadata [[V0]]#2 +// CHECK: bufferization.dealloc ([[ALLOC]], [[BASE0]], [[BASE1]], [[BASE2]] :{{.*}}) if (%true{{[0-9_]*}}, [[V0]]#3, [[V0]]#4, [[V0]]#5) + +// TODO: better alias analysis could simplify the dealloc inside the body further + +// ----- + +// Memref allocated in `then` region and passed back to the parent if op. +#set = affine_set<() : (0 >= 0)> +func.func @test_affine_if_1(%arg0: memref<10xf32>) -> memref<10xf32> { + %0 = affine.if #set() -> memref<10xf32> { + %alloc = memref.alloc() : memref<10xf32> + affine.yield %alloc : memref<10xf32> + } else { + affine.yield %arg0 : memref<10xf32> + } + return %0 : memref<10xf32> +} + +// CHECK-LABEL: func @test_affine_if_1 +// CHECK-SAME: ([[ARG0:%.*]]: memref<10xf32>) +// CHECK: [[V0:%.+]]:2 = affine.if +// CHECK: [[ALLOC:%.+]] = memref.alloc() +// CHECK: affine.yield [[ALLOC]], %true +// CHECK: affine.yield [[ARG0]], %false +// CHECK: [[V1:%.+]] = scf.if [[V0]]#1 +// CHECK: scf.yield [[V0]]#0 +// CHECK: [[CLONE:%.+]] = bufferization.clone [[V0]]#0 +// CHECK: scf.yield [[CLONE]] +// CHECK: [[BASE:%[a-zA-Z0-9_]+]],{{.*}} = memref.extract_strided_metadata [[V0]]#0 +// CHECK: bufferization.dealloc ([[BASE]] :{{.*}}) if ([[V0]]#1) retain ([[V1]] : +// CHECK: return [[V1]] + +// TODO: the dealloc could be optimized away since the memref to be deallocated +// either aliases with V1 or the condition is false + +// ----- + +// Memref allocated before parent IfOp and used in `then` region. +// Expected result: deallocation should happen after affine.if op. +#set = affine_set<() : (0 >= 0)> +func.func @test_affine_if_2() -> memref<10xf32> { + %alloc0 = memref.alloc() : memref<10xf32> + %0 = affine.if #set() -> memref<10xf32> { + affine.yield %alloc0 : memref<10xf32> + } else { + %alloc = memref.alloc() : memref<10xf32> + affine.yield %alloc : memref<10xf32> + } + return %0 : memref<10xf32> +} +// CHECK-LABEL: func @test_affine_if_2 +// CHECK: [[ALLOC:%.+]] = memref.alloc() +// CHECK: [[V0:%.+]]:2 = affine.if +// CHECK: affine.yield [[ALLOC]], %false +// CHECK: [[ALLOC1:%.+]] = memref.alloc() +// CHECK: affine.yield [[ALLOC1]], %true +// CHECK: [[V1:%.+]] = scf.if [[V0]]#1 +// CHECK: scf.yield [[V0]]#0 +// CHECK: [[CLONE:%.+]] = bufferization.clone [[V0]]#0 +// CHECK: scf.yield [[CLONE]] +// CHECK: [[BASE:%[a-zA-Z0-9_]+]],{{.*}} = memref.extract_strided_metadata [[V0]]#0 +// CHECK: bufferization.dealloc ([[ALLOC]], [[BASE]] :{{.*}}) if (%true{{[0-9_]*}}, [[V0]]#1) retain ([[V1]] : +// CHECK: return [[V1]] + +// ----- + +// Memref allocated before parent IfOp and used in `else` region. +// Expected result: deallocation should happen after affine.if op. +#set = affine_set<() : (0 >= 0)> +func.func @test_affine_if_3() -> memref<10xf32> { + %alloc0 = memref.alloc() : memref<10xf32> + %0 = affine.if #set() -> memref<10xf32> { + %alloc = memref.alloc() : memref<10xf32> + affine.yield %alloc : memref<10xf32> + } else { + affine.yield %alloc0 : memref<10xf32> + } + return %0 : memref<10xf32> +} + +// CHECK-LABEL: func @test_affine_if_3 +// CHECK: [[ALLOC:%.+]] = memref.alloc() +// CHECK: [[V0:%.+]]:2 = affine.if +// CHECK: [[ALLOC1:%.+]] = memref.alloc() +// CHECK: affine.yield [[ALLOC1]], %true +// CHECK: affine.yield [[ALLOC]], %false +// CHECK: [[V1:%.+]] = scf.if [[V0]]#1 +// CHECK: scf.yield [[V0]]#0 +// CHECK: [[CLONE:%.+]] = bufferization.clone [[V0]]#0 +// CHECK: scf.yield [[CLONE]] +// CHECK: [[BASE:%[a-zA-Z0-9_]+]],{{.*}} = memref.extract_strided_metadata [[V0]]#0 +// CHECK: bufferization.dealloc ([[ALLOC]], [[BASE]] :{{.*}}) if (%true{{[0-9_]*}}, [[V0]]#1) retain ([[V1]] +// CHECK: return [[V1]] diff --git a/mlir/test/Dialect/Bufferization/Transforms/BufferDeallocation/dealloc-subviews.mlir b/mlir/test/Dialect/Bufferization/Transforms/BufferDeallocation/dealloc-subviews.mlir new file mode 100644 --- /dev/null +++ b/mlir/test/Dialect/Bufferization/Transforms/BufferDeallocation/dealloc-subviews.mlir @@ -0,0 +1,23 @@ +// DEFINE: %{canonicalize} = -canonicalize=enable-patterns="bufferization-skip-extract-metadata-of-alloc,bufferization-erase-always-false-dealloc,bufferization-erase-empty-dealloc,bufferization-dealloc-remove-duplicate-retained-memrefs,bufferization-dealloc-remove-duplicate-dealloc-memrefs",region-simplify=false + +// RUN: mlir-opt -verify-diagnostics -buffer-deallocation \ +// RUN: %{canonicalize} --buffer-deallocation-simplification %{canonicalize} -split-input-file %s | FileCheck %s +// RUN: mlir-opt -verify-diagnostics -buffer-deallocation=private-function-dynamic-ownership=true -split-input-file %s > /dev/null + +// CHECK-LABEL: func @subview +func.func @subview(%arg0 : index, %arg1 : index, %arg2 : memref) { + %0 = memref.alloc() : memref<64x4xf32, strided<[4, 1], offset: 0>> + %1 = memref.subview %0[%arg0, %arg1][%arg0, %arg1][%arg0, %arg1] : + memref<64x4xf32, strided<[4, 1], offset: 0>> + to memref> + test.copy(%1, %arg2) : + (memref>, memref) + return +} + +// CHECK: %[[ALLOC:.*]] = memref.alloc() +// CHECK-NEXT: memref.subview +// CHECK-NEXT: test.copy +// CHECK-NEXT: bufferization.dealloc (%[[ALLOC]] : +// CHECK-SAME: if (%true) +// CHECK-NEXT: return diff --git a/mlir/test/Dialect/Bufferization/Transforms/BufferDeallocation/invalid-buffer-deallocation.mlir b/mlir/test/Dialect/Bufferization/Transforms/BufferDeallocation/invalid-buffer-deallocation.mlir new file mode 100644 --- /dev/null +++ b/mlir/test/Dialect/Bufferization/Transforms/BufferDeallocation/invalid-buffer-deallocation.mlir @@ -0,0 +1,65 @@ +// RUN: mlir-opt -verify-diagnostics -buffer-deallocation -split-input-file %s + + +// Test Case: explicit control-flow loop with a dynamically allocated buffer. +// The BufferDeallocation transformation should fail on this explicit +// control-flow loop since they are not supported. + +// expected-error@+1 {{Only structured control-flow loops are supported}} +func.func @loop_dynalloc( + %arg0 : i32, + %arg1 : i32, + %arg2: memref, + %arg3: memref) { + %const0 = arith.constant 0 : i32 + cf.br ^loopHeader(%const0, %arg2 : i32, memref) + +^loopHeader(%i : i32, %buff : memref): + %lessThan = arith.cmpi slt, %i, %arg1 : i32 + cf.cond_br %lessThan, + ^loopBody(%i, %buff : i32, memref), + ^exit(%buff : memref) + +^loopBody(%val : i32, %buff2: memref): + %const1 = arith.constant 1 : i32 + %inc = arith.addi %val, %const1 : i32 + %size = arith.index_cast %inc : i32 to index + %alloc1 = memref.alloc(%size) : memref + cf.br ^loopHeader(%inc, %alloc1 : i32, memref) + +^exit(%buff3 : memref): + test.copy(%buff3, %arg3) : (memref, memref) + return +} + +// ----- + +// Test Case: explicit control-flow loop with a dynamically allocated buffer. +// The BufferDeallocation transformation should fail on this explicit +// control-flow loop since they are not supported. + +// expected-error@+1 {{Only structured control-flow loops are supported}} +func.func @do_loop_alloc( + %arg0 : i32, + %arg1 : i32, + %arg2: memref<2xf32>, + %arg3: memref<2xf32>) { + %const0 = arith.constant 0 : i32 + cf.br ^loopBody(%const0, %arg2 : i32, memref<2xf32>) + +^loopBody(%val : i32, %buff2: memref<2xf32>): + %const1 = arith.constant 1 : i32 + %inc = arith.addi %val, %const1 : i32 + %alloc1 = memref.alloc() : memref<2xf32> + cf.br ^loopHeader(%inc, %alloc1 : i32, memref<2xf32>) + +^loopHeader(%i : i32, %buff : memref<2xf32>): + %lessThan = arith.cmpi slt, %i, %arg1 : i32 + cf.cond_br %lessThan, + ^loopBody(%i, %buff : i32, memref<2xf32>), + ^exit(%buff : memref<2xf32>) + +^exit(%buff3 : memref<2xf32>): + test.copy(%buff3, %arg3) : (memref<2xf32>, memref<2xf32>) + return +} diff --git a/mlir/test/Dialect/Bufferization/Transforms/buffer-deallocation.mlir b/mlir/test/Dialect/Bufferization/Transforms/buffer-deallocation.mlir deleted file mode 100644 --- a/mlir/test/Dialect/Bufferization/Transforms/buffer-deallocation.mlir +++ /dev/null @@ -1,1462 +0,0 @@ -// RUN: mlir-opt -verify-diagnostics -buffer-deallocation -split-input-file %s | FileCheck %s - -// This file checks the behaviour of BufferDeallocation pass for moving and -// inserting missing DeallocOps in their correct positions. Furthermore, -// copies and their corresponding AllocOps are inserted. - -// Test Case: -// bb0 -// / \ -// bb1 bb2 <- Initial position of AllocOp -// \ / -// bb3 -// BufferDeallocation expected behavior: bb2 contains an AllocOp which is -// passed to bb3. In the latter block, there should be an deallocation. -// Since bb1 does not contain an adequate alloc and the alloc in bb2 is not -// moved to bb0, we need to insert allocs and copies. - -// CHECK-LABEL: func @condBranch -func.func @condBranch(%arg0: i1, %arg1: memref<2xf32>, %arg2: memref<2xf32>) { - cf.cond_br %arg0, ^bb1, ^bb2 -^bb1: - cf.br ^bb3(%arg1 : memref<2xf32>) -^bb2: - %0 = memref.alloc() : memref<2xf32> - test.buffer_based in(%arg1: memref<2xf32>) out(%0: memref<2xf32>) - cf.br ^bb3(%0 : memref<2xf32>) -^bb3(%1: memref<2xf32>): - test.copy(%1, %arg2) : (memref<2xf32>, memref<2xf32>) - return -} - -// CHECK-NEXT: cf.cond_br -// CHECK: %[[ALLOC0:.*]] = bufferization.clone -// CHECK-NEXT: cf.br ^bb3(%[[ALLOC0]] -// CHECK: %[[ALLOC1:.*]] = memref.alloc -// CHECK-NEXT: test.buffer_based -// CHECK-NEXT: %[[ALLOC2:.*]] = bufferization.clone %[[ALLOC1]] -// CHECK-NEXT: memref.dealloc %[[ALLOC1]] -// CHECK-NEXT: cf.br ^bb3(%[[ALLOC2]] -// CHECK: test.copy -// CHECK-NEXT: memref.dealloc -// CHECK-NEXT: return - -// ----- - -// Test Case: -// bb0 -// / \ -// bb1 bb2 <- Initial position of AllocOp -// \ / -// bb3 -// BufferDeallocation expected behavior: The existing AllocOp has a dynamic -// dependency to block argument %0 in bb2. Since the dynamic type is passed -// to bb3 via the block argument %2, it is currently required to allocate a -// temporary buffer for %2 that gets copies of %arg0 and %1 with their -// appropriate shape dimensions. The copy buffer deallocation will be applied -// to %2 in block bb3. - -// CHECK-LABEL: func @condBranchDynamicType -func.func @condBranchDynamicType( - %arg0: i1, - %arg1: memref, - %arg2: memref, - %arg3: index) { - cf.cond_br %arg0, ^bb1, ^bb2(%arg3: index) -^bb1: - cf.br ^bb3(%arg1 : memref) -^bb2(%0: index): - %1 = memref.alloc(%0) : memref - test.buffer_based in(%arg1: memref) out(%1: memref) - cf.br ^bb3(%1 : memref) -^bb3(%2: memref): - test.copy(%2, %arg2) : (memref, memref) - return -} - -// CHECK-NEXT: cf.cond_br -// CHECK: %[[ALLOC0:.*]] = bufferization.clone -// CHECK-NEXT: cf.br ^bb3(%[[ALLOC0]] -// CHECK: ^bb2(%[[IDX:.*]]:{{.*}}) -// CHECK-NEXT: %[[ALLOC1:.*]] = memref.alloc(%[[IDX]]) -// CHECK-NEXT: test.buffer_based -// CHECK-NEXT: %[[ALLOC2:.*]] = bufferization.clone -// CHECK-NEXT: memref.dealloc %[[ALLOC1]] -// CHECK-NEXT: cf.br ^bb3 -// CHECK-NEXT: ^bb3(%[[ALLOC3:.*]]:{{.*}}) -// CHECK: test.copy(%[[ALLOC3]], -// CHECK-NEXT: memref.dealloc %[[ALLOC3]] -// CHECK-NEXT: return - -// ----- - -// Test case: See above. - -// CHECK-LABEL: func @condBranchUnrankedType -func.func @condBranchUnrankedType( - %arg0: i1, - %arg1: memref<*xf32>, - %arg2: memref<*xf32>, - %arg3: index) { - cf.cond_br %arg0, ^bb1, ^bb2(%arg3: index) -^bb1: - cf.br ^bb3(%arg1 : memref<*xf32>) -^bb2(%0: index): - %1 = memref.alloc(%0) : memref - %2 = memref.cast %1 : memref to memref<*xf32> - test.buffer_based in(%arg1: memref<*xf32>) out(%2: memref<*xf32>) - cf.br ^bb3(%2 : memref<*xf32>) -^bb3(%3: memref<*xf32>): - test.copy(%3, %arg2) : (memref<*xf32>, memref<*xf32>) - return -} - -// CHECK-NEXT: cf.cond_br -// CHECK: %[[ALLOC0:.*]] = bufferization.clone -// CHECK-NEXT: cf.br ^bb3(%[[ALLOC0]] -// CHECK: ^bb2(%[[IDX:.*]]:{{.*}}) -// CHECK-NEXT: %[[ALLOC1:.*]] = memref.alloc(%[[IDX]]) -// CHECK: test.buffer_based -// CHECK-NEXT: %[[ALLOC2:.*]] = bufferization.clone -// CHECK-NEXT: memref.dealloc %[[ALLOC1]] -// CHECK-NEXT: cf.br ^bb3 -// CHECK-NEXT: ^bb3(%[[ALLOC3:.*]]:{{.*}}) -// CHECK: test.copy(%[[ALLOC3]], -// CHECK-NEXT: memref.dealloc %[[ALLOC3]] -// CHECK-NEXT: return - -// ----- - -// Test Case: -// bb0 -// / \ -// bb1 bb2 <- Initial position of AllocOp -// | / \ -// | bb3 bb4 -// | \ / -// \ bb5 -// \ / -// bb6 -// | -// bb7 -// BufferDeallocation expected behavior: The existing AllocOp has a dynamic -// dependency to block argument %0 in bb2. Since the dynamic type is passed to -// bb5 via the block argument %2 and to bb6 via block argument %3, it is -// currently required to allocate temporary buffers for %2 and %3 that gets -// copies of %1 and %arg0 1 with their appropriate shape dimensions. The copy -// buffer deallocations will be applied to %2 in block bb5 and to %3 in block -// bb6. Furthermore, there should be no copy inserted for %4. - -// CHECK-LABEL: func @condBranchDynamicTypeNested -func.func @condBranchDynamicTypeNested( - %arg0: i1, - %arg1: memref, - %arg2: memref, - %arg3: index) { - cf.cond_br %arg0, ^bb1, ^bb2(%arg3: index) -^bb1: - cf.br ^bb6(%arg1 : memref) -^bb2(%0: index): - %1 = memref.alloc(%0) : memref - test.buffer_based in(%arg1: memref) out(%1: memref) - cf.cond_br %arg0, ^bb3, ^bb4 -^bb3: - cf.br ^bb5(%1 : memref) -^bb4: - cf.br ^bb5(%1 : memref) -^bb5(%2: memref): - cf.br ^bb6(%2 : memref) -^bb6(%3: memref): - cf.br ^bb7(%3 : memref) -^bb7(%4: memref): - test.copy(%4, %arg2) : (memref, memref) - return -} - -// CHECK-NEXT: cf.cond_br{{.*}} -// CHECK-NEXT: ^bb1 -// CHECK-NEXT: %[[ALLOC0:.*]] = bufferization.clone -// CHECK-NEXT: cf.br ^bb6(%[[ALLOC0]] -// CHECK: ^bb2(%[[IDX:.*]]:{{.*}}) -// CHECK-NEXT: %[[ALLOC1:.*]] = memref.alloc(%[[IDX]]) -// CHECK-NEXT: test.buffer_based -// CHECK: cf.cond_br -// CHECK: ^bb3: -// CHECK-NEXT: cf.br ^bb5(%[[ALLOC1]]{{.*}}) -// CHECK: ^bb4: -// CHECK-NEXT: cf.br ^bb5(%[[ALLOC1]]{{.*}}) -// CHECK-NEXT: ^bb5(%[[ALLOC2:.*]]:{{.*}}) -// CHECK-NEXT: %[[ALLOC3:.*]] = bufferization.clone %[[ALLOC2]] -// CHECK-NEXT: memref.dealloc %[[ALLOC1]] -// CHECK-NEXT: cf.br ^bb6(%[[ALLOC3]]{{.*}}) -// CHECK-NEXT: ^bb6(%[[ALLOC4:.*]]:{{.*}}) -// CHECK-NEXT: cf.br ^bb7(%[[ALLOC4]]{{.*}}) -// CHECK-NEXT: ^bb7(%[[ALLOC5:.*]]:{{.*}}) -// CHECK: test.copy(%[[ALLOC5]], -// CHECK-NEXT: memref.dealloc %[[ALLOC4]] -// CHECK-NEXT: return - -// ----- - -// Test Case: Existing AllocOp with no users. -// BufferDeallocation expected behavior: It should insert a DeallocOp right -// before ReturnOp. - -// CHECK-LABEL: func @emptyUsesValue -func.func @emptyUsesValue(%arg0: memref<4xf32>) { - %0 = memref.alloc() : memref<4xf32> - return -} -// CHECK-NEXT: %[[ALLOC:.*]] = memref.alloc() -// CHECK-NEXT: memref.dealloc %[[ALLOC]] -// CHECK-NEXT: return - -// ----- - -// Test Case: -// bb0 -// / \ -// | bb1 <- Initial position of AllocOp -// \ / -// bb2 -// BufferDeallocation expected behavior: It should insert a DeallocOp at the -// exit block after CopyOp since %1 is an alias for %0 and %arg1. Furthermore, -// we have to insert a copy and an alloc in the beginning of the function. - -// CHECK-LABEL: func @criticalEdge -func.func @criticalEdge(%arg0: i1, %arg1: memref<2xf32>, %arg2: memref<2xf32>) { - cf.cond_br %arg0, ^bb1, ^bb2(%arg1 : memref<2xf32>) -^bb1: - %0 = memref.alloc() : memref<2xf32> - test.buffer_based in(%arg1: memref<2xf32>) out(%0: memref<2xf32>) - cf.br ^bb2(%0 : memref<2xf32>) -^bb2(%1: memref<2xf32>): - test.copy(%1, %arg2) : (memref<2xf32>, memref<2xf32>) - return -} - -// CHECK-NEXT: %[[ALLOC0:.*]] = bufferization.clone -// CHECK-NEXT: cf.cond_br -// CHECK: %[[ALLOC1:.*]] = memref.alloc() -// CHECK-NEXT: test.buffer_based -// CHECK-NEXT: %[[ALLOC2:.*]] = bufferization.clone %[[ALLOC1]] -// CHECK-NEXT: memref.dealloc %[[ALLOC1]] -// CHECK: test.copy -// CHECK-NEXT: memref.dealloc -// CHECK-NEXT: return - -// ----- - -// Test Case: -// bb0 <- Initial position of AllocOp -// / \ -// | bb1 -// \ / -// bb2 -// BufferDeallocation expected behavior: It only inserts a DeallocOp at the -// exit block after CopyOp since %1 is an alias for %0 and %arg1. - -// CHECK-LABEL: func @invCriticalEdge -func.func @invCriticalEdge(%arg0: i1, %arg1: memref<2xf32>, %arg2: memref<2xf32>) { - %0 = memref.alloc() : memref<2xf32> - test.buffer_based in(%arg1: memref<2xf32>) out(%0: memref<2xf32>) - cf.cond_br %arg0, ^bb1, ^bb2(%arg1 : memref<2xf32>) -^bb1: - cf.br ^bb2(%0 : memref<2xf32>) -^bb2(%1: memref<2xf32>): - test.copy(%1, %arg2) : (memref<2xf32>, memref<2xf32>) - return -} - -// CHECK: dealloc -// CHECK-NEXT: return - -// ----- - -// Test Case: -// bb0 <- Initial position of the first AllocOp -// / \ -// bb1 bb2 -// \ / -// bb3 <- Initial position of the second AllocOp -// BufferDeallocation expected behavior: It only inserts two missing -// DeallocOps in the exit block. %5 is an alias for %0. Therefore, the -// DeallocOp for %0 should occur after the last BufferBasedOp. The Dealloc for -// %7 should happen after CopyOp. - -// CHECK-LABEL: func @ifElse -func.func @ifElse(%arg0: i1, %arg1: memref<2xf32>, %arg2: memref<2xf32>) { - %0 = memref.alloc() : memref<2xf32> - test.buffer_based in(%arg1: memref<2xf32>) out(%0: memref<2xf32>) - cf.cond_br %arg0, - ^bb1(%arg1, %0 : memref<2xf32>, memref<2xf32>), - ^bb2(%0, %arg1 : memref<2xf32>, memref<2xf32>) -^bb1(%1: memref<2xf32>, %2: memref<2xf32>): - cf.br ^bb3(%1, %2 : memref<2xf32>, memref<2xf32>) -^bb2(%3: memref<2xf32>, %4: memref<2xf32>): - cf.br ^bb3(%3, %4 : memref<2xf32>, memref<2xf32>) -^bb3(%5: memref<2xf32>, %6: memref<2xf32>): - %7 = memref.alloc() : memref<2xf32> - test.buffer_based in(%5: memref<2xf32>) out(%7: memref<2xf32>) - test.copy(%7, %arg2) : (memref<2xf32>, memref<2xf32>) - return -} - -// CHECK-NEXT: %[[FIRST_ALLOC:.*]] = memref.alloc() -// CHECK-NEXT: test.buffer_based -// CHECK: %[[SECOND_ALLOC:.*]] = memref.alloc() -// CHECK-NEXT: test.buffer_based -// CHECK: memref.dealloc %[[FIRST_ALLOC]] -// CHECK: test.copy -// CHECK-NEXT: memref.dealloc %[[SECOND_ALLOC]] -// CHECK-NEXT: return - -// ----- - -// Test Case: No users for buffer in if-else CFG -// bb0 <- Initial position of AllocOp -// / \ -// bb1 bb2 -// \ / -// bb3 -// BufferDeallocation expected behavior: It only inserts a missing DeallocOp -// in the exit block since %5 or %6 are the latest aliases of %0. - -// CHECK-LABEL: func @ifElseNoUsers -func.func @ifElseNoUsers(%arg0: i1, %arg1: memref<2xf32>, %arg2: memref<2xf32>) { - %0 = memref.alloc() : memref<2xf32> - test.buffer_based in(%arg1: memref<2xf32>) out(%0: memref<2xf32>) - cf.cond_br %arg0, - ^bb1(%arg1, %0 : memref<2xf32>, memref<2xf32>), - ^bb2(%0, %arg1 : memref<2xf32>, memref<2xf32>) -^bb1(%1: memref<2xf32>, %2: memref<2xf32>): - cf.br ^bb3(%1, %2 : memref<2xf32>, memref<2xf32>) -^bb2(%3: memref<2xf32>, %4: memref<2xf32>): - cf.br ^bb3(%3, %4 : memref<2xf32>, memref<2xf32>) -^bb3(%5: memref<2xf32>, %6: memref<2xf32>): - test.copy(%arg1, %arg2) : (memref<2xf32>, memref<2xf32>) - return -} - -// CHECK-NEXT: %[[FIRST_ALLOC:.*]] = memref.alloc() -// CHECK: test.copy -// CHECK-NEXT: memref.dealloc %[[FIRST_ALLOC]] -// CHECK-NEXT: return - -// ----- - -// Test Case: -// bb0 <- Initial position of the first AllocOp -// / \ -// bb1 bb2 -// | / \ -// | bb3 bb4 -// \ \ / -// \ / -// bb5 <- Initial position of the second AllocOp -// BufferDeallocation expected behavior: Two missing DeallocOps should be -// inserted in the exit block. - -// CHECK-LABEL: func @ifElseNested -func.func @ifElseNested(%arg0: i1, %arg1: memref<2xf32>, %arg2: memref<2xf32>) { - %0 = memref.alloc() : memref<2xf32> - test.buffer_based in(%arg1: memref<2xf32>) out(%0: memref<2xf32>) - cf.cond_br %arg0, - ^bb1(%arg1, %0 : memref<2xf32>, memref<2xf32>), - ^bb2(%0, %arg1 : memref<2xf32>, memref<2xf32>) -^bb1(%1: memref<2xf32>, %2: memref<2xf32>): - cf.br ^bb5(%1, %2 : memref<2xf32>, memref<2xf32>) -^bb2(%3: memref<2xf32>, %4: memref<2xf32>): - cf.cond_br %arg0, ^bb3(%3 : memref<2xf32>), ^bb4(%4 : memref<2xf32>) -^bb3(%5: memref<2xf32>): - cf.br ^bb5(%5, %3 : memref<2xf32>, memref<2xf32>) -^bb4(%6: memref<2xf32>): - cf.br ^bb5(%3, %6 : memref<2xf32>, memref<2xf32>) -^bb5(%7: memref<2xf32>, %8: memref<2xf32>): - %9 = memref.alloc() : memref<2xf32> - test.buffer_based in(%7: memref<2xf32>) out(%9: memref<2xf32>) - test.copy(%9, %arg2) : (memref<2xf32>, memref<2xf32>) - return -} - -// CHECK-NEXT: %[[FIRST_ALLOC:.*]] = memref.alloc() -// CHECK-NEXT: test.buffer_based -// CHECK: %[[SECOND_ALLOC:.*]] = memref.alloc() -// CHECK-NEXT: test.buffer_based -// CHECK: memref.dealloc %[[FIRST_ALLOC]] -// CHECK: test.copy -// CHECK-NEXT: memref.dealloc %[[SECOND_ALLOC]] -// CHECK-NEXT: return - -// ----- - -// Test Case: Dead operations in a single block. -// BufferDeallocation expected behavior: It only inserts the two missing -// DeallocOps after the last BufferBasedOp. - -// CHECK-LABEL: func @redundantOperations -func.func @redundantOperations(%arg0: memref<2xf32>) { - %0 = memref.alloc() : memref<2xf32> - test.buffer_based in(%arg0: memref<2xf32>) out(%0: memref<2xf32>) - %1 = memref.alloc() : memref<2xf32> - test.buffer_based in(%0: memref<2xf32>) out(%1: memref<2xf32>) - return -} - -// CHECK: (%[[ARG0:.*]]: {{.*}}) -// CHECK-NEXT: %[[FIRST_ALLOC:.*]] = memref.alloc() -// CHECK-NEXT: test.buffer_based in(%[[ARG0]]{{.*}}out(%[[FIRST_ALLOC]] -// CHECK: %[[SECOND_ALLOC:.*]] = memref.alloc() -// CHECK-NEXT: test.buffer_based in(%[[FIRST_ALLOC]]{{.*}}out(%[[SECOND_ALLOC]] -// CHECK: dealloc -// CHECK-NEXT: dealloc -// CHECK-NEXT: return - -// ----- - -// Test Case: -// bb0 -// / \ -// Initial pos of the 1st AllocOp -> bb1 bb2 <- Initial pos of the 2nd AllocOp -// \ / -// bb3 -// BufferDeallocation expected behavior: We need to introduce a copy for each -// buffer since the buffers are passed to bb3. The both missing DeallocOps are -// inserted in the respective block of the allocs. The copy is freed in the exit -// block. - -// CHECK-LABEL: func @moving_alloc_and_inserting_missing_dealloc -func.func @moving_alloc_and_inserting_missing_dealloc( - %cond: i1, - %arg0: memref<2xf32>, - %arg1: memref<2xf32>) { - cf.cond_br %cond, ^bb1, ^bb2 -^bb1: - %0 = memref.alloc() : memref<2xf32> - test.buffer_based in(%arg0: memref<2xf32>) out(%0: memref<2xf32>) - cf.br ^exit(%0 : memref<2xf32>) -^bb2: - %1 = memref.alloc() : memref<2xf32> - test.buffer_based in(%arg0: memref<2xf32>) out(%1: memref<2xf32>) - cf.br ^exit(%1 : memref<2xf32>) -^exit(%arg2: memref<2xf32>): - test.copy(%arg2, %arg1) : (memref<2xf32>, memref<2xf32>) - return -} - -// CHECK-NEXT: cf.cond_br{{.*}} -// CHECK-NEXT: ^bb1 -// CHECK: %[[ALLOC0:.*]] = memref.alloc() -// CHECK-NEXT: test.buffer_based -// CHECK-NEXT: %[[ALLOC1:.*]] = bufferization.clone %[[ALLOC0]] -// CHECK-NEXT: memref.dealloc %[[ALLOC0]] -// CHECK-NEXT: cf.br ^bb3(%[[ALLOC1]] -// CHECK-NEXT: ^bb2 -// CHECK-NEXT: %[[ALLOC2:.*]] = memref.alloc() -// CHECK-NEXT: test.buffer_based -// CHECK-NEXT: %[[ALLOC3:.*]] = bufferization.clone %[[ALLOC2]] -// CHECK-NEXT: memref.dealloc %[[ALLOC2]] -// CHECK-NEXT: cf.br ^bb3(%[[ALLOC3]] -// CHECK-NEXT: ^bb3(%[[ALLOC4:.*]]:{{.*}}) -// CHECK: test.copy -// CHECK-NEXT: memref.dealloc %[[ALLOC4]] -// CHECK-NEXT: return - -// ----- - -// Test Case: Invalid position of the DeallocOp. There is a user after -// deallocation. -// bb0 -// / \ -// bb1 bb2 <- Initial position of AllocOp -// \ / -// bb3 -// BufferDeallocation expected behavior: The existing DeallocOp should be -// moved to exit block. - -// CHECK-LABEL: func @moving_invalid_dealloc_op_complex -func.func @moving_invalid_dealloc_op_complex( - %cond: i1, - %arg0: memref<2xf32>, - %arg1: memref<2xf32>) { - %1 = memref.alloc() : memref<2xf32> - cf.cond_br %cond, ^bb1, ^bb2 -^bb1: - cf.br ^exit(%arg0 : memref<2xf32>) -^bb2: - test.buffer_based in(%arg0: memref<2xf32>) out(%1: memref<2xf32>) - memref.dealloc %1 : memref<2xf32> - cf.br ^exit(%1 : memref<2xf32>) -^exit(%arg2: memref<2xf32>): - test.copy(%arg2, %arg1) : (memref<2xf32>, memref<2xf32>) - return -} - -// CHECK-NEXT: %[[ALLOC0:.*]] = memref.alloc() -// CHECK-NEXT: cf.cond_br -// CHECK: test.copy -// CHECK-NEXT: memref.dealloc %[[ALLOC0]] -// CHECK-NEXT: return - -// ----- - -// Test Case: Inserting missing DeallocOp in a single block. - -// CHECK-LABEL: func @inserting_missing_dealloc_simple -func.func @inserting_missing_dealloc_simple( - %arg0 : memref<2xf32>, - %arg1: memref<2xf32>) { - %0 = memref.alloc() : memref<2xf32> - test.buffer_based in(%arg0: memref<2xf32>) out(%0: memref<2xf32>) - test.copy(%0, %arg1) : (memref<2xf32>, memref<2xf32>) - return -} - -// CHECK-NEXT: %[[ALLOC0:.*]] = memref.alloc() -// CHECK: test.copy -// CHECK-NEXT: memref.dealloc %[[ALLOC0]] - -// ----- - -// Test Case: Moving invalid DeallocOp (there is a user after deallocation) in a -// single block. - -// CHECK-LABEL: func @moving_invalid_dealloc_op -func.func @moving_invalid_dealloc_op(%arg0 : memref<2xf32>, %arg1: memref<2xf32>) { - %0 = memref.alloc() : memref<2xf32> - test.buffer_based in(%arg0: memref<2xf32>) out(%0: memref<2xf32>) - memref.dealloc %0 : memref<2xf32> - test.copy(%0, %arg1) : (memref<2xf32>, memref<2xf32>) - return -} - -// CHECK-NEXT: %[[ALLOC0:.*]] = memref.alloc() -// CHECK: test.copy -// CHECK-NEXT: memref.dealloc %[[ALLOC0]] - -// ----- - -// Test Case: Nested regions - This test defines a BufferBasedOp inside the -// region of a RegionBufferBasedOp. -// BufferDeallocation expected behavior: The AllocOp for the BufferBasedOp -// should remain inside the region of the RegionBufferBasedOp and it should insert -// the missing DeallocOp in the same region. The missing DeallocOp should be -// inserted after CopyOp. - -// CHECK-LABEL: func @nested_regions_and_cond_branch -func.func @nested_regions_and_cond_branch( - %arg0: i1, - %arg1: memref<2xf32>, - %arg2: memref<2xf32>) { - cf.cond_br %arg0, ^bb1, ^bb2 -^bb1: - cf.br ^bb3(%arg1 : memref<2xf32>) -^bb2: - %0 = memref.alloc() : memref<2xf32> - test.region_buffer_based in(%arg1: memref<2xf32>) out(%0: memref<2xf32>) { - ^bb0(%gen1_arg0: f32, %gen1_arg1: f32): - %1 = memref.alloc() : memref<2xf32> - test.buffer_based in(%arg1: memref<2xf32>) out(%1: memref<2xf32>) - %tmp1 = math.exp %gen1_arg0 : f32 - test.region_yield %tmp1 : f32 - } - cf.br ^bb3(%0 : memref<2xf32>) -^bb3(%1: memref<2xf32>): - test.copy(%1, %arg2) : (memref<2xf32>, memref<2xf32>) - return -} -// CHECK: (%[[cond:.*]]: {{.*}}, %[[ARG1:.*]]: {{.*}}, %{{.*}}: {{.*}}) -// CHECK-NEXT: cf.cond_br %[[cond]], ^[[BB1:.*]], ^[[BB2:.*]] -// CHECK: %[[ALLOC0:.*]] = bufferization.clone %[[ARG1]] -// CHECK: ^[[BB2]]: -// CHECK: %[[ALLOC1:.*]] = memref.alloc() -// CHECK-NEXT: test.region_buffer_based in(%[[ARG1]]{{.*}}out(%[[ALLOC1]] -// CHECK: %[[ALLOC2:.*]] = memref.alloc() -// CHECK-NEXT: test.buffer_based in(%[[ARG1]]{{.*}}out(%[[ALLOC2]] -// CHECK: memref.dealloc %[[ALLOC2]] -// CHECK-NEXT: %{{.*}} = math.exp -// CHECK: %[[ALLOC3:.*]] = bufferization.clone %[[ALLOC1]] -// CHECK-NEXT: memref.dealloc %[[ALLOC1]] -// CHECK: ^[[BB3:.*]]({{.*}}): -// CHECK: test.copy -// CHECK-NEXT: memref.dealloc - -// ----- - -// Test Case: buffer deallocation escaping -// BufferDeallocation expected behavior: It must not dealloc %arg1 and %x -// since they are operands of return operation and should escape from -// deallocating. It should dealloc %y after CopyOp. - -// CHECK-LABEL: func @memref_in_function_results -func.func @memref_in_function_results( - %arg0: memref<5xf32>, - %arg1: memref<10xf32>, - %arg2: memref<5xf32>) -> (memref<10xf32>, memref<15xf32>) { - %x = memref.alloc() : memref<15xf32> - %y = memref.alloc() : memref<5xf32> - test.buffer_based in(%arg0: memref<5xf32>) out(%y: memref<5xf32>) - test.copy(%y, %arg2) : (memref<5xf32>, memref<5xf32>) - return %arg1, %x : memref<10xf32>, memref<15xf32> -} -// CHECK: (%[[ARG0:.*]]: memref<5xf32>, %[[ARG1:.*]]: memref<10xf32>, -// CHECK-SAME: %[[RESULT:.*]]: memref<5xf32>) -// CHECK: %[[X:.*]] = memref.alloc() -// CHECK: %[[Y:.*]] = memref.alloc() -// CHECK: test.copy -// CHECK: memref.dealloc %[[Y]] -// CHECK: return %[[ARG1]], %[[X]] - -// ----- - -// Test Case: nested region control flow -// The alloc %1 flows through both if branches until it is finally returned. -// Hence, it does not require a specific dealloc operation. However, %3 -// requires a dealloc. - -// CHECK-LABEL: func @nested_region_control_flow -func.func @nested_region_control_flow( - %arg0 : index, - %arg1 : index) -> memref { - %0 = arith.cmpi eq, %arg0, %arg1 : index - %1 = memref.alloc(%arg0, %arg0) : memref - %2 = scf.if %0 -> (memref) { - scf.yield %1 : memref - } else { - %3 = memref.alloc(%arg0, %arg1) : memref - scf.yield %1 : memref - } - return %2 : memref -} - -// CHECK: %[[ALLOC0:.*]] = memref.alloc(%arg0, %arg0) -// CHECK-NEXT: %[[ALLOC1:.*]] = scf.if -// CHECK: scf.yield %[[ALLOC0]] -// CHECK: %[[ALLOC2:.*]] = memref.alloc(%arg0, %arg1) -// CHECK-NEXT: memref.dealloc %[[ALLOC2]] -// CHECK-NEXT: scf.yield %[[ALLOC0]] -// CHECK: return %[[ALLOC1]] - -// ----- - -// Test Case: nested region control flow with a nested buffer allocation in a -// divergent branch. -// Buffer deallocation places a copy for both %1 and %3, since they are -// returned in the end. - -// CHECK-LABEL: func @nested_region_control_flow_div -func.func @nested_region_control_flow_div( - %arg0 : index, - %arg1 : index) -> memref { - %0 = arith.cmpi eq, %arg0, %arg1 : index - %1 = memref.alloc(%arg0, %arg0) : memref - %2 = scf.if %0 -> (memref) { - scf.yield %1 : memref - } else { - %3 = memref.alloc(%arg0, %arg1) : memref - scf.yield %3 : memref - } - return %2 : memref -} - -// CHECK: %[[ALLOC0:.*]] = memref.alloc(%arg0, %arg0) -// CHECK-NEXT: %[[ALLOC1:.*]] = scf.if -// CHECK-NEXT: %[[ALLOC2:.*]] = bufferization.clone %[[ALLOC0]] -// CHECK: scf.yield %[[ALLOC2]] -// CHECK: %[[ALLOC3:.*]] = memref.alloc(%arg0, %arg1) -// CHECK-NEXT: %[[ALLOC4:.*]] = bufferization.clone %[[ALLOC3]] -// CHECK: memref.dealloc %[[ALLOC3]] -// CHECK: scf.yield %[[ALLOC4]] -// CHECK: memref.dealloc %[[ALLOC0]] -// CHECK-NEXT: return %[[ALLOC1]] - -// ----- - -// Test Case: nested region control flow within a region interface. -// No copies are required in this case since the allocation finally escapes -// the method. - -// CHECK-LABEL: func @inner_region_control_flow -func.func @inner_region_control_flow(%arg0 : index) -> memref { - %0 = memref.alloc(%arg0, %arg0) : memref - %1 = test.region_if %0 : memref -> (memref) then { - ^bb0(%arg1 : memref): - test.region_if_yield %arg1 : memref - } else { - ^bb0(%arg1 : memref): - test.region_if_yield %arg1 : memref - } join { - ^bb0(%arg1 : memref): - test.region_if_yield %arg1 : memref - } - return %1 : memref -} - -// CHECK: %[[ALLOC0:.*]] = memref.alloc(%arg0, %arg0) -// CHECK-NEXT: %[[ALLOC1:.*]] = test.region_if -// CHECK-NEXT: ^bb0(%[[ALLOC2:.*]]:{{.*}}): -// CHECK-NEXT: test.region_if_yield %[[ALLOC2]] -// CHECK: ^bb0(%[[ALLOC3:.*]]:{{.*}}): -// CHECK-NEXT: test.region_if_yield %[[ALLOC3]] -// CHECK: ^bb0(%[[ALLOC4:.*]]:{{.*}}): -// CHECK-NEXT: test.region_if_yield %[[ALLOC4]] -// CHECK: return %[[ALLOC1]] - -// ----- - -// CHECK-LABEL: func @subview -func.func @subview(%arg0 : index, %arg1 : index, %arg2 : memref) { - %0 = memref.alloc() : memref<64x4xf32, strided<[4, 1], offset: 0>> - %1 = memref.subview %0[%arg0, %arg1][%arg0, %arg1][%arg0, %arg1] : - memref<64x4xf32, strided<[4, 1], offset: 0>> - to memref> - test.copy(%1, %arg2) : - (memref>, memref) - return -} - -// CHECK-NEXT: %[[ALLOC:.*]] = memref.alloc() -// CHECK-NEXT: memref.subview -// CHECK-NEXT: test.copy -// CHECK-NEXT: memref.dealloc %[[ALLOC]] -// CHECK-NEXT: return - -// ----- - -// Test Case: In the presence of AllocaOps only the AllocOps has top be freed. -// Therefore, all allocas are not handled. - -// CHECK-LABEL: func @condBranchAlloca -func.func @condBranchAlloca(%arg0: i1, %arg1: memref<2xf32>, %arg2: memref<2xf32>) { - cf.cond_br %arg0, ^bb1, ^bb2 -^bb1: - cf.br ^bb3(%arg1 : memref<2xf32>) -^bb2: - %0 = memref.alloca() : memref<2xf32> - test.buffer_based in(%arg1: memref<2xf32>) out(%0: memref<2xf32>) - cf.br ^bb3(%0 : memref<2xf32>) -^bb3(%1: memref<2xf32>): - test.copy(%1, %arg2) : (memref<2xf32>, memref<2xf32>) - return -} - -// CHECK-NEXT: cf.cond_br -// CHECK: %[[ALLOCA:.*]] = memref.alloca() -// CHECK: cf.br ^bb3(%[[ALLOCA:.*]]) -// CHECK-NEXT: ^bb3 -// CHECK-NEXT: test.copy -// CHECK-NEXT: return - -// ----- - -// Test Case: In the presence of AllocaOps only the AllocOps has top be freed. -// Therefore, all allocas are not handled. In this case, only alloc %0 has a -// dealloc. - -// CHECK-LABEL: func @ifElseAlloca -func.func @ifElseAlloca(%arg0: i1, %arg1: memref<2xf32>, %arg2: memref<2xf32>) { - %0 = memref.alloc() : memref<2xf32> - test.buffer_based in(%arg1: memref<2xf32>) out(%0: memref<2xf32>) - cf.cond_br %arg0, - ^bb1(%arg1, %0 : memref<2xf32>, memref<2xf32>), - ^bb2(%0, %arg1 : memref<2xf32>, memref<2xf32>) -^bb1(%1: memref<2xf32>, %2: memref<2xf32>): - cf.br ^bb3(%1, %2 : memref<2xf32>, memref<2xf32>) -^bb2(%3: memref<2xf32>, %4: memref<2xf32>): - cf.br ^bb3(%3, %4 : memref<2xf32>, memref<2xf32>) -^bb3(%5: memref<2xf32>, %6: memref<2xf32>): - %7 = memref.alloca() : memref<2xf32> - test.buffer_based in(%5: memref<2xf32>) out(%7: memref<2xf32>) - test.copy(%7, %arg2) : (memref<2xf32>, memref<2xf32>) - return -} - -// CHECK-NEXT: %[[ALLOC:.*]] = memref.alloc() -// CHECK-NEXT: test.buffer_based -// CHECK: %[[ALLOCA:.*]] = memref.alloca() -// CHECK-NEXT: test.buffer_based -// CHECK: memref.dealloc %[[ALLOC]] -// CHECK: test.copy -// CHECK-NEXT: return - -// ----- - -// CHECK-LABEL: func @ifElseNestedAlloca -func.func @ifElseNestedAlloca( - %arg0: i1, - %arg1: memref<2xf32>, - %arg2: memref<2xf32>) { - %0 = memref.alloca() : memref<2xf32> - test.buffer_based in(%arg1: memref<2xf32>) out(%0: memref<2xf32>) - cf.cond_br %arg0, - ^bb1(%arg1, %0 : memref<2xf32>, memref<2xf32>), - ^bb2(%0, %arg1 : memref<2xf32>, memref<2xf32>) -^bb1(%1: memref<2xf32>, %2: memref<2xf32>): - cf.br ^bb5(%1, %2 : memref<2xf32>, memref<2xf32>) -^bb2(%3: memref<2xf32>, %4: memref<2xf32>): - cf.cond_br %arg0, ^bb3(%3 : memref<2xf32>), ^bb4(%4 : memref<2xf32>) -^bb3(%5: memref<2xf32>): - cf.br ^bb5(%5, %3 : memref<2xf32>, memref<2xf32>) -^bb4(%6: memref<2xf32>): - cf.br ^bb5(%3, %6 : memref<2xf32>, memref<2xf32>) -^bb5(%7: memref<2xf32>, %8: memref<2xf32>): - %9 = memref.alloc() : memref<2xf32> - test.buffer_based in(%7: memref<2xf32>) out(%9: memref<2xf32>) - test.copy(%9, %arg2) : (memref<2xf32>, memref<2xf32>) - return -} - -// CHECK-NEXT: %[[ALLOCA:.*]] = memref.alloca() -// CHECK-NEXT: test.buffer_based -// CHECK: %[[ALLOC:.*]] = memref.alloc() -// CHECK-NEXT: test.buffer_based -// CHECK: test.copy -// CHECK-NEXT: memref.dealloc %[[ALLOC]] -// CHECK-NEXT: return - -// ----- - -// CHECK-LABEL: func @nestedRegionsAndCondBranchAlloca -func.func @nestedRegionsAndCondBranchAlloca( - %arg0: i1, - %arg1: memref<2xf32>, - %arg2: memref<2xf32>) { - cf.cond_br %arg0, ^bb1, ^bb2 -^bb1: - cf.br ^bb3(%arg1 : memref<2xf32>) -^bb2: - %0 = memref.alloc() : memref<2xf32> - test.region_buffer_based in(%arg1: memref<2xf32>) out(%0: memref<2xf32>) { - ^bb0(%gen1_arg0: f32, %gen1_arg1: f32): - %1 = memref.alloca() : memref<2xf32> - test.buffer_based in(%arg1: memref<2xf32>) out(%1: memref<2xf32>) - %tmp1 = math.exp %gen1_arg0 : f32 - test.region_yield %tmp1 : f32 - } - cf.br ^bb3(%0 : memref<2xf32>) -^bb3(%1: memref<2xf32>): - test.copy(%1, %arg2) : (memref<2xf32>, memref<2xf32>) - return -} -// CHECK: (%[[cond:.*]]: {{.*}}, %[[ARG1:.*]]: {{.*}}, %{{.*}}: {{.*}}) -// CHECK-NEXT: cf.cond_br %[[cond]], ^[[BB1:.*]], ^[[BB2:.*]] -// CHECK: ^[[BB1]]: -// CHECK: %[[ALLOC0:.*]] = bufferization.clone -// CHECK: ^[[BB2]]: -// CHECK: %[[ALLOC1:.*]] = memref.alloc() -// CHECK-NEXT: test.region_buffer_based in(%[[ARG1]]{{.*}}out(%[[ALLOC1]] -// CHECK: %[[ALLOCA:.*]] = memref.alloca() -// CHECK-NEXT: test.buffer_based in(%[[ARG1]]{{.*}}out(%[[ALLOCA]] -// CHECK: %{{.*}} = math.exp -// CHECK: %[[ALLOC2:.*]] = bufferization.clone %[[ALLOC1]] -// CHECK-NEXT: memref.dealloc %[[ALLOC1]] -// CHECK: ^[[BB3:.*]]({{.*}}): -// CHECK: test.copy -// CHECK-NEXT: memref.dealloc - -// ----- - -// CHECK-LABEL: func @nestedRegionControlFlowAlloca -func.func @nestedRegionControlFlowAlloca( - %arg0 : index, - %arg1 : index) -> memref { - %0 = arith.cmpi eq, %arg0, %arg1 : index - %1 = memref.alloc(%arg0, %arg0) : memref - %2 = scf.if %0 -> (memref) { - scf.yield %1 : memref - } else { - %3 = memref.alloca(%arg0, %arg1) : memref - scf.yield %1 : memref - } - return %2 : memref -} - -// CHECK: %[[ALLOC0:.*]] = memref.alloc(%arg0, %arg0) -// CHECK-NEXT: %[[ALLOC1:.*]] = scf.if -// CHECK: scf.yield %[[ALLOC0]] -// CHECK: %[[ALLOCA:.*]] = memref.alloca(%arg0, %arg1) -// CHECK-NEXT: scf.yield %[[ALLOC0]] -// CHECK: return %[[ALLOC1]] - -// ----- - -// Test Case: structured control-flow loop using a nested alloc. -// The iteration argument %iterBuf has to be freed before yielding %3 to avoid -// memory leaks. - -// CHECK-LABEL: func @loop_alloc -func.func @loop_alloc( - %lb: index, - %ub: index, - %step: index, - %buf: memref<2xf32>, - %res: memref<2xf32>) { - %0 = memref.alloc() : memref<2xf32> - %1 = scf.for %i = %lb to %ub step %step - iter_args(%iterBuf = %buf) -> memref<2xf32> { - %2 = arith.cmpi eq, %i, %ub : index - %3 = memref.alloc() : memref<2xf32> - scf.yield %3 : memref<2xf32> - } - test.copy(%1, %res) : (memref<2xf32>, memref<2xf32>) - return -} - -// CHECK: %[[ALLOC0:.*]] = memref.alloc() -// CHECK-NEXT: memref.dealloc %[[ALLOC0]] -// CHECK-NEXT: %[[ALLOC1:.*]] = bufferization.clone %arg3 -// CHECK: %[[ALLOC2:.*]] = scf.for {{.*}} iter_args -// CHECK-SAME: (%[[IALLOC:.*]] = %[[ALLOC1]] -// CHECK: arith.cmpi -// CHECK: memref.dealloc %[[IALLOC]] -// CHECK: %[[ALLOC3:.*]] = memref.alloc() -// CHECK: %[[ALLOC4:.*]] = bufferization.clone %[[ALLOC3]] -// CHECK: memref.dealloc %[[ALLOC3]] -// CHECK: scf.yield %[[ALLOC4]] -// CHECK: } -// CHECK: test.copy(%[[ALLOC2]], %arg4) -// CHECK-NEXT: memref.dealloc %[[ALLOC2]] - -// ----- - -// Test Case: structured control-flow loop with a nested if operation. -// The loop yields buffers that have been defined outside of the loop and the -// backedges only use the iteration arguments (or one of its aliases). -// Therefore, we do not have to (and are not allowed to) free any buffers -// that are passed via the backedges. - -// CHECK-LABEL: func @loop_nested_if_no_alloc -func.func @loop_nested_if_no_alloc( - %lb: index, - %ub: index, - %step: index, - %buf: memref<2xf32>, - %res: memref<2xf32>) { - %0 = memref.alloc() : memref<2xf32> - %1 = scf.for %i = %lb to %ub step %step - iter_args(%iterBuf = %buf) -> memref<2xf32> { - %2 = arith.cmpi eq, %i, %ub : index - %3 = scf.if %2 -> (memref<2xf32>) { - scf.yield %0 : memref<2xf32> - } else { - scf.yield %iterBuf : memref<2xf32> - } - scf.yield %3 : memref<2xf32> - } - test.copy(%1, %res) : (memref<2xf32>, memref<2xf32>) - return -} - -// CHECK: %[[ALLOC0:.*]] = memref.alloc() -// CHECK-NEXT: %[[ALLOC1:.*]] = scf.for {{.*}} iter_args(%[[IALLOC:.*]] = -// CHECK: %[[ALLOC2:.*]] = scf.if -// CHECK: scf.yield %[[ALLOC0]] -// CHECK: scf.yield %[[IALLOC]] -// CHECK: scf.yield %[[ALLOC2]] -// CHECK: test.copy(%[[ALLOC1]], %arg4) -// CHECK: memref.dealloc %[[ALLOC0]] - -// ----- - -// Test Case: structured control-flow loop with a nested if operation using -// a deeply nested buffer allocation. -// Since the innermost allocation happens in a divergent branch, we have to -// introduce additional copies for the nested if operation. Since the loop's -// yield operation "returns" %3, it will return a newly allocated buffer. -// Therefore, we have to free the iteration argument %iterBuf before -// "returning" %3. - -// CHECK-LABEL: func @loop_nested_if_alloc -func.func @loop_nested_if_alloc( - %lb: index, - %ub: index, - %step: index, - %buf: memref<2xf32>) -> memref<2xf32> { - %0 = memref.alloc() : memref<2xf32> - %1 = scf.for %i = %lb to %ub step %step - iter_args(%iterBuf = %buf) -> memref<2xf32> { - %2 = arith.cmpi eq, %i, %ub : index - %3 = scf.if %2 -> (memref<2xf32>) { - %4 = memref.alloc() : memref<2xf32> - scf.yield %4 : memref<2xf32> - } else { - scf.yield %0 : memref<2xf32> - } - scf.yield %3 : memref<2xf32> - } - return %1 : memref<2xf32> -} - -// CHECK: %[[ALLOC0:.*]] = memref.alloc() -// CHECK-NEXT: %[[ALLOC1:.*]] = bufferization.clone %arg3 -// CHECK-NEXT: %[[ALLOC2:.*]] = scf.for {{.*}} iter_args -// CHECK-SAME: (%[[IALLOC:.*]] = %[[ALLOC1]] -// CHECK: memref.dealloc %[[IALLOC]] -// CHECK: %[[ALLOC3:.*]] = scf.if - -// CHECK: %[[ALLOC4:.*]] = memref.alloc() -// CHECK-NEXT: %[[ALLOC5:.*]] = bufferization.clone %[[ALLOC4]] -// CHECK-NEXT: memref.dealloc %[[ALLOC4]] -// CHECK-NEXT: scf.yield %[[ALLOC5]] - -// CHECK: %[[ALLOC6:.*]] = bufferization.clone %[[ALLOC0]] -// CHECK-NEXT: scf.yield %[[ALLOC6]] - -// CHECK: %[[ALLOC7:.*]] = bufferization.clone %[[ALLOC3]] -// CHECK-NEXT: memref.dealloc %[[ALLOC3]] -// CHECK-NEXT: scf.yield %[[ALLOC7]] - -// CHECK: memref.dealloc %[[ALLOC0]] -// CHECK-NEXT: return %[[ALLOC2]] - -// ----- - -// Test Case: several nested structured control-flow loops with a deeply nested -// buffer allocation inside an if operation. -// Same behavior is an loop_nested_if_alloc: we have to insert deallocations -// before each yield in all loops recursively. - -// CHECK-LABEL: func @loop_nested_alloc -func.func @loop_nested_alloc( - %lb: index, - %ub: index, - %step: index, - %buf: memref<2xf32>, - %res: memref<2xf32>) { - %0 = memref.alloc() : memref<2xf32> - %1 = scf.for %i = %lb to %ub step %step - iter_args(%iterBuf = %buf) -> memref<2xf32> { - %2 = scf.for %i2 = %lb to %ub step %step - iter_args(%iterBuf2 = %iterBuf) -> memref<2xf32> { - %3 = scf.for %i3 = %lb to %ub step %step - iter_args(%iterBuf3 = %iterBuf2) -> memref<2xf32> { - %4 = memref.alloc() : memref<2xf32> - %5 = arith.cmpi eq, %i, %ub : index - %6 = scf.if %5 -> (memref<2xf32>) { - %7 = memref.alloc() : memref<2xf32> - scf.yield %7 : memref<2xf32> - } else { - scf.yield %iterBuf3 : memref<2xf32> - } - scf.yield %6 : memref<2xf32> - } - scf.yield %3 : memref<2xf32> - } - scf.yield %2 : memref<2xf32> - } - test.copy(%1, %res) : (memref<2xf32>, memref<2xf32>) - return -} - -// CHECK: %[[ALLOC0:.*]] = memref.alloc() -// CHECK-NEXT: memref.dealloc %[[ALLOC0]] -// CHECK-NEXT: %[[ALLOC1:.*]] = bufferization.clone %arg3 -// CHECK-NEXT: %[[VAL_7:.*]] = scf.for {{.*}} iter_args -// CHECK-SAME: (%[[IALLOC0:.*]] = %[[ALLOC1]]) -// CHECK-NEXT: %[[ALLOC2:.*]] = bufferization.clone %[[IALLOC0]] -// CHECK-NEXT: memref.dealloc %[[IALLOC0]] -// CHECK-NEXT: %[[ALLOC3:.*]] = scf.for {{.*}} iter_args -// CHECK-SAME: (%[[IALLOC1:.*]] = %[[ALLOC2]]) -// CHECK-NEXT: %[[ALLOC5:.*]] = bufferization.clone %[[IALLOC1]] -// CHECK-NEXT: memref.dealloc %[[IALLOC1]] - -// CHECK: %[[ALLOC6:.*]] = scf.for {{.*}} iter_args -// CHECK-SAME: (%[[IALLOC2:.*]] = %[[ALLOC5]]) -// CHECK: %[[ALLOC8:.*]] = memref.alloc() -// CHECK-NEXT: memref.dealloc %[[ALLOC8]] -// CHECK: %[[ALLOC9:.*]] = scf.if - -// CHECK: %[[ALLOC11:.*]] = memref.alloc() -// CHECK-NEXT: %[[ALLOC12:.*]] = bufferization.clone %[[ALLOC11]] -// CHECK-NEXT: memref.dealloc %[[ALLOC11]] -// CHECK-NEXT: scf.yield %[[ALLOC12]] - -// CHECK: %[[ALLOC13:.*]] = bufferization.clone %[[IALLOC2]] -// CHECK-NEXT: scf.yield %[[ALLOC13]] - -// CHECK: memref.dealloc %[[IALLOC2]] -// CHECK-NEXT: %[[ALLOC10:.*]] = bufferization.clone %[[ALLOC9]] -// CHECK-NEXT: memref.dealloc %[[ALLOC9]] -// CHECK-NEXT: scf.yield %[[ALLOC10]] - -// CHECK: %[[ALLOC7:.*]] = bufferization.clone %[[ALLOC6]] -// CHECK-NEXT: memref.dealloc %[[ALLOC6]] -// CHECK-NEXT: scf.yield %[[ALLOC7]] - -// CHECK: %[[ALLOC4:.*]] = bufferization.clone %[[ALLOC3]] -// CHECK-NEXT: memref.dealloc %[[ALLOC3]] -// CHECK-NEXT: scf.yield %[[ALLOC4]] - -// CHECK: test.copy(%[[VAL_7]], %arg4) -// CHECK-NEXT: memref.dealloc %[[VAL_7]] - -// ----- - -// CHECK-LABEL: func @affine_loop -func.func @affine_loop() { - %buffer = memref.alloc() : memref<1024xf32> - %sum_init_0 = arith.constant 0.0 : f32 - %res = affine.for %i = 0 to 10 step 2 iter_args(%sum_iter = %sum_init_0) -> f32 { - %t = affine.load %buffer[%i] : memref<1024xf32> - %sum_next = arith.addf %sum_iter, %t : f32 - affine.yield %sum_next : f32 - } - // CHECK: %[[M:.*]] = memref.alloc - // CHECK: affine.for - // CHECK: } - // CHECK-NEXT: memref.dealloc %[[M]] - return -} - -// ----- - -// Test Case: explicit control-flow loop with a dynamically allocated buffer. -// The BufferDeallocation transformation should fail on this explicit -// control-flow loop since they are not supported. - -// expected-error@+1 {{Only structured control-flow loops are supported}} -func.func @loop_dynalloc( - %arg0 : i32, - %arg1 : i32, - %arg2: memref, - %arg3: memref) { - %const0 = arith.constant 0 : i32 - cf.br ^loopHeader(%const0, %arg2 : i32, memref) - -^loopHeader(%i : i32, %buff : memref): - %lessThan = arith.cmpi slt, %i, %arg1 : i32 - cf.cond_br %lessThan, - ^loopBody(%i, %buff : i32, memref), - ^exit(%buff : memref) - -^loopBody(%val : i32, %buff2: memref): - %const1 = arith.constant 1 : i32 - %inc = arith.addi %val, %const1 : i32 - %size = arith.index_cast %inc : i32 to index - %alloc1 = memref.alloc(%size) : memref - cf.br ^loopHeader(%inc, %alloc1 : i32, memref) - -^exit(%buff3 : memref): - test.copy(%buff3, %arg3) : (memref, memref) - return -} - -// ----- - -// Test Case: explicit control-flow loop with a dynamically allocated buffer. -// The BufferDeallocation transformation should fail on this explicit -// control-flow loop since they are not supported. - -// expected-error@+1 {{Only structured control-flow loops are supported}} -func.func @do_loop_alloc( - %arg0 : i32, - %arg1 : i32, - %arg2: memref<2xf32>, - %arg3: memref<2xf32>) { - %const0 = arith.constant 0 : i32 - cf.br ^loopBody(%const0, %arg2 : i32, memref<2xf32>) - -^loopBody(%val : i32, %buff2: memref<2xf32>): - %const1 = arith.constant 1 : i32 - %inc = arith.addi %val, %const1 : i32 - %alloc1 = memref.alloc() : memref<2xf32> - cf.br ^loopHeader(%inc, %alloc1 : i32, memref<2xf32>) - -^loopHeader(%i : i32, %buff : memref<2xf32>): - %lessThan = arith.cmpi slt, %i, %arg1 : i32 - cf.cond_br %lessThan, - ^loopBody(%i, %buff : i32, memref<2xf32>), - ^exit(%buff : memref<2xf32>) - -^exit(%buff3 : memref<2xf32>): - test.copy(%buff3, %arg3) : (memref<2xf32>, memref<2xf32>) - return -} - -// ----- - -// CHECK-LABEL: func @assumingOp( -func.func @assumingOp( - %arg0: !shape.witness, - %arg2: memref<2xf32>, - %arg3: memref<2xf32>) { - // Confirm the alloc will be dealloc'ed in the block. - %1 = shape.assuming %arg0 -> memref<2xf32> { - %0 = memref.alloc() : memref<2xf32> - shape.assuming_yield %arg2 : memref<2xf32> - } - // Confirm the alloc will be returned and dealloc'ed after its use. - %3 = shape.assuming %arg0 -> memref<2xf32> { - %2 = memref.alloc() : memref<2xf32> - shape.assuming_yield %2 : memref<2xf32> - } - test.copy(%3, %arg3) : (memref<2xf32>, memref<2xf32>) - return -} - -// CHECK-SAME: %[[ARG0:.*]]: !shape.witness, -// CHECK-SAME: %[[ARG1:.*]]: {{.*}}, -// CHECK-SAME: %[[ARG2:.*]]: {{.*}} -// CHECK: %[[UNUSED_RESULT:.*]] = shape.assuming %[[ARG0]] -// CHECK-NEXT: %[[ALLOC0:.*]] = memref.alloc() -// CHECK-NEXT: memref.dealloc %[[ALLOC0]] -// CHECK-NEXT: shape.assuming_yield %[[ARG1]] -// CHECK: %[[ASSUMING_RESULT:.*]] = shape.assuming %[[ARG0]] -// CHECK-NEXT: %[[TMP_ALLOC:.*]] = memref.alloc() -// CHECK-NEXT: %[[RETURNING_ALLOC:.*]] = bufferization.clone %[[TMP_ALLOC]] -// CHECK-NEXT: memref.dealloc %[[TMP_ALLOC]] -// CHECK-NEXT: shape.assuming_yield %[[RETURNING_ALLOC]] -// CHECK: test.copy(%[[ASSUMING_RESULT:.*]], %[[ARG2]]) -// CHECK-NEXT: memref.dealloc %[[ASSUMING_RESULT]] - -// ----- - -// Test Case: The op "test.bar" does not implement the RegionBranchOpInterface. -// This is not allowed in buffer deallocation. - -func.func @noRegionBranchOpInterface() { -// expected-error@+1 {{All operations with attached regions need to implement the RegionBranchOpInterface.}} - %0 = "test.bar"() ({ -// expected-error@+1 {{All operations with attached regions need to implement the RegionBranchOpInterface.}} - %1 = "test.bar"() ({ - "test.yield"() : () -> () - }) : () -> (i32) - "test.yield"() : () -> () - }) : () -> (i32) - "test.terminator"() : () -> () -} - -// ----- - -// CHECK-LABEL: func @dealloc_existing_clones -// CHECK: (%[[ARG0:.*]]: memref, %[[ARG1:.*]]: memref) -// CHECK: %[[RES0:.*]] = bufferization.clone %[[ARG0]] -// CHECK: %[[RES1:.*]] = bufferization.clone %[[ARG1]] -// CHECK-NOT: memref.dealloc %[[RES0]] -// CHECK: memref.dealloc %[[RES1]] -// CHECK: return %[[RES0]] -func.func @dealloc_existing_clones(%arg0: memref, %arg1: memref) -> memref { - %0 = bufferization.clone %arg0 : memref to memref - %1 = bufferization.clone %arg1 : memref to memref - return %0 : memref -} - -// ----- - -// CHECK-LABEL: func @while_two_arg -func.func @while_two_arg(%arg0: index) { - %a = memref.alloc(%arg0) : memref -// CHECK: %[[WHILE:.*]]:2 = scf.while (%[[ARG1:.*]] = %[[ALLOC:.*]], %[[ARG2:.*]] = %[[CLONE:.*]]) - scf.while (%arg1 = %a, %arg2 = %a) : (memref, memref) -> (memref, memref) { -// CHECK-NEXT: make_condition - %0 = "test.make_condition"() : () -> i1 -// CHECK-NEXT: bufferization.clone %[[ARG2]] -// CHECK-NEXT: memref.dealloc %[[ARG2]] - scf.condition(%0) %arg1, %arg2 : memref, memref - } do { - ^bb0(%arg1: memref, %arg2: memref): -// CHECK: %[[ALLOC2:.*]] = memref.alloc - %b = memref.alloc(%arg0) : memref -// CHECK: memref.dealloc %[[ARG2]] -// CHECK: %[[CLONE2:.*]] = bufferization.clone %[[ALLOC2]] -// CHECK: memref.dealloc %[[ALLOC2]] - scf.yield %arg1, %b : memref, memref - } -// CHECK: } -// CHECK-NEXT: memref.dealloc %[[WHILE]]#1 -// CHECK-NEXT: memref.dealloc %[[ALLOC]] -// CHECK-NEXT: return - return -} - -// ----- - -func.func @while_three_arg(%arg0: index) { -// CHECK: %[[ALLOC:.*]] = memref.alloc - %a = memref.alloc(%arg0) : memref -// CHECK-NEXT: %[[CLONE1:.*]] = bufferization.clone %[[ALLOC]] -// CHECK-NEXT: %[[CLONE2:.*]] = bufferization.clone %[[ALLOC]] -// CHECK-NEXT: %[[CLONE3:.*]] = bufferization.clone %[[ALLOC]] -// CHECK-NEXT: memref.dealloc %[[ALLOC]] -// CHECK-NEXT: %[[WHILE:.*]]:3 = scf.while -// FIXME: This is non-deterministic -// CHECK-SAME-DAG: [[CLONE1]] -// CHECK-SAME-DAG: [[CLONE2]] -// CHECK-SAME-DAG: [[CLONE3]] - scf.while (%arg1 = %a, %arg2 = %a, %arg3 = %a) : (memref, memref, memref) -> (memref, memref, memref) { - %0 = "test.make_condition"() : () -> i1 - scf.condition(%0) %arg1, %arg2, %arg3 : memref, memref, memref - } do { - ^bb0(%arg1: memref, %arg2: memref, %arg3: memref): - %b = memref.alloc(%arg0) : memref - %q = memref.alloc(%arg0) : memref - scf.yield %q, %b, %arg2: memref, memref, memref - } -// CHECK-DAG: memref.dealloc %[[WHILE]]#0 -// CHECK-DAG: memref.dealloc %[[WHILE]]#1 -// CHECK-DAG: memref.dealloc %[[WHILE]]#2 -// CHECK-NEXT: return - return -} - -// ----- - -func.func @select_aliases(%arg0: index, %arg1: memref, %arg2: i1) { - // CHECK: memref.alloc - // CHECK: memref.alloc - // CHECK: arith.select - // CHECK: test.copy - // CHECK: memref.dealloc - // CHECK: memref.dealloc - %0 = memref.alloc(%arg0) : memref - %1 = memref.alloc(%arg0) : memref - %2 = arith.select %arg2, %0, %1 : memref - test.copy(%2, %arg1) : (memref, memref) - return -} - -// ----- - -func.func @f(%arg0: memref) -> memref { - return %arg0 : memref -} - -// CHECK-LABEL: func @function_call -// CHECK: memref.alloc -// CHECK: memref.alloc -// CHECK: call -// CHECK: test.copy -// CHECK: memref.dealloc -// CHECK: memref.dealloc -func.func @function_call() { - %alloc = memref.alloc() : memref - %alloc2 = memref.alloc() : memref - %ret = call @f(%alloc) : (memref) -> memref - test.copy(%ret, %alloc2) : (memref, memref) - return -} - -// ----- - -// Memref allocated in `then` region and passed back to the parent if op. -#set = affine_set<() : (0 >= 0)> -// CHECK-LABEL: func @test_affine_if_1 -// CHECK-SAME: %[[ARG0:.*]]: memref<10xf32>) -> memref<10xf32> { -func.func @test_affine_if_1(%arg0: memref<10xf32>) -> memref<10xf32> { - %0 = affine.if #set() -> memref<10xf32> { - %alloc = memref.alloc() : memref<10xf32> - affine.yield %alloc : memref<10xf32> - } else { - affine.yield %arg0 : memref<10xf32> - } - return %0 : memref<10xf32> -} -// CHECK-NEXT: %[[IF:.*]] = affine.if -// CHECK-NEXT: %[[MEMREF:.*]] = memref.alloc() : memref<10xf32> -// CHECK-NEXT: %[[CLONED:.*]] = bufferization.clone %[[MEMREF]] : memref<10xf32> to memref<10xf32> -// CHECK-NEXT: memref.dealloc %[[MEMREF]] : memref<10xf32> -// CHECK-NEXT: affine.yield %[[CLONED]] : memref<10xf32> -// CHECK-NEXT: } else { -// CHECK-NEXT: %[[ARG0_CLONE:.*]] = bufferization.clone %[[ARG0]] : memref<10xf32> to memref<10xf32> -// CHECK-NEXT: affine.yield %[[ARG0_CLONE]] : memref<10xf32> -// CHECK-NEXT: } -// CHECK-NEXT: return %[[IF]] : memref<10xf32> - -// ----- - -// Memref allocated before parent IfOp and used in `then` region. -// Expected result: deallocation should happen after affine.if op. -#set = affine_set<() : (0 >= 0)> -// CHECK-LABEL: func @test_affine_if_2() -> memref<10xf32> { -func.func @test_affine_if_2() -> memref<10xf32> { - %alloc0 = memref.alloc() : memref<10xf32> - %0 = affine.if #set() -> memref<10xf32> { - affine.yield %alloc0 : memref<10xf32> - } else { - %alloc = memref.alloc() : memref<10xf32> - affine.yield %alloc : memref<10xf32> - } - return %0 : memref<10xf32> -} -// CHECK-NEXT: %[[ALLOC:.*]] = memref.alloc() : memref<10xf32> -// CHECK-NEXT: %[[IF_RES:.*]] = affine.if {{.*}} -> memref<10xf32> { -// CHECK-NEXT: %[[ALLOC_CLONE:.*]] = bufferization.clone %[[ALLOC]] : memref<10xf32> to memref<10xf32> -// CHECK-NEXT: affine.yield %[[ALLOC_CLONE]] : memref<10xf32> -// CHECK-NEXT: } else { -// CHECK-NEXT: %[[ALLOC2:.*]] = memref.alloc() : memref<10xf32> -// CHECK-NEXT: %[[ALLOC2_CLONE:.*]] = bufferization.clone %[[ALLOC2]] : memref<10xf32> to memref<10xf32> -// CHECK-NEXT: memref.dealloc %[[ALLOC2]] : memref<10xf32> -// CHECK-NEXT: affine.yield %[[ALLOC2_CLONE]] : memref<10xf32> -// CHECK-NEXT: } -// CHECK-NEXT: memref.dealloc %[[ALLOC]] : memref<10xf32> -// CHECK-NEXT: return %[[IF_RES]] : memref<10xf32> - -// ----- - -// Memref allocated before parent IfOp and used in `else` region. -// Expected result: deallocation should happen after affine.if op. -#set = affine_set<() : (0 >= 0)> -// CHECK-LABEL: func @test_affine_if_3() -> memref<10xf32> { -func.func @test_affine_if_3() -> memref<10xf32> { - %alloc0 = memref.alloc() : memref<10xf32> - %0 = affine.if #set() -> memref<10xf32> { - %alloc = memref.alloc() : memref<10xf32> - affine.yield %alloc : memref<10xf32> - } else { - affine.yield %alloc0 : memref<10xf32> - } - return %0 : memref<10xf32> -} -// CHECK-NEXT: %[[ALLOC:.*]] = memref.alloc() : memref<10xf32> -// CHECK-NEXT: %[[IFRES:.*]] = affine.if {{.*}} -> memref<10xf32> { -// CHECK-NEXT: memref.alloc -// CHECK-NEXT: bufferization.clone -// CHECK-NEXT: memref.dealloc -// CHECK-NEXT: affine.yield -// CHECK-NEXT: } else { -// CHECK-NEXT: bufferization.clone -// CHECK-NEXT: affine.yield -// CHECK-NEXT: } -// CHECK-NEXT: memref.dealloc %[[ALLOC]] : memref<10xf32> -// CHECK-NEXT: return %[[IFRES]] : memref<10xf32> - -// ----- - -// Memref allocated before parent IfOp and not used later. -// Expected result: deallocation should happen before affine.if op. -#set = affine_set<() : (0 >= 0)> -// CHECK-LABEL: func @test_affine_if_4({{.*}}: memref<10xf32>) -> memref<10xf32> { -func.func @test_affine_if_4(%arg0 : memref<10xf32>) -> memref<10xf32> { - %alloc0 = memref.alloc() : memref<10xf32> - %0 = affine.if #set() -> memref<10xf32> { - affine.yield %arg0 : memref<10xf32> - } else { - %alloc = memref.alloc() : memref<10xf32> - affine.yield %alloc : memref<10xf32> - } - return %0 : memref<10xf32> -} -// CHECK-NEXT: %[[ALLOC:.*]] = memref.alloc() : memref<10xf32> -// CHECK-NEXT: memref.dealloc %[[ALLOC]] : memref<10xf32> -// CHECK-NEXT: affine.if - -// ----- - -// Ensure we free the realloc, not the alloc. - -// CHECK-LABEL: func @auto_dealloc() -func.func @auto_dealloc() { - %c10 = arith.constant 10 : index - %c100 = arith.constant 100 : index - %alloc = memref.alloc(%c10) : memref - %realloc = memref.realloc %alloc(%c100) : memref to memref - return -} -// CHECK-DAG: %[[C10:.*]] = arith.constant 10 : index -// CHECK-DAG: %[[C100:.*]] = arith.constant 100 : index -// CHECK-NEXT: %[[A:.*]] = memref.alloc(%[[C10]]) : memref -// CHECK-NEXT: %[[R:.*]] = memref.realloc %alloc(%[[C100]]) : memref to memref -// CHECK-NEXT: memref.dealloc %[[R]] : memref -// CHECK-NEXT: return - - diff --git a/utils/bazel/llvm-project-overlay/mlir/BUILD.bazel b/utils/bazel/llvm-project-overlay/mlir/BUILD.bazel --- a/utils/bazel/llvm-project-overlay/mlir/BUILD.bazel +++ b/utils/bazel/llvm-project-overlay/mlir/BUILD.bazel @@ -11959,6 +11959,7 @@ ":BufferizationDialect", ":BufferizationEnumsIncGen", ":BufferizationPassIncGen", + ":ControlFlowDialect", ":ControlFlowInterfaces", ":FuncDialect", ":IR", @@ -11966,6 +11967,7 @@ ":MemRefDialect", ":MemRefUtils", ":Pass", + ":SCFDialect", ":SideEffectInterfaces", ":TensorDialect", ":Transforms",