26#include <mlir/Dialect/SCF/Transforms/Patterns.h>
27#include <mlir/IR/Attributes.h>
28#include <mlir/IR/BuiltinAttributes.h>
29#include <mlir/IR/BuiltinTypes.h>
30#include <mlir/IR/MLIRContext.h>
31#include <mlir/IR/Operation.h>
32#include <mlir/IR/PatternMatch.h>
33#include <mlir/Transforms/DialectConversion.h>
35#include <llvm/ADT/SmallVector.h>
45 if (!tyConv.isLegal(op)) {
50 mlir::DictionaryAttr dictAttr = op->getAttrDictionary();
51 for (mlir::NamedAttribute n : dictAttr.getValue()) {
52 if (mlir::TypeAttr tyAttr = llvm::dyn_cast<mlir::TypeAttr>(n.getValue())) {
53 mlir::Type t = tyAttr.getValue();
54 if (mlir::FunctionType funcTy = llvm::dyn_cast<mlir::FunctionType>(t)) {
55 if (!tyConv.isSignatureLegal(funcTy)) {
59 if (!tyConv.isLegal(t)) {
70template <
typename OpClass,
typename Rewriter,
typename... Args>
72 mlir::DictionaryAttr attrs = op->getDiscardableAttrDictionary();
74 newOp->setDiscardableAttrs(attrs);
79static struct OpClassesWithStructTypes {
89 array::ExtractArrayOp,
90 constrain::EmitEqualityOp,
91 constrain::EmitContainmentOp,
92 component::MemberDefOp,
93 component::MemberReadOp,
94 component::MemberWriteOp,
95 component::CreateStructOp,
100 global::GlobalWriteOp,
101 polymorphic::UnifiableCastOp,
102 polymorphic::ConstReadOp
105 WithGeneralBuilder {};
112 const std::tuple<function::CallOp, array::CreateArrayOp> NoGeneralBuilder {};
114} OpClassesWithStructTypes;
124template <
typename OpClass>
125class GeneralTypeReplacePattern :
public mlir::OpConversionPattern<OpClass> {
129 GeneralTypeReplacePattern(mlir::TypeConverter &converter, mlir::MLIRContext *ctx)
130 : mlir::OpConversionPattern<OpClass>(converter, ctx, 0) {}
132 mlir::LogicalResult matchAndRewrite(
133 OpClass op,
typename OpClass::Adaptor adaptor, mlir::ConversionPatternRewriter &rewriter
135 const mlir::TypeConverter *converter = mlir::OpConversionPattern<OpClass>::getTypeConverter();
138 mlir::SmallVector<mlir::Type> newResultTypes;
139 if (mlir::failed(converter->convertTypes(op->getResultTypes(), newResultTypes))) {
140 return op->emitError(
"Could not convert Op result types.");
145 adaptor.getAttributes().empty() ||
147 adaptor.getAttributes(), [d = op->getAttrDictionary()](mlir::NamedAttribute a) {
148 return d.contains(a.getName());
153 mlir::SmallVector<mlir::NamedAttribute> newAttrs(op->getAttrDictionary().getValue());
154 for (mlir::NamedAttribute &n : newAttrs) {
155 if (mlir::TypeAttr t = llvm::dyn_cast<mlir::TypeAttr>(n.getValue())) {
156 if (mlir::Type newType = converter->convertType(t.getValue())) {
157 n.setValue(mlir::TypeAttr::get(newType));
159 return op->emitError().append(
"Could not convert type in attribute: ", t);
165 rewriter, op, mlir::TypeRange(newResultTypes), adaptor.getOperands(),
166 mlir::ArrayRef(newAttrs)
168 return mlir::success();
173class CreateArrayOpClassReplacePattern :
public mlir::OpConversionPattern<array::CreateArrayOp> {
177 CreateArrayOpClassReplacePattern(mlir::TypeConverter &converter, mlir::MLIRContext *ctx)
178 : mlir::OpConversionPattern<array::CreateArrayOp>(converter, ctx, 0) {}
180 mlir::LogicalResult match(array::CreateArrayOp op)
const override {
181 if (getTypeConverter()->convertType(op.getType())) {
182 return mlir::success();
184 return op->emitError(
"Could not convert Op result type.");
188 array::CreateArrayOp op, OpAdaptor adapter, mlir::ConversionPatternRewriter &rewriter
190 mlir::Type newType = getTypeConverter()->convertType(op.getType());
191 assert(llvm::isa<array::ArrayType>(newType) &&
"CreateArrayOp must produce ArrayType result");
192 mlir::DenseI32ArrayAttr numDimsPerMap = op.getNumDimsPerMapAttr();
195 rewriter, op, llvm::cast<array::ArrayType>(newType), adapter.getElements()
199 rewriter, op, llvm::cast<array::ArrayType>(newType), adapter.getMapOperands(),
207class CallOpClassReplacePattern :
public mlir::OpConversionPattern<function::CallOp> {
209 CallOpClassReplacePattern(mlir::TypeConverter &converter, mlir::MLIRContext *ctx)
210 : mlir::OpConversionPattern<function::CallOp>(converter, ctx, 0) {}
212 mlir::LogicalResult matchAndRewrite(
213 function::CallOp op, OpAdaptor adapter, mlir::ConversionPatternRewriter &rewriter
215 mlir::SmallVector<mlir::Type> newResultTypes;
216 if (mlir::failed(getTypeConverter()->convertTypes(op.getResultTypes(), newResultTypes))) {
217 return op->emitError(
"Could not convert Op result types.");
220 rewriter, op, newResultTypes, op.getCalleeAttr(), adapter.getMapOperands(),
221 op.getNumDimsPerMapAttr(), adapter.getArgOperands()
223 return mlir::success();
227template <
typename I,
typename NextOpClass,
typename... OtherOpClasses>
228inline void applyToMoreTypes(I inserter) {
229 std::apply(inserter, std::tuple<NextOpClass, OtherOpClasses...> {});
231template <
typename I>
inline void applyToMoreTypes(I) {}
239template <
typename... AdditionalOpClasses>
241 mlir::TypeConverter &tyConv, mlir::MLIRContext *ctx, mlir::ConversionTarget &target
243 mlir::RewritePatternSet patterns(ctx);
244 auto inserter = [&](
auto... opClasses) {
245 patterns.add<GeneralTypeReplacePattern<
decltype(opClasses)>...>(tyConv, ctx);
247 std::apply(inserter, OpClassesWithStructTypes.WithGeneralBuilder);
248 applyToMoreTypes<
decltype(inserter), AdditionalOpClasses...>(inserter);
250 patterns.add<CreateArrayOpClassReplacePattern, CallOpClassReplacePattern>(tyConv, ctx);
252 mlir::populateFunctionOpInterfaceTypeConversionPattern<function::FuncDefOp>(patterns, tyConv);
253 mlir::scf::populateSCFStructuralTypeConversionsAndLegality(tyConv, patterns, target);
OpClass replaceOpWithNewOp(Rewriter &rewriter, mlir::Operation *op, Args &&...args)
Wrapper for PatternRewriter::replaceOpWithNewOp() that automatically copies discardable attributes (i...
bool isNullOrEmpty(mlir::ArrayAttr a)
mlir::RewritePatternSet newGeneralRewritePatternSet(mlir::TypeConverter &tyConv, mlir::MLIRContext *ctx, mlir::ConversionTarget &target)
Return a new RewritePatternSet covering all LLZK op types that may contain a StructType.
bool defaultLegalityCheck(const mlir::TypeConverter &tyConv, mlir::Operation *op)
Check whether an op is legal with respect to the given type converter, including TypeAttr attributes ...