Files
clang-p2996/mlir/lib/Dialect/Shape/Transforms/ShapeToShapeLowering.cpp
River Riddle 3fffffa882 [mlir][Pattern] Add a new FrozenRewritePatternList class
This class represents a rewrite pattern list that has been frozen, and thus immutable. This replaces the uses of OwningRewritePatternList in pattern driver related API, such as dialect conversion. When PDL becomes more prevalent, this API will allow for optimizing a set of patterns once without the need to do this per run of a pass.

Differential Revision: https://reviews.llvm.org/D89104
2020-10-26 18:01:06 -07:00

83 lines
2.8 KiB
C++

//===- ShapeToShapeLowering.cpp - Prepare for lowering to Standard --------===//
//
// 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 "PassDetail.h"
#include "mlir/Dialect/Shape/IR/Shape.h"
#include "mlir/Dialect/Shape/Transforms/Passes.h"
#include "mlir/Dialect/StandardOps/IR/Ops.h"
#include "mlir/IR/Builders.h"
#include "mlir/IR/PatternMatch.h"
#include "mlir/Pass/Pass.h"
#include "mlir/Transforms/DialectConversion.h"
using namespace mlir;
using namespace mlir::shape;
namespace {
/// Converts `shape.num_elements` to `shape.reduce`.
struct NumElementsOpConverter : public OpRewritePattern<NumElementsOp> {
public:
using OpRewritePattern::OpRewritePattern;
LogicalResult matchAndRewrite(NumElementsOp op,
PatternRewriter &rewriter) const final;
};
} // namespace
LogicalResult
NumElementsOpConverter::matchAndRewrite(NumElementsOp op,
PatternRewriter &rewriter) const {
auto loc = op.getLoc();
Type valueType = op.getResult().getType();
Value init = op.getDialect()
->materializeConstant(rewriter, rewriter.getIndexAttr(1),
valueType, loc)
->getResult(0);
ReduceOp reduce = rewriter.create<ReduceOp>(loc, op.shape(), init);
// Generate reduce operator.
Block *body = reduce.getBody();
OpBuilder b = OpBuilder::atBlockEnd(body);
Value product = b.create<MulOp>(loc, valueType, body->getArgument(1),
body->getArgument(2));
b.create<shape::YieldOp>(loc, product);
rewriter.replaceOp(op, reduce.result());
return success();
}
namespace {
struct ShapeToShapeLowering
: public ShapeToShapeLoweringBase<ShapeToShapeLowering> {
void runOnFunction() override;
};
} // namespace
void ShapeToShapeLowering::runOnFunction() {
MLIRContext &ctx = getContext();
OwningRewritePatternList patterns;
populateShapeRewritePatterns(&ctx, patterns);
ConversionTarget target(getContext());
target.addLegalDialect<ShapeDialect, StandardOpsDialect>();
target.addIllegalOp<NumElementsOp>();
if (failed(mlir::applyPartialConversion(getFunction(), target,
std::move(patterns))))
signalPassFailure();
}
void mlir::populateShapeRewritePatterns(MLIRContext *context,
OwningRewritePatternList &patterns) {
patterns.insert<NumElementsOpConverter>(context);
}
std::unique_ptr<Pass> mlir::createShapeToShapeLowering() {
return std::make_unique<ShapeToShapeLowering>();
}