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