[flang-commits] [flang] [flang][fir]: Add conversion of `fir.iterate_while` to `scf.while`. (PR #152439)
Ming Yan via flang-commits
flang-commits at lists.llvm.org
Fri Aug 8 07:25:06 PDT 2025
================
@@ -88,6 +89,85 @@ struct DoLoopConversion : public OpRewritePattern<fir::DoLoopOp> {
}
};
+struct IterWhileConversion : public OpRewritePattern<fir::IterWhileOp> {
+ using OpRewritePattern<fir::IterWhileOp>::OpRewritePattern;
+
+ LogicalResult matchAndRewrite(fir::IterWhileOp iterWhileOp,
+ PatternRewriter &rewriter) const override {
+
+ Location loc = iterWhileOp.getLoc();
+ Value lowerBound = iterWhileOp.getLowerBound();
+ Value upperBound = iterWhileOp.getUpperBound();
+ Value step = iterWhileOp.getStep();
+
+ Value okInit = iterWhileOp.getIterateIn();
+ ValueRange iterArgs = iterWhileOp.getInitArgs();
+
+ SmallVector<Value> initVals;
+ initVals.push_back(lowerBound);
+ initVals.push_back(okInit);
+ initVals.append(iterArgs.begin(), iterArgs.end());
+
+ SmallVector<Type> loopTypes;
+ loopTypes.push_back(lowerBound.getType());
+ loopTypes.push_back(okInit.getType());
+ for (auto val : iterArgs)
+ loopTypes.push_back(val.getType());
+
+ auto scfWhileOp = scf::WhileOp::create(rewriter, loc, loopTypes, initVals);
+
+ auto &beforeBlock = *rewriter.createBlock(
+ &scfWhileOp.getBefore(), scfWhileOp.getBefore().end(), loopTypes,
+ SmallVector<Location>(loopTypes.size(), loc));
+
+ auto &afterBlock = *rewriter.createBlock(
+ &scfWhileOp.getAfter(), scfWhileOp.getAfter().end(), loopTypes,
+ SmallVector<Location>(loopTypes.size(), loc));
+
+ auto beforeArgs = scfWhileOp.getBefore().getArguments();
+ auto beforeIv = beforeArgs[0];
+ auto beforeOk = beforeArgs[1];
+
+ rewriter.setInsertionPointToStart(&beforeBlock);
+
+ Value inductionCmp = mlir::arith::CmpIOp::create(
+ rewriter, loc, mlir::arith::CmpIPredicate::sle, beforeIv, upperBound);
+ Value cond =
+ mlir::arith::AndIOp::create(rewriter, loc, inductionCmp, beforeOk);
+
+ mlir::scf::ConditionOp::create(rewriter, loc, cond, beforeArgs);
+
+ auto afterArgs = scfWhileOp.getAfter().getArguments();
+
+ SmallVector<Value> argReplacements;
+ for (auto [oldArg, newVal] :
+ llvm::zip(iterWhileOp.getBody()->getArguments(), afterArgs))
+ argReplacements.push_back(newVal);
+
+ auto resultOp = cast<fir::ResultOp>(iterWhileOp.getBody()->getTerminator());
+ SmallVector<Value> results(resultOp->getOperands().begin(),
+ resultOp->getOperands().end());
+
+ rewriter.inlineBlockBefore(iterWhileOp.getBody(), &afterBlock,
+ afterBlock.begin(), argReplacements);
+
+ Value afterIv = afterArgs[0];
+
+ rewriter.setInsertionPointToStart(&afterBlock);
+
+ results[0] = mlir::arith::AddIOp::create(rewriter, loc, afterIv, step);
+
+ Operation *movedTerminator = afterBlock.getTerminator();
+ rewriter.setInsertionPoint(movedTerminator);
+
+ mlir::scf::YieldOp::create(rewriter, loc, results);
----------------
NexMing wrote:
Please use `movedTerminator` Location.
I think we can use
`rewriter.replaceOpWithNewOp<scf::YieldOp>(resultOp, resultOp.getOperands())`
https://github.com/llvm/llvm-project/pull/152439
More information about the flang-commits
mailing list