MLIR  20.0.0git
LoopPipelining.cpp
Go to the documentation of this file.
1 //===- LoopPipelining.cpp - Code to perform loop software pipelining-------===//
2 //
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6 //
7 //===----------------------------------------------------------------------===//
8 //
9 // This file implements loop software pipelining
10 //
11 //===----------------------------------------------------------------------===//
12 
18 #include "mlir/IR/IRMapping.h"
19 #include "mlir/IR/PatternMatch.h"
21 #include "llvm/ADT/MapVector.h"
22 #include "llvm/Support/Debug.h"
23 #include "llvm/Support/MathExtras.h"
24 
25 #define DEBUG_TYPE "scf-loop-pipelining"
26 #define DBGS() (llvm::dbgs() << "[" DEBUG_TYPE "]: ")
27 #define LDBG(X) LLVM_DEBUG(DBGS() << X << "\n")
28 
29 using namespace mlir;
30 using namespace mlir::scf;
31 
32 namespace {
33 
34 /// Helper to keep internal information during pipelining transformation.
35 struct LoopPipelinerInternal {
36  /// Coarse liverange information for ops used across stages.
37  struct LiverangeInfo {
38  unsigned lastUseStage = 0;
39  unsigned defStage = 0;
40  };
41 
42 protected:
43  ForOp forOp;
44  unsigned maxStage = 0;
46  std::vector<Operation *> opOrder;
47  Value ub;
48  Value lb;
49  Value step;
50  bool dynamicLoop;
51  PipeliningOption::AnnotationlFnType annotateFn = nullptr;
52  bool peelEpilogue;
53  PipeliningOption::PredicateOpFn predicateFn = nullptr;
54 
55  // When peeling the kernel we generate several version of each value for
56  // different stage of the prologue. This map tracks the mapping between
57  // original Values in the loop and the different versions
58  // peeled from the loop.
60 
61  /// Assign a value to `valueMapping`, this means `val` represents the version
62  /// `idx` of `key` in the epilogue.
63  void setValueMapping(Value key, Value el, int64_t idx);
64 
65  /// Return the defining op of the given value, if the Value is an argument of
66  /// the loop return the associated defining op in the loop and its distance to
67  /// the Value.
68  std::pair<Operation *, int64_t> getDefiningOpAndDistance(Value value);
69 
70  /// Return true if the schedule is possible and return false otherwise. A
71  /// schedule is correct if all definitions are scheduled before uses.
72  bool verifySchedule();
73 
74 public:
75  /// Initalize the information for the given `op`, return true if it
76  /// satisfies the pre-condition to apply pipelining.
77  bool initializeLoopInfo(ForOp op, const PipeliningOption &options);
78  /// Emits the prologue, this creates `maxStage - 1` part which will contain
79  /// operations from stages [0; i], where i is the part index.
80  LogicalResult emitPrologue(RewriterBase &rewriter);
81  /// Gather liverange information for Values that are used in a different stage
82  /// than its definition.
83  llvm::MapVector<Value, LiverangeInfo> analyzeCrossStageValues();
84  scf::ForOp createKernelLoop(
85  const llvm::MapVector<Value, LiverangeInfo> &crossStageValues,
86  RewriterBase &rewriter,
87  llvm::DenseMap<std::pair<Value, unsigned>, unsigned> &loopArgMap);
88  /// Emits the pipelined kernel. This clones loop operations following user
89  /// order and remaps operands defined in a different stage as their use.
90  LogicalResult createKernel(
91  scf::ForOp newForOp,
92  const llvm::MapVector<Value, LiverangeInfo> &crossStageValues,
93  const llvm::DenseMap<std::pair<Value, unsigned>, unsigned> &loopArgMap,
94  RewriterBase &rewriter);
95  /// Emits the epilogue, this creates `maxStage - 1` part which will contain
96  /// operations from stages [i; maxStage], where i is the part index.
97  LogicalResult emitEpilogue(RewriterBase &rewriter,
98  llvm::SmallVector<Value> &returnValues);
99 };
100 
101 bool LoopPipelinerInternal::initializeLoopInfo(
102  ForOp op, const PipeliningOption &options) {
103  LDBG("Start initializeLoopInfo");
104  forOp = op;
105  ub = forOp.getUpperBound();
106  lb = forOp.getLowerBound();
107  step = forOp.getStep();
108 
109  dynamicLoop = true;
110  auto upperBoundCst = getConstantIntValue(ub);
111  auto lowerBoundCst = getConstantIntValue(lb);
112  auto stepCst = getConstantIntValue(step);
113  if (!upperBoundCst || !lowerBoundCst || !stepCst) {
114  if (!options.supportDynamicLoops) {
115  LDBG("--dynamic loop not supported -> BAIL");
116  return false;
117  }
118  } else {
119  int64_t ubImm = upperBoundCst.value();
120  int64_t lbImm = lowerBoundCst.value();
121  int64_t stepImm = stepCst.value();
122  int64_t numIteration = llvm::divideCeilSigned(ubImm - lbImm, stepImm);
123  if (numIteration > maxStage) {
124  dynamicLoop = false;
125  } else if (!options.supportDynamicLoops) {
126  LDBG("--fewer loop iterations than pipeline stages -> BAIL");
127  return false;
128  }
129  }
130  peelEpilogue = options.peelEpilogue;
131  predicateFn = options.predicateFn;
132  if ((!peelEpilogue || dynamicLoop) && predicateFn == nullptr) {
133  LDBG("--no epilogue or predicate set -> BAIL");
134  return false;
135  }
136  std::vector<std::pair<Operation *, unsigned>> schedule;
137  options.getScheduleFn(forOp, schedule);
138  if (schedule.empty()) {
139  LDBG("--empty schedule -> BAIL");
140  return false;
141  }
142 
143  opOrder.reserve(schedule.size());
144  for (auto &opSchedule : schedule) {
145  maxStage = std::max(maxStage, opSchedule.second);
146  stages[opSchedule.first] = opSchedule.second;
147  opOrder.push_back(opSchedule.first);
148  }
149 
150  // All operations need to have a stage.
151  for (Operation &op : forOp.getBody()->without_terminator()) {
152  if (!stages.contains(&op)) {
153  op.emitOpError("not assigned a pipeline stage");
154  LDBG("--op not assigned a pipeline stage: " << op << " -> BAIL");
155  return false;
156  }
157  }
158 
159  if (!verifySchedule()) {
160  LDBG("--invalid schedule: " << op << " -> BAIL");
161  return false;
162  }
163 
164  // Currently, we do not support assigning stages to ops in nested regions. The
165  // block of all operations assigned a stage should be the single `scf.for`
166  // body block.
167  for (const auto &[op, stageNum] : stages) {
168  (void)stageNum;
169  if (op == forOp.getBody()->getTerminator()) {
170  op->emitError("terminator should not be assigned a stage");
171  LDBG("--terminator should not be assigned stage: " << *op << " -> BAIL");
172  return false;
173  }
174  if (op->getBlock() != forOp.getBody()) {
175  op->emitOpError("the owning Block of all operations assigned a stage "
176  "should be the loop body block");
177  LDBG("--the owning Block of all operations assigned a stage "
178  "should be the loop body block: "
179  << *op << " -> BAIL");
180  return false;
181  }
182  }
183 
184  // Support only loop-carried dependencies with a distance of one iteration or
185  // those defined outside of the loop. This means that any dependency within a
186  // loop should either be on the immediately preceding iteration, the current
187  // iteration, or on variables whose values are set before entering the loop.
188  if (llvm::any_of(forOp.getBody()->getTerminator()->getOperands(),
189  [this](Value operand) {
190  Operation *def = operand.getDefiningOp();
191  return !def ||
192  (!stages.contains(def) && forOp->isAncestor(def));
193  })) {
194  LDBG("--only support loop carried dependency with a distance of 1 or "
195  "defined outside of the loop -> BAIL");
196  return false;
197  }
198  annotateFn = options.annotateFn;
199  return true;
200 }
201 
202 /// Find operands of all the nested operations within `op`.
203 static SetVector<Value> getNestedOperands(Operation *op) {
204  SetVector<Value> operands;
205  op->walk([&](Operation *nestedOp) {
206  for (Value operand : nestedOp->getOperands()) {
207  operands.insert(operand);
208  }
209  });
210  return operands;
211 }
212 
213 /// Compute unrolled cycles of each op (consumer) and verify that each op is
214 /// scheduled after its operands (producers) while adjusting for the distance
215 /// between producer and consumer.
216 bool LoopPipelinerInternal::verifySchedule() {
217  int64_t numCylesPerIter = opOrder.size();
218  // Pre-compute the unrolled cycle of each op.
219  DenseMap<Operation *, int64_t> unrolledCyles;
220  for (int64_t cycle = 0; cycle < numCylesPerIter; cycle++) {
221  Operation *def = opOrder[cycle];
222  auto it = stages.find(def);
223  assert(it != stages.end());
224  int64_t stage = it->second;
225  unrolledCyles[def] = cycle + stage * numCylesPerIter;
226  }
227  for (Operation *consumer : opOrder) {
228  int64_t consumerCycle = unrolledCyles[consumer];
229  for (Value operand : getNestedOperands(consumer)) {
230  auto [producer, distance] = getDefiningOpAndDistance(operand);
231  if (!producer)
232  continue;
233  auto it = unrolledCyles.find(producer);
234  // Skip producer coming from outside the loop.
235  if (it == unrolledCyles.end())
236  continue;
237  int64_t producerCycle = it->second;
238  if (consumerCycle < producerCycle - numCylesPerIter * distance) {
239  consumer->emitError("operation scheduled before its operands");
240  return false;
241  }
242  }
243  }
244  return true;
245 }
246 
247 /// Clone `op` and call `callback` on the cloned op's oeprands as well as any
248 /// operands of nested ops that:
249 /// 1) aren't defined within the new op or
250 /// 2) are block arguments.
251 static Operation *
252 cloneAndUpdateOperands(RewriterBase &rewriter, Operation *op,
253  function_ref<void(OpOperand *newOperand)> callback) {
254  Operation *clone = rewriter.clone(*op);
255  clone->walk<WalkOrder::PreOrder>([&](Operation *nested) {
256  // 'clone' itself will be visited first.
257  for (OpOperand &operand : nested->getOpOperands()) {
258  Operation *def = operand.get().getDefiningOp();
259  if ((def && !clone->isAncestor(def)) || isa<BlockArgument>(operand.get()))
260  callback(&operand);
261  }
262  });
263  return clone;
264 }
265 
266 LogicalResult LoopPipelinerInternal::emitPrologue(RewriterBase &rewriter) {
267  // Initialize the iteration argument to the loop initial values.
268  for (auto [arg, operand] :
269  llvm::zip(forOp.getRegionIterArgs(), forOp.getInitsMutable())) {
270  setValueMapping(arg, operand.get(), 0);
271  }
272  auto yield = cast<scf::YieldOp>(forOp.getBody()->getTerminator());
273  Location loc = forOp.getLoc();
274  SmallVector<Value> predicates(maxStage);
275  for (int64_t i = 0; i < maxStage; i++) {
276  if (dynamicLoop) {
277  Type t = ub.getType();
278  // pred = ub > lb + (i * step)
279  Value iv = rewriter.create<arith::AddIOp>(
280  loc, lb,
281  rewriter.create<arith::MulIOp>(
282  loc, step,
283  rewriter.create<arith::ConstantOp>(
284  loc, rewriter.getIntegerAttr(t, i))));
285  predicates[i] = rewriter.create<arith::CmpIOp>(
286  loc, arith::CmpIPredicate::slt, iv, ub);
287  }
288 
289  // special handling for induction variable as the increment is implicit.
290  // iv = lb + i * step
291  Type t = lb.getType();
292  Value iv = rewriter.create<arith::AddIOp>(
293  loc, lb,
294  rewriter.create<arith::MulIOp>(
295  loc, step,
296  rewriter.create<arith::ConstantOp>(loc,
297  rewriter.getIntegerAttr(t, i))));
298  setValueMapping(forOp.getInductionVar(), iv, i);
299  for (Operation *op : opOrder) {
300  if (stages[op] > i)
301  continue;
302  Operation *newOp =
303  cloneAndUpdateOperands(rewriter, op, [&](OpOperand *newOperand) {
304  auto it = valueMapping.find(newOperand->get());
305  if (it != valueMapping.end()) {
306  Value replacement = it->second[i - stages[op]];
307  newOperand->set(replacement);
308  }
309  });
310  int predicateIdx = i - stages[op];
311  if (predicates[predicateIdx]) {
312  OpBuilder::InsertionGuard insertGuard(rewriter);
313  newOp = predicateFn(rewriter, newOp, predicates[predicateIdx]);
314  if (newOp == nullptr)
315  return failure();
316  }
317  if (annotateFn)
318  annotateFn(newOp, PipeliningOption::PipelinerPart::Prologue, i);
319  for (unsigned destId : llvm::seq(unsigned(0), op->getNumResults())) {
320  Value source = newOp->getResult(destId);
321  // If the value is a loop carried dependency update the loop argument
322  for (OpOperand &operand : yield->getOpOperands()) {
323  if (operand.get() != op->getResult(destId))
324  continue;
325  if (predicates[predicateIdx] &&
326  !forOp.getResult(operand.getOperandNumber()).use_empty()) {
327  // If the value is used outside the loop, we need to make sure we
328  // return the correct version of it.
329  Value prevValue = valueMapping
330  [forOp.getRegionIterArgs()[operand.getOperandNumber()]]
331  [i - stages[op]];
332  source = rewriter.create<arith::SelectOp>(
333  loc, predicates[predicateIdx], source, prevValue);
334  }
335  setValueMapping(forOp.getRegionIterArgs()[operand.getOperandNumber()],
336  source, i - stages[op] + 1);
337  }
338  setValueMapping(op->getResult(destId), newOp->getResult(destId),
339  i - stages[op]);
340  }
341  }
342  }
343  return success();
344 }
345 
346 llvm::MapVector<Value, LoopPipelinerInternal::LiverangeInfo>
347 LoopPipelinerInternal::analyzeCrossStageValues() {
348  llvm::MapVector<Value, LoopPipelinerInternal::LiverangeInfo> crossStageValues;
349  for (Operation *op : opOrder) {
350  unsigned stage = stages[op];
351 
352  auto analyzeOperand = [&](OpOperand &operand) {
353  auto [def, distance] = getDefiningOpAndDistance(operand.get());
354  if (!def)
355  return;
356  auto defStage = stages.find(def);
357  if (defStage == stages.end() || defStage->second == stage ||
358  defStage->second == stage + distance)
359  return;
360  assert(stage > defStage->second);
361  LiverangeInfo &info = crossStageValues[operand.get()];
362  info.defStage = defStage->second;
363  info.lastUseStage = std::max(info.lastUseStage, stage);
364  };
365 
366  for (OpOperand &operand : op->getOpOperands())
367  analyzeOperand(operand);
368  visitUsedValuesDefinedAbove(op->getRegions(), [&](OpOperand *operand) {
369  analyzeOperand(*operand);
370  });
371  }
372  return crossStageValues;
373 }
374 
375 std::pair<Operation *, int64_t>
376 LoopPipelinerInternal::getDefiningOpAndDistance(Value value) {
377  int64_t distance = 0;
378  if (auto arg = dyn_cast<BlockArgument>(value)) {
379  if (arg.getOwner() != forOp.getBody())
380  return {nullptr, 0};
381  // Ignore induction variable.
382  if (arg.getArgNumber() == 0)
383  return {nullptr, 0};
384  distance++;
385  value =
386  forOp.getBody()->getTerminator()->getOperand(arg.getArgNumber() - 1);
387  }
388  Operation *def = value.getDefiningOp();
389  if (!def)
390  return {nullptr, 0};
391  return {def, distance};
392 }
393 
394 scf::ForOp LoopPipelinerInternal::createKernelLoop(
395  const llvm::MapVector<Value, LoopPipelinerInternal::LiverangeInfo>
396  &crossStageValues,
397  RewriterBase &rewriter,
398  llvm::DenseMap<std::pair<Value, unsigned>, unsigned> &loopArgMap) {
399  // Creates the list of initial values associated to values used across
400  // stages. The initial values come from the prologue created above.
401  // Keep track of the kernel argument associated to each version of the
402  // values passed to the kernel.
403  llvm::SmallVector<Value> newLoopArg;
404  // For existing loop argument initialize them with the right version from the
405  // prologue.
406  for (const auto &retVal :
407  llvm::enumerate(forOp.getBody()->getTerminator()->getOperands())) {
408  Operation *def = retVal.value().getDefiningOp();
409  assert(def && "Only support loop carried dependencies of distance of 1 or "
410  "outside the loop");
411  auto defStage = stages.find(def);
412  if (defStage != stages.end()) {
413  Value valueVersion =
414  valueMapping[forOp.getRegionIterArgs()[retVal.index()]]
415  [maxStage - defStage->second];
416  assert(valueVersion);
417  newLoopArg.push_back(valueVersion);
418  } else
419  newLoopArg.push_back(forOp.getInitArgs()[retVal.index()]);
420  }
421  for (auto escape : crossStageValues) {
422  LiverangeInfo &info = escape.second;
423  Value value = escape.first;
424  for (unsigned stageIdx = 0; stageIdx < info.lastUseStage - info.defStage;
425  stageIdx++) {
426  Value valueVersion =
427  valueMapping[value][maxStage - info.lastUseStage + stageIdx];
428  assert(valueVersion);
429  newLoopArg.push_back(valueVersion);
430  loopArgMap[std::make_pair(value, info.lastUseStage - info.defStage -
431  stageIdx)] = newLoopArg.size() - 1;
432  }
433  }
434 
435  // Create the new kernel loop. When we peel the epilgue we need to peel
436  // `numStages - 1` iterations. Then we adjust the upper bound to remove those
437  // iterations.
438  Value newUb = forOp.getUpperBound();
439  if (peelEpilogue) {
440  Type t = ub.getType();
441  Location loc = forOp.getLoc();
442  // newUb = ub - maxStage * step
443  Value maxStageValue = rewriter.create<arith::ConstantOp>(
444  loc, rewriter.getIntegerAttr(t, maxStage));
445  Value maxStageByStep =
446  rewriter.create<arith::MulIOp>(loc, step, maxStageValue);
447  newUb = rewriter.create<arith::SubIOp>(loc, ub, maxStageByStep);
448  }
449  auto newForOp =
450  rewriter.create<scf::ForOp>(forOp.getLoc(), forOp.getLowerBound(), newUb,
451  forOp.getStep(), newLoopArg);
452  // When there are no iter args, the loop body terminator will be created.
453  // Since we always create it below, remove the terminator if it was created.
454  if (!newForOp.getBody()->empty())
455  rewriter.eraseOp(newForOp.getBody()->getTerminator());
456  return newForOp;
457 }
458 
459 LogicalResult LoopPipelinerInternal::createKernel(
460  scf::ForOp newForOp,
461  const llvm::MapVector<Value, LoopPipelinerInternal::LiverangeInfo>
462  &crossStageValues,
463  const llvm::DenseMap<std::pair<Value, unsigned>, unsigned> &loopArgMap,
464  RewriterBase &rewriter) {
465  valueMapping.clear();
466 
467  // Create the kernel, we clone instruction based on the order given by
468  // user and remap operands coming from a previous stages.
469  rewriter.setInsertionPoint(newForOp.getBody(), newForOp.getBody()->begin());
470  IRMapping mapping;
471  mapping.map(forOp.getInductionVar(), newForOp.getInductionVar());
472  for (const auto &arg : llvm::enumerate(forOp.getRegionIterArgs())) {
473  mapping.map(arg.value(), newForOp.getRegionIterArgs()[arg.index()]);
474  }
475  SmallVector<Value> predicates(maxStage + 1, nullptr);
476  if (!peelEpilogue) {
477  // Create a predicate for each stage except the last stage.
478  Location loc = newForOp.getLoc();
479  Type t = ub.getType();
480  for (unsigned i = 0; i < maxStage; i++) {
481  // c = ub - (maxStage - i) * step
482  Value c = rewriter.create<arith::SubIOp>(
483  loc, ub,
484  rewriter.create<arith::MulIOp>(
485  loc, step,
486  rewriter.create<arith::ConstantOp>(
487  loc, rewriter.getIntegerAttr(t, int64_t(maxStage - i)))));
488 
489  Value pred = rewriter.create<arith::CmpIOp>(
490  newForOp.getLoc(), arith::CmpIPredicate::slt,
491  newForOp.getInductionVar(), c);
492  predicates[i] = pred;
493  }
494  }
495  for (Operation *op : opOrder) {
496  int64_t useStage = stages[op];
497  auto *newOp = rewriter.clone(*op, mapping);
498  SmallVector<OpOperand *> operands;
499  // Collect all the operands for the cloned op and its nested ops.
500  op->walk([&operands](Operation *nestedOp) {
501  for (OpOperand &operand : nestedOp->getOpOperands()) {
502  operands.push_back(&operand);
503  }
504  });
505  for (OpOperand *operand : operands) {
506  Operation *nestedNewOp = mapping.lookup(operand->getOwner());
507  // Special case for the induction variable uses. We replace it with a
508  // version incremented based on the stage where it is used.
509  if (operand->get() == forOp.getInductionVar()) {
510  rewriter.setInsertionPoint(newOp);
511 
512  // offset = (maxStage - stages[op]) * step
513  Type t = step.getType();
514  Value offset = rewriter.create<arith::MulIOp>(
515  forOp.getLoc(), step,
516  rewriter.create<arith::ConstantOp>(
517  forOp.getLoc(),
518  rewriter.getIntegerAttr(t, maxStage - stages[op])));
519  Value iv = rewriter.create<arith::AddIOp>(
520  forOp.getLoc(), newForOp.getInductionVar(), offset);
521  nestedNewOp->setOperand(operand->getOperandNumber(), iv);
522  rewriter.setInsertionPointAfter(newOp);
523  continue;
524  }
525  Value source = operand->get();
526  auto arg = dyn_cast<BlockArgument>(source);
527  if (arg && arg.getOwner() == forOp.getBody()) {
528  Value ret = forOp.getBody()->getTerminator()->getOperand(
529  arg.getArgNumber() - 1);
530  Operation *dep = ret.getDefiningOp();
531  if (!dep)
532  continue;
533  auto stageDep = stages.find(dep);
534  if (stageDep == stages.end() || stageDep->second == useStage)
535  continue;
536  // If the value is a loop carried value coming from stage N + 1 remap,
537  // it will become a direct use.
538  if (stageDep->second == useStage + 1) {
539  nestedNewOp->setOperand(operand->getOperandNumber(),
540  mapping.lookupOrDefault(ret));
541  continue;
542  }
543  source = ret;
544  }
545  // For operands defined in a previous stage we need to remap it to use
546  // the correct region argument. We look for the right version of the
547  // Value based on the stage where it is used.
548  Operation *def = source.getDefiningOp();
549  if (!def)
550  continue;
551  auto stageDef = stages.find(def);
552  if (stageDef == stages.end() || stageDef->second == useStage)
553  continue;
554  auto remap = loopArgMap.find(
555  std::make_pair(operand->get(), useStage - stageDef->second));
556  assert(remap != loopArgMap.end());
557  nestedNewOp->setOperand(operand->getOperandNumber(),
558  newForOp.getRegionIterArgs()[remap->second]);
559  }
560 
561  if (predicates[useStage]) {
562  OpBuilder::InsertionGuard insertGuard(rewriter);
563  newOp = predicateFn(rewriter, newOp, predicates[useStage]);
564  if (!newOp)
565  return failure();
566  // Remap the results to the new predicated one.
567  for (auto values : llvm::zip(op->getResults(), newOp->getResults()))
568  mapping.map(std::get<0>(values), std::get<1>(values));
569  }
570  if (annotateFn)
571  annotateFn(newOp, PipeliningOption::PipelinerPart::Kernel, 0);
572  }
573 
574  // Collect the Values that need to be returned by the forOp. For each
575  // value we need to have `LastUseStage - DefStage` number of versions
576  // returned.
577  // We create a mapping between original values and the associated loop
578  // returned values that will be needed by the epilogue.
579  llvm::SmallVector<Value> yieldOperands;
580  for (OpOperand &yieldOperand :
581  forOp.getBody()->getTerminator()->getOpOperands()) {
582  Value source = mapping.lookupOrDefault(yieldOperand.get());
583  // When we don't peel the epilogue and the yield value is used outside the
584  // loop we need to make sure we return the version from numStages -
585  // defStage.
586  if (!peelEpilogue &&
587  !forOp.getResult(yieldOperand.getOperandNumber()).use_empty()) {
588  Operation *def = getDefiningOpAndDistance(yieldOperand.get()).first;
589  if (def) {
590  auto defStage = stages.find(def);
591  if (defStage != stages.end() && defStage->second < maxStage) {
592  Value pred = predicates[defStage->second];
593  source = rewriter.create<arith::SelectOp>(
594  pred.getLoc(), pred, source,
595  newForOp.getBody()
596  ->getArguments()[yieldOperand.getOperandNumber() + 1]);
597  }
598  }
599  }
600  yieldOperands.push_back(source);
601  }
602 
603  for (auto &it : crossStageValues) {
604  int64_t version = maxStage - it.second.lastUseStage + 1;
605  unsigned numVersionReturned = it.second.lastUseStage - it.second.defStage;
606  // add the original version to yield ops.
607  // If there is a live range spanning across more than 2 stages we need to
608  // add extra arg.
609  for (unsigned i = 1; i < numVersionReturned; i++) {
610  setValueMapping(it.first, newForOp->getResult(yieldOperands.size()),
611  version++);
612  yieldOperands.push_back(
613  newForOp.getBody()->getArguments()[yieldOperands.size() + 1 +
614  newForOp.getNumInductionVars()]);
615  }
616  setValueMapping(it.first, newForOp->getResult(yieldOperands.size()),
617  version++);
618  yieldOperands.push_back(mapping.lookupOrDefault(it.first));
619  }
620  // Map the yield operand to the forOp returned value.
621  for (const auto &retVal :
622  llvm::enumerate(forOp.getBody()->getTerminator()->getOperands())) {
623  Operation *def = retVal.value().getDefiningOp();
624  assert(def && "Only support loop carried dependencies of distance of 1 or "
625  "defined outside the loop");
626  auto defStage = stages.find(def);
627  if (defStage == stages.end()) {
628  for (unsigned int stage = 1; stage <= maxStage; stage++)
629  setValueMapping(forOp.getRegionIterArgs()[retVal.index()],
630  retVal.value(), stage);
631  } else if (defStage->second > 0) {
632  setValueMapping(forOp.getRegionIterArgs()[retVal.index()],
633  newForOp->getResult(retVal.index()),
634  maxStage - defStage->second + 1);
635  }
636  }
637  rewriter.create<scf::YieldOp>(forOp.getLoc(), yieldOperands);
638  return success();
639 }
640 
641 LogicalResult
642 LoopPipelinerInternal::emitEpilogue(RewriterBase &rewriter,
643  llvm::SmallVector<Value> &returnValues) {
644  Location loc = forOp.getLoc();
645  // Emit different versions of the induction variable. They will be
646  // removed by dead code if not used.
647 
648  // bounds_range = ub - lb
649  // total_iterations = (bounds_range + step - 1) / step
650  Type t = lb.getType();
651  Value minus1 =
652  rewriter.create<arith::ConstantOp>(loc, rewriter.getIntegerAttr(t, -1));
653  Value boundsRange = rewriter.create<arith::SubIOp>(loc, ub, lb);
654  Value rangeIncr = rewriter.create<arith::AddIOp>(loc, boundsRange, step);
655  Value rangeDecr = rewriter.create<arith::AddIOp>(loc, rangeIncr, minus1);
656  Value totalIterations = rewriter.create<arith::DivUIOp>(loc, rangeDecr, step);
657 
658  SmallVector<Value> predicates(maxStage + 1);
659  for (int64_t i = 0; i < maxStage; i++) {
660  // iterI = total_iters - 1 - i
661  // May go negative...
662  Value minusI =
663  rewriter.create<arith::ConstantOp>(loc, rewriter.getIntegerAttr(t, -i));
664  Value iterI = rewriter.create<arith::AddIOp>(
665  loc, rewriter.create<arith::AddIOp>(loc, totalIterations, minus1),
666  minusI);
667  // newLastIter = lb + step * iterI
668  Value newlastIter = rewriter.create<arith::AddIOp>(
669  loc, lb, rewriter.create<arith::MulIOp>(loc, step, iterI));
670 
671  setValueMapping(forOp.getInductionVar(), newlastIter, maxStage - i);
672 
673  if (dynamicLoop) {
674  // pred = iterI >= lb
675  predicates[i + 1] = rewriter.create<arith::CmpIOp>(
676  loc, arith::CmpIPredicate::sge, iterI, lb);
677  }
678  }
679 
680  // Emit `maxStage - 1` epilogue part that includes operations from stages
681  // [i; maxStage].
682  for (int64_t i = 1; i <= maxStage; i++) {
683  SmallVector<std::pair<Value, unsigned>> returnMap(returnValues.size());
684  for (Operation *op : opOrder) {
685  if (stages[op] < i)
686  continue;
687  unsigned currentVersion = maxStage - stages[op] + i;
688  unsigned nextVersion = currentVersion + 1;
689  Operation *newOp =
690  cloneAndUpdateOperands(rewriter, op, [&](OpOperand *newOperand) {
691  auto it = valueMapping.find(newOperand->get());
692  if (it != valueMapping.end()) {
693  Value replacement = it->second[currentVersion];
694  newOperand->set(replacement);
695  }
696  });
697  if (dynamicLoop) {
698  OpBuilder::InsertionGuard insertGuard(rewriter);
699  newOp = predicateFn(rewriter, newOp, predicates[currentVersion]);
700  if (!newOp)
701  return failure();
702  }
703  if (annotateFn)
704  annotateFn(newOp, PipeliningOption::PipelinerPart::Epilogue, i - 1);
705 
706  for (auto [opRes, newRes] :
707  llvm::zip(op->getResults(), newOp->getResults())) {
708  setValueMapping(opRes, newRes, currentVersion);
709  // If the value is a loop carried dependency update the loop argument
710  // mapping and keep track of the last version to replace the original
711  // forOp uses.
712  for (OpOperand &operand :
713  forOp.getBody()->getTerminator()->getOpOperands()) {
714  if (operand.get() != opRes)
715  continue;
716  // If the version is greater than maxStage it means it maps to the
717  // original forOp returned value.
718  unsigned ri = operand.getOperandNumber();
719  returnValues[ri] = newRes;
720  Value mapVal = forOp.getRegionIterArgs()[ri];
721  returnMap[ri] = std::make_pair(mapVal, currentVersion);
722  if (nextVersion <= maxStage)
723  setValueMapping(mapVal, newRes, nextVersion);
724  }
725  }
726  }
727  if (dynamicLoop) {
728  // Select return values from this stage (live outs) based on predication.
729  // If the stage is valid select the peeled value, else use previous stage
730  // value.
731  for (auto pair : llvm::enumerate(returnValues)) {
732  unsigned ri = pair.index();
733  auto [mapVal, currentVersion] = returnMap[ri];
734  if (mapVal) {
735  unsigned nextVersion = currentVersion + 1;
736  Value pred = predicates[currentVersion];
737  Value prevValue = valueMapping[mapVal][currentVersion];
738  auto selOp = rewriter.create<arith::SelectOp>(loc, pred, pair.value(),
739  prevValue);
740  returnValues[ri] = selOp;
741  if (nextVersion <= maxStage)
742  setValueMapping(mapVal, selOp, nextVersion);
743  }
744  }
745  }
746  }
747  return success();
748 }
749 
750 void LoopPipelinerInternal::setValueMapping(Value key, Value el, int64_t idx) {
751  auto it = valueMapping.find(key);
752  // If the value is not in the map yet add a vector big enough to store all
753  // versions.
754  if (it == valueMapping.end())
755  it =
756  valueMapping
757  .insert(std::make_pair(key, llvm::SmallVector<Value>(maxStage + 1)))
758  .first;
759  it->second[idx] = el;
760 }
761 
762 } // namespace
763 
764 FailureOr<ForOp> mlir::scf::pipelineForLoop(RewriterBase &rewriter, ForOp forOp,
765  const PipeliningOption &options,
766  bool *modifiedIR) {
767  if (modifiedIR)
768  *modifiedIR = false;
769  LoopPipelinerInternal pipeliner;
770  if (!pipeliner.initializeLoopInfo(forOp, options))
771  return failure();
772 
773  if (modifiedIR)
774  *modifiedIR = true;
775 
776  // 1. Emit prologue.
777  if (failed(pipeliner.emitPrologue(rewriter)))
778  return failure();
779 
780  // 2. Track values used across stages. When a value cross stages it will
781  // need to be passed as loop iteration arguments.
782  // We first collect the values that are used in a different stage than where
783  // they are defined.
784  llvm::MapVector<Value, LoopPipelinerInternal::LiverangeInfo>
785  crossStageValues = pipeliner.analyzeCrossStageValues();
786 
787  // Mapping between original loop values used cross stage and the block
788  // arguments associated after pipelining. A Value may map to several
789  // arguments if its liverange spans across more than 2 stages.
790  llvm::DenseMap<std::pair<Value, unsigned>, unsigned> loopArgMap;
791  // 3. Create the new kernel loop and return the block arguments mapping.
792  ForOp newForOp =
793  pipeliner.createKernelLoop(crossStageValues, rewriter, loopArgMap);
794  // Create the kernel block, order ops based on user choice and remap
795  // operands.
796  if (failed(pipeliner.createKernel(newForOp, crossStageValues, loopArgMap,
797  rewriter)))
798  return failure();
799 
800  llvm::SmallVector<Value> returnValues =
801  newForOp.getResults().take_front(forOp->getNumResults());
802  if (options.peelEpilogue) {
803  // 4. Emit the epilogue after the new forOp.
804  rewriter.setInsertionPointAfter(newForOp);
805  if (failed(pipeliner.emitEpilogue(rewriter, returnValues)))
806  return failure();
807  }
808  // 5. Erase the original loop and replace the uses with the epilogue output.
809  if (forOp->getNumResults() > 0)
810  rewriter.replaceOp(forOp, returnValues);
811  else
812  rewriter.eraseOp(forOp);
813 
814  return newForOp;
815 }
816 
818  RewritePatternSet &patterns, const PipeliningOption &options) {
819  patterns.add<ForLoopPipeliningPattern>(options, patterns.getContext());
820 }
#define LDBG(X)
static llvm::ManagedStatic< PassManagerOptions > options
static Value max(ImplicitLocOpBuilder &builder, Value value, Value bound)
IntegerAttr getIntegerAttr(Type type, int64_t value)
Definition: Builders.cpp:250
This is a utility class for mapping one set of IR entities to another.
Definition: IRMapping.h:26
auto lookupOrDefault(T from) const
Lookup a mapped value within the map.
Definition: IRMapping.h:65
auto lookup(T from) const
Lookup a mapped value within the map.
Definition: IRMapping.h:72
void map(Value from, Value to)
Inserts a new mapping for 'from' to 'to'.
Definition: IRMapping.h:30
IRValueT get() const
Return the current value being used by this operand.
Definition: UseDefLists.h:160
This class defines the main interface for locations in MLIR and acts as a non-nullable wrapper around...
Definition: Location.h:63
RAII guard to reset the insertion point of the builder when destroyed.
Definition: Builders.h:353
Operation * clone(Operation &op, IRMapping &mapper)
Creates a deep copy of the specified operation, remapping any operands that use values outside of the...
Definition: Builders.cpp:567
void setInsertionPoint(Block *block, Block::iterator insertPoint)
Set the insertion point to the specified location.
Definition: Builders.h:403
Operation * create(const OperationState &state)
Creates an operation given the fields represented as an OperationState.
Definition: Builders.cpp:476
void setInsertionPointAfter(Operation *op)
Sets the insertion point to the node after the specified operation, which will cause subsequent inser...
Definition: Builders.h:417
This class represents an operand of an operation.
Definition: Value.h:267
Operation is the basic unit of execution within MLIR.
Definition: Operation.h:88
void setOperand(unsigned idx, Value value)
Definition: Operation.h:346
OpResult getResult(unsigned idx)
Get the 'idx'th result of this operation.
Definition: Operation.h:402
std::enable_if_t< llvm::function_traits< std::decay_t< FnT > >::num_args==1, RetT > walk(FnT &&callback)
Walk the operation by calling the callback for each nested operation (including this one),...
Definition: Operation.h:793
InFlightDiagnostic emitError(const Twine &message={})
Emit an error about fatal conditions with this operation, reporting up to any diagnostic handlers tha...
Definition: Operation.cpp:268
Block * getBlock()
Returns the operation block that contains this operation.
Definition: Operation.h:213
MutableArrayRef< Region > getRegions()
Returns the regions held by this operation.
Definition: Operation.h:672
MutableArrayRef< OpOperand > getOpOperands()
Definition: Operation.h:378
operand_range getOperands()
Returns an iterator on the underlying Value's.
Definition: Operation.h:373
bool isAncestor(Operation *other)
Return true if this operation is an ancestor of the other operation.
Definition: Operation.h:263
result_range getResults()
Definition: Operation.h:410
InFlightDiagnostic emitOpError(const Twine &message={})
Emit an error with the op name prefixed, like "'dim' op " which is convenient for verifiers.
Definition: Operation.cpp:671
unsigned getNumResults()
Return the number of results held by this operation.
Definition: Operation.h:399
MLIRContext * getContext() const
Definition: PatternMatch.h:823
RewritePatternSet & add(ConstructorArg &&arg, ConstructorArgs &&...args)
Add an instance of each of the pattern types 'Ts' to the pattern list with the given arguments.
Definition: PatternMatch.h:847
This class coordinates the application of a rewrite on a set of IR, providing a way for clients to tr...
Definition: PatternMatch.h:400
virtual void replaceOp(Operation *op, ValueRange newValues)
Replace the results of the given (original) operation with the specified list of values (replacements...
virtual void eraseOp(Operation *op)
This method erases an operation that is known to have no uses.
Instances of the Type class are uniqued, have an immutable identifier and an optional mutable compone...
Definition: Types.h:74
This class represents an instance of an SSA value in the MLIR system, representing a computable value...
Definition: Value.h:96
Location getLoc() const
Return the location of this value.
Definition: Value.cpp:26
Operation * getDefiningOp() const
If this value is the result of an operation, return the operation that defines it.
Definition: Value.cpp:20
constexpr void enumerate(std::tuple< Tys... > &tuple, CallbackT &&callback)
Definition: Matchers.h:344
void populateSCFLoopPipeliningPatterns(RewritePatternSet &patterns, const PipeliningOption &options)
Populate patterns for SCF software pipelining transformation.
FailureOr< ForOp > pipelineForLoop(RewriterBase &rewriter, ForOp forOp, const PipeliningOption &options, bool *modifiedIR=nullptr)
Generate a pipelined version of the scf.for loop based on the schedule given as option.
Include the generated interface declarations.
std::optional< int64_t > getConstantIntValue(OpFoldResult ofr)
If ofr is a constant integer or an IntegerAttr, return the integer.
Operation * clone(OpBuilder &b, Operation *op, TypeRange newResultTypes, ValueRange newOperands)
void visitUsedValuesDefinedAbove(Region &region, Region &limit, function_ref< void(OpOperand *)> callback)
Calls callback for each use of a value within region or its descendants that was defined at the ances...
Definition: RegionUtils.cpp:40
Options to dictate how loops should be pipelined.
Definition: Transforms.h:123
std::function< void(Operation *, PipelinerPart, unsigned)> AnnotationlFnType
Lambda called by the pipeliner to allow the user to annotate the IR while it is generated.
Definition: Transforms.h:141
std::function< Operation *(RewriterBase &, Operation *, Value)> PredicateOpFn
Definition: Transforms.h:164