2019-01-03 04:52:41 +08:00
|
|
|
//===- LowerAffine.cpp - Lower affine constructs to primitives ------------===//
|
2018-12-31 08:22:50 +08:00
|
|
|
//
|
|
|
|
// Copyright 2019 The MLIR Authors.
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
//
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
// =============================================================================
|
|
|
|
//
|
2019-01-03 04:52:41 +08:00
|
|
|
// This file lowers affine constructs (If and For statements, AffineApply
|
2019-07-12 21:48:45 +08:00
|
|
|
// operations) within a function into their standard If and For equivalent ops.
|
2018-12-31 08:22:50 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2019-05-28 15:49:23 +08:00
|
|
|
#include "mlir/Transforms/LowerAffine.h"
|
2019-01-29 13:23:53 +08:00
|
|
|
#include "mlir/AffineOps/AffineOps.h"
|
2019-07-15 17:50:09 +08:00
|
|
|
#include "mlir/Dialect/LoopOps/LoopOps.h"
|
2019-01-03 04:52:41 +08:00
|
|
|
#include "mlir/IR/AffineExprVisitor.h"
|
2019-06-11 23:33:18 +08:00
|
|
|
#include "mlir/IR/BlockAndValueMapping.h"
|
2018-12-31 08:22:50 +08:00
|
|
|
#include "mlir/IR/Builders.h"
|
2019-02-02 08:42:18 +08:00
|
|
|
#include "mlir/IR/IntegerSet.h"
|
2019-01-03 04:52:41 +08:00
|
|
|
#include "mlir/IR/MLIRContext.h"
|
2019-02-20 09:17:46 +08:00
|
|
|
#include "mlir/Pass/Pass.h"
|
2019-03-02 05:48:24 +08:00
|
|
|
#include "mlir/StandardOps/Ops.h"
|
2019-01-03 04:52:41 +08:00
|
|
|
#include "mlir/Support/Functional.h"
|
2019-06-11 23:33:18 +08:00
|
|
|
#include "mlir/Transforms/DialectConversion.h"
|
2018-12-31 08:22:50 +08:00
|
|
|
#include "mlir/Transforms/Passes.h"
|
2019-06-11 23:33:18 +08:00
|
|
|
|
2018-12-31 08:22:50 +08:00
|
|
|
using namespace mlir;
|
|
|
|
|
|
|
|
namespace {
|
2019-03-28 05:02:02 +08:00
|
|
|
// Visit affine expressions recursively and build the sequence of operations
|
2019-01-03 04:52:41 +08:00
|
|
|
// that correspond to it. Visitation functions return an Value of the
|
|
|
|
// expression subtree they visited or `nullptr` on error.
|
|
|
|
class AffineApplyExpander
|
|
|
|
: public AffineExprVisitor<AffineApplyExpander, Value *> {
|
2018-12-31 08:22:50 +08:00
|
|
|
public:
|
2019-01-03 04:52:41 +08:00
|
|
|
// This internal class expects arguments to be non-null, checks must be
|
|
|
|
// performed at the call site.
|
2019-06-21 06:10:35 +08:00
|
|
|
AffineApplyExpander(OpBuilder &builder, ArrayRef<Value *> dimValues,
|
2019-01-03 04:52:41 +08:00
|
|
|
ArrayRef<Value *> symbolValues, Location loc)
|
2019-06-21 06:10:35 +08:00
|
|
|
: builder(builder), dimValues(dimValues), symbolValues(symbolValues),
|
2019-01-03 04:52:41 +08:00
|
|
|
loc(loc) {}
|
|
|
|
|
|
|
|
template <typename OpTy> Value *buildBinaryExpr(AffineBinaryOpExpr expr) {
|
|
|
|
auto lhs = visit(expr.getLHS());
|
|
|
|
auto rhs = visit(expr.getRHS());
|
|
|
|
if (!lhs || !rhs)
|
|
|
|
return nullptr;
|
|
|
|
auto op = builder.create<OpTy>(loc, lhs, rhs);
|
2019-03-26 02:13:31 +08:00
|
|
|
return op.getResult();
|
2019-01-03 04:52:41 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
Value *visitAddExpr(AffineBinaryOpExpr expr) {
|
|
|
|
return buildBinaryExpr<AddIOp>(expr);
|
|
|
|
}
|
|
|
|
|
|
|
|
Value *visitMulExpr(AffineBinaryOpExpr expr) {
|
|
|
|
return buildBinaryExpr<MulIOp>(expr);
|
|
|
|
}
|
|
|
|
|
2019-01-10 17:44:32 +08:00
|
|
|
// Euclidean modulo operation: negative RHS is not allowed.
|
|
|
|
// Remainder of the euclidean integer division is always non-negative.
|
|
|
|
//
|
|
|
|
// Implemented as
|
|
|
|
//
|
|
|
|
// a mod b =
|
|
|
|
// let remainder = srem a, b;
|
|
|
|
// negative = a < 0 in
|
|
|
|
// select negative, remainder + b, remainder.
|
|
|
|
Value *visitModExpr(AffineBinaryOpExpr expr) {
|
|
|
|
auto rhsConst = expr.getRHS().dyn_cast<AffineConstantExpr>();
|
|
|
|
if (!rhsConst) {
|
2019-06-26 12:31:54 +08:00
|
|
|
emitError(
|
2019-01-10 17:44:32 +08:00
|
|
|
loc,
|
|
|
|
"semi-affine expressions (modulo by non-const) are not supported");
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
if (rhsConst.getValue() <= 0) {
|
2019-06-26 12:31:54 +08:00
|
|
|
emitError(loc, "modulo by non-positive value is not supported");
|
2019-01-10 17:44:32 +08:00
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto lhs = visit(expr.getLHS());
|
|
|
|
auto rhs = visit(expr.getRHS());
|
|
|
|
assert(lhs && rhs && "unexpected affine expr lowering failure");
|
|
|
|
|
|
|
|
Value *remainder = builder.create<RemISOp>(loc, lhs, rhs);
|
|
|
|
Value *zeroCst = builder.create<ConstantIndexOp>(loc, 0);
|
|
|
|
Value *isRemainderNegative =
|
|
|
|
builder.create<CmpIOp>(loc, CmpIPredicate::SLT, remainder, zeroCst);
|
|
|
|
Value *correctedRemainder = builder.create<AddIOp>(loc, remainder, rhs);
|
|
|
|
Value *result = builder.create<SelectOp>(loc, isRemainderNegative,
|
|
|
|
correctedRemainder, remainder);
|
|
|
|
return result;
|
2019-01-03 04:52:41 +08:00
|
|
|
}
|
|
|
|
|
2019-01-10 17:44:32 +08:00
|
|
|
// Floor division operation (rounds towards negative infinity).
|
|
|
|
//
|
|
|
|
// For positive divisors, it can be implemented without branching and with a
|
2019-03-28 05:02:02 +08:00
|
|
|
// single division operation as
|
2019-01-10 17:44:32 +08:00
|
|
|
//
|
|
|
|
// a floordiv b =
|
|
|
|
// let negative = a < 0 in
|
|
|
|
// let absolute = negative ? -a - 1 : a in
|
|
|
|
// let quotient = absolute / b in
|
|
|
|
// negative ? -quotient - 1 : quotient
|
|
|
|
Value *visitFloorDivExpr(AffineBinaryOpExpr expr) {
|
|
|
|
auto rhsConst = expr.getRHS().dyn_cast<AffineConstantExpr>();
|
|
|
|
if (!rhsConst) {
|
2019-06-26 12:31:54 +08:00
|
|
|
emitError(
|
2019-01-10 17:44:32 +08:00
|
|
|
loc,
|
|
|
|
"semi-affine expressions (division by non-const) are not supported");
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
if (rhsConst.getValue() <= 0) {
|
2019-06-26 12:31:54 +08:00
|
|
|
emitError(loc, "division by non-positive value is not supported");
|
2019-01-10 17:44:32 +08:00
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto lhs = visit(expr.getLHS());
|
|
|
|
auto rhs = visit(expr.getRHS());
|
|
|
|
assert(lhs && rhs && "unexpected affine expr lowering failure");
|
|
|
|
|
|
|
|
Value *zeroCst = builder.create<ConstantIndexOp>(loc, 0);
|
|
|
|
Value *noneCst = builder.create<ConstantIndexOp>(loc, -1);
|
|
|
|
Value *negative =
|
|
|
|
builder.create<CmpIOp>(loc, CmpIPredicate::SLT, lhs, zeroCst);
|
|
|
|
Value *negatedDecremented = builder.create<SubIOp>(loc, noneCst, lhs);
|
|
|
|
Value *dividend =
|
|
|
|
builder.create<SelectOp>(loc, negative, negatedDecremented, lhs);
|
|
|
|
Value *quotient = builder.create<DivISOp>(loc, dividend, rhs);
|
|
|
|
Value *correctedQuotient = builder.create<SubIOp>(loc, noneCst, quotient);
|
|
|
|
Value *result =
|
|
|
|
builder.create<SelectOp>(loc, negative, correctedQuotient, quotient);
|
|
|
|
return result;
|
2019-01-03 04:52:41 +08:00
|
|
|
}
|
|
|
|
|
2019-01-10 17:44:32 +08:00
|
|
|
// Ceiling division operation (rounds towards positive infinity).
|
|
|
|
//
|
|
|
|
// For positive divisors, it can be implemented without branching and with a
|
2019-03-28 05:02:02 +08:00
|
|
|
// single division operation as
|
2019-01-10 17:44:32 +08:00
|
|
|
//
|
|
|
|
// a ceildiv b =
|
|
|
|
// let negative = a <= 0 in
|
|
|
|
// let absolute = negative ? -a : a - 1 in
|
|
|
|
// let quotient = absolute / b in
|
|
|
|
// negative ? -quotient : quotient + 1
|
|
|
|
Value *visitCeilDivExpr(AffineBinaryOpExpr expr) {
|
|
|
|
auto rhsConst = expr.getRHS().dyn_cast<AffineConstantExpr>();
|
|
|
|
if (!rhsConst) {
|
2019-06-26 12:31:54 +08:00
|
|
|
emitError(loc) << "semi-affine expressions (division by non-const) are "
|
|
|
|
"not supported";
|
2019-01-10 17:44:32 +08:00
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
if (rhsConst.getValue() <= 0) {
|
2019-06-26 12:31:54 +08:00
|
|
|
emitError(loc, "division by non-positive value is not supported");
|
2019-01-10 17:44:32 +08:00
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
auto lhs = visit(expr.getLHS());
|
|
|
|
auto rhs = visit(expr.getRHS());
|
|
|
|
assert(lhs && rhs && "unexpected affine expr lowering failure");
|
|
|
|
|
|
|
|
Value *zeroCst = builder.create<ConstantIndexOp>(loc, 0);
|
|
|
|
Value *oneCst = builder.create<ConstantIndexOp>(loc, 1);
|
|
|
|
Value *nonPositive =
|
|
|
|
builder.create<CmpIOp>(loc, CmpIPredicate::SLE, lhs, zeroCst);
|
|
|
|
Value *negated = builder.create<SubIOp>(loc, zeroCst, lhs);
|
|
|
|
Value *decremented = builder.create<SubIOp>(loc, lhs, oneCst);
|
|
|
|
Value *dividend =
|
|
|
|
builder.create<SelectOp>(loc, nonPositive, negated, decremented);
|
|
|
|
Value *quotient = builder.create<DivISOp>(loc, dividend, rhs);
|
|
|
|
Value *negatedQuotient = builder.create<SubIOp>(loc, zeroCst, quotient);
|
|
|
|
Value *incrementedQuotient = builder.create<AddIOp>(loc, quotient, oneCst);
|
|
|
|
Value *result = builder.create<SelectOp>(loc, nonPositive, negatedQuotient,
|
|
|
|
incrementedQuotient);
|
|
|
|
return result;
|
2019-01-03 04:52:41 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
Value *visitConstantExpr(AffineConstantExpr expr) {
|
|
|
|
auto valueAttr =
|
|
|
|
builder.getIntegerAttr(builder.getIndexType(), expr.getValue());
|
|
|
|
auto op =
|
2019-01-15 05:36:02 +08:00
|
|
|
builder.create<ConstantOp>(loc, builder.getIndexType(), valueAttr);
|
2019-03-26 02:13:31 +08:00
|
|
|
return op.getResult();
|
2019-01-03 04:52:41 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
Value *visitDimExpr(AffineDimExpr expr) {
|
|
|
|
assert(expr.getPosition() < dimValues.size() &&
|
|
|
|
"affine dim position out of range");
|
|
|
|
return dimValues[expr.getPosition()];
|
|
|
|
}
|
|
|
|
|
|
|
|
Value *visitSymbolExpr(AffineSymbolExpr expr) {
|
|
|
|
assert(expr.getPosition() < symbolValues.size() &&
|
|
|
|
"symbol dim position out of range");
|
|
|
|
return symbolValues[expr.getPosition()];
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
2019-06-05 10:18:23 +08:00
|
|
|
OpBuilder &builder;
|
2019-01-03 04:52:41 +08:00
|
|
|
ArrayRef<Value *> dimValues;
|
|
|
|
ArrayRef<Value *> symbolValues;
|
|
|
|
|
|
|
|
Location loc;
|
|
|
|
};
|
|
|
|
} // namespace
|
|
|
|
|
2019-03-28 05:02:02 +08:00
|
|
|
// Create a sequence of operations that implement the `expr` applied to the
|
2019-01-03 04:52:41 +08:00
|
|
|
// given dimension and symbol values.
|
2019-06-14 16:56:19 +08:00
|
|
|
mlir::Value *mlir::expandAffineExpr(OpBuilder &builder, Location loc,
|
|
|
|
AffineExpr expr,
|
|
|
|
ArrayRef<Value *> dimValues,
|
|
|
|
ArrayRef<Value *> symbolValues) {
|
2019-06-21 06:10:35 +08:00
|
|
|
return AffineApplyExpander(builder, dimValues, symbolValues, loc).visit(expr);
|
2019-01-03 04:52:41 +08:00
|
|
|
}
|
|
|
|
|
2019-03-28 05:02:02 +08:00
|
|
|
// Create a sequence of operations that implement the `affineMap` applied to
|
2019-01-03 04:52:41 +08:00
|
|
|
// the given `operands` (as it it were an AffineApplyOp).
|
|
|
|
Optional<SmallVector<Value *, 8>> static expandAffineMap(
|
2019-06-21 06:10:35 +08:00
|
|
|
OpBuilder &builder, Location loc, AffineMap affineMap,
|
2019-01-03 04:52:41 +08:00
|
|
|
ArrayRef<Value *> operands) {
|
|
|
|
auto numDims = affineMap.getNumDims();
|
|
|
|
auto expanded = functional::map(
|
2019-06-21 06:10:35 +08:00
|
|
|
[numDims, &builder, loc, operands](AffineExpr expr) {
|
|
|
|
return expandAffineExpr(builder, loc, expr,
|
2019-01-03 04:52:41 +08:00
|
|
|
operands.take_front(numDims),
|
|
|
|
operands.drop_front(numDims));
|
|
|
|
},
|
|
|
|
affineMap.getResults());
|
|
|
|
if (llvm::all_of(expanded, [](Value *v) { return v; }))
|
|
|
|
return expanded;
|
|
|
|
return None;
|
|
|
|
}
|
|
|
|
|
2018-12-31 08:22:50 +08:00
|
|
|
// Given a range of values, emit the code that reduces them with "min" or "max"
|
|
|
|
// depending on the provided comparison predicate. The predicate defines which
|
|
|
|
// comparison to perform, "lt" for "min", "gt" for "max" and is used for the
|
|
|
|
// `cmpi` operation followed by the `select` operation:
|
|
|
|
//
|
|
|
|
// %cond = cmpi "predicate" %v0, %v1
|
|
|
|
// %result = select %cond, %v0, %v1
|
|
|
|
//
|
|
|
|
// Multiple values are scanned in a linear sequence. This creates a data
|
|
|
|
// dependences that wouldn't exist in a tree reduction, but is easier to
|
|
|
|
// recognize as a reduction by the subsequent passes.
|
2019-01-03 04:36:58 +08:00
|
|
|
static Value *buildMinMaxReductionSeq(Location loc, CmpIPredicate predicate,
|
|
|
|
ArrayRef<Value *> values,
|
2019-06-05 10:18:23 +08:00
|
|
|
OpBuilder &builder) {
|
2018-12-31 08:22:50 +08:00
|
|
|
assert(!llvm::empty(values) && "empty min/max chain");
|
|
|
|
|
|
|
|
auto valueIt = values.begin();
|
|
|
|
Value *value = *valueIt++;
|
|
|
|
for (; valueIt != values.end(); ++valueIt) {
|
|
|
|
auto cmpOp = builder.create<CmpIOp>(loc, predicate, value, *valueIt);
|
2019-03-26 02:13:31 +08:00
|
|
|
value = builder.create<SelectOp>(loc, cmpOp.getResult(), value, *valueIt);
|
2018-12-31 08:22:50 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
2019-06-14 16:56:19 +08:00
|
|
|
// Emit instructions that correspond to the affine map in the lower bound
|
|
|
|
// applied to the respective operands, and compute the maximum value across
|
|
|
|
// the results.
|
|
|
|
Value *mlir::lowerAffineLowerBound(AffineForOp op, OpBuilder &builder) {
|
|
|
|
SmallVector<Value *, 8> boundOperands(op.getLowerBoundOperands());
|
2019-06-21 06:10:35 +08:00
|
|
|
auto lbValues = expandAffineMap(builder, op.getLoc(), op.getLowerBoundMap(),
|
2019-06-14 16:56:19 +08:00
|
|
|
boundOperands);
|
|
|
|
if (!lbValues)
|
|
|
|
return nullptr;
|
|
|
|
return buildMinMaxReductionSeq(op.getLoc(), CmpIPredicate::SGT, *lbValues,
|
|
|
|
builder);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Emit instructions that correspond to the affine map in the upper bound
|
|
|
|
// applied to the respective operands, and compute the minimum value across
|
|
|
|
// the results.
|
|
|
|
Value *mlir::lowerAffineUpperBound(AffineForOp op, OpBuilder &builder) {
|
|
|
|
SmallVector<Value *, 8> boundOperands(op.getUpperBoundOperands());
|
2019-06-21 06:10:35 +08:00
|
|
|
auto ubValues = expandAffineMap(builder, op.getLoc(), op.getUpperBoundMap(),
|
2019-06-14 16:56:19 +08:00
|
|
|
boundOperands);
|
|
|
|
if (!ubValues)
|
|
|
|
return nullptr;
|
|
|
|
return buildMinMaxReductionSeq(op.getLoc(), CmpIPredicate::SLT, *ubValues,
|
|
|
|
builder);
|
|
|
|
}
|
|
|
|
|
2019-06-11 23:33:18 +08:00
|
|
|
namespace {
|
|
|
|
// Affine terminators are removed.
|
2019-07-15 22:35:00 +08:00
|
|
|
class AffineTerminatorLowering : public OpRewritePattern<AffineTerminatorOp> {
|
2019-06-11 23:33:18 +08:00
|
|
|
public:
|
2019-07-15 22:35:00 +08:00
|
|
|
using OpRewritePattern<AffineTerminatorOp>::OpRewritePattern;
|
2019-06-11 23:33:18 +08:00
|
|
|
|
2019-07-15 22:35:00 +08:00
|
|
|
PatternMatchResult matchAndRewrite(AffineTerminatorOp op,
|
|
|
|
PatternRewriter &rewriter) const override {
|
2019-07-15 17:50:09 +08:00
|
|
|
rewriter.replaceOpWithNewOp<loop::TerminatorOp>(op);
|
2019-06-11 23:33:18 +08:00
|
|
|
return matchSuccess();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2019-07-15 22:35:00 +08:00
|
|
|
class AffineForLowering : public OpRewritePattern<AffineForOp> {
|
2019-06-11 23:33:18 +08:00
|
|
|
public:
|
2019-07-15 22:35:00 +08:00
|
|
|
using OpRewritePattern<AffineForOp>::OpRewritePattern;
|
|
|
|
|
|
|
|
PatternMatchResult matchAndRewrite(AffineForOp op,
|
|
|
|
PatternRewriter &rewriter) const override {
|
|
|
|
Location loc = op.getLoc();
|
|
|
|
Value *lowerBound = lowerAffineLowerBound(op, rewriter);
|
|
|
|
Value *upperBound = lowerAffineUpperBound(op, rewriter);
|
|
|
|
Value *step = rewriter.create<ConstantIndexOp>(loc, op.getStep());
|
2019-07-15 17:50:09 +08:00
|
|
|
auto f = rewriter.create<loop::ForOp>(loc, lowerBound, upperBound, step);
|
2019-07-12 21:48:45 +08:00
|
|
|
f.region().getBlocks().clear();
|
2019-07-17 03:20:15 +08:00
|
|
|
rewriter.inlineRegionBefore(op.region(), f.region(), f.region().end());
|
2019-06-11 23:33:18 +08:00
|
|
|
rewriter.replaceOp(op, {});
|
|
|
|
return matchSuccess();
|
|
|
|
}
|
|
|
|
};
|
2018-12-31 08:22:50 +08:00
|
|
|
|
2019-07-15 22:35:00 +08:00
|
|
|
class AffineIfLowering : public OpRewritePattern<AffineIfOp> {
|
2019-06-11 23:33:18 +08:00
|
|
|
public:
|
2019-07-15 22:35:00 +08:00
|
|
|
using OpRewritePattern<AffineIfOp>::OpRewritePattern;
|
2019-06-11 23:33:18 +08:00
|
|
|
|
2019-07-15 22:35:00 +08:00
|
|
|
PatternMatchResult matchAndRewrite(AffineIfOp op,
|
|
|
|
PatternRewriter &rewriter) const override {
|
|
|
|
auto loc = op.getLoc();
|
2019-06-11 23:33:18 +08:00
|
|
|
|
|
|
|
// Now we just have to handle the condition logic.
|
2019-07-15 22:35:00 +08:00
|
|
|
auto integerSet = op.getIntegerSet();
|
2019-06-11 23:33:18 +08:00
|
|
|
Value *zeroConstant = rewriter.create<ConstantIndexOp>(loc, 0);
|
2019-07-17 03:20:15 +08:00
|
|
|
SmallVector<Value *, 8> operands(op.getOperation()->getOperands());
|
2019-07-15 22:35:00 +08:00
|
|
|
auto operandsRef = llvm::makeArrayRef(operands);
|
2019-06-11 23:33:18 +08:00
|
|
|
|
2019-07-12 21:48:45 +08:00
|
|
|
// Calculate cond as a conjunction without short-circuiting.
|
|
|
|
Value *cond = nullptr;
|
2019-06-11 23:33:18 +08:00
|
|
|
for (unsigned i = 0, e = integerSet.getNumConstraints(); i < e; ++i) {
|
|
|
|
AffineExpr constraintExpr = integerSet.getConstraint(i);
|
|
|
|
bool isEquality = integerSet.isEq(i);
|
|
|
|
|
|
|
|
// Build and apply an affine expression
|
|
|
|
auto numDims = integerSet.getNumDims();
|
2019-06-14 16:56:19 +08:00
|
|
|
Value *affResult = expandAffineExpr(rewriter, loc, constraintExpr,
|
2019-07-15 22:35:00 +08:00
|
|
|
operandsRef.take_front(numDims),
|
|
|
|
operandsRef.drop_front(numDims));
|
2019-06-11 23:33:18 +08:00
|
|
|
if (!affResult)
|
|
|
|
return matchFailure();
|
2019-07-12 21:48:45 +08:00
|
|
|
auto pred = isEquality ? CmpIPredicate::EQ : CmpIPredicate::SGE;
|
|
|
|
Value *cmpVal =
|
|
|
|
rewriter.create<CmpIOp>(loc, pred, affResult, zeroConstant);
|
|
|
|
cond =
|
|
|
|
cond ? rewriter.create<AndOp>(loc, cond, cmpVal).getResult() : cmpVal;
|
|
|
|
}
|
|
|
|
cond = cond ? cond
|
|
|
|
: rewriter.create<ConstantIntOp>(loc, /*value=*/1, /*width=*/1);
|
|
|
|
|
2019-07-17 03:20:15 +08:00
|
|
|
bool hasElseRegion = !op.elseRegion().empty();
|
2019-07-15 17:50:09 +08:00
|
|
|
auto ifOp = rewriter.create<loop::IfOp>(loc, cond, hasElseRegion);
|
2019-07-17 03:20:15 +08:00
|
|
|
rewriter.inlineRegionBefore(op.thenRegion(), &ifOp.thenRegion().back());
|
2019-07-12 21:48:45 +08:00
|
|
|
ifOp.thenRegion().back().erase();
|
|
|
|
if (hasElseRegion) {
|
2019-07-17 03:20:15 +08:00
|
|
|
rewriter.inlineRegionBefore(op.elseRegion(), &ifOp.elseRegion().back());
|
2019-07-12 21:48:45 +08:00
|
|
|
ifOp.elseRegion().back().erase();
|
2019-06-11 23:33:18 +08:00
|
|
|
}
|
2018-12-31 08:22:50 +08:00
|
|
|
|
2019-06-11 23:33:18 +08:00
|
|
|
// Ok, we're done!
|
|
|
|
rewriter.replaceOp(op, {});
|
|
|
|
return matchSuccess();
|
2018-12-31 08:22:50 +08:00
|
|
|
}
|
2019-06-11 23:33:18 +08:00
|
|
|
};
|
2018-12-31 08:22:50 +08:00
|
|
|
|
2019-02-07 03:08:18 +08:00
|
|
|
// Convert an "affine.apply" operation into a sequence of arithmetic
|
2019-06-11 23:33:18 +08:00
|
|
|
// operations using the StandardOps dialect.
|
2019-07-15 22:35:00 +08:00
|
|
|
class AffineApplyLowering : public OpRewritePattern<AffineApplyOp> {
|
2019-06-11 23:33:18 +08:00
|
|
|
public:
|
2019-07-15 22:35:00 +08:00
|
|
|
using OpRewritePattern<AffineApplyOp>::OpRewritePattern;
|
2019-06-11 23:33:18 +08:00
|
|
|
|
|
|
|
virtual PatternMatchResult
|
2019-07-15 22:35:00 +08:00
|
|
|
matchAndRewrite(AffineApplyOp op, PatternRewriter &rewriter) const override {
|
|
|
|
auto maybeExpandedMap =
|
|
|
|
expandAffineMap(rewriter, op.getLoc(), op.getAffineMap(),
|
|
|
|
llvm::to_vector<8>(op.getOperands()));
|
2019-06-11 23:33:18 +08:00
|
|
|
if (!maybeExpandedMap)
|
|
|
|
return matchFailure();
|
|
|
|
rewriter.replaceOp(op, *maybeExpandedMap);
|
|
|
|
return matchSuccess();
|
2019-02-02 08:42:18 +08:00
|
|
|
}
|
2019-06-11 23:33:18 +08:00
|
|
|
};
|
2019-07-01 23:32:44 +08:00
|
|
|
|
|
|
|
// Apply the affine map from an 'affine.load' operation to its operands, and
|
|
|
|
// feed the results to a newly created 'std.load' operation (which replaces the
|
|
|
|
// original 'affine.load').
|
2019-07-15 22:35:00 +08:00
|
|
|
class AffineLoadLowering : public OpRewritePattern<AffineLoadOp> {
|
2019-07-01 23:32:44 +08:00
|
|
|
public:
|
2019-07-15 22:35:00 +08:00
|
|
|
using OpRewritePattern<AffineLoadOp>::OpRewritePattern;
|
2019-07-01 23:32:44 +08:00
|
|
|
|
|
|
|
virtual PatternMatchResult
|
2019-07-15 22:35:00 +08:00
|
|
|
matchAndRewrite(AffineLoadOp op, PatternRewriter &rewriter) const override {
|
2019-07-01 23:32:44 +08:00
|
|
|
// Expand affine map from 'affineLoadOp'.
|
2019-07-15 22:35:00 +08:00
|
|
|
SmallVector<Value *, 8> indices(op.getIndices());
|
2019-07-01 23:32:44 +08:00
|
|
|
auto maybeExpandedMap =
|
2019-07-15 22:35:00 +08:00
|
|
|
expandAffineMap(rewriter, op.getLoc(), op.getAffineMap(), indices);
|
2019-07-01 23:32:44 +08:00
|
|
|
if (!maybeExpandedMap)
|
|
|
|
return matchFailure();
|
2019-07-15 22:35:00 +08:00
|
|
|
|
2019-07-01 23:32:44 +08:00
|
|
|
// Build std.load memref[expandedMap.results].
|
2019-07-15 22:35:00 +08:00
|
|
|
rewriter.replaceOpWithNewOp<LoadOp>(op, op.getMemRef(), *maybeExpandedMap);
|
2019-07-01 23:32:44 +08:00
|
|
|
return matchSuccess();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
// Apply the affine map from an 'affine.store' operation to its operands, and
|
|
|
|
// feed the results to a newly created 'std.store' operation (which replaces the
|
|
|
|
// original 'affine.store').
|
2019-07-15 22:35:00 +08:00
|
|
|
class AffineStoreLowering : public OpRewritePattern<AffineStoreOp> {
|
2019-07-01 23:32:44 +08:00
|
|
|
public:
|
2019-07-15 22:35:00 +08:00
|
|
|
using OpRewritePattern<AffineStoreOp>::OpRewritePattern;
|
2019-07-01 23:32:44 +08:00
|
|
|
|
|
|
|
virtual PatternMatchResult
|
2019-07-15 22:35:00 +08:00
|
|
|
matchAndRewrite(AffineStoreOp op, PatternRewriter &rewriter) const override {
|
2019-07-01 23:32:44 +08:00
|
|
|
// Expand affine map from 'affineStoreOp'.
|
2019-07-15 22:35:00 +08:00
|
|
|
SmallVector<Value *, 8> indices(op.getIndices());
|
2019-07-01 23:32:44 +08:00
|
|
|
auto maybeExpandedMap =
|
2019-07-15 22:35:00 +08:00
|
|
|
expandAffineMap(rewriter, op.getLoc(), op.getAffineMap(), indices);
|
2019-07-01 23:32:44 +08:00
|
|
|
if (!maybeExpandedMap)
|
|
|
|
return matchFailure();
|
2019-07-15 22:35:00 +08:00
|
|
|
|
2019-07-01 23:32:44 +08:00
|
|
|
// Build std.store valutToStore, memref[expandedMap.results].
|
2019-07-15 22:35:00 +08:00
|
|
|
rewriter.replaceOpWithNewOp<StoreOp>(op, op.getValueToStore(),
|
|
|
|
op.getMemRef(), *maybeExpandedMap);
|
2019-07-01 23:32:44 +08:00
|
|
|
return matchSuccess();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
// Apply the affine maps from an 'affine.dma_start' operation to each of their
|
|
|
|
// respective map operands, and feed the results to a newly created
|
|
|
|
// 'std.dma_start' operation (which replaces the original 'affine.dma_start').
|
2019-07-15 22:35:00 +08:00
|
|
|
class AffineDmaStartLowering : public OpRewritePattern<AffineDmaStartOp> {
|
2019-07-01 23:32:44 +08:00
|
|
|
public:
|
2019-07-15 22:35:00 +08:00
|
|
|
using OpRewritePattern<AffineDmaStartOp>::OpRewritePattern;
|
2019-07-01 23:32:44 +08:00
|
|
|
|
|
|
|
virtual PatternMatchResult
|
2019-07-15 22:35:00 +08:00
|
|
|
matchAndRewrite(AffineDmaStartOp op,
|
2019-07-01 23:32:44 +08:00
|
|
|
PatternRewriter &rewriter) const override {
|
2019-07-15 22:35:00 +08:00
|
|
|
SmallVector<Value *, 8> operands(op.getOperands());
|
|
|
|
auto operandsRef = llvm::makeArrayRef(operands);
|
|
|
|
|
2019-07-01 23:32:44 +08:00
|
|
|
// Expand affine map for DMA source memref.
|
|
|
|
auto maybeExpandedSrcMap = expandAffineMap(
|
2019-07-15 22:35:00 +08:00
|
|
|
rewriter, op.getLoc(), op.getSrcMap(),
|
|
|
|
operandsRef.drop_front(op.getSrcMemRefOperandIndex() + 1));
|
2019-07-01 23:32:44 +08:00
|
|
|
if (!maybeExpandedSrcMap)
|
|
|
|
return matchFailure();
|
|
|
|
// Expand affine map for DMA destination memref.
|
|
|
|
auto maybeExpandedDstMap = expandAffineMap(
|
2019-07-15 22:35:00 +08:00
|
|
|
rewriter, op.getLoc(), op.getDstMap(),
|
|
|
|
operandsRef.drop_front(op.getDstMemRefOperandIndex() + 1));
|
2019-07-01 23:32:44 +08:00
|
|
|
if (!maybeExpandedDstMap)
|
|
|
|
return matchFailure();
|
|
|
|
// Expand affine map for DMA tag memref.
|
|
|
|
auto maybeExpandedTagMap = expandAffineMap(
|
2019-07-15 22:35:00 +08:00
|
|
|
rewriter, op.getLoc(), op.getTagMap(),
|
|
|
|
operandsRef.drop_front(op.getTagMemRefOperandIndex() + 1));
|
2019-07-01 23:32:44 +08:00
|
|
|
if (!maybeExpandedTagMap)
|
|
|
|
return matchFailure();
|
|
|
|
|
|
|
|
// Build std.dma_start operation with affine map results.
|
|
|
|
rewriter.replaceOpWithNewOp<DmaStartOp>(
|
2019-07-15 22:35:00 +08:00
|
|
|
op, op.getSrcMemRef(), *maybeExpandedSrcMap, op.getDstMemRef(),
|
|
|
|
*maybeExpandedDstMap, op.getNumElements(), op.getTagMemRef(),
|
|
|
|
*maybeExpandedTagMap, op.getStride(), op.getNumElementsPerStride());
|
2019-07-01 23:32:44 +08:00
|
|
|
return matchSuccess();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
// Apply the affine map from an 'affine.dma_wait' operation tag memref,
|
|
|
|
// and feed the results to a newly created 'std.dma_wait' operation (which
|
|
|
|
// replaces the original 'affine.dma_wait').
|
2019-07-15 22:35:00 +08:00
|
|
|
class AffineDmaWaitLowering : public OpRewritePattern<AffineDmaWaitOp> {
|
2019-07-01 23:32:44 +08:00
|
|
|
public:
|
2019-07-15 22:35:00 +08:00
|
|
|
using OpRewritePattern<AffineDmaWaitOp>::OpRewritePattern;
|
2019-07-01 23:32:44 +08:00
|
|
|
|
|
|
|
virtual PatternMatchResult
|
2019-07-15 22:35:00 +08:00
|
|
|
matchAndRewrite(AffineDmaWaitOp op,
|
2019-07-01 23:32:44 +08:00
|
|
|
PatternRewriter &rewriter) const override {
|
|
|
|
// Expand affine map for DMA tag memref.
|
2019-07-15 22:35:00 +08:00
|
|
|
SmallVector<Value *, 8> indices(op.getTagIndices());
|
2019-07-01 23:32:44 +08:00
|
|
|
auto maybeExpandedTagMap =
|
2019-07-15 22:35:00 +08:00
|
|
|
expandAffineMap(rewriter, op.getLoc(), op.getTagMap(), indices);
|
2019-07-01 23:32:44 +08:00
|
|
|
if (!maybeExpandedTagMap)
|
|
|
|
return matchFailure();
|
|
|
|
|
|
|
|
// Build std.dma_wait operation with affine map results.
|
|
|
|
rewriter.replaceOpWithNewOp<DmaWaitOp>(
|
2019-07-15 22:35:00 +08:00
|
|
|
op, op.getTagMemRef(), *maybeExpandedTagMap, op.getNumElements());
|
2019-07-01 23:32:44 +08:00
|
|
|
return matchSuccess();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2019-06-11 23:33:18 +08:00
|
|
|
} // end namespace
|
2019-05-28 15:49:23 +08:00
|
|
|
|
2019-07-16 03:52:44 +08:00
|
|
|
void mlir::populateAffineToStdConversionPatterns(
|
|
|
|
OwningRewritePatternList &patterns, MLIRContext *ctx) {
|
2019-07-01 23:32:44 +08:00
|
|
|
RewriteListBuilder<AffineApplyLowering, AffineDmaStartLowering,
|
|
|
|
AffineDmaWaitLowering, AffineLoadLowering,
|
|
|
|
AffineStoreLowering, AffineForLowering, AffineIfLowering,
|
2019-07-16 03:52:44 +08:00
|
|
|
AffineTerminatorLowering>::build(patterns, ctx);
|
2019-05-28 15:49:23 +08:00
|
|
|
}
|
|
|
|
|
2019-06-11 23:33:18 +08:00
|
|
|
namespace {
|
|
|
|
class LowerAffinePass : public FunctionPass<LowerAffinePass> {
|
2019-07-16 03:52:44 +08:00
|
|
|
void runOnFunction() override {
|
|
|
|
OwningRewritePatternList patterns;
|
|
|
|
populateAffineToStdConversionPatterns(patterns, &getContext());
|
|
|
|
ConversionTarget target(getContext());
|
|
|
|
target.addLegalDialect<loop::LoopOpsDialect, StandardOpsDialect>();
|
2019-07-17 02:57:45 +08:00
|
|
|
if (failed(
|
|
|
|
applyPartialConversion(getFunction(), target, std::move(patterns))))
|
2019-07-16 03:52:44 +08:00
|
|
|
signalPassFailure();
|
|
|
|
}
|
2019-06-11 23:33:18 +08:00
|
|
|
};
|
|
|
|
} // namespace
|
2018-12-31 08:22:50 +08:00
|
|
|
|
2019-03-28 05:02:02 +08:00
|
|
|
/// Lowers If and For operations within a function into their lower level CFG
|
2018-12-31 08:22:50 +08:00
|
|
|
/// equivalent blocks.
|
2019-02-28 02:59:29 +08:00
|
|
|
FunctionPassBase *mlir::createLowerAffinePass() {
|
|
|
|
return new LowerAffinePass();
|
|
|
|
}
|
2018-12-31 08:22:50 +08:00
|
|
|
|
2019-01-03 04:52:41 +08:00
|
|
|
static PassRegistration<LowerAffinePass>
|
|
|
|
pass("lower-affine",
|
2019-03-28 05:02:02 +08:00
|
|
|
"Lower If, For, AffineApply operations to primitive equivalents");
|