MLIR  20.0.0git
RegionUtils.cpp
Go to the documentation of this file.
1 //===- RegionUtils.cpp - Region-related transformation utilities ----------===//
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 
11 #include "mlir/IR/Block.h"
12 #include "mlir/IR/BuiltinOps.h"
13 #include "mlir/IR/IRMapping.h"
14 #include "mlir/IR/Operation.h"
15 #include "mlir/IR/PatternMatch.h"
17 #include "mlir/IR/Value.h"
21 
22 #include "llvm/ADT/DepthFirstIterator.h"
23 #include "llvm/ADT/PostOrderIterator.h"
24 #include "llvm/ADT/STLExtras.h"
25 #include "llvm/ADT/SmallSet.h"
26 
27 #include <deque>
28 #include <iterator>
29 
30 using namespace mlir;
31 
33  Region &region) {
34  for (auto &use : llvm::make_early_inc_range(orig.getUses())) {
35  if (region.isAncestor(use.getOwner()->getParentRegion()))
36  use.set(replacement);
37  }
38 }
39 
41  Region &region, Region &limit, function_ref<void(OpOperand *)> callback) {
42  assert(limit.isAncestor(&region) &&
43  "expected isolation limit to be an ancestor of the given region");
44 
45  // Collect proper ancestors of `limit` upfront to avoid traversing the region
46  // tree for every value.
47  SmallPtrSet<Region *, 4> properAncestors;
48  for (auto *reg = limit.getParentRegion(); reg != nullptr;
49  reg = reg->getParentRegion()) {
50  properAncestors.insert(reg);
51  }
52 
53  region.walk([callback, &properAncestors](Operation *op) {
54  for (OpOperand &operand : op->getOpOperands())
55  // Callback on values defined in a proper ancestor of region.
56  if (properAncestors.count(operand.get().getParentRegion()))
57  callback(&operand);
58  });
59 }
60 
62  MutableArrayRef<Region> regions, function_ref<void(OpOperand *)> callback) {
63  for (Region &region : regions)
64  visitUsedValuesDefinedAbove(region, region, callback);
65 }
66 
68  SetVector<Value> &values) {
69  visitUsedValuesDefinedAbove(region, limit, [&](OpOperand *operand) {
70  values.insert(operand->get());
71  });
72 }
73 
75  SetVector<Value> &values) {
76  for (Region &region : regions)
77  getUsedValuesDefinedAbove(region, region, values);
78 }
79 
80 //===----------------------------------------------------------------------===//
81 // Make block isolated from above.
82 //===----------------------------------------------------------------------===//
83 
85  RewriterBase &rewriter, Region &region,
86  llvm::function_ref<bool(Operation *)> cloneOperationIntoRegion) {
87 
88  // Get initial list of values used within region but defined above.
89  llvm::SetVector<Value> initialCapturedValues;
90  mlir::getUsedValuesDefinedAbove(region, initialCapturedValues);
91 
92  std::deque<Value> worklist(initialCapturedValues.begin(),
93  initialCapturedValues.end());
94  llvm::DenseSet<Value> visited;
95  llvm::DenseSet<Operation *> visitedOps;
96 
97  llvm::SetVector<Value> finalCapturedValues;
98  SmallVector<Operation *> clonedOperations;
99  while (!worklist.empty()) {
100  Value currValue = worklist.front();
101  worklist.pop_front();
102  if (visited.count(currValue))
103  continue;
104  visited.insert(currValue);
105 
106  Operation *definingOp = currValue.getDefiningOp();
107  if (!definingOp || visitedOps.count(definingOp)) {
108  finalCapturedValues.insert(currValue);
109  continue;
110  }
111  visitedOps.insert(definingOp);
112 
113  if (!cloneOperationIntoRegion(definingOp)) {
114  // Defining operation isnt cloned, so add the current value to final
115  // captured values list.
116  finalCapturedValues.insert(currValue);
117  continue;
118  }
119 
120  // Add all operands of the operation to the worklist and mark the op as to
121  // be cloned.
122  for (Value operand : definingOp->getOperands()) {
123  if (visited.count(operand))
124  continue;
125  worklist.push_back(operand);
126  }
127  clonedOperations.push_back(definingOp);
128  }
129 
130  // The operations to be cloned need to be ordered in topological order
131  // so that they can be cloned into the region without violating use-def
132  // chains.
133  mlir::computeTopologicalSorting(clonedOperations);
134 
135  OpBuilder::InsertionGuard g(rewriter);
136  // Collect types of existing block
137  Block *entryBlock = &region.front();
138  SmallVector<Type> newArgTypes =
139  llvm::to_vector(entryBlock->getArgumentTypes());
140  SmallVector<Location> newArgLocs = llvm::to_vector(llvm::map_range(
141  entryBlock->getArguments(), [](BlockArgument b) { return b.getLoc(); }));
142 
143  // Append the types of the captured values.
144  for (auto value : finalCapturedValues) {
145  newArgTypes.push_back(value.getType());
146  newArgLocs.push_back(value.getLoc());
147  }
148 
149  // Create a new entry block.
150  Block *newEntryBlock =
151  rewriter.createBlock(&region, region.begin(), newArgTypes, newArgLocs);
152  auto newEntryBlockArgs = newEntryBlock->getArguments();
153 
154  // Create a mapping between the captured values and the new arguments added.
155  IRMapping map;
156  auto replaceIfFn = [&](OpOperand &use) {
157  return use.getOwner()->getBlock()->getParent() == &region;
158  };
159  for (auto [arg, capturedVal] :
160  llvm::zip(newEntryBlockArgs.take_back(finalCapturedValues.size()),
161  finalCapturedValues)) {
162  map.map(capturedVal, arg);
163  rewriter.replaceUsesWithIf(capturedVal, arg, replaceIfFn);
164  }
165  rewriter.setInsertionPointToStart(newEntryBlock);
166  for (auto *clonedOp : clonedOperations) {
167  Operation *newOp = rewriter.clone(*clonedOp, map);
168  rewriter.replaceOpUsesWithIf(clonedOp, newOp->getResults(), replaceIfFn);
169  }
170  rewriter.mergeBlocks(
171  entryBlock, newEntryBlock,
172  newEntryBlock->getArguments().take_front(entryBlock->getNumArguments()));
173  return llvm::to_vector(finalCapturedValues);
174 }
175 
176 //===----------------------------------------------------------------------===//
177 // Unreachable Block Elimination
178 //===----------------------------------------------------------------------===//
179 
180 /// Erase the unreachable blocks within the provided regions. Returns success
181 /// if any blocks were erased, failure otherwise.
182 // TODO: We could likely merge this with the DCE algorithm below.
184  MutableArrayRef<Region> regions) {
185  // Set of blocks found to be reachable within a given region.
186  llvm::df_iterator_default_set<Block *, 16> reachable;
187  // If any blocks were found to be dead.
188  bool erasedDeadBlocks = false;
189 
190  SmallVector<Region *, 1> worklist;
191  worklist.reserve(regions.size());
192  for (Region &region : regions)
193  worklist.push_back(&region);
194  while (!worklist.empty()) {
195  Region *region = worklist.pop_back_val();
196  if (region->empty())
197  continue;
198 
199  // If this is a single block region, just collect the nested regions.
200  if (std::next(region->begin()) == region->end()) {
201  for (Operation &op : region->front())
202  for (Region &region : op.getRegions())
203  worklist.push_back(&region);
204  continue;
205  }
206 
207  // Mark all reachable blocks.
208  reachable.clear();
209  for (Block *block : depth_first_ext(&region->front(), reachable))
210  (void)block /* Mark all reachable blocks */;
211 
212  // Collect all of the dead blocks and push the live regions onto the
213  // worklist.
214  for (Block &block : llvm::make_early_inc_range(*region)) {
215  if (!reachable.count(&block)) {
216  block.dropAllDefinedValueUses();
217  rewriter.eraseBlock(&block);
218  erasedDeadBlocks = true;
219  continue;
220  }
221 
222  // Walk any regions within this block.
223  for (Operation &op : block)
224  for (Region &region : op.getRegions())
225  worklist.push_back(&region);
226  }
227  }
228 
229  return success(erasedDeadBlocks);
230 }
231 
232 //===----------------------------------------------------------------------===//
233 // Dead Code Elimination
234 //===----------------------------------------------------------------------===//
235 
236 namespace {
237 /// Data structure used to track which values have already been proved live.
238 ///
239 /// Because Operation's can have multiple results, this data structure tracks
240 /// liveness for both Value's and Operation's to avoid having to look through
241 /// all Operation results when analyzing a use.
242 ///
243 /// This data structure essentially tracks the dataflow lattice.
244 /// The set of values/ops proved live increases monotonically to a fixed-point.
245 class LiveMap {
246 public:
247  /// Value methods.
248  bool wasProvenLive(Value value) {
249  // TODO: For results that are removable, e.g. for region based control flow,
250  // we could allow for these values to be tracked independently.
251  if (OpResult result = dyn_cast<OpResult>(value))
252  return wasProvenLive(result.getOwner());
253  return wasProvenLive(cast<BlockArgument>(value));
254  }
255  bool wasProvenLive(BlockArgument arg) { return liveValues.count(arg); }
256  void setProvedLive(Value value) {
257  // TODO: For results that are removable, e.g. for region based control flow,
258  // we could allow for these values to be tracked independently.
259  if (OpResult result = dyn_cast<OpResult>(value))
260  return setProvedLive(result.getOwner());
261  setProvedLive(cast<BlockArgument>(value));
262  }
263  void setProvedLive(BlockArgument arg) {
264  changed |= liveValues.insert(arg).second;
265  }
266 
267  /// Operation methods.
268  bool wasProvenLive(Operation *op) { return liveOps.count(op); }
269  void setProvedLive(Operation *op) { changed |= liveOps.insert(op).second; }
270 
271  /// Methods for tracking if we have reached a fixed-point.
272  void resetChanged() { changed = false; }
273  bool hasChanged() { return changed; }
274 
275 private:
276  bool changed = false;
277  DenseSet<Value> liveValues;
278  DenseSet<Operation *> liveOps;
279 };
280 } // namespace
281 
282 static bool isUseSpeciallyKnownDead(OpOperand &use, LiveMap &liveMap) {
283  Operation *owner = use.getOwner();
284  unsigned operandIndex = use.getOperandNumber();
285  // This pass generally treats all uses of an op as live if the op itself is
286  // considered live. However, for successor operands to terminators we need a
287  // finer-grained notion where we deduce liveness for operands individually.
288  // The reason for this is easiest to think about in terms of a classical phi
289  // node based SSA IR, where each successor operand is really an operand to a
290  // *separate* phi node, rather than all operands to the branch itself as with
291  // the block argument representation that MLIR uses.
292  //
293  // And similarly, because each successor operand is really an operand to a phi
294  // node, rather than to the terminator op itself, a terminator op can't e.g.
295  // "print" the value of a successor operand.
296  if (owner->hasTrait<OpTrait::IsTerminator>()) {
297  if (BranchOpInterface branchInterface = dyn_cast<BranchOpInterface>(owner))
298  if (auto arg = branchInterface.getSuccessorBlockArgument(operandIndex))
299  return !liveMap.wasProvenLive(*arg);
300  return false;
301  }
302  return false;
303 }
304 
305 static void processValue(Value value, LiveMap &liveMap) {
306  bool provedLive = llvm::any_of(value.getUses(), [&](OpOperand &use) {
307  if (isUseSpeciallyKnownDead(use, liveMap))
308  return false;
309  return liveMap.wasProvenLive(use.getOwner());
310  });
311  if (provedLive)
312  liveMap.setProvedLive(value);
313 }
314 
315 static void propagateLiveness(Region &region, LiveMap &liveMap);
316 
317 static void propagateTerminatorLiveness(Operation *op, LiveMap &liveMap) {
318  // Terminators are always live.
319  liveMap.setProvedLive(op);
320 
321  // Check to see if we can reason about the successor operands and mutate them.
322  BranchOpInterface branchInterface = dyn_cast<BranchOpInterface>(op);
323  if (!branchInterface) {
324  for (Block *successor : op->getSuccessors())
325  for (BlockArgument arg : successor->getArguments())
326  liveMap.setProvedLive(arg);
327  return;
328  }
329 
330  // If we can't reason about the operand to a successor, conservatively mark
331  // it as live.
332  for (unsigned i = 0, e = op->getNumSuccessors(); i != e; ++i) {
333  SuccessorOperands successorOperands =
334  branchInterface.getSuccessorOperands(i);
335  for (unsigned opI = 0, opE = successorOperands.getProducedOperandCount();
336  opI != opE; ++opI)
337  liveMap.setProvedLive(op->getSuccessor(i)->getArgument(opI));
338  }
339 }
340 
341 static void propagateLiveness(Operation *op, LiveMap &liveMap) {
342  // Recurse on any regions the op has.
343  for (Region &region : op->getRegions())
344  propagateLiveness(region, liveMap);
345 
346  // Process terminator operations.
347  if (op->hasTrait<OpTrait::IsTerminator>())
348  return propagateTerminatorLiveness(op, liveMap);
349 
350  // Don't reprocess live operations.
351  if (liveMap.wasProvenLive(op))
352  return;
353 
354  // Process the op itself.
355  if (!wouldOpBeTriviallyDead(op))
356  return liveMap.setProvedLive(op);
357 
358  // If the op isn't intrinsically alive, check it's results.
359  for (Value value : op->getResults())
360  processValue(value, liveMap);
361 }
362 
363 static void propagateLiveness(Region &region, LiveMap &liveMap) {
364  if (region.empty())
365  return;
366 
367  for (Block *block : llvm::post_order(&region.front())) {
368  // We process block arguments after the ops in the block, to promote
369  // faster convergence to a fixed point (we try to visit uses before defs).
370  for (Operation &op : llvm::reverse(block->getOperations()))
371  propagateLiveness(&op, liveMap);
372 
373  // We currently do not remove entry block arguments, so there is no need to
374  // track their liveness.
375  // TODO: We could track these and enable removing dead operands/arguments
376  // from region control flow operations.
377  if (block->isEntryBlock())
378  continue;
379 
380  for (Value value : block->getArguments()) {
381  if (!liveMap.wasProvenLive(value))
382  processValue(value, liveMap);
383  }
384  }
385 }
386 
388  LiveMap &liveMap) {
389  BranchOpInterface branchOp = dyn_cast<BranchOpInterface>(terminator);
390  if (!branchOp)
391  return;
392 
393  for (unsigned succI = 0, succE = terminator->getNumSuccessors();
394  succI < succE; succI++) {
395  // Iterating successors in reverse is not strictly needed, since we
396  // aren't erasing any successors. But it is slightly more efficient
397  // since it will promote later operands of the terminator being erased
398  // first, reducing the quadratic-ness.
399  unsigned succ = succE - succI - 1;
400  SuccessorOperands succOperands = branchOp.getSuccessorOperands(succ);
401  Block *successor = terminator->getSuccessor(succ);
402 
403  for (unsigned argI = 0, argE = succOperands.size(); argI < argE; ++argI) {
404  // Iterating args in reverse is needed for correctness, to avoid
405  // shifting later args when earlier args are erased.
406  unsigned arg = argE - argI - 1;
407  if (!liveMap.wasProvenLive(successor->getArgument(arg)))
408  succOperands.erase(arg);
409  }
410  }
411 }
412 
413 static LogicalResult deleteDeadness(RewriterBase &rewriter,
414  MutableArrayRef<Region> regions,
415  LiveMap &liveMap) {
416  bool erasedAnything = false;
417  for (Region &region : regions) {
418  if (region.empty())
419  continue;
420  bool hasSingleBlock = llvm::hasSingleElement(region);
421 
422  // Delete every operation that is not live. Graph regions may have cycles
423  // in the use-def graph, so we must explicitly dropAllUses() from each
424  // operation as we erase it. Visiting the operations in post-order
425  // guarantees that in SSA CFG regions value uses are removed before defs,
426  // which makes dropAllUses() a no-op.
427  for (Block *block : llvm::post_order(&region.front())) {
428  if (!hasSingleBlock)
429  eraseTerminatorSuccessorOperands(block->getTerminator(), liveMap);
430  for (Operation &childOp :
431  llvm::make_early_inc_range(llvm::reverse(block->getOperations()))) {
432  if (!liveMap.wasProvenLive(&childOp)) {
433  erasedAnything = true;
434  childOp.dropAllUses();
435  rewriter.eraseOp(&childOp);
436  } else {
437  erasedAnything |= succeeded(
438  deleteDeadness(rewriter, childOp.getRegions(), liveMap));
439  }
440  }
441  }
442  // Delete block arguments.
443  // The entry block has an unknown contract with their enclosing block, so
444  // skip it.
445  for (Block &block : llvm::drop_begin(region.getBlocks(), 1)) {
446  block.eraseArguments(
447  [&](BlockArgument arg) { return !liveMap.wasProvenLive(arg); });
448  }
449  }
450  return success(erasedAnything);
451 }
452 
453 // This function performs a simple dead code elimination algorithm over the
454 // given regions.
455 //
456 // The overall goal is to prove that Values are dead, which allows deleting ops
457 // and block arguments.
458 //
459 // This uses an optimistic algorithm that assumes everything is dead until
460 // proved otherwise, allowing it to delete recursively dead cycles.
461 //
462 // This is a simple fixed-point dataflow analysis algorithm on a lattice
463 // {Dead,Alive}. Because liveness flows backward, we generally try to
464 // iterate everything backward to speed up convergence to the fixed-point. This
465 // allows for being able to delete recursively dead cycles of the use-def graph,
466 // including block arguments.
467 //
468 // This function returns success if any operations or arguments were deleted,
469 // failure otherwise.
470 LogicalResult mlir::runRegionDCE(RewriterBase &rewriter,
471  MutableArrayRef<Region> regions) {
472  LiveMap liveMap;
473  do {
474  liveMap.resetChanged();
475 
476  for (Region &region : regions)
477  propagateLiveness(region, liveMap);
478  } while (liveMap.hasChanged());
479 
480  return deleteDeadness(rewriter, regions, liveMap);
481 }
482 
483 //===----------------------------------------------------------------------===//
484 // Block Merging
485 //===----------------------------------------------------------------------===//
486 
487 //===----------------------------------------------------------------------===//
488 // BlockEquivalenceData
489 
490 namespace {
491 /// This class contains the information for comparing the equivalencies of two
492 /// blocks. Blocks are considered equivalent if they contain the same operations
493 /// in the same order. The only allowed divergence is for operands that come
494 /// from sources outside of the parent block, i.e. the uses of values produced
495 /// within the block must be equivalent.
496 /// e.g.,
497 /// Equivalent:
498 /// ^bb1(%arg0: i32)
499 /// return %arg0, %foo : i32, i32
500 /// ^bb2(%arg1: i32)
501 /// return %arg1, %bar : i32, i32
502 /// Not Equivalent:
503 /// ^bb1(%arg0: i32)
504 /// return %foo, %arg0 : i32, i32
505 /// ^bb2(%arg1: i32)
506 /// return %arg1, %bar : i32, i32
507 struct BlockEquivalenceData {
508  BlockEquivalenceData(Block *block);
509 
510  /// Return the order index for the given value that is within the block of
511  /// this data.
512  unsigned getOrderOf(Value value) const;
513 
514  /// The block this data refers to.
515  Block *block;
516  /// A hash value for this block.
517  llvm::hash_code hash;
518  /// A map of result producing operations to their relative orders within this
519  /// block. The order of an operation is the number of defined values that are
520  /// produced within the block before this operation.
521  DenseMap<Operation *, unsigned> opOrderIndex;
522 };
523 } // namespace
524 
525 BlockEquivalenceData::BlockEquivalenceData(Block *block)
526  : block(block), hash(0) {
527  unsigned orderIt = block->getNumArguments();
528  for (Operation &op : *block) {
529  if (unsigned numResults = op.getNumResults()) {
530  opOrderIndex.try_emplace(&op, orderIt);
531  orderIt += numResults;
532  }
533  auto opHash = OperationEquivalence::computeHash(
537  hash = llvm::hash_combine(hash, opHash);
538  }
539 }
540 
541 unsigned BlockEquivalenceData::getOrderOf(Value value) const {
542  assert(value.getParentBlock() == block && "expected value of this block");
543 
544  // Arguments use the argument number as the order index.
545  if (BlockArgument arg = dyn_cast<BlockArgument>(value))
546  return arg.getArgNumber();
547 
548  // Otherwise, the result order is offset from the parent op's order.
549  OpResult result = cast<OpResult>(value);
550  auto opOrderIt = opOrderIndex.find(result.getDefiningOp());
551  assert(opOrderIt != opOrderIndex.end() && "expected op to have an order");
552  return opOrderIt->second + result.getResultNumber();
553 }
554 
555 //===----------------------------------------------------------------------===//
556 // BlockMergeCluster
557 
558 namespace {
559 /// This class represents a cluster of blocks to be merged together.
560 class BlockMergeCluster {
561 public:
562  BlockMergeCluster(BlockEquivalenceData &&leaderData)
563  : leaderData(std::move(leaderData)) {}
564 
565  /// Attempt to add the given block to this cluster. Returns success if the
566  /// block was merged, failure otherwise.
567  LogicalResult addToCluster(BlockEquivalenceData &blockData);
568 
569  /// Try to merge all of the blocks within this cluster into the leader block.
570  LogicalResult merge(RewriterBase &rewriter);
571 
572 private:
573  /// The equivalence data for the leader of the cluster.
574  BlockEquivalenceData leaderData;
575 
576  /// The set of blocks that can be merged into the leader.
577  llvm::SmallSetVector<Block *, 1> blocksToMerge;
578 
579  /// A set of operand+index pairs that correspond to operands that need to be
580  /// replaced by arguments when the cluster gets merged.
581  std::set<std::pair<int, int>> operandsToMerge;
582 };
583 } // namespace
584 
585 LogicalResult BlockMergeCluster::addToCluster(BlockEquivalenceData &blockData) {
586  if (leaderData.hash != blockData.hash)
587  return failure();
588  Block *leaderBlock = leaderData.block, *mergeBlock = blockData.block;
589  if (leaderBlock->getArgumentTypes() != mergeBlock->getArgumentTypes())
590  return failure();
591 
592  // A set of operands that mismatch between the leader and the new block.
593  SmallVector<std::pair<int, int>, 8> mismatchedOperands;
594  auto lhsIt = leaderBlock->begin(), lhsE = leaderBlock->end();
595  auto rhsIt = blockData.block->begin(), rhsE = blockData.block->end();
596  for (int opI = 0; lhsIt != lhsE && rhsIt != rhsE; ++lhsIt, ++rhsIt, ++opI) {
597  // Check that the operations are equivalent.
598  if (!OperationEquivalence::isEquivalentTo(
599  &*lhsIt, &*rhsIt, OperationEquivalence::ignoreValueEquivalence,
600  /*markEquivalent=*/nullptr,
601  OperationEquivalence::Flags::IgnoreLocations))
602  return failure();
603 
604  // Compare the operands of the two operations. If the operand is within
605  // the block, it must refer to the same operation.
606  auto lhsOperands = lhsIt->getOperands(), rhsOperands = rhsIt->getOperands();
607  for (int operand : llvm::seq<int>(0, lhsIt->getNumOperands())) {
608  Value lhsOperand = lhsOperands[operand];
609  Value rhsOperand = rhsOperands[operand];
610  if (lhsOperand == rhsOperand)
611  continue;
612  // Check that the types of the operands match.
613  if (lhsOperand.getType() != rhsOperand.getType())
614  return failure();
615 
616  // Check that these uses are both external, or both internal.
617  bool lhsIsInBlock = lhsOperand.getParentBlock() == leaderBlock;
618  bool rhsIsInBlock = rhsOperand.getParentBlock() == mergeBlock;
619  if (lhsIsInBlock != rhsIsInBlock)
620  return failure();
621  // Let the operands differ if they are defined in a different block. These
622  // will become new arguments if the blocks get merged.
623  if (!lhsIsInBlock) {
624 
625  // Check whether the operands aren't the result of an immediate
626  // predecessors terminator. In that case we are not able to use it as a
627  // successor operand when branching to the merged block as it does not
628  // dominate its producing operation.
629  auto isValidSuccessorArg = [](Block *block, Value operand) {
630  if (operand.getDefiningOp() !=
631  operand.getParentBlock()->getTerminator())
632  return true;
633  return !llvm::is_contained(block->getPredecessors(),
634  operand.getParentBlock());
635  };
636 
637  if (!isValidSuccessorArg(leaderBlock, lhsOperand) ||
638  !isValidSuccessorArg(mergeBlock, rhsOperand))
639  return failure();
640 
641  mismatchedOperands.emplace_back(opI, operand);
642  continue;
643  }
644 
645  // Otherwise, these operands must have the same logical order within the
646  // parent block.
647  if (leaderData.getOrderOf(lhsOperand) != blockData.getOrderOf(rhsOperand))
648  return failure();
649  }
650 
651  // If the lhs or rhs has external uses, the blocks cannot be merged as the
652  // merged version of this operation will not be either the lhs or rhs
653  // alone (thus semantically incorrect), but some mix dependending on which
654  // block preceeded this.
655  // TODO allow merging of operations when one block does not dominate the
656  // other
657  if (rhsIt->isUsedOutsideOfBlock(mergeBlock) ||
658  lhsIt->isUsedOutsideOfBlock(leaderBlock)) {
659  return failure();
660  }
661  }
662  // Make sure that the block sizes are equivalent.
663  if (lhsIt != lhsE || rhsIt != rhsE)
664  return failure();
665 
666  // If we get here, the blocks are equivalent and can be merged.
667  operandsToMerge.insert(mismatchedOperands.begin(), mismatchedOperands.end());
668  blocksToMerge.insert(blockData.block);
669  return success();
670 }
671 
672 /// Returns true if the predecessor terminators of the given block can not have
673 /// their operands updated.
674 static bool ableToUpdatePredOperands(Block *block) {
675  for (auto it = block->pred_begin(), e = block->pred_end(); it != e; ++it) {
676  if (!isa<BranchOpInterface>((*it)->getTerminator()))
677  return false;
678  }
679  return true;
680 }
681 
682 /// Prunes the redundant list of new arguments. E.g., if we are passing an
683 /// argument list like [x, y, z, x] this would return [x, y, z] and it would
684 /// update the `block` (to whom the argument are passed to) accordingly. The new
685 /// arguments are passed as arguments at the back of the block, hence we need to
686 /// know how many `numOldArguments` were before, in order to correctly replace
687 /// the new arguments in the block
689  const SmallVector<SmallVector<Value, 8>, 2> &newArguments,
690  RewriterBase &rewriter, unsigned numOldArguments, Block *block) {
691 
692  SmallVector<SmallVector<Value, 8>, 2> newArgumentsPruned(
693  newArguments.size(), SmallVector<Value, 8>());
694 
695  if (newArguments.empty())
696  return newArguments;
697 
698  // `newArguments` is a 2D array of size `numLists` x `numArgs`
699  unsigned numLists = newArguments.size();
700  unsigned numArgs = newArguments[0].size();
701 
702  // Map that for each arg index contains the index that we can use in place of
703  // the original index. E.g., if we have newArgs = [x, y, z, x], we will have
704  // idxToReplacement[3] = 0
705  llvm::DenseMap<unsigned, unsigned> idxToReplacement;
706 
707  // This is a useful data structure to track the first appearance of a Value
708  // on a given list of arguments
709  DenseMap<Value, unsigned> firstValueToIdx;
710  for (unsigned j = 0; j < numArgs; ++j) {
711  Value newArg = newArguments[0][j];
712  if (!firstValueToIdx.contains(newArg))
713  firstValueToIdx[newArg] = j;
714  }
715 
716  // Go through the first list of arguments (list 0).
717  for (unsigned j = 0; j < numArgs; ++j) {
718  // Look back to see if there are possible redundancies in list 0. Please
719  // note that we are using a map to annotate when an argument was seen first
720  // to avoid a O(N^2) algorithm. This has the drawback that if we have two
721  // lists like:
722  // list0: [%a, %a, %a]
723  // list1: [%c, %b, %b]
724  // We cannot simplify it, because firstValueToIdx[%a] = 0, but we cannot
725  // point list1[1](==%b) or list1[2](==%b) to list1[0](==%c). However, since
726  // the number of arguments can be potentially unbounded we cannot afford a
727  // O(N^2) algorithm (to search to all the possible pairs) and we need to
728  // accept the trade-off.
729  unsigned k = firstValueToIdx[newArguments[0][j]];
730  if (k == j)
731  continue;
732 
733  bool shouldReplaceJ = true;
734  unsigned replacement = k;
735  // If a possible redundancy is found, then scan the other lists: we
736  // can prune the arguments if and only if they are redundant in every
737  // list.
738  for (unsigned i = 1; i < numLists; ++i)
739  shouldReplaceJ =
740  shouldReplaceJ && (newArguments[i][k] == newArguments[i][j]);
741  // Save the replacement.
742  if (shouldReplaceJ)
743  idxToReplacement[j] = replacement;
744  }
745 
746  // Populate the pruned argument list.
747  for (unsigned i = 0; i < numLists; ++i)
748  for (unsigned j = 0; j < numArgs; ++j)
749  if (!idxToReplacement.contains(j))
750  newArgumentsPruned[i].push_back(newArguments[i][j]);
751 
752  // Replace the block's redundant arguments.
753  SmallVector<unsigned> toErase;
754  for (auto [idx, arg] : llvm::enumerate(block->getArguments())) {
755  if (idxToReplacement.contains(idx)) {
756  Value oldArg = block->getArgument(numOldArguments + idx);
757  Value newArg =
758  block->getArgument(numOldArguments + idxToReplacement[idx]);
759  rewriter.replaceAllUsesWith(oldArg, newArg);
760  toErase.push_back(numOldArguments + idx);
761  }
762  }
763 
764  // Erase the block's redundant arguments.
765  for (unsigned idxToErase : llvm::reverse(toErase))
766  block->eraseArgument(idxToErase);
767  return newArgumentsPruned;
768 }
769 
770 LogicalResult BlockMergeCluster::merge(RewriterBase &rewriter) {
771  // Don't consider clusters that don't have blocks to merge.
772  if (blocksToMerge.empty())
773  return failure();
774 
775  Block *leaderBlock = leaderData.block;
776  if (!operandsToMerge.empty()) {
777  // If the cluster has operands to merge, verify that the predecessor
778  // terminators of each of the blocks can have their successor operands
779  // updated.
780  // TODO: We could try and sub-partition this cluster if only some blocks
781  // cause the mismatch.
782  if (!ableToUpdatePredOperands(leaderBlock) ||
783  !llvm::all_of(blocksToMerge, ableToUpdatePredOperands))
784  return failure();
785 
786  // Collect the iterators for each of the blocks to merge. We will walk all
787  // of the iterators at once to avoid operand index invalidation.
788  SmallVector<Block::iterator, 2> blockIterators;
789  blockIterators.reserve(blocksToMerge.size() + 1);
790  blockIterators.push_back(leaderBlock->begin());
791  for (Block *mergeBlock : blocksToMerge)
792  blockIterators.push_back(mergeBlock->begin());
793 
794  // Update each of the predecessor terminators with the new arguments.
795  SmallVector<SmallVector<Value, 8>, 2> newArguments(
796  1 + blocksToMerge.size(),
797  SmallVector<Value, 8>(operandsToMerge.size()));
798  unsigned curOpIndex = 0;
799  unsigned numOldArguments = leaderBlock->getNumArguments();
800  for (const auto &it : llvm::enumerate(operandsToMerge)) {
801  unsigned nextOpOffset = it.value().first - curOpIndex;
802  curOpIndex = it.value().first;
803 
804  // Process the operand for each of the block iterators.
805  for (unsigned i = 0, e = blockIterators.size(); i != e; ++i) {
806  Block::iterator &blockIter = blockIterators[i];
807  std::advance(blockIter, nextOpOffset);
808  auto &operand = blockIter->getOpOperand(it.value().second);
809  newArguments[i][it.index()] = operand.get();
810 
811  // Update the operand and insert an argument if this is the leader.
812  if (i == 0) {
813  Value operandVal = operand.get();
814  operand.set(leaderBlock->addArgument(operandVal.getType(),
815  operandVal.getLoc()));
816  }
817  }
818  }
819 
820  // Prune redundant arguments and update the leader block argument list
821  newArguments = pruneRedundantArguments(newArguments, rewriter,
822  numOldArguments, leaderBlock);
823 
824  // Update the predecessors for each of the blocks.
825  auto updatePredecessors = [&](Block *block, unsigned clusterIndex) {
826  for (auto predIt = block->pred_begin(), predE = block->pred_end();
827  predIt != predE; ++predIt) {
828  auto branch = cast<BranchOpInterface>((*predIt)->getTerminator());
829  unsigned succIndex = predIt.getSuccessorIndex();
830  branch.getSuccessorOperands(succIndex).append(
831  newArguments[clusterIndex]);
832  }
833  };
834  updatePredecessors(leaderBlock, /*clusterIndex=*/0);
835  for (unsigned i = 0, e = blocksToMerge.size(); i != e; ++i)
836  updatePredecessors(blocksToMerge[i], /*clusterIndex=*/i + 1);
837  }
838 
839  // Replace all uses of the merged blocks with the leader and erase them.
840  for (Block *block : blocksToMerge) {
841  block->replaceAllUsesWith(leaderBlock);
842  rewriter.eraseBlock(block);
843  }
844  return success();
845 }
846 
847 /// Identify identical blocks within the given region and merge them, inserting
848 /// new block arguments as necessary. Returns success if any blocks were merged,
849 /// failure otherwise.
850 static LogicalResult mergeIdenticalBlocks(RewriterBase &rewriter,
851  Region &region) {
852  if (region.empty() || llvm::hasSingleElement(region))
853  return failure();
854 
855  // Identify sets of blocks, other than the entry block, that branch to the
856  // same successors. We will use these groups to create clusters of equivalent
857  // blocks.
859  for (Block &block : llvm::drop_begin(region, 1))
860  matchingSuccessors[block.getSuccessors()].push_back(&block);
861 
862  bool mergedAnyBlocks = false;
863  for (ArrayRef<Block *> blocks : llvm::make_second_range(matchingSuccessors)) {
864  if (blocks.size() == 1)
865  continue;
866 
868  for (Block *block : blocks) {
869  BlockEquivalenceData data(block);
870 
871  // Don't allow merging if this block has any regions.
872  // TODO: Add support for regions if necessary.
873  bool hasNonEmptyRegion = llvm::any_of(*block, [](Operation &op) {
874  return llvm::any_of(op.getRegions(),
875  [](Region &region) { return !region.empty(); });
876  });
877  if (hasNonEmptyRegion)
878  continue;
879 
880  // Don't allow merging if this block's arguments are used outside of the
881  // original block.
882  bool argHasExternalUsers = llvm::any_of(
883  block->getArguments(), [block](mlir::BlockArgument &arg) {
884  return arg.isUsedOutsideOfBlock(block);
885  });
886  if (argHasExternalUsers)
887  continue;
888 
889  // Try to add this block to an existing cluster.
890  bool addedToCluster = false;
891  for (auto &cluster : clusters)
892  if ((addedToCluster = succeeded(cluster.addToCluster(data))))
893  break;
894  if (!addedToCluster)
895  clusters.emplace_back(std::move(data));
896  }
897  for (auto &cluster : clusters)
898  mergedAnyBlocks |= succeeded(cluster.merge(rewriter));
899  }
900 
901  return success(mergedAnyBlocks);
902 }
903 
904 /// Identify identical blocks within the given regions and merge them, inserting
905 /// new block arguments as necessary.
906 static LogicalResult mergeIdenticalBlocks(RewriterBase &rewriter,
907  MutableArrayRef<Region> regions) {
908  llvm::SmallSetVector<Region *, 1> worklist;
909  for (auto &region : regions)
910  worklist.insert(&region);
911  bool anyChanged = false;
912  while (!worklist.empty()) {
913  Region *region = worklist.pop_back_val();
914  if (succeeded(mergeIdenticalBlocks(rewriter, *region))) {
915  worklist.insert(region);
916  anyChanged = true;
917  }
918 
919  // Add any nested regions to the worklist.
920  for (Block &block : *region)
921  for (auto &op : block)
922  for (auto &nestedRegion : op.getRegions())
923  worklist.insert(&nestedRegion);
924  }
925 
926  return success(anyChanged);
927 }
928 
929 /// If a block's argument is always the same across different invocations, then
930 /// drop the argument and use the value directly inside the block
931 static LogicalResult dropRedundantArguments(RewriterBase &rewriter,
932  Block &block) {
933  SmallVector<size_t> argsToErase;
934 
935  // Go through the arguments of the block.
936  for (auto [argIdx, blockOperand] : llvm::enumerate(block.getArguments())) {
937  bool sameArg = true;
938  Value commonValue;
939 
940  // Go through the block predecessor and flag if they pass to the block
941  // different values for the same argument.
942  for (Block::pred_iterator predIt = block.pred_begin(),
943  predE = block.pred_end();
944  predIt != predE; ++predIt) {
945  auto branch = dyn_cast<BranchOpInterface>((*predIt)->getTerminator());
946  if (!branch) {
947  sameArg = false;
948  break;
949  }
950  unsigned succIndex = predIt.getSuccessorIndex();
951  SuccessorOperands succOperands = branch.getSuccessorOperands(succIndex);
952  auto branchOperands = succOperands.getForwardedOperands();
953  if (!commonValue) {
954  commonValue = branchOperands[argIdx];
955  continue;
956  }
957  if (branchOperands[argIdx] != commonValue) {
958  sameArg = false;
959  break;
960  }
961  }
962 
963  // If they are passing the same value, drop the argument.
964  if (commonValue && sameArg) {
965  argsToErase.push_back(argIdx);
966 
967  // Remove the argument from the block.
968  rewriter.replaceAllUsesWith(blockOperand, commonValue);
969  }
970  }
971 
972  // Remove the arguments.
973  for (size_t argIdx : llvm::reverse(argsToErase)) {
974  block.eraseArgument(argIdx);
975 
976  // Remove the argument from the branch ops.
977  for (auto predIt = block.pred_begin(), predE = block.pred_end();
978  predIt != predE; ++predIt) {
979  auto branch = cast<BranchOpInterface>((*predIt)->getTerminator());
980  unsigned succIndex = predIt.getSuccessorIndex();
981  SuccessorOperands succOperands = branch.getSuccessorOperands(succIndex);
982  succOperands.erase(argIdx);
983  }
984  }
985  return success(!argsToErase.empty());
986 }
987 
988 /// This optimization drops redundant argument to blocks. I.e., if a given
989 /// argument to a block receives the same value from each of the block
990 /// predecessors, we can remove the argument from the block and use directly the
991 /// original value. This is a simple example:
992 ///
993 /// %cond = llvm.call @rand() : () -> i1
994 /// %val0 = llvm.mlir.constant(1 : i64) : i64
995 /// %val1 = llvm.mlir.constant(2 : i64) : i64
996 /// %val2 = llvm.mlir.constant(3 : i64) : i64
997 /// llvm.cond_br %cond, ^bb1(%val0 : i64, %val1 : i64), ^bb2(%val0 : i64, %val2
998 /// : i64)
999 ///
1000 /// ^bb1(%arg0 : i64, %arg1 : i64):
1001 /// llvm.call @foo(%arg0, %arg1)
1002 ///
1003 /// The previous IR can be rewritten as:
1004 /// %cond = llvm.call @rand() : () -> i1
1005 /// %val0 = llvm.mlir.constant(1 : i64) : i64
1006 /// %val1 = llvm.mlir.constant(2 : i64) : i64
1007 /// %val2 = llvm.mlir.constant(3 : i64) : i64
1008 /// llvm.cond_br %cond, ^bb1(%val1 : i64), ^bb2(%val2 : i64)
1009 ///
1010 /// ^bb1(%arg0 : i64):
1011 /// llvm.call @foo(%val0, %arg0)
1012 ///
1013 static LogicalResult dropRedundantArguments(RewriterBase &rewriter,
1014  MutableArrayRef<Region> regions) {
1015  llvm::SmallSetVector<Region *, 1> worklist;
1016  for (Region &region : regions)
1017  worklist.insert(&region);
1018  bool anyChanged = false;
1019  while (!worklist.empty()) {
1020  Region *region = worklist.pop_back_val();
1021 
1022  // Add any nested regions to the worklist.
1023  for (Block &block : *region) {
1024  anyChanged =
1025  succeeded(dropRedundantArguments(rewriter, block)) || anyChanged;
1026 
1027  for (Operation &op : block)
1028  for (Region &nestedRegion : op.getRegions())
1029  worklist.insert(&nestedRegion);
1030  }
1031  }
1032  return success(anyChanged);
1033 }
1034 
1035 //===----------------------------------------------------------------------===//
1036 // Region Simplification
1037 //===----------------------------------------------------------------------===//
1038 
1039 /// Run a set of structural simplifications over the given regions. This
1040 /// includes transformations like unreachable block elimination, dead argument
1041 /// elimination, as well as some other DCE. This function returns success if any
1042 /// of the regions were simplified, failure otherwise.
1043 LogicalResult mlir::simplifyRegions(RewriterBase &rewriter,
1044  MutableArrayRef<Region> regions,
1045  bool mergeBlocks) {
1046  bool eliminatedBlocks = succeeded(eraseUnreachableBlocks(rewriter, regions));
1047  bool eliminatedOpsOrArgs = succeeded(runRegionDCE(rewriter, regions));
1048  bool mergedIdenticalBlocks = false;
1049  bool droppedRedundantArguments = false;
1050  if (mergeBlocks) {
1051  mergedIdenticalBlocks = succeeded(mergeIdenticalBlocks(rewriter, regions));
1052  droppedRedundantArguments =
1053  succeeded(dropRedundantArguments(rewriter, regions));
1054  }
1055  return success(eliminatedBlocks || eliminatedOpsOrArgs ||
1056  mergedIdenticalBlocks || droppedRedundantArguments);
1057 }
static LogicalResult mergeIdenticalBlocks(RewriterBase &rewriter, Region &region)
Identify identical blocks within the given region and merge them, inserting new block arguments as ne...
static void propagateLiveness(Region &region, LiveMap &liveMap)
static void processValue(Value value, LiveMap &liveMap)
static bool ableToUpdatePredOperands(Block *block)
Returns true if the predecessor terminators of the given block can not have their operands updated.
static void eraseTerminatorSuccessorOperands(Operation *terminator, LiveMap &liveMap)
static LogicalResult dropRedundantArguments(RewriterBase &rewriter, Block &block)
If a block's argument is always the same across different invocations, then drop the argument and use...
static SmallVector< SmallVector< Value, 8 >, 2 > pruneRedundantArguments(const SmallVector< SmallVector< Value, 8 >, 2 > &newArguments, RewriterBase &rewriter, unsigned numOldArguments, Block *block)
Prunes the redundant list of new arguments.
static void propagateTerminatorLiveness(Operation *op, LiveMap &liveMap)
static bool isUseSpeciallyKnownDead(OpOperand &use, LiveMap &liveMap)
static LogicalResult deleteDeadness(RewriterBase &rewriter, MutableArrayRef< Region > regions, LiveMap &liveMap)
This class represents an argument of a Block.
Definition: Value.h:319
unsigned getArgNumber() const
Returns the number of this argument.
Definition: Value.h:331
Block represents an ordered list of Operations.
Definition: Block.h:31
OpListType::iterator iterator
Definition: Block.h:138
ValueTypeRange< BlockArgListType > getArgumentTypes()
Return a range containing the types of the arguments for this block.
Definition: Block.cpp:148
BlockArgument getArgument(unsigned i)
Definition: Block.h:127
unsigned getNumArguments()
Definition: Block.h:126
pred_iterator pred_begin()
Definition: Block.h:231
SuccessorRange getSuccessors()
Definition: Block.h:265
iterator_range< pred_iterator > getPredecessors()
Definition: Block.h:235
BlockArgument addArgument(Type type, Location loc)
Add one value to the argument list.
Definition: Block.cpp:152
BlockArgListType getArguments()
Definition: Block.h:85
iterator end()
Definition: Block.h:142
iterator begin()
Definition: Block.h:141
void eraseArgument(unsigned index)
Erase the argument at 'index' and remove it from the argument list.
Definition: Block.cpp:192
pred_iterator pred_end()
Definition: Block.h:234
This is a utility class for mapping one set of IR entities to another.
Definition: IRMapping.h:26
void map(Value from, Value to)
Inserts a new mapping for 'from' to 'to'.
Definition: IRMapping.h:30
void replaceAllUsesWith(ValueT &&newValue)
Replace all uses of 'this' value with the new value, updating anything in the IR that uses 'this' to ...
Definition: UseDefLists.h:211
IRValueT get() const
Return the current value being used by this operand.
Definition: UseDefLists.h:160
RAII guard to reset the insertion point of the builder when destroyed.
Definition: Builders.h:354
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:571
void setInsertionPointToStart(Block *block)
Sets the insertion point to the start of the specified block.
Definition: Builders.h:437
Block * createBlock(Region *parent, Region::iterator insertPt={}, TypeRange argTypes=std::nullopt, ArrayRef< Location > locs=std::nullopt)
Add new block with 'argTypes' arguments and set the insertion point to the end of it.
Definition: Builders.cpp:453
This class represents an operand of an operation.
Definition: Value.h:267
unsigned getOperandNumber()
Return which operand this is in the OpOperand list of the Operation.
Definition: Value.cpp:216
This is a value defined by a result of an operation.
Definition: Value.h:457
unsigned getResultNumber() const
Returns the number of this result.
Definition: Value.h:469
This class provides the API for ops that are known to be terminators.
Definition: OpDefinition.h:764
Operation is the basic unit of execution within MLIR.
Definition: Operation.h:88
bool hasTrait()
Returns true if the operation was registered with a particular trait, e.g.
Definition: Operation.h:745
Block * getSuccessor(unsigned index)
Definition: Operation.h:704
unsigned getNumSuccessors()
Definition: Operation.h:702
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
SuccessorRange getSuccessors()
Definition: Operation.h:699
result_range getResults()
Definition: Operation.h:410
unsigned getNumResults()
Return the number of results held by this operation.
Definition: Operation.h:399
Implement a predecessor iterator for blocks.
Definition: BlockSupport.h:51
This class contains a list of basic blocks and a link to the parent operation it is attached to.
Definition: Region.h:26
Region * getParentRegion()
Return the region containing this region or nullptr if the region is attached to a top-level operatio...
Definition: Region.cpp:45
bool isAncestor(Region *other)
Return true if this region is ancestor of the other region.
Definition: Region.h:222
bool empty()
Definition: Region.h:60
iterator end()
Definition: Region.h:56
iterator begin()
Definition: Region.h:55
BlockListType & getBlocks()
Definition: Region.h:45
Block & front()
Definition: Region.h:65
RetT walk(FnT &&callback)
Walk all nested operations, blocks or regions (including this region), depending on the type of callb...
Definition: Region.h:285
This class coordinates the application of a rewrite on a set of IR, providing a way for clients to tr...
Definition: PatternMatch.h:400
void replaceOpUsesWithIf(Operation *from, ValueRange to, function_ref< bool(OpOperand &)> functor, bool *allUsesReplaced=nullptr)
Definition: PatternMatch.h:679
virtual void eraseBlock(Block *block)
This method erases all operations in a block.
void replaceAllUsesWith(Value from, Value to)
Find uses of from and replace them with to.
Definition: PatternMatch.h:638
void mergeBlocks(Block *source, Block *dest, ValueRange argValues=std::nullopt)
Inline the operations of block 'source' into the end of block 'dest'.
virtual void eraseOp(Operation *op)
This method erases an operation that is known to have no uses.
void replaceUsesWithIf(Value from, Value to, function_ref< bool(OpOperand &)> functor, bool *allUsesReplaced=nullptr)
Find uses of from and replace them with to if the functor returns true.
This class models how operands are forwarded to block arguments in control flow.
void erase(unsigned subStart, unsigned subLen=1)
Erase operands forwarded to the successor.
unsigned getProducedOperandCount() const
Returns the amount of operands that are produced internally by the operation.
unsigned size() const
Returns the amount of operands passed to the successor.
OperandRange getForwardedOperands() const
Get the range of operands that are simply forwarded to the successor.
This class represents an instance of an SSA value in the MLIR system, representing a computable value...
Definition: Value.h:96
Type getType() const
Return the type of this value.
Definition: Value.h:129
use_range getUses() const
Returns a range of all uses, which is useful for iterating over all uses.
Definition: Value.h:212
Block * getParentBlock()
Return the Block in which this Value is defined.
Definition: Value.cpp:48
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
Operation * getOwner() const
Return the owner of this operand.
Definition: UseDefLists.h:38
constexpr void enumerate(std::tuple< Tys... > &tuple, CallbackT &&callback)
Definition: Matchers.h:344
Include the generated interface declarations.
void replaceAllUsesInRegionWith(Value orig, Value replacement, Region &region)
Replace all uses of orig within the given region with replacement.
Definition: RegionUtils.cpp:32
bool computeTopologicalSorting(MutableArrayRef< Operation * > ops, function_ref< bool(Value, Operation *)> isOperandReady=nullptr)
Compute a topological ordering of the given ops.
bool wouldOpBeTriviallyDead(Operation *op)
Return true if the given operation would be dead if unused, and has no side effects on memory that wo...
LogicalResult eraseUnreachableBlocks(RewriterBase &rewriter, MutableArrayRef< Region > regions)
Erase the unreachable blocks within the provided regions.
SmallVector< Value > makeRegionIsolatedFromAbove(RewriterBase &rewriter, Region &region, llvm::function_ref< bool(Operation *)> cloneOperationIntoRegion=[](Operation *) { return false;})
Make a region isolated from above.
Definition: RegionUtils.cpp:84
void getUsedValuesDefinedAbove(Region &region, Region &limit, SetVector< Value > &values)
Fill values with a list of values defined at the ancestors of the limit region and used within region...
Definition: RegionUtils.cpp:67
LogicalResult runRegionDCE(RewriterBase &rewriter, MutableArrayRef< Region > regions)
This function returns success if any operations or arguments were deleted, failure otherwise.
LogicalResult simplifyRegions(RewriterBase &rewriter, MutableArrayRef< Region > regions, bool mergeBlocks=true)
Run a set of structural simplifications over the given regions.
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
static llvm::hash_code ignoreHashValue(Value)
Helper that can be used with computeHash above to ignore operation operands/result mapping.
static llvm::hash_code computeHash(Operation *op, function_ref< llvm::hash_code(Value)> hashOperands=[](Value v) { return hash_value(v);}, function_ref< llvm::hash_code(Value)> hashResults=[](Value v) { return hash_value(v);}, Flags flags=Flags::None)
Compute a hash for the given operation.
Eliminates variable at the specified position using Fourier-Motzkin variable elimination.