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