diff --git a/mlir/include/mlir/Dialect/Affine/IR/AffineOps.h b/mlir/include/mlir/Dialect/Affine/IR/AffineOps.h --- a/mlir/include/mlir/Dialect/Affine/IR/AffineOps.h +++ b/mlir/include/mlir/Dialect/Affine/IR/AffineOps.h @@ -381,6 +381,21 @@ void extractForInductionVars(ArrayRef forInsts, SmallVectorImpl *ivs); +/// Builds a perfect nest of affine "for" loops, i.e. each loop except the +/// innermost only contains another loop and a terminator. The loops iterate +/// from "lbs" to "ubs" with "steps". The body of the innermost loop is +/// populated by calling "bodyBuilderFn" and providing it with an OpBuilder, a +/// Location and a list of loop induction variables. +void buildAffineLoopNest(OpBuilder &builder, Location loc, + ArrayRef lbs, ArrayRef ubs, + ArrayRef steps, + function_ref + bodyBuilderFn = nullptr); +void buildAffineLoopNest(OpBuilder &builder, Location loc, ValueRange lbs, + ValueRange ubs, ArrayRef steps, + function_ref + bodyBuilderFn = nullptr); + /// AffineBound represents a lower or upper bound in the for operation. /// This class does not own the underlying operands. Instead, it refers /// to the operands stored in the AffineForOp. Its life span should not exceed diff --git a/mlir/lib/Dialect/Affine/EDSC/Builders.cpp b/mlir/lib/Dialect/Affine/EDSC/Builders.cpp --- a/mlir/lib/Dialect/Affine/EDSC/Builders.cpp +++ b/mlir/lib/Dialect/Affine/EDSC/Builders.cpp @@ -68,51 +68,22 @@ ValueRange lbs, ValueRange ubs, ArrayRef steps, function_ref bodyBuilderFn) { assert(ScopedContext::getContext() && "EDSC ScopedContext not set up"); - assert(lbs.size() == ubs.size() && "Mismatch in number of arguments"); - assert(lbs.size() == steps.size() && "Mismatch in number of arguments"); - - // If there are no loops to be constructed, construct the body anyway. - if (lbs.empty()) { - if (bodyBuilderFn) - bodyBuilderFn(ValueRange()); - return; - } - // Fetch the builder and location. + // Wrap the body builder function into an interface compatible with the main + // builder. + auto wrappedBuilderFn = [&](OpBuilder &nestedBuilder, Location nestedLoc, + ValueRange ivs) { + ScopedContext context(nestedBuilder, nestedLoc); + bodyBuilderFn(ivs); + }; + function_ref wrapper; + if (bodyBuilderFn) + wrapper = wrappedBuilderFn; + + // Extract the builder, location and construct the loop nest. OpBuilder &builder = ScopedContext::getBuilderRef(); - OpBuilder::InsertionGuard guard(builder); Location loc = ScopedContext::getLocation(); - AffineMap identity = builder.getDimIdentityMap(); - - // Create the loops iteratively and store the induction variables. - SmallVector ivs; - ivs.reserve(lbs.size()); - for (unsigned i = 0, e = lbs.size(); i < e; ++i) { - // Callback for creating the loop body, always creates the terminator. - auto loopBody = [&](OpBuilder &nestedBuilder, Location nestedLoc, - Value iv) { - ivs.push_back(iv); - // In the innermost loop, call the body builder. - if (i == e - 1 && bodyBuilderFn) { - ScopedContext nestedContext(nestedBuilder, loc); - OpBuilder::InsertionGuard nestedGuard(nestedBuilder); - bodyBuilderFn(ivs); - } - nestedBuilder.create(nestedLoc); - }; - - // Create the loop. If the bounds are known to be constants, use the - // constant form of the loop. - auto lbConst = lbs[i].getDefiningOp(); - auto ubConst = ubs[i].getDefiningOp(); - auto loop = lbConst && ubConst - ? builder.create(loc, lbConst.getValue(), - ubConst.getValue(), steps[i], - loopBody) - : builder.create(loc, lbs[i], identity, ubs[i], - identity, steps[i], loopBody); - builder.setInsertionPointToStart(loop.getBody()); - } + buildAffineLoopNest(builder, loc, lbs, ubs, steps, wrapper); } void mlir::edsc::affineLoopBuilder(ValueRange lbs, ValueRange ubs, int64_t step, diff --git a/mlir/lib/Dialect/Affine/IR/AffineOps.cpp b/mlir/lib/Dialect/Affine/IR/AffineOps.cpp --- a/mlir/lib/Dialect/Affine/IR/AffineOps.cpp +++ b/mlir/lib/Dialect/Affine/IR/AffineOps.cpp @@ -1688,6 +1688,86 @@ ivs->push_back(forInst.getInductionVar()); } +/// Builds an affine loop nest, using "loopCreatorFn" to create individual loop +/// operations. +template +void buildAffineLoopNestImpl( + OpBuilder &builder, Location loc, BoundListTy lbs, BoundListTy ubs, + ArrayRef steps, + function_ref bodyBuilderFn, + LoopCreatorTy &&loopCreatorFn) { + assert(lbs.size() == ubs.size() && "Mismatch in number of arguments"); + assert(lbs.size() == steps.size() && "Mismatch in number of arguments"); + + // If there are no loops to be constructed, construct the body anyway. + OpBuilder::InsertionGuard guard(builder); + if (lbs.empty()) { + if (bodyBuilderFn) + bodyBuilderFn(builder, loc, ValueRange()); + return; + } + + // Create the loops iteratively and store the induction variables. + SmallVector ivs; + ivs.reserve(lbs.size()); + for (unsigned i = 0, e = lbs.size(); i < e; ++i) { + // Callback for creating the loop body, always creates the terminator. + auto loopBody = [&](OpBuilder &nestedBuilder, Location nestedLoc, + Value iv) { + ivs.push_back(iv); + // In the innermost loop, call the body builder. + if (i == e - 1 && bodyBuilderFn) { + OpBuilder::InsertionGuard nestedGuard(nestedBuilder); + bodyBuilderFn(nestedBuilder, nestedLoc, ivs); + } + nestedBuilder.create(nestedLoc); + }; + + // Delegate actual loop creation to the callback in order to dispatch + // between constant- and variable-bound loops. + auto loop = loopCreatorFn(builder, loc, lbs[i], ubs[i], steps[i], loopBody); + builder.setInsertionPointToStart(loop.getBody()); + } +} + +/// Creates an affine loop from the bounds known to be constants. +static AffineForOp buildAffineLoopFromConstants( + OpBuilder &builder, Location loc, int64_t lb, int64_t ub, int64_t step, + function_ref bodyBuilderFn) { + return builder.create(loc, lb, ub, step, bodyBuilderFn); +} + +/// Creates an affine loop from the bounds that may or may not be constants. +static AffineForOp buildAffineLoopFromValues( + OpBuilder &builder, Location loc, Value lb, Value ub, int64_t step, + function_ref bodyBuilderFn) { + auto lbConst = lb.getDefiningOp(); + auto ubConst = ub.getDefiningOp(); + if (lbConst && ubConst) + return buildAffineLoopFromConstants(builder, loc, lbConst.getValue(), + ubConst.getValue(), step, + bodyBuilderFn); + return builder.create(loc, lb, builder.getDimIdentityMap(), ub, + builder.getDimIdentityMap(), step, + bodyBuilderFn); +} + +void mlir::buildAffineLoopNest( + OpBuilder &builder, Location loc, ArrayRef lbs, + ArrayRef ubs, ArrayRef steps, + function_ref bodyBuilderFn) { + buildAffineLoopNestImpl(builder, loc, lbs, ubs, steps, bodyBuilderFn, + buildAffineLoopFromConstants); +} + +void mlir::buildAffineLoopNest( + OpBuilder &builder, Location loc, ValueRange lbs, ValueRange ubs, + ArrayRef steps, + function_ref bodyBuilderFn) { + buildAffineLoopNestImpl(builder, loc, lbs, ubs, steps, bodyBuilderFn, + buildAffineLoopFromValues); +} + //===----------------------------------------------------------------------===// // AffineIfOp //===----------------------------------------------------------------------===//