//===- LegalizeToF32.cpp - Legalize functions on small floats ----------===// // // 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 // //===----------------------------------------------------------------------===// // // This file implements legalizing math operations on small floating-point // types through arith.extf and arith.truncf. // //===----------------------------------------------------------------------===// #include "mlir/Dialect/Arith/IR/Arith.h" #include "mlir/Dialect/Math/IR/Math.h" #include "mlir/Dialect/Math/Transforms/Passes.h" #include "mlir/IR/Diagnostics.h" #include "mlir/IR/PatternMatch.h" #include "mlir/IR/TypeUtilities.h" #include "mlir/Transforms/DialectConversion.h" #include "llvm/ADT/STLExtras.h" namespace mlir::math { #define GEN_PASS_DEF_MATHLEGALIZETOF32 #include "mlir/Dialect/Math/Transforms/Passes.h.inc" } // namespace mlir::math using namespace mlir; namespace { struct LegalizeToF32RewritePattern final : ConversionPattern { LegalizeToF32RewritePattern(TypeConverter &converter, MLIRContext *context) : ConversionPattern(converter, MatchAnyOpTypeTag{}, 1, context) {} LogicalResult matchAndRewrite(Operation *op, ArrayRef operands, ConversionPatternRewriter &rewriter) const override; }; struct LegalizeToF32Pass final : mlir::math::impl::MathLegalizeToF32Base { void runOnOperation() override; }; } // namespace void mlir::math::populateLegalizeToF32TypeConverter( TypeConverter &typeConverter) { typeConverter.addConversion( [](Type type) -> std::optional { return type; }); typeConverter.addConversion([](FloatType type) -> std::optional { if (type.getWidth() < 32) return Float32Type::get(type.getContext()); return std::nullopt; }); typeConverter.addConversion([](ShapedType type) -> std::optional { if (auto elemTy = dyn_cast(type.getElementType())) return type.clone(Float32Type::get(type.getContext())); return std::nullopt; }); typeConverter.addTargetMaterialization( [](OpBuilder &b, Type target, ValueRange input, Location loc) { return b.create(loc, target, input); }); } void mlir::math::populateLegalizeToF32ConversionTarget( ConversionTarget &target, TypeConverter &typeConverter) { target.addDynamicallyLegalDialect( [&typeConverter](Operation *op) -> bool { return typeConverter.isLegal(op); }); target.addLegalOp(); target.addLegalOp(); } LogicalResult LegalizeToF32RewritePattern::matchAndRewrite( Operation *op, ArrayRef operands, ConversionPatternRewriter &rewriter) const { Location loc = op->getLoc(); const TypeConverter *converter = getTypeConverter(); if (converter->isLegal(op)) return rewriter.notifyMatchFailure(loc, "op already legal"); OperationState newOp(loc, op->getName()); newOp.addOperands(operands); SmallVector newResultTypes; if (failed(converter->convertTypes(op->getResultTypes(), newResultTypes))) return rewriter.notifyMatchFailure(loc, "couldn't convert return types"); newOp.addTypes(newResultTypes); newOp.addAttributes(op->getAttrs()); Operation *legalized = rewriter.create(newOp); SmallVector results = legalized->getResults(); for (auto [result, newType, origType] : llvm::zip_equal(results, newResultTypes, op->getResultTypes())) { if (newType != origType) result = rewriter.create(loc, origType, result); } rewriter.replaceOp(op, results); return success(); } void mlir::math::populateLegalizeToF32Patterns(RewritePatternSet &patterns, TypeConverter &typeConverter) { patterns.add(typeConverter, patterns.getContext()); } void LegalizeToF32Pass::runOnOperation() { Operation *op = getOperation(); MLIRContext &ctx = getContext(); TypeConverter typeConverter; math::populateLegalizeToF32TypeConverter(typeConverter); ConversionTarget target(ctx); math::populateLegalizeToF32ConversionTarget(target, typeConverter); RewritePatternSet patterns(&ctx); math::populateLegalizeToF32Patterns(patterns, typeConverter); if (failed(applyPartialConversion(op, target, std::move(patterns)))) return signalPassFailure(); }