160f443bbSAlex Zinenko //===- ParallelLoopTiling.cpp - Tiles scf.parallel ---------------===// 2c25b20c0SAlex Zinenko // 3c25b20c0SAlex Zinenko // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 4c25b20c0SAlex Zinenko // See https://llvm.org/LICENSE.txt for license information. 5c25b20c0SAlex Zinenko // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 6c25b20c0SAlex Zinenko // 7c25b20c0SAlex Zinenko //===----------------------------------------------------------------------===// 8c25b20c0SAlex Zinenko // 9c25b20c0SAlex Zinenko // This file implements loop tiling on parallel loops. 10c25b20c0SAlex Zinenko // 11c25b20c0SAlex Zinenko //===----------------------------------------------------------------------===// 12c25b20c0SAlex Zinenko 13c25b20c0SAlex Zinenko #include "PassDetail.h" 14c25b20c0SAlex Zinenko #include "mlir/Dialect/Affine/IR/AffineOps.h" 15c25b20c0SAlex Zinenko #include "mlir/Dialect/SCF/Passes.h" 16c25b20c0SAlex Zinenko #include "mlir/Dialect/SCF/SCF.h" 17c25b20c0SAlex Zinenko #include "mlir/Dialect/SCF/Transforms.h" 18c25b20c0SAlex Zinenko #include "mlir/Dialect/StandardOps/IR/Ops.h" 19c25b20c0SAlex Zinenko #include "mlir/Transforms/RegionUtils.h" 20c25b20c0SAlex Zinenko #include "llvm/Support/CommandLine.h" 21c25b20c0SAlex Zinenko 22c25b20c0SAlex Zinenko using namespace mlir; 23c25b20c0SAlex Zinenko using namespace mlir::scf; 24c25b20c0SAlex Zinenko 25c25b20c0SAlex Zinenko /// Tile a parallel loop of the form 2660f443bbSAlex Zinenko /// scf.parallel (%i0, %i1) = (%arg0, %arg1) to (%arg2, %arg3) 27c25b20c0SAlex Zinenko /// step (%arg4, %arg5) 28c25b20c0SAlex Zinenko /// 29c25b20c0SAlex Zinenko /// into 3060f443bbSAlex Zinenko /// scf.parallel (%i0, %i1) = (%arg0, %arg1) to (%arg2, %arg3) 31c25b20c0SAlex Zinenko /// step (%arg4*tileSize[0], 32c25b20c0SAlex Zinenko /// %arg5*tileSize[1]) 331e60678cSStephan Herhut /// scf.parallel (%j0, %j1) = (0, 0) to (min(tileSize[0], %arg2-%i0) 341e60678cSStephan Herhut /// min(tileSize[1], %arg3-%i1)) 35c25b20c0SAlex Zinenko /// step (%arg4, %arg5) 361e60678cSStephan Herhut /// 371e60678cSStephan Herhut /// where the uses of %i0 and %i1 in the loop body are replaced by 381e60678cSStephan Herhut /// %i0 + j0 and %i1 + %j1. 391e60678cSStephan Herhut // 40c25b20c0SAlex Zinenko /// The old loop is replaced with the new one. 41c25b20c0SAlex Zinenko void mlir::scf::tileParallelLoop(ParallelOp op, ArrayRef<int64_t> tileSizes) { 42c25b20c0SAlex Zinenko OpBuilder b(op); 43c25b20c0SAlex Zinenko auto zero = b.create<ConstantIndexOp>(op.getLoc(), 0); 44c25b20c0SAlex Zinenko SmallVector<Value, 2> tileSizeConstants; 45c25b20c0SAlex Zinenko tileSizeConstants.reserve(op.upperBound().size()); 46c25b20c0SAlex Zinenko for (size_t i = 0, end = op.upperBound().size(); i != end; ++i) { 47c25b20c0SAlex Zinenko if (i < tileSizes.size()) 48c25b20c0SAlex Zinenko tileSizeConstants.push_back( 49c25b20c0SAlex Zinenko b.create<ConstantIndexOp>(op.getLoc(), tileSizes[i])); 50c25b20c0SAlex Zinenko else 51c25b20c0SAlex Zinenko // Just pick 1 for the remaining dimensions. 52c25b20c0SAlex Zinenko tileSizeConstants.push_back(b.create<ConstantIndexOp>(op.getLoc(), 1)); 53c25b20c0SAlex Zinenko } 54c25b20c0SAlex Zinenko 55c25b20c0SAlex Zinenko // Create the outer loop with adjusted steps. 56c25b20c0SAlex Zinenko SmallVector<Value, 2> newSteps; 57c25b20c0SAlex Zinenko newSteps.reserve(op.step().size()); 58c25b20c0SAlex Zinenko for (auto step : llvm::zip(op.step(), tileSizeConstants)) { 59c25b20c0SAlex Zinenko newSteps.push_back( 60c25b20c0SAlex Zinenko b.create<MulIOp>(op.getLoc(), std::get<0>(step), std::get<1>(step))); 61c25b20c0SAlex Zinenko } 62c25b20c0SAlex Zinenko auto outerLoop = b.create<ParallelOp>(op.getLoc(), op.lowerBound(), 63c25b20c0SAlex Zinenko op.upperBound(), newSteps); 64c25b20c0SAlex Zinenko b.setInsertionPointToStart(outerLoop.getBody()); 65c25b20c0SAlex Zinenko 66c25b20c0SAlex Zinenko // Compute min(size, dim - offset) to avoid out-of-bounds accesses. 67c25b20c0SAlex Zinenko // FIXME: Instead of using min, we want to replicate the tail. This would give 68c25b20c0SAlex Zinenko // the inner loop constant bounds for easy vectorization. 69c25b20c0SAlex Zinenko auto minMap = AffineMap::get( 70c25b20c0SAlex Zinenko /*dimCount=*/3, /*symbolCount=*/0, 71c25b20c0SAlex Zinenko {getAffineDimExpr(/*position=*/0, b.getContext()), 72c25b20c0SAlex Zinenko getAffineDimExpr(/*position=*/1, b.getContext()) - 73c25b20c0SAlex Zinenko getAffineDimExpr(/*position=*/2, b.getContext())}, 74c25b20c0SAlex Zinenko b.getContext()); 75c25b20c0SAlex Zinenko 76c25b20c0SAlex Zinenko // Create the inner loop with adjusted bounds. 77c25b20c0SAlex Zinenko SmallVector<Value, 2> newBounds; 78c25b20c0SAlex Zinenko newBounds.reserve(op.upperBound().size()); 79c25b20c0SAlex Zinenko for (auto bounds : llvm::zip(tileSizeConstants, outerLoop.upperBound(), 80c25b20c0SAlex Zinenko outerLoop.getInductionVars())) { 81c25b20c0SAlex Zinenko newBounds.push_back(b.create<AffineMinOp>( 82c25b20c0SAlex Zinenko op.getLoc(), b.getIndexType(), minMap, 83c25b20c0SAlex Zinenko ValueRange{std::get<0>(bounds), std::get<1>(bounds), 84c25b20c0SAlex Zinenko std::get<2>(bounds)})); 85c25b20c0SAlex Zinenko } 86c25b20c0SAlex Zinenko auto innerLoop = b.create<ParallelOp>( 87c25b20c0SAlex Zinenko op.getLoc(), SmallVector<Value, 2>(newBounds.size(), zero), newBounds, 88c25b20c0SAlex Zinenko op.step()); 89c25b20c0SAlex Zinenko 90c25b20c0SAlex Zinenko // Steal the body of the old parallel loop and erase it. 91c25b20c0SAlex Zinenko innerLoop.region().takeBody(op.region()); 921e60678cSStephan Herhut 931e60678cSStephan Herhut // Insert computation for new index vectors and replace uses. 941e60678cSStephan Herhut b.setInsertionPointToStart(innerLoop.getBody()); 951e60678cSStephan Herhut for (auto ivs : 961e60678cSStephan Herhut llvm::zip(innerLoop.getInductionVars(), outerLoop.getInductionVars())) { 971e60678cSStephan Herhut Value inner_index = std::get<0>(ivs); 981e60678cSStephan Herhut AddIOp newIndex = 991e60678cSStephan Herhut b.create<AddIOp>(op.getLoc(), std::get<0>(ivs), std::get<1>(ivs)); 1001e60678cSStephan Herhut inner_index.replaceAllUsesExcept( 1011e60678cSStephan Herhut newIndex, SmallPtrSet<Operation *, 1>{newIndex.getOperation()}); 1021e60678cSStephan Herhut } 1031e60678cSStephan Herhut 104c25b20c0SAlex Zinenko op.erase(); 105c25b20c0SAlex Zinenko } 106c25b20c0SAlex Zinenko 107c25b20c0SAlex Zinenko /// Get a list of most nested parallel loops. Assumes that ParallelOps are only 108c25b20c0SAlex Zinenko /// directly nested. 109c25b20c0SAlex Zinenko static bool getInnermostNestedLoops(Block *block, 110c25b20c0SAlex Zinenko SmallVectorImpl<ParallelOp> &loops) { 111c25b20c0SAlex Zinenko bool hasInnerLoop = false; 112c25b20c0SAlex Zinenko for (auto parallelOp : block->getOps<ParallelOp>()) { 113c25b20c0SAlex Zinenko hasInnerLoop = true; 114c25b20c0SAlex Zinenko if (!getInnermostNestedLoops(parallelOp.getBody(), loops)) 115c25b20c0SAlex Zinenko loops.push_back(parallelOp); 116c25b20c0SAlex Zinenko } 117c25b20c0SAlex Zinenko return hasInnerLoop; 118c25b20c0SAlex Zinenko } 119c25b20c0SAlex Zinenko 120c25b20c0SAlex Zinenko namespace { 121c25b20c0SAlex Zinenko struct ParallelLoopTiling 122*4bcd08ebSStephan Herhut : public SCFParallelLoopTilingBase<ParallelLoopTiling> { 123c25b20c0SAlex Zinenko ParallelLoopTiling() = default; 124c25b20c0SAlex Zinenko explicit ParallelLoopTiling(ArrayRef<int64_t> tileSizes) { 125c25b20c0SAlex Zinenko this->tileSizes = tileSizes; 126c25b20c0SAlex Zinenko } 127c25b20c0SAlex Zinenko 128c25b20c0SAlex Zinenko void runOnFunction() override { 129c25b20c0SAlex Zinenko SmallVector<ParallelOp, 2> mostNestedParallelOps; 130c25b20c0SAlex Zinenko for (Block &block : getFunction()) { 131c25b20c0SAlex Zinenko getInnermostNestedLoops(&block, mostNestedParallelOps); 132c25b20c0SAlex Zinenko } 133c25b20c0SAlex Zinenko for (ParallelOp pLoop : mostNestedParallelOps) { 134c25b20c0SAlex Zinenko tileParallelLoop(pLoop, tileSizes); 135c25b20c0SAlex Zinenko } 136c25b20c0SAlex Zinenko } 137c25b20c0SAlex Zinenko }; 138c25b20c0SAlex Zinenko } // namespace 139c25b20c0SAlex Zinenko 140c25b20c0SAlex Zinenko std::unique_ptr<Pass> 141c25b20c0SAlex Zinenko mlir::createParallelLoopTilingPass(ArrayRef<int64_t> tileSizes) { 142c25b20c0SAlex Zinenko return std::make_unique<ParallelLoopTiling>(tileSizes); 143c25b20c0SAlex Zinenko } 144