#include "mlir/Dialect/SCF/Transforms/Passes.h"
#include "mlir/Dialect/Arith/IR/Arith.h"
#include "mlir/Dialect/SCF/IR/SCF.h"
#include "mlir/Dialect/SCF/Transforms/Transforms.h"
#include "mlir/IR/PatternMatch.h"
#include "mlir/Transforms/GreedyPatternRewriteDriver.h"
namespace mlir {
#define GEN_PASS_DEF_SCFFORTOWHILELOOP
#include "mlir/Dialect/SCF/Transforms/Passes.h.inc"
}
using namespace llvm;
using namespace mlir;
using scf::ForOp;
using scf::WhileOp;
namespace {
struct ForLoopLoweringPattern : public OpRewritePattern<ForOp> {
using OpRewritePattern<ForOp>::OpRewritePattern;
LogicalResult matchAndRewrite(ForOp forOp,
PatternRewriter &rewriter) const override {
SmallVector<Type> lcvTypes;
SmallVector<Location> lcvLocs;
lcvTypes.push_back(forOp.getInductionVar().getType());
lcvLocs.push_back(forOp.getInductionVar().getLoc());
for (Value value : forOp.getInitArgs()) {
lcvTypes.push_back(value.getType());
lcvLocs.push_back(value.getLoc());
}
SmallVector<Value> initArgs;
initArgs.push_back(forOp.getLowerBound());
llvm::append_range(initArgs, forOp.getInitArgs());
auto whileOp = rewriter.create<WhileOp>(forOp.getLoc(), lcvTypes, initArgs,
forOp->getAttrs());
auto *beforeBlock = rewriter.createBlock(
&whileOp.getBefore(), whileOp.getBefore().begin(), lcvTypes, lcvLocs);
rewriter.setInsertionPointToStart(whileOp.getBeforeBody());
auto cmpOp = rewriter.create<arith::CmpIOp>(
whileOp.getLoc(), arith::CmpIPredicate::slt,
beforeBlock->getArgument(0), forOp.getUpperBound());
rewriter.create<scf::ConditionOp>(whileOp.getLoc(), cmpOp.getResult(),
beforeBlock->getArguments());
auto *afterBlock = rewriter.createBlock(
&whileOp.getAfter(), whileOp.getAfter().begin(), lcvTypes, lcvLocs);
rewriter.setInsertionPointToEnd(afterBlock);
auto ivIncOp = rewriter.create<arith::AddIOp>(
whileOp.getLoc(), afterBlock->getArgument(0), forOp.getStep());
for (const auto &barg : enumerate(forOp.getBody(0)->getArguments()))
rewriter.replaceAllUsesWith(barg.value(),
afterBlock->getArgument(barg.index()));
for (auto &arg : llvm::make_early_inc_range(*forOp.getBody()))
rewriter.moveOpBefore(&arg, afterBlock, afterBlock->end());
for (auto yieldOp : afterBlock->getOps<scf::YieldOp>()) {
SmallVector<Value> yieldOperands = yieldOp.getOperands();
yieldOperands.insert(yieldOperands.begin(), ivIncOp.getResult());
rewriter.modifyOpInPlace(yieldOp,
[&]() { yieldOp->setOperands(yieldOperands); });
}
for (const auto &arg : llvm::enumerate(forOp.getResults()))
rewriter.replaceAllUsesWith(arg.value(),
whileOp.getResult(arg.index() + 1));
rewriter.eraseOp(forOp);
return success();
}
};
struct ForToWhileLoop : public impl::SCFForToWhileLoopBase<ForToWhileLoop> {
void runOnOperation() override {
auto *parentOp = getOperation();
MLIRContext *ctx = parentOp->getContext();
RewritePatternSet patterns(ctx);
patterns.add<ForLoopLoweringPattern>(ctx);
(void)applyPatternsAndFoldGreedily(parentOp, std::move(patterns));
}
};
}
std::unique_ptr<Pass> mlir::createForToWhileLoopPass() {
return std::make_unique<ForToWhileLoop>();
}