mirror of
https://github.com/capstone-engine/llvm-capstone.git
synced 2025-01-14 20:22:30 +00:00
8067ced144
- Generic visitors invoke operation callbacks before/in-between/after visiting the regions attached to an operation and use a `WalkStage` to indicate which regions have been visited. - This can be useful for cases where we need to visit the operation in between visiting regions attached to the operation. Differential Revision: https://reviews.llvm.org/D116230
208 lines
7.3 KiB
C++
208 lines
7.3 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) {}
|
|
|
|
/// Walk all of the regions/blocks/operations nested under and including the
|
|
/// given operation. Regions, blocks and operations at the same nesting level
|
|
/// are visited in lexicographical order. The walk order for enclosing regions,
|
|
/// blocks and operations with respect to their nested ones is specified by
|
|
/// 'order'. These methods are invoked for void-returning callbacks. A callback
|
|
/// on a block or operation is allowed to erase that block or operation only if
|
|
/// the walk is in post-order. See non-void method for pre-order erasure.
|
|
void detail::walk(Operation *op, function_ref<void(Region *)> callback,
|
|
WalkOrder order) {
|
|
// We don't use early increment for regions because they can't be erased from
|
|
// a callback.
|
|
for (auto ®ion : op->getRegions()) {
|
|
if (order == WalkOrder::PreOrder)
|
|
callback(®ion);
|
|
for (auto &block : region) {
|
|
for (auto &nestedOp : block)
|
|
walk(&nestedOp, callback, order);
|
|
}
|
|
if (order == WalkOrder::PostOrder)
|
|
callback(®ion);
|
|
}
|
|
}
|
|
|
|
void detail::walk(Operation *op, function_ref<void(Block *)> callback,
|
|
WalkOrder order) {
|
|
for (auto ®ion : op->getRegions()) {
|
|
// Early increment here in the case where the block is erased.
|
|
for (auto &block : llvm::make_early_inc_range(region)) {
|
|
if (order == WalkOrder::PreOrder)
|
|
callback(&block);
|
|
for (auto &nestedOp : block)
|
|
walk(&nestedOp, callback, order);
|
|
if (order == WalkOrder::PostOrder)
|
|
callback(&block);
|
|
}
|
|
}
|
|
}
|
|
|
|
void detail::walk(Operation *op, function_ref<void(Operation *)> callback,
|
|
WalkOrder order) {
|
|
if (order == WalkOrder::PreOrder)
|
|
callback(op);
|
|
|
|
// TODO: This walk should be iterative over the operations.
|
|
for (auto ®ion : op->getRegions()) {
|
|
for (auto &block : region) {
|
|
// Early increment here in the case where the operation is erased.
|
|
for (auto &nestedOp : llvm::make_early_inc_range(block))
|
|
walk(&nestedOp, callback, order);
|
|
}
|
|
}
|
|
|
|
if (order == WalkOrder::PostOrder)
|
|
callback(op);
|
|
}
|
|
|
|
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);
|
|
}
|
|
|
|
/// Walk all of the regions/blocks/operations nested under and including the
|
|
/// given operation. These functions walk operations until an interrupt result
|
|
/// is returned by the callback. Walks on regions, blocks and operations may
|
|
/// also be skipped if the callback returns a skip result. Regions, blocks and
|
|
/// operations at the same nesting level are visited in lexicographical order.
|
|
/// The walk order for enclosing regions, blocks and operations with respect to
|
|
/// their nested ones is specified by 'order'. A callback on a block or
|
|
/// operation is allowed to erase that block or operation if either:
|
|
/// * the walk is in post-order, or
|
|
/// * the walk is in pre-order and the walk is skipped after the erasure.
|
|
WalkResult detail::walk(Operation *op,
|
|
function_ref<WalkResult(Region *)> callback,
|
|
WalkOrder order) {
|
|
// We don't use early increment for regions because they can't be erased from
|
|
// a callback.
|
|
for (auto ®ion : op->getRegions()) {
|
|
if (order == WalkOrder::PreOrder) {
|
|
WalkResult result = callback(®ion);
|
|
if (result.wasSkipped())
|
|
continue;
|
|
if (result.wasInterrupted())
|
|
return WalkResult::interrupt();
|
|
}
|
|
for (auto &block : region) {
|
|
for (auto &nestedOp : block)
|
|
walk(&nestedOp, callback, order);
|
|
}
|
|
if (order == WalkOrder::PostOrder) {
|
|
if (callback(®ion).wasInterrupted())
|
|
return WalkResult::interrupt();
|
|
// We don't check if this region was skipped because its walk already
|
|
// finished and the walk will continue with the next region.
|
|
}
|
|
}
|
|
return WalkResult::advance();
|
|
}
|
|
|
|
WalkResult detail::walk(Operation *op,
|
|
function_ref<WalkResult(Block *)> callback,
|
|
WalkOrder order) {
|
|
for (auto ®ion : op->getRegions()) {
|
|
// Early increment here in the case where the block is erased.
|
|
for (auto &block : llvm::make_early_inc_range(region)) {
|
|
if (order == WalkOrder::PreOrder) {
|
|
WalkResult result = callback(&block);
|
|
if (result.wasSkipped())
|
|
continue;
|
|
if (result.wasInterrupted())
|
|
return WalkResult::interrupt();
|
|
}
|
|
for (auto &nestedOp : block)
|
|
walk(&nestedOp, callback, order);
|
|
if (order == WalkOrder::PostOrder) {
|
|
if (callback(&block).wasInterrupted())
|
|
return WalkResult::interrupt();
|
|
// We don't check if this block was skipped because its walk already
|
|
// finished and the walk will continue with the next block.
|
|
}
|
|
}
|
|
}
|
|
return WalkResult::advance();
|
|
}
|
|
|
|
WalkResult detail::walk(Operation *op,
|
|
function_ref<WalkResult(Operation *)> callback,
|
|
WalkOrder order) {
|
|
if (order == WalkOrder::PreOrder) {
|
|
WalkResult result = callback(op);
|
|
// If skipped, caller will continue the walk on the next operation.
|
|
if (result.wasSkipped())
|
|
return WalkResult::advance();
|
|
if (result.wasInterrupted())
|
|
return WalkResult::interrupt();
|
|
}
|
|
|
|
// TODO: This walk should be iterative over the operations.
|
|
for (auto ®ion : op->getRegions()) {
|
|
for (auto &block : region) {
|
|
// Early increment here in the case where the operation is erased.
|
|
for (auto &nestedOp : llvm::make_early_inc_range(block)) {
|
|
if (walk(&nestedOp, callback, order).wasInterrupted())
|
|
return WalkResult::interrupt();
|
|
}
|
|
}
|
|
}
|
|
|
|
if (order == WalkOrder::PostOrder)
|
|
return callback(op);
|
|
return WalkResult::advance();
|
|
}
|
|
|
|
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);
|
|
}
|