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