mirror of
https://github.com/llvm/llvm-project.git
synced 2025-04-25 01:56:04 +00:00

This allows users to provide custom `Iterator` templates. A new iterator will be added in a subsequent change. Also rename `makeRange` to `makeIterable` and add a test case for the reverse iterator. Differential Revision: https://reviews.llvm.org/D144887
65 lines
1.9 KiB
C++
65 lines
1.9 KiB
C++
//===- Visitors.cpp - MLIR Visitor Utilities ------------------------------===//
|
|
//
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "mlir/IR/Visitors.h"
|
|
#include "mlir/IR/Operation.h"
|
|
|
|
using namespace mlir;
|
|
|
|
WalkStage::WalkStage(Operation *op)
|
|
: numRegions(op->getNumRegions()), nextRegion(0) {}
|
|
|
|
MutableArrayRef<Region> ForwardIterator::makeIterable(Operation &range) {
|
|
return range.getRegions();
|
|
}
|
|
|
|
void detail::walk(Operation *op,
|
|
function_ref<void(Operation *, const WalkStage &)> callback) {
|
|
WalkStage stage(op);
|
|
|
|
for (Region ®ion : op->getRegions()) {
|
|
// Invoke callback on the parent op before visiting each child region.
|
|
callback(op, stage);
|
|
stage.advance();
|
|
|
|
for (Block &block : region) {
|
|
for (Operation &nestedOp : block)
|
|
walk(&nestedOp, callback);
|
|
}
|
|
}
|
|
|
|
// Invoke callback after all regions have been visited.
|
|
callback(op, stage);
|
|
}
|
|
|
|
WalkResult detail::walk(
|
|
Operation *op,
|
|
function_ref<WalkResult(Operation *, const WalkStage &)> callback) {
|
|
WalkStage stage(op);
|
|
|
|
for (Region ®ion : op->getRegions()) {
|
|
// Invoke callback on the parent op before visiting each child region.
|
|
WalkResult result = callback(op, stage);
|
|
|
|
if (result.wasSkipped())
|
|
return WalkResult::advance();
|
|
if (result.wasInterrupted())
|
|
return WalkResult::interrupt();
|
|
|
|
stage.advance();
|
|
|
|
for (Block &block : region) {
|
|
// Early increment here in the case where the operation is erased.
|
|
for (Operation &nestedOp : llvm::make_early_inc_range(block))
|
|
if (walk(&nestedOp, callback).wasInterrupted())
|
|
return WalkResult::interrupt();
|
|
}
|
|
}
|
|
return callback(op, stage);
|
|
}
|