1044d5b5dSValentin Clement //===-- CodeGen.cpp -- bridge to lower to LLVM ----------------------------===// 2044d5b5dSValentin Clement // 3044d5b5dSValentin Clement // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 4044d5b5dSValentin Clement // See https://llvm.org/LICENSE.txt for license information. 5044d5b5dSValentin Clement // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 6044d5b5dSValentin Clement // 7044d5b5dSValentin Clement //===----------------------------------------------------------------------===// 8044d5b5dSValentin Clement // 9044d5b5dSValentin Clement // Coding style: https://mlir.llvm.org/getting_started/DeveloperGuide/ 10044d5b5dSValentin Clement // 11044d5b5dSValentin Clement //===----------------------------------------------------------------------===// 12044d5b5dSValentin Clement 13044d5b5dSValentin Clement #include "flang/Optimizer/CodeGen/CodeGen.h" 14044d5b5dSValentin Clement #include "PassDetail.h" 15044d5b5dSValentin Clement #include "flang/Optimizer/Dialect/FIROps.h" 16044d5b5dSValentin Clement #include "flang/Optimizer/Dialect/FIRType.h" 177b5132daSValentin Clement #include "flang/Optimizer/Support/FIRContext.h" 18044d5b5dSValentin Clement #include "mlir/Conversion/ArithmeticToLLVM/ArithmeticToLLVM.h" 19044d5b5dSValentin Clement #include "mlir/Conversion/LLVMCommon/Pattern.h" 20044d5b5dSValentin Clement #include "mlir/Conversion/LLVMCommon/TypeConverter.h" 21044d5b5dSValentin Clement #include "mlir/Conversion/StandardToLLVM/ConvertStandardToLLVM.h" 22044d5b5dSValentin Clement #include "mlir/IR/BuiltinTypes.h" 233ae8e442SValentin Clement #include "mlir/IR/Matchers.h" 24044d5b5dSValentin Clement #include "mlir/Pass/Pass.h" 25044d5b5dSValentin Clement #include "llvm/ADT/ArrayRef.h" 26044d5b5dSValentin Clement 27044d5b5dSValentin Clement #define DEBUG_TYPE "flang-codegen" 28044d5b5dSValentin Clement 29044d5b5dSValentin Clement // fir::LLVMTypeConverter for converting to LLVM IR dialect types. 30044d5b5dSValentin Clement #include "TypeConverter.h" 31044d5b5dSValentin Clement 32044d5b5dSValentin Clement namespace { 33044d5b5dSValentin Clement /// FIR conversion pattern template 34044d5b5dSValentin Clement template <typename FromOp> 35044d5b5dSValentin Clement class FIROpConversion : public mlir::ConvertOpToLLVMPattern<FromOp> { 36044d5b5dSValentin Clement public: 37044d5b5dSValentin Clement explicit FIROpConversion(fir::LLVMTypeConverter &lowering) 38044d5b5dSValentin Clement : mlir::ConvertOpToLLVMPattern<FromOp>(lowering) {} 39044d5b5dSValentin Clement 40044d5b5dSValentin Clement protected: 41044d5b5dSValentin Clement mlir::Type convertType(mlir::Type ty) const { 42044d5b5dSValentin Clement return lowerTy().convertType(ty); 43044d5b5dSValentin Clement } 44044d5b5dSValentin Clement 45*df3b9810SValentin Clement mlir::LLVM::ConstantOp 46*df3b9810SValentin Clement genConstantOffset(mlir::Location loc, 47*df3b9810SValentin Clement mlir::ConversionPatternRewriter &rewriter, 48*df3b9810SValentin Clement int offset) const { 49*df3b9810SValentin Clement auto ity = lowerTy().offsetType(); 50*df3b9810SValentin Clement auto cattr = rewriter.getI32IntegerAttr(offset); 51*df3b9810SValentin Clement return rewriter.create<mlir::LLVM::ConstantOp>(loc, ity, cattr); 52*df3b9810SValentin Clement } 53*df3b9810SValentin Clement 54*df3b9810SValentin Clement /// Construct code sequence to get the rank from a box. 55*df3b9810SValentin Clement mlir::Value getRankFromBox(mlir::Location loc, mlir::Value box, 56*df3b9810SValentin Clement mlir::Type resultTy, 57*df3b9810SValentin Clement mlir::ConversionPatternRewriter &rewriter) const { 58*df3b9810SValentin Clement mlir::LLVM::ConstantOp c0 = genConstantOffset(loc, rewriter, 0); 59*df3b9810SValentin Clement mlir::LLVM::ConstantOp cRank = 60*df3b9810SValentin Clement genConstantOffset(loc, rewriter, kRankPosInBox); 61*df3b9810SValentin Clement auto pty = mlir::LLVM::LLVMPointerType::get(resultTy); 62*df3b9810SValentin Clement auto p = rewriter.create<mlir::LLVM::GEPOp>( 63*df3b9810SValentin Clement loc, pty, mlir::ValueRange{box, c0, cRank}); 64*df3b9810SValentin Clement return rewriter.create<mlir::LLVM::LoadOp>(loc, resultTy, p); 65*df3b9810SValentin Clement } 66*df3b9810SValentin Clement 67*df3b9810SValentin Clement /// Method to construct code sequence to get the triple for dimension `dim` 68*df3b9810SValentin Clement /// from a box. 69*df3b9810SValentin Clement SmallVector<mlir::Value, 3> 70*df3b9810SValentin Clement getDimsFromBox(mlir::Location loc, ArrayRef<mlir::Type> retTys, 71*df3b9810SValentin Clement mlir::Value box, mlir::Value dim, 72*df3b9810SValentin Clement mlir::ConversionPatternRewriter &rewriter) const { 73*df3b9810SValentin Clement mlir::LLVM::ConstantOp c0 = genConstantOffset(loc, rewriter, 0); 74*df3b9810SValentin Clement mlir::LLVM::ConstantOp cDims = 75*df3b9810SValentin Clement genConstantOffset(loc, rewriter, kDimsPosInBox); 76*df3b9810SValentin Clement mlir::LLVM::LoadOp l0 = 77*df3b9810SValentin Clement loadFromOffset(loc, box, c0, cDims, dim, 0, retTys[0], rewriter); 78*df3b9810SValentin Clement mlir::LLVM::LoadOp l1 = 79*df3b9810SValentin Clement loadFromOffset(loc, box, c0, cDims, dim, 1, retTys[1], rewriter); 80*df3b9810SValentin Clement mlir::LLVM::LoadOp l2 = 81*df3b9810SValentin Clement loadFromOffset(loc, box, c0, cDims, dim, 2, retTys[2], rewriter); 82*df3b9810SValentin Clement return {l0.getResult(), l1.getResult(), l2.getResult()}; 83*df3b9810SValentin Clement } 84*df3b9810SValentin Clement 85*df3b9810SValentin Clement mlir::LLVM::LoadOp 86*df3b9810SValentin Clement loadFromOffset(mlir::Location loc, mlir::Value a, mlir::LLVM::ConstantOp c0, 87*df3b9810SValentin Clement mlir::LLVM::ConstantOp cDims, mlir::Value dim, int off, 88*df3b9810SValentin Clement mlir::Type ty, 89*df3b9810SValentin Clement mlir::ConversionPatternRewriter &rewriter) const { 90*df3b9810SValentin Clement auto pty = mlir::LLVM::LLVMPointerType::get(ty); 91*df3b9810SValentin Clement mlir::LLVM::ConstantOp c = genConstantOffset(loc, rewriter, off); 92*df3b9810SValentin Clement mlir::LLVM::GEPOp p = genGEP(loc, pty, rewriter, a, c0, cDims, dim, c); 93*df3b9810SValentin Clement return rewriter.create<mlir::LLVM::LoadOp>(loc, ty, p); 94*df3b9810SValentin Clement } 95*df3b9810SValentin Clement 96*df3b9810SValentin Clement /// Read base address from a fir.box. Returned address has type ty. 97*df3b9810SValentin Clement mlir::Value 98*df3b9810SValentin Clement loadBaseAddrFromBox(mlir::Location loc, mlir::Type ty, mlir::Value box, 99*df3b9810SValentin Clement mlir::ConversionPatternRewriter &rewriter) const { 100*df3b9810SValentin Clement mlir::LLVM::ConstantOp c0 = genConstantOffset(loc, rewriter, 0); 101*df3b9810SValentin Clement mlir::LLVM::ConstantOp cAddr = 102*df3b9810SValentin Clement genConstantOffset(loc, rewriter, kAddrPosInBox); 103*df3b9810SValentin Clement auto pty = mlir::LLVM::LLVMPointerType::get(ty); 104*df3b9810SValentin Clement mlir::LLVM::GEPOp p = genGEP(loc, pty, rewriter, box, c0, cAddr); 105*df3b9810SValentin Clement return rewriter.create<mlir::LLVM::LoadOp>(loc, ty, p); 106*df3b9810SValentin Clement } 107*df3b9810SValentin Clement 108*df3b9810SValentin Clement mlir::Value 109*df3b9810SValentin Clement loadElementSizeFromBox(mlir::Location loc, mlir::Type ty, mlir::Value box, 110*df3b9810SValentin Clement mlir::ConversionPatternRewriter &rewriter) const { 111*df3b9810SValentin Clement mlir::LLVM::ConstantOp c0 = genConstantOffset(loc, rewriter, 0); 112*df3b9810SValentin Clement mlir::LLVM::ConstantOp cElemLen = 113*df3b9810SValentin Clement genConstantOffset(loc, rewriter, kElemLenPosInBox); 114*df3b9810SValentin Clement auto pty = mlir::LLVM::LLVMPointerType::get(ty); 115*df3b9810SValentin Clement mlir::LLVM::GEPOp p = genGEP(loc, pty, rewriter, box, c0, cElemLen); 116*df3b9810SValentin Clement return rewriter.create<mlir::LLVM::LoadOp>(loc, ty, p); 117*df3b9810SValentin Clement } 118*df3b9810SValentin Clement 119*df3b9810SValentin Clement template <typename... ARGS> 120*df3b9810SValentin Clement mlir::LLVM::GEPOp genGEP(mlir::Location loc, mlir::Type ty, 121*df3b9810SValentin Clement mlir::ConversionPatternRewriter &rewriter, 122*df3b9810SValentin Clement mlir::Value base, ARGS... args) const { 123*df3b9810SValentin Clement SmallVector<mlir::Value> cv{args...}; 124*df3b9810SValentin Clement return rewriter.create<mlir::LLVM::GEPOp>(loc, ty, base, cv); 125*df3b9810SValentin Clement } 126*df3b9810SValentin Clement 127044d5b5dSValentin Clement fir::LLVMTypeConverter &lowerTy() const { 128044d5b5dSValentin Clement return *static_cast<fir::LLVMTypeConverter *>(this->getTypeConverter()); 129044d5b5dSValentin Clement } 130044d5b5dSValentin Clement }; 131044d5b5dSValentin Clement 1323ae8e442SValentin Clement /// FIR conversion pattern template 1333ae8e442SValentin Clement template <typename FromOp> 1343ae8e442SValentin Clement class FIROpAndTypeConversion : public FIROpConversion<FromOp> { 1353ae8e442SValentin Clement public: 1363ae8e442SValentin Clement using FIROpConversion<FromOp>::FIROpConversion; 1373ae8e442SValentin Clement using OpAdaptor = typename FromOp::Adaptor; 1383ae8e442SValentin Clement 1393ae8e442SValentin Clement mlir::LogicalResult 1403ae8e442SValentin Clement matchAndRewrite(FromOp op, OpAdaptor adaptor, 1413ae8e442SValentin Clement mlir::ConversionPatternRewriter &rewriter) const final { 1423ae8e442SValentin Clement mlir::Type ty = this->convertType(op.getType()); 1433ae8e442SValentin Clement return doRewrite(op, ty, adaptor, rewriter); 1443ae8e442SValentin Clement } 1453ae8e442SValentin Clement 1463ae8e442SValentin Clement virtual mlir::LogicalResult 1473ae8e442SValentin Clement doRewrite(FromOp addr, mlir::Type ty, OpAdaptor adaptor, 1483ae8e442SValentin Clement mlir::ConversionPatternRewriter &rewriter) const = 0; 1493ae8e442SValentin Clement }; 1503ae8e442SValentin Clement 1510c4a7a52SValentin Clement // Lower `fir.address_of` operation to `llvm.address_of` operation. 152044d5b5dSValentin Clement struct AddrOfOpConversion : public FIROpConversion<fir::AddrOfOp> { 153044d5b5dSValentin Clement using FIROpConversion::FIROpConversion; 154044d5b5dSValentin Clement 155044d5b5dSValentin Clement mlir::LogicalResult 156044d5b5dSValentin Clement matchAndRewrite(fir::AddrOfOp addr, OpAdaptor adaptor, 157044d5b5dSValentin Clement mlir::ConversionPatternRewriter &rewriter) const override { 158044d5b5dSValentin Clement auto ty = convertType(addr.getType()); 159044d5b5dSValentin Clement rewriter.replaceOpWithNewOp<mlir::LLVM::AddressOfOp>( 160044d5b5dSValentin Clement addr, ty, addr.symbol().getRootReference().getValue()); 161044d5b5dSValentin Clement return success(); 162044d5b5dSValentin Clement } 163044d5b5dSValentin Clement }; 164044d5b5dSValentin Clement 165*df3b9810SValentin Clement /// Lower `fir.box_addr` to the sequence of operations to extract the first 166*df3b9810SValentin Clement /// element of the box. 167*df3b9810SValentin Clement struct BoxAddrOpConversion : public FIROpConversion<fir::BoxAddrOp> { 168*df3b9810SValentin Clement using FIROpConversion::FIROpConversion; 169*df3b9810SValentin Clement 170*df3b9810SValentin Clement mlir::LogicalResult 171*df3b9810SValentin Clement matchAndRewrite(fir::BoxAddrOp boxaddr, OpAdaptor adaptor, 172*df3b9810SValentin Clement mlir::ConversionPatternRewriter &rewriter) const override { 173*df3b9810SValentin Clement mlir::Value a = adaptor.getOperands()[0]; 174*df3b9810SValentin Clement auto loc = boxaddr.getLoc(); 175*df3b9810SValentin Clement mlir::Type ty = convertType(boxaddr.getType()); 176*df3b9810SValentin Clement if (auto argty = boxaddr.val().getType().dyn_cast<fir::BoxType>()) { 177*df3b9810SValentin Clement rewriter.replaceOp(boxaddr, loadBaseAddrFromBox(loc, ty, a, rewriter)); 178*df3b9810SValentin Clement } else { 179*df3b9810SValentin Clement auto c0attr = rewriter.getI32IntegerAttr(0); 180*df3b9810SValentin Clement auto c0 = mlir::ArrayAttr::get(boxaddr.getContext(), c0attr); 181*df3b9810SValentin Clement rewriter.replaceOpWithNewOp<mlir::LLVM::ExtractValueOp>(boxaddr, ty, a, 182*df3b9810SValentin Clement c0); 183*df3b9810SValentin Clement } 184*df3b9810SValentin Clement return success(); 185*df3b9810SValentin Clement } 186*df3b9810SValentin Clement }; 187*df3b9810SValentin Clement 188*df3b9810SValentin Clement /// Lower `fir.box_dims` to a sequence of operations to extract the requested 189*df3b9810SValentin Clement /// dimension infomartion from the boxed value. 190*df3b9810SValentin Clement /// Result in a triple set of GEPs and loads. 191*df3b9810SValentin Clement struct BoxDimsOpConversion : public FIROpConversion<fir::BoxDimsOp> { 192*df3b9810SValentin Clement using FIROpConversion::FIROpConversion; 193*df3b9810SValentin Clement 194*df3b9810SValentin Clement mlir::LogicalResult 195*df3b9810SValentin Clement matchAndRewrite(fir::BoxDimsOp boxdims, OpAdaptor adaptor, 196*df3b9810SValentin Clement mlir::ConversionPatternRewriter &rewriter) const override { 197*df3b9810SValentin Clement SmallVector<mlir::Type, 3> resultTypes = { 198*df3b9810SValentin Clement convertType(boxdims.getResult(0).getType()), 199*df3b9810SValentin Clement convertType(boxdims.getResult(1).getType()), 200*df3b9810SValentin Clement convertType(boxdims.getResult(2).getType()), 201*df3b9810SValentin Clement }; 202*df3b9810SValentin Clement auto results = 203*df3b9810SValentin Clement getDimsFromBox(boxdims.getLoc(), resultTypes, adaptor.getOperands()[0], 204*df3b9810SValentin Clement adaptor.getOperands()[1], rewriter); 205*df3b9810SValentin Clement rewriter.replaceOp(boxdims, results); 206*df3b9810SValentin Clement return success(); 207*df3b9810SValentin Clement } 208*df3b9810SValentin Clement }; 209*df3b9810SValentin Clement 210*df3b9810SValentin Clement /// Lower `fir.box_elesize` to a sequence of operations ro extract the size of 211*df3b9810SValentin Clement /// an element in the boxed value. 212*df3b9810SValentin Clement struct BoxEleSizeOpConversion : public FIROpConversion<fir::BoxEleSizeOp> { 213*df3b9810SValentin Clement using FIROpConversion::FIROpConversion; 214*df3b9810SValentin Clement 215*df3b9810SValentin Clement mlir::LogicalResult 216*df3b9810SValentin Clement matchAndRewrite(fir::BoxEleSizeOp boxelesz, OpAdaptor adaptor, 217*df3b9810SValentin Clement mlir::ConversionPatternRewriter &rewriter) const override { 218*df3b9810SValentin Clement mlir::Value a = adaptor.getOperands()[0]; 219*df3b9810SValentin Clement auto loc = boxelesz.getLoc(); 220*df3b9810SValentin Clement auto ty = convertType(boxelesz.getType()); 221*df3b9810SValentin Clement rewriter.replaceOp(boxelesz, loadElementSizeFromBox(loc, ty, a, rewriter)); 222*df3b9810SValentin Clement return success(); 223*df3b9810SValentin Clement } 224*df3b9810SValentin Clement }; 225*df3b9810SValentin Clement 226*df3b9810SValentin Clement /// Lower `fir.box_rank` to the sequence of operation to extract the rank from 227*df3b9810SValentin Clement /// the box. 228*df3b9810SValentin Clement struct BoxRankOpConversion : public FIROpConversion<fir::BoxRankOp> { 229*df3b9810SValentin Clement using FIROpConversion::FIROpConversion; 230*df3b9810SValentin Clement 231*df3b9810SValentin Clement mlir::LogicalResult 232*df3b9810SValentin Clement matchAndRewrite(fir::BoxRankOp boxrank, OpAdaptor adaptor, 233*df3b9810SValentin Clement mlir::ConversionPatternRewriter &rewriter) const override { 234*df3b9810SValentin Clement mlir::Value a = adaptor.getOperands()[0]; 235*df3b9810SValentin Clement auto loc = boxrank.getLoc(); 236*df3b9810SValentin Clement mlir::Type ty = convertType(boxrank.getType()); 237*df3b9810SValentin Clement auto result = getRankFromBox(loc, a, ty, rewriter); 238*df3b9810SValentin Clement rewriter.replaceOp(boxrank, result); 239*df3b9810SValentin Clement return success(); 240*df3b9810SValentin Clement } 241*df3b9810SValentin Clement }; 242*df3b9810SValentin Clement 243ddd11b9aSAndrzej Warzynski // `fir.call` -> `llvm.call` 244ddd11b9aSAndrzej Warzynski struct CallOpConversion : public FIROpConversion<fir::CallOp> { 245ddd11b9aSAndrzej Warzynski using FIROpConversion::FIROpConversion; 246ddd11b9aSAndrzej Warzynski 247ddd11b9aSAndrzej Warzynski mlir::LogicalResult 248ddd11b9aSAndrzej Warzynski matchAndRewrite(fir::CallOp call, OpAdaptor adaptor, 249ddd11b9aSAndrzej Warzynski mlir::ConversionPatternRewriter &rewriter) const override { 250ddd11b9aSAndrzej Warzynski SmallVector<mlir::Type> resultTys; 251ddd11b9aSAndrzej Warzynski for (auto r : call.getResults()) 252ddd11b9aSAndrzej Warzynski resultTys.push_back(convertType(r.getType())); 253ddd11b9aSAndrzej Warzynski rewriter.replaceOpWithNewOp<mlir::LLVM::CallOp>( 254ddd11b9aSAndrzej Warzynski call, resultTys, adaptor.getOperands(), call->getAttrs()); 255ddd11b9aSAndrzej Warzynski return success(); 256ddd11b9aSAndrzej Warzynski } 257ddd11b9aSAndrzej Warzynski }; 258ddd11b9aSAndrzej Warzynski 259092cee5fSValentin Clement static mlir::Type getComplexEleTy(mlir::Type complex) { 260092cee5fSValentin Clement if (auto cc = complex.dyn_cast<mlir::ComplexType>()) 261092cee5fSValentin Clement return cc.getElementType(); 262092cee5fSValentin Clement return complex.cast<fir::ComplexType>().getElementType(); 263092cee5fSValentin Clement } 264092cee5fSValentin Clement 265092cee5fSValentin Clement /// convert value of from-type to value of to-type 266092cee5fSValentin Clement struct ConvertOpConversion : public FIROpConversion<fir::ConvertOp> { 267092cee5fSValentin Clement using FIROpConversion::FIROpConversion; 268092cee5fSValentin Clement 269092cee5fSValentin Clement static bool isFloatingPointTy(mlir::Type ty) { 270092cee5fSValentin Clement return ty.isa<mlir::FloatType>(); 271092cee5fSValentin Clement } 272092cee5fSValentin Clement 273092cee5fSValentin Clement mlir::LogicalResult 274092cee5fSValentin Clement matchAndRewrite(fir::ConvertOp convert, OpAdaptor adaptor, 275092cee5fSValentin Clement mlir::ConversionPatternRewriter &rewriter) const override { 276092cee5fSValentin Clement auto fromTy = convertType(convert.value().getType()); 277092cee5fSValentin Clement auto toTy = convertType(convert.res().getType()); 278092cee5fSValentin Clement mlir::Value op0 = adaptor.getOperands()[0]; 279092cee5fSValentin Clement if (fromTy == toTy) { 280092cee5fSValentin Clement rewriter.replaceOp(convert, op0); 281092cee5fSValentin Clement return success(); 282092cee5fSValentin Clement } 283092cee5fSValentin Clement auto loc = convert.getLoc(); 284092cee5fSValentin Clement auto convertFpToFp = [&](mlir::Value val, unsigned fromBits, 285092cee5fSValentin Clement unsigned toBits, mlir::Type toTy) -> mlir::Value { 286092cee5fSValentin Clement if (fromBits == toBits) { 287092cee5fSValentin Clement // TODO: Converting between two floating-point representations with the 288092cee5fSValentin Clement // same bitwidth is not allowed for now. 289092cee5fSValentin Clement mlir::emitError(loc, 290092cee5fSValentin Clement "cannot implicitly convert between two floating-point " 291092cee5fSValentin Clement "representations of the same bitwidth"); 292092cee5fSValentin Clement return {}; 293092cee5fSValentin Clement } 294092cee5fSValentin Clement if (fromBits > toBits) 295092cee5fSValentin Clement return rewriter.create<mlir::LLVM::FPTruncOp>(loc, toTy, val); 296092cee5fSValentin Clement return rewriter.create<mlir::LLVM::FPExtOp>(loc, toTy, val); 297092cee5fSValentin Clement }; 298092cee5fSValentin Clement // Complex to complex conversion. 299092cee5fSValentin Clement if (fir::isa_complex(convert.value().getType()) && 300092cee5fSValentin Clement fir::isa_complex(convert.res().getType())) { 301092cee5fSValentin Clement // Special case: handle the conversion of a complex such that both the 302092cee5fSValentin Clement // real and imaginary parts are converted together. 303092cee5fSValentin Clement auto zero = mlir::ArrayAttr::get(convert.getContext(), 304092cee5fSValentin Clement rewriter.getI32IntegerAttr(0)); 305092cee5fSValentin Clement auto one = mlir::ArrayAttr::get(convert.getContext(), 306092cee5fSValentin Clement rewriter.getI32IntegerAttr(1)); 307092cee5fSValentin Clement auto ty = convertType(getComplexEleTy(convert.value().getType())); 308092cee5fSValentin Clement auto rp = rewriter.create<mlir::LLVM::ExtractValueOp>(loc, ty, op0, zero); 309092cee5fSValentin Clement auto ip = rewriter.create<mlir::LLVM::ExtractValueOp>(loc, ty, op0, one); 310092cee5fSValentin Clement auto nt = convertType(getComplexEleTy(convert.res().getType())); 311092cee5fSValentin Clement auto fromBits = mlir::LLVM::getPrimitiveTypeSizeInBits(ty); 312092cee5fSValentin Clement auto toBits = mlir::LLVM::getPrimitiveTypeSizeInBits(nt); 313092cee5fSValentin Clement auto rc = convertFpToFp(rp, fromBits, toBits, nt); 314092cee5fSValentin Clement auto ic = convertFpToFp(ip, fromBits, toBits, nt); 315092cee5fSValentin Clement auto un = rewriter.create<mlir::LLVM::UndefOp>(loc, toTy); 316092cee5fSValentin Clement auto i1 = 317092cee5fSValentin Clement rewriter.create<mlir::LLVM::InsertValueOp>(loc, toTy, un, rc, zero); 318092cee5fSValentin Clement rewriter.replaceOpWithNewOp<mlir::LLVM::InsertValueOp>(convert, toTy, i1, 319092cee5fSValentin Clement ic, one); 320092cee5fSValentin Clement return mlir::success(); 321092cee5fSValentin Clement } 322092cee5fSValentin Clement // Floating point to floating point conversion. 323092cee5fSValentin Clement if (isFloatingPointTy(fromTy)) { 324092cee5fSValentin Clement if (isFloatingPointTy(toTy)) { 325092cee5fSValentin Clement auto fromBits = mlir::LLVM::getPrimitiveTypeSizeInBits(fromTy); 326092cee5fSValentin Clement auto toBits = mlir::LLVM::getPrimitiveTypeSizeInBits(toTy); 327092cee5fSValentin Clement auto v = convertFpToFp(op0, fromBits, toBits, toTy); 328092cee5fSValentin Clement rewriter.replaceOp(convert, v); 329092cee5fSValentin Clement return mlir::success(); 330092cee5fSValentin Clement } 331092cee5fSValentin Clement if (toTy.isa<mlir::IntegerType>()) { 332092cee5fSValentin Clement rewriter.replaceOpWithNewOp<mlir::LLVM::FPToSIOp>(convert, toTy, op0); 333092cee5fSValentin Clement return mlir::success(); 334092cee5fSValentin Clement } 335092cee5fSValentin Clement } else if (fromTy.isa<mlir::IntegerType>()) { 336092cee5fSValentin Clement // Integer to integer conversion. 337092cee5fSValentin Clement if (toTy.isa<mlir::IntegerType>()) { 338092cee5fSValentin Clement auto fromBits = mlir::LLVM::getPrimitiveTypeSizeInBits(fromTy); 339092cee5fSValentin Clement auto toBits = mlir::LLVM::getPrimitiveTypeSizeInBits(toTy); 340092cee5fSValentin Clement assert(fromBits != toBits); 341092cee5fSValentin Clement if (fromBits > toBits) { 342092cee5fSValentin Clement rewriter.replaceOpWithNewOp<mlir::LLVM::TruncOp>(convert, toTy, op0); 343092cee5fSValentin Clement return mlir::success(); 344092cee5fSValentin Clement } 345092cee5fSValentin Clement rewriter.replaceOpWithNewOp<mlir::LLVM::SExtOp>(convert, toTy, op0); 346092cee5fSValentin Clement return mlir::success(); 347092cee5fSValentin Clement } 348092cee5fSValentin Clement // Integer to floating point conversion. 349092cee5fSValentin Clement if (isFloatingPointTy(toTy)) { 350092cee5fSValentin Clement rewriter.replaceOpWithNewOp<mlir::LLVM::SIToFPOp>(convert, toTy, op0); 351092cee5fSValentin Clement return mlir::success(); 352092cee5fSValentin Clement } 353092cee5fSValentin Clement // Integer to pointer conversion. 354092cee5fSValentin Clement if (toTy.isa<mlir::LLVM::LLVMPointerType>()) { 355092cee5fSValentin Clement rewriter.replaceOpWithNewOp<mlir::LLVM::IntToPtrOp>(convert, toTy, op0); 356092cee5fSValentin Clement return mlir::success(); 357092cee5fSValentin Clement } 358092cee5fSValentin Clement } else if (fromTy.isa<mlir::LLVM::LLVMPointerType>()) { 359092cee5fSValentin Clement // Pointer to integer conversion. 360092cee5fSValentin Clement if (toTy.isa<mlir::IntegerType>()) { 361092cee5fSValentin Clement rewriter.replaceOpWithNewOp<mlir::LLVM::PtrToIntOp>(convert, toTy, op0); 362092cee5fSValentin Clement return mlir::success(); 363092cee5fSValentin Clement } 364092cee5fSValentin Clement // Pointer to pointer conversion. 365092cee5fSValentin Clement if (toTy.isa<mlir::LLVM::LLVMPointerType>()) { 366092cee5fSValentin Clement rewriter.replaceOpWithNewOp<mlir::LLVM::BitcastOp>(convert, toTy, op0); 367092cee5fSValentin Clement return mlir::success(); 368092cee5fSValentin Clement } 369092cee5fSValentin Clement } 370092cee5fSValentin Clement return emitError(loc) << "cannot convert " << fromTy << " to " << toTy; 371092cee5fSValentin Clement } 372092cee5fSValentin Clement }; 373092cee5fSValentin Clement 3740c4a7a52SValentin Clement /// Lower `fir.has_value` operation to `llvm.return` operation. 375044d5b5dSValentin Clement struct HasValueOpConversion : public FIROpConversion<fir::HasValueOp> { 376044d5b5dSValentin Clement using FIROpConversion::FIROpConversion; 377044d5b5dSValentin Clement 378044d5b5dSValentin Clement mlir::LogicalResult 379044d5b5dSValentin Clement matchAndRewrite(fir::HasValueOp op, OpAdaptor adaptor, 380044d5b5dSValentin Clement mlir::ConversionPatternRewriter &rewriter) const override { 381044d5b5dSValentin Clement rewriter.replaceOpWithNewOp<LLVM::ReturnOp>(op, adaptor.getOperands()); 382044d5b5dSValentin Clement return success(); 383044d5b5dSValentin Clement } 384044d5b5dSValentin Clement }; 385044d5b5dSValentin Clement 3860c4a7a52SValentin Clement /// Lower `fir.global` operation to `llvm.global` operation. 3870c4a7a52SValentin Clement /// `fir.insert_on_range` operations are replaced with constant dense attribute 3880c4a7a52SValentin Clement /// if they are applied on the full range. 389044d5b5dSValentin Clement struct GlobalOpConversion : public FIROpConversion<fir::GlobalOp> { 390044d5b5dSValentin Clement using FIROpConversion::FIROpConversion; 391044d5b5dSValentin Clement 392044d5b5dSValentin Clement mlir::LogicalResult 393044d5b5dSValentin Clement matchAndRewrite(fir::GlobalOp global, OpAdaptor adaptor, 394044d5b5dSValentin Clement mlir::ConversionPatternRewriter &rewriter) const override { 395044d5b5dSValentin Clement auto tyAttr = convertType(global.getType()); 396044d5b5dSValentin Clement if (global.getType().isa<fir::BoxType>()) 397044d5b5dSValentin Clement tyAttr = tyAttr.cast<mlir::LLVM::LLVMPointerType>().getElementType(); 398044d5b5dSValentin Clement auto loc = global.getLoc(); 399044d5b5dSValentin Clement mlir::Attribute initAttr{}; 400044d5b5dSValentin Clement if (global.initVal()) 401044d5b5dSValentin Clement initAttr = global.initVal().getValue(); 402044d5b5dSValentin Clement auto linkage = convertLinkage(global.linkName()); 403044d5b5dSValentin Clement auto isConst = global.constant().hasValue(); 404044d5b5dSValentin Clement auto g = rewriter.create<mlir::LLVM::GlobalOp>( 405044d5b5dSValentin Clement loc, tyAttr, isConst, linkage, global.sym_name(), initAttr); 406044d5b5dSValentin Clement auto &gr = g.getInitializerRegion(); 407044d5b5dSValentin Clement rewriter.inlineRegionBefore(global.region(), gr, gr.end()); 408044d5b5dSValentin Clement if (!gr.empty()) { 409044d5b5dSValentin Clement // Replace insert_on_range with a constant dense attribute if the 410044d5b5dSValentin Clement // initialization is on the full range. 411044d5b5dSValentin Clement auto insertOnRangeOps = gr.front().getOps<fir::InsertOnRangeOp>(); 412044d5b5dSValentin Clement for (auto insertOp : insertOnRangeOps) { 413044d5b5dSValentin Clement if (isFullRange(insertOp.coor(), insertOp.getType())) { 414044d5b5dSValentin Clement auto seqTyAttr = convertType(insertOp.getType()); 415044d5b5dSValentin Clement auto *op = insertOp.val().getDefiningOp(); 416044d5b5dSValentin Clement auto constant = mlir::dyn_cast<mlir::arith::ConstantOp>(op); 417044d5b5dSValentin Clement if (!constant) { 418044d5b5dSValentin Clement auto convertOp = mlir::dyn_cast<fir::ConvertOp>(op); 419044d5b5dSValentin Clement if (!convertOp) 420044d5b5dSValentin Clement continue; 421044d5b5dSValentin Clement constant = cast<mlir::arith::ConstantOp>( 422044d5b5dSValentin Clement convertOp.value().getDefiningOp()); 423044d5b5dSValentin Clement } 424044d5b5dSValentin Clement mlir::Type vecType = mlir::VectorType::get( 425044d5b5dSValentin Clement insertOp.getType().getShape(), constant.getType()); 426044d5b5dSValentin Clement auto denseAttr = mlir::DenseElementsAttr::get( 427044d5b5dSValentin Clement vecType.cast<ShapedType>(), constant.value()); 428044d5b5dSValentin Clement rewriter.setInsertionPointAfter(insertOp); 429044d5b5dSValentin Clement rewriter.replaceOpWithNewOp<mlir::arith::ConstantOp>( 430044d5b5dSValentin Clement insertOp, seqTyAttr, denseAttr); 431044d5b5dSValentin Clement } 432044d5b5dSValentin Clement } 433044d5b5dSValentin Clement } 434044d5b5dSValentin Clement rewriter.eraseOp(global); 435044d5b5dSValentin Clement return success(); 436044d5b5dSValentin Clement } 437044d5b5dSValentin Clement 438044d5b5dSValentin Clement bool isFullRange(mlir::ArrayAttr indexes, fir::SequenceType seqTy) const { 439044d5b5dSValentin Clement auto extents = seqTy.getShape(); 440044d5b5dSValentin Clement if (indexes.size() / 2 != extents.size()) 441044d5b5dSValentin Clement return false; 442044d5b5dSValentin Clement for (unsigned i = 0; i < indexes.size(); i += 2) { 443044d5b5dSValentin Clement if (indexes[i].cast<IntegerAttr>().getInt() != 0) 444044d5b5dSValentin Clement return false; 445044d5b5dSValentin Clement if (indexes[i + 1].cast<IntegerAttr>().getInt() != extents[i / 2] - 1) 446044d5b5dSValentin Clement return false; 447044d5b5dSValentin Clement } 448044d5b5dSValentin Clement return true; 449044d5b5dSValentin Clement } 450044d5b5dSValentin Clement 4510c4a7a52SValentin Clement // TODO: String comparaison should be avoided. Replace linkName with an 4520c4a7a52SValentin Clement // enumeration. 453044d5b5dSValentin Clement mlir::LLVM::Linkage convertLinkage(Optional<StringRef> optLinkage) const { 454044d5b5dSValentin Clement if (optLinkage.hasValue()) { 455044d5b5dSValentin Clement auto name = optLinkage.getValue(); 456044d5b5dSValentin Clement if (name == "internal") 457044d5b5dSValentin Clement return mlir::LLVM::Linkage::Internal; 458044d5b5dSValentin Clement if (name == "linkonce") 459044d5b5dSValentin Clement return mlir::LLVM::Linkage::Linkonce; 460044d5b5dSValentin Clement if (name == "common") 461044d5b5dSValentin Clement return mlir::LLVM::Linkage::Common; 462044d5b5dSValentin Clement if (name == "weak") 463044d5b5dSValentin Clement return mlir::LLVM::Linkage::Weak; 464044d5b5dSValentin Clement } 465044d5b5dSValentin Clement return mlir::LLVM::Linkage::External; 466044d5b5dSValentin Clement } 467044d5b5dSValentin Clement }; 468044d5b5dSValentin Clement 4698c239909SValentin Clement template <typename OP> 4708c239909SValentin Clement void selectMatchAndRewrite(fir::LLVMTypeConverter &lowering, OP select, 4718c239909SValentin Clement typename OP::Adaptor adaptor, 4728c239909SValentin Clement mlir::ConversionPatternRewriter &rewriter) { 4738c239909SValentin Clement unsigned conds = select.getNumConditions(); 4748c239909SValentin Clement auto cases = select.getCases().getValue(); 4758c239909SValentin Clement mlir::Value selector = adaptor.selector(); 4768c239909SValentin Clement auto loc = select.getLoc(); 4778c239909SValentin Clement assert(conds > 0 && "select must have cases"); 4788c239909SValentin Clement 4798c239909SValentin Clement llvm::SmallVector<mlir::Block *> destinations; 4808c239909SValentin Clement llvm::SmallVector<mlir::ValueRange> destinationsOperands; 4818c239909SValentin Clement mlir::Block *defaultDestination; 4828c239909SValentin Clement mlir::ValueRange defaultOperands; 4838c239909SValentin Clement llvm::SmallVector<int32_t> caseValues; 4848c239909SValentin Clement 4858c239909SValentin Clement for (unsigned t = 0; t != conds; ++t) { 4868c239909SValentin Clement mlir::Block *dest = select.getSuccessor(t); 4878c239909SValentin Clement auto destOps = select.getSuccessorOperands(adaptor.getOperands(), t); 4888c239909SValentin Clement const mlir::Attribute &attr = cases[t]; 4898c239909SValentin Clement if (auto intAttr = attr.template dyn_cast<mlir::IntegerAttr>()) { 4908c239909SValentin Clement destinations.push_back(dest); 4918c239909SValentin Clement destinationsOperands.push_back(destOps.hasValue() ? *destOps 4928c239909SValentin Clement : ValueRange()); 4938c239909SValentin Clement caseValues.push_back(intAttr.getInt()); 4948c239909SValentin Clement continue; 4958c239909SValentin Clement } 4968c239909SValentin Clement assert(attr.template dyn_cast_or_null<mlir::UnitAttr>()); 4978c239909SValentin Clement assert((t + 1 == conds) && "unit must be last"); 4988c239909SValentin Clement defaultDestination = dest; 4998c239909SValentin Clement defaultOperands = destOps.hasValue() ? *destOps : ValueRange(); 5008c239909SValentin Clement } 5018c239909SValentin Clement 5028c239909SValentin Clement // LLVM::SwitchOp takes a i32 type for the selector. 5038c239909SValentin Clement if (select.getSelector().getType() != rewriter.getI32Type()) 5048c239909SValentin Clement selector = 5058c239909SValentin Clement rewriter.create<LLVM::TruncOp>(loc, rewriter.getI32Type(), selector); 5068c239909SValentin Clement 5078c239909SValentin Clement rewriter.replaceOpWithNewOp<mlir::LLVM::SwitchOp>( 5088c239909SValentin Clement select, selector, 5098c239909SValentin Clement /*defaultDestination=*/defaultDestination, 5108c239909SValentin Clement /*defaultOperands=*/defaultOperands, 5118c239909SValentin Clement /*caseValues=*/caseValues, 5128c239909SValentin Clement /*caseDestinations=*/destinations, 5138c239909SValentin Clement /*caseOperands=*/destinationsOperands, 5148c239909SValentin Clement /*branchWeights=*/ArrayRef<int32_t>()); 5158c239909SValentin Clement } 5168c239909SValentin Clement 5178c239909SValentin Clement /// conversion of fir::SelectOp to an if-then-else ladder 5188c239909SValentin Clement struct SelectOpConversion : public FIROpConversion<fir::SelectOp> { 5198c239909SValentin Clement using FIROpConversion::FIROpConversion; 5208c239909SValentin Clement 5218c239909SValentin Clement mlir::LogicalResult 5228c239909SValentin Clement matchAndRewrite(fir::SelectOp op, OpAdaptor adaptor, 5238c239909SValentin Clement mlir::ConversionPatternRewriter &rewriter) const override { 5248c239909SValentin Clement selectMatchAndRewrite<fir::SelectOp>(lowerTy(), op, adaptor, rewriter); 5258c239909SValentin Clement return success(); 5268c239909SValentin Clement } 5278c239909SValentin Clement }; 5288c239909SValentin Clement 529e3349fa1SAndrzej Warzynski /// `fir.load` --> `llvm.load` 530e3349fa1SAndrzej Warzynski struct LoadOpConversion : public FIROpConversion<fir::LoadOp> { 531e3349fa1SAndrzej Warzynski using FIROpConversion::FIROpConversion; 532e3349fa1SAndrzej Warzynski 533e3349fa1SAndrzej Warzynski mlir::LogicalResult 534e3349fa1SAndrzej Warzynski matchAndRewrite(fir::LoadOp load, OpAdaptor adaptor, 535e3349fa1SAndrzej Warzynski mlir::ConversionPatternRewriter &rewriter) const override { 536e3349fa1SAndrzej Warzynski // fir.box is a special case because it is considered as an ssa values in 537e3349fa1SAndrzej Warzynski // fir, but it is lowered as a pointer to a descriptor. So fir.ref<fir.box> 538e3349fa1SAndrzej Warzynski // and fir.box end up being the same llvm types and loading a 539e3349fa1SAndrzej Warzynski // fir.ref<fir.box> is actually a no op in LLVM. 540e3349fa1SAndrzej Warzynski if (load.getType().isa<fir::BoxType>()) { 541e3349fa1SAndrzej Warzynski rewriter.replaceOp(load, adaptor.getOperands()[0]); 542e3349fa1SAndrzej Warzynski } else { 543e3349fa1SAndrzej Warzynski mlir::Type ty = convertType(load.getType()); 544e3349fa1SAndrzej Warzynski ArrayRef<NamedAttribute> at = load->getAttrs(); 545e3349fa1SAndrzej Warzynski rewriter.replaceOpWithNewOp<mlir::LLVM::LoadOp>( 546e3349fa1SAndrzej Warzynski load, ty, adaptor.getOperands(), at); 547e3349fa1SAndrzej Warzynski } 548e3349fa1SAndrzej Warzynski return success(); 549e3349fa1SAndrzej Warzynski } 550e3349fa1SAndrzej Warzynski }; 551e3349fa1SAndrzej Warzynski 5528c239909SValentin Clement /// conversion of fir::SelectRankOp to an if-then-else ladder 5538c239909SValentin Clement struct SelectRankOpConversion : public FIROpConversion<fir::SelectRankOp> { 5548c239909SValentin Clement using FIROpConversion::FIROpConversion; 5558c239909SValentin Clement 5568c239909SValentin Clement mlir::LogicalResult 5578c239909SValentin Clement matchAndRewrite(fir::SelectRankOp op, OpAdaptor adaptor, 5588c239909SValentin Clement mlir::ConversionPatternRewriter &rewriter) const override { 5598c239909SValentin Clement selectMatchAndRewrite<fir::SelectRankOp>(lowerTy(), op, adaptor, rewriter); 5608c239909SValentin Clement return success(); 5618c239909SValentin Clement } 5628c239909SValentin Clement }; 5638c239909SValentin Clement 564e3349fa1SAndrzej Warzynski /// `fir.store` --> `llvm.store` 565e3349fa1SAndrzej Warzynski struct StoreOpConversion : public FIROpConversion<fir::StoreOp> { 566e3349fa1SAndrzej Warzynski using FIROpConversion::FIROpConversion; 567e3349fa1SAndrzej Warzynski 568e3349fa1SAndrzej Warzynski mlir::LogicalResult 569e3349fa1SAndrzej Warzynski matchAndRewrite(fir::StoreOp store, OpAdaptor adaptor, 570e3349fa1SAndrzej Warzynski mlir::ConversionPatternRewriter &rewriter) const override { 571e3349fa1SAndrzej Warzynski if (store.value().getType().isa<fir::BoxType>()) { 572e3349fa1SAndrzej Warzynski // fir.box value is actually in memory, load it first before storing it. 573e3349fa1SAndrzej Warzynski mlir::Location loc = store.getLoc(); 574e3349fa1SAndrzej Warzynski mlir::Type boxPtrTy = adaptor.getOperands()[0].getType(); 575e3349fa1SAndrzej Warzynski auto val = rewriter.create<mlir::LLVM::LoadOp>( 576e3349fa1SAndrzej Warzynski loc, boxPtrTy.cast<mlir::LLVM::LLVMPointerType>().getElementType(), 577e3349fa1SAndrzej Warzynski adaptor.getOperands()[0]); 578e3349fa1SAndrzej Warzynski rewriter.replaceOpWithNewOp<mlir::LLVM::StoreOp>( 579e3349fa1SAndrzej Warzynski store, val, adaptor.getOperands()[1]); 580e3349fa1SAndrzej Warzynski } else { 581e3349fa1SAndrzej Warzynski rewriter.replaceOpWithNewOp<mlir::LLVM::StoreOp>( 582e3349fa1SAndrzej Warzynski store, adaptor.getOperands()[0], adaptor.getOperands()[1]); 583e3349fa1SAndrzej Warzynski } 584e3349fa1SAndrzej Warzynski return success(); 585e3349fa1SAndrzej Warzynski } 586e3349fa1SAndrzej Warzynski }; 587e3349fa1SAndrzej Warzynski 588e3349fa1SAndrzej Warzynski /// convert to LLVM IR dialect `undef` 589044d5b5dSValentin Clement struct UndefOpConversion : public FIROpConversion<fir::UndefOp> { 590044d5b5dSValentin Clement using FIROpConversion::FIROpConversion; 591044d5b5dSValentin Clement 592044d5b5dSValentin Clement mlir::LogicalResult 593044d5b5dSValentin Clement matchAndRewrite(fir::UndefOp undef, OpAdaptor, 594044d5b5dSValentin Clement mlir::ConversionPatternRewriter &rewriter) const override { 595044d5b5dSValentin Clement rewriter.replaceOpWithNewOp<mlir::LLVM::UndefOp>( 596044d5b5dSValentin Clement undef, convertType(undef.getType())); 597044d5b5dSValentin Clement return success(); 598044d5b5dSValentin Clement } 599044d5b5dSValentin Clement }; 600a7a61359SValentin Clement 601e3349fa1SAndrzej Warzynski /// `fir.unreachable` --> `llvm.unreachable` 60232e08248SAndrzej Warzynski struct UnreachableOpConversion : public FIROpConversion<fir::UnreachableOp> { 60332e08248SAndrzej Warzynski using FIROpConversion::FIROpConversion; 60432e08248SAndrzej Warzynski 60532e08248SAndrzej Warzynski mlir::LogicalResult 60632e08248SAndrzej Warzynski matchAndRewrite(fir::UnreachableOp unreach, OpAdaptor adaptor, 60732e08248SAndrzej Warzynski mlir::ConversionPatternRewriter &rewriter) const override { 60832e08248SAndrzej Warzynski rewriter.replaceOpWithNewOp<mlir::LLVM::UnreachableOp>(unreach); 60932e08248SAndrzej Warzynski return success(); 61032e08248SAndrzej Warzynski } 61132e08248SAndrzej Warzynski }; 61232e08248SAndrzej Warzynski 613a7a61359SValentin Clement struct ZeroOpConversion : public FIROpConversion<fir::ZeroOp> { 614a7a61359SValentin Clement using FIROpConversion::FIROpConversion; 615a7a61359SValentin Clement 616a7a61359SValentin Clement mlir::LogicalResult 617a7a61359SValentin Clement matchAndRewrite(fir::ZeroOp zero, OpAdaptor, 618a7a61359SValentin Clement mlir::ConversionPatternRewriter &rewriter) const override { 619a7a61359SValentin Clement auto ty = convertType(zero.getType()); 620a7a61359SValentin Clement if (ty.isa<mlir::LLVM::LLVMPointerType>()) { 621a7a61359SValentin Clement rewriter.replaceOpWithNewOp<mlir::LLVM::NullOp>(zero, ty); 622a7a61359SValentin Clement } else if (ty.isa<mlir::IntegerType>()) { 623a7a61359SValentin Clement rewriter.replaceOpWithNewOp<mlir::LLVM::ConstantOp>( 624a7a61359SValentin Clement zero, ty, mlir::IntegerAttr::get(zero.getType(), 0)); 625a7a61359SValentin Clement } else if (mlir::LLVM::isCompatibleFloatingPointType(ty)) { 626a7a61359SValentin Clement rewriter.replaceOpWithNewOp<mlir::LLVM::ConstantOp>( 627a7a61359SValentin Clement zero, ty, mlir::FloatAttr::get(zero.getType(), 0.0)); 628a7a61359SValentin Clement } else { 629a7a61359SValentin Clement // TODO: create ConstantAggregateZero for FIR aggregate/array types. 63052d813edSValentin Clement return rewriter.notifyMatchFailure( 63152d813edSValentin Clement zero, 632a7a61359SValentin Clement "conversion of fir.zero with aggregate type not implemented yet"); 633a7a61359SValentin Clement } 634a7a61359SValentin Clement return success(); 635a7a61359SValentin Clement } 636a7a61359SValentin Clement }; 63732e08248SAndrzej Warzynski 63854c56347SValentin Clement // Code shared between insert_value and extract_value Ops. 63954c56347SValentin Clement struct ValueOpCommon { 64054c56347SValentin Clement // Translate the arguments pertaining to any multidimensional array to 64154c56347SValentin Clement // row-major order for LLVM-IR. 64254c56347SValentin Clement static void toRowMajor(SmallVectorImpl<mlir::Attribute> &attrs, 64354c56347SValentin Clement mlir::Type ty) { 64454c56347SValentin Clement assert(ty && "type is null"); 64554c56347SValentin Clement const auto end = attrs.size(); 64654c56347SValentin Clement for (std::remove_const_t<decltype(end)> i = 0; i < end; ++i) { 64754c56347SValentin Clement if (auto seq = ty.dyn_cast<mlir::LLVM::LLVMArrayType>()) { 64854c56347SValentin Clement const auto dim = getDimension(seq); 64954c56347SValentin Clement if (dim > 1) { 65054c56347SValentin Clement auto ub = std::min(i + dim, end); 65154c56347SValentin Clement std::reverse(attrs.begin() + i, attrs.begin() + ub); 65254c56347SValentin Clement i += dim - 1; 65354c56347SValentin Clement } 65454c56347SValentin Clement ty = getArrayElementType(seq); 65554c56347SValentin Clement } else if (auto st = ty.dyn_cast<mlir::LLVM::LLVMStructType>()) { 65654c56347SValentin Clement ty = st.getBody()[attrs[i].cast<mlir::IntegerAttr>().getInt()]; 65754c56347SValentin Clement } else { 65854c56347SValentin Clement llvm_unreachable("index into invalid type"); 65954c56347SValentin Clement } 66054c56347SValentin Clement } 66154c56347SValentin Clement } 66254c56347SValentin Clement 66354c56347SValentin Clement static llvm::SmallVector<mlir::Attribute> 66454c56347SValentin Clement collectIndices(mlir::ConversionPatternRewriter &rewriter, 66554c56347SValentin Clement mlir::ArrayAttr arrAttr) { 66654c56347SValentin Clement llvm::SmallVector<mlir::Attribute> attrs; 66754c56347SValentin Clement for (auto i = arrAttr.begin(), e = arrAttr.end(); i != e; ++i) { 66854c56347SValentin Clement if (i->isa<mlir::IntegerAttr>()) { 66954c56347SValentin Clement attrs.push_back(*i); 67054c56347SValentin Clement } else { 67154c56347SValentin Clement auto fieldName = i->cast<mlir::StringAttr>().getValue(); 67254c56347SValentin Clement ++i; 67354c56347SValentin Clement auto ty = i->cast<mlir::TypeAttr>().getValue(); 67454c56347SValentin Clement auto index = ty.cast<fir::RecordType>().getFieldIndex(fieldName); 67554c56347SValentin Clement attrs.push_back(mlir::IntegerAttr::get(rewriter.getI32Type(), index)); 67654c56347SValentin Clement } 67754c56347SValentin Clement } 67854c56347SValentin Clement return attrs; 67954c56347SValentin Clement } 68054c56347SValentin Clement 68154c56347SValentin Clement private: 68254c56347SValentin Clement static unsigned getDimension(mlir::LLVM::LLVMArrayType ty) { 68354c56347SValentin Clement unsigned result = 1; 68454c56347SValentin Clement for (auto eleTy = ty.getElementType().dyn_cast<mlir::LLVM::LLVMArrayType>(); 68554c56347SValentin Clement eleTy; 68654c56347SValentin Clement eleTy = eleTy.getElementType().dyn_cast<mlir::LLVM::LLVMArrayType>()) 68754c56347SValentin Clement ++result; 68854c56347SValentin Clement return result; 68954c56347SValentin Clement } 69054c56347SValentin Clement 69154c56347SValentin Clement static mlir::Type getArrayElementType(mlir::LLVM::LLVMArrayType ty) { 69254c56347SValentin Clement auto eleTy = ty.getElementType(); 69354c56347SValentin Clement while (auto arrTy = eleTy.dyn_cast<mlir::LLVM::LLVMArrayType>()) 69454c56347SValentin Clement eleTy = arrTy.getElementType(); 69554c56347SValentin Clement return eleTy; 69654c56347SValentin Clement } 69754c56347SValentin Clement }; 69854c56347SValentin Clement 69954c56347SValentin Clement /// Extract a subobject value from an ssa-value of aggregate type 70054c56347SValentin Clement struct ExtractValueOpConversion 70154c56347SValentin Clement : public FIROpAndTypeConversion<fir::ExtractValueOp>, 70254c56347SValentin Clement public ValueOpCommon { 70354c56347SValentin Clement using FIROpAndTypeConversion::FIROpAndTypeConversion; 70454c56347SValentin Clement 70554c56347SValentin Clement mlir::LogicalResult 70654c56347SValentin Clement doRewrite(fir::ExtractValueOp extractVal, mlir::Type ty, OpAdaptor adaptor, 70754c56347SValentin Clement mlir::ConversionPatternRewriter &rewriter) const override { 70854c56347SValentin Clement auto attrs = collectIndices(rewriter, extractVal.coor()); 70954c56347SValentin Clement toRowMajor(attrs, adaptor.getOperands()[0].getType()); 71054c56347SValentin Clement auto position = mlir::ArrayAttr::get(extractVal.getContext(), attrs); 71154c56347SValentin Clement rewriter.replaceOpWithNewOp<mlir::LLVM::ExtractValueOp>( 71254c56347SValentin Clement extractVal, ty, adaptor.getOperands()[0], position); 71354c56347SValentin Clement return success(); 71454c56347SValentin Clement } 71554c56347SValentin Clement }; 71654c56347SValentin Clement 71754c56347SValentin Clement /// InsertValue is the generalized instruction for the composition of new 71854c56347SValentin Clement /// aggregate type values. 71954c56347SValentin Clement struct InsertValueOpConversion 72054c56347SValentin Clement : public FIROpAndTypeConversion<fir::InsertValueOp>, 72154c56347SValentin Clement public ValueOpCommon { 72254c56347SValentin Clement using FIROpAndTypeConversion::FIROpAndTypeConversion; 72354c56347SValentin Clement 72454c56347SValentin Clement mlir::LogicalResult 72554c56347SValentin Clement doRewrite(fir::InsertValueOp insertVal, mlir::Type ty, OpAdaptor adaptor, 72654c56347SValentin Clement mlir::ConversionPatternRewriter &rewriter) const override { 72754c56347SValentin Clement auto attrs = collectIndices(rewriter, insertVal.coor()); 72854c56347SValentin Clement toRowMajor(attrs, adaptor.getOperands()[0].getType()); 72954c56347SValentin Clement auto position = mlir::ArrayAttr::get(insertVal.getContext(), attrs); 73054c56347SValentin Clement rewriter.replaceOpWithNewOp<mlir::LLVM::InsertValueOp>( 73154c56347SValentin Clement insertVal, ty, adaptor.getOperands()[0], adaptor.getOperands()[1], 73254c56347SValentin Clement position); 73354c56347SValentin Clement return success(); 73454c56347SValentin Clement } 73554c56347SValentin Clement }; 73654c56347SValentin Clement 7373ae8e442SValentin Clement /// InsertOnRange inserts a value into a sequence over a range of offsets. 7383ae8e442SValentin Clement struct InsertOnRangeOpConversion 7393ae8e442SValentin Clement : public FIROpAndTypeConversion<fir::InsertOnRangeOp> { 7403ae8e442SValentin Clement using FIROpAndTypeConversion::FIROpAndTypeConversion; 7413ae8e442SValentin Clement 7423ae8e442SValentin Clement // Increments an array of subscripts in a row major fasion. 7433ae8e442SValentin Clement void incrementSubscripts(const SmallVector<uint64_t> &dims, 7443ae8e442SValentin Clement SmallVector<uint64_t> &subscripts) const { 7453ae8e442SValentin Clement for (size_t i = dims.size(); i > 0; --i) { 7463ae8e442SValentin Clement if (++subscripts[i - 1] < dims[i - 1]) { 7473ae8e442SValentin Clement return; 7483ae8e442SValentin Clement } 7493ae8e442SValentin Clement subscripts[i - 1] = 0; 7503ae8e442SValentin Clement } 7513ae8e442SValentin Clement } 7523ae8e442SValentin Clement 7533ae8e442SValentin Clement mlir::LogicalResult 7543ae8e442SValentin Clement doRewrite(fir::InsertOnRangeOp range, mlir::Type ty, OpAdaptor adaptor, 7553ae8e442SValentin Clement mlir::ConversionPatternRewriter &rewriter) const override { 7563ae8e442SValentin Clement 7573ae8e442SValentin Clement llvm::SmallVector<uint64_t> dims; 7583ae8e442SValentin Clement auto type = adaptor.getOperands()[0].getType(); 7593ae8e442SValentin Clement 7603ae8e442SValentin Clement // Iteratively extract the array dimensions from the type. 7613ae8e442SValentin Clement while (auto t = type.dyn_cast<mlir::LLVM::LLVMArrayType>()) { 7623ae8e442SValentin Clement dims.push_back(t.getNumElements()); 7633ae8e442SValentin Clement type = t.getElementType(); 7643ae8e442SValentin Clement } 7653ae8e442SValentin Clement 7663ae8e442SValentin Clement SmallVector<uint64_t> lBounds; 7673ae8e442SValentin Clement SmallVector<uint64_t> uBounds; 7683ae8e442SValentin Clement 7693ae8e442SValentin Clement // Extract integer value from the attribute 7703ae8e442SValentin Clement SmallVector<int64_t> coordinates = llvm::to_vector<4>( 7713ae8e442SValentin Clement llvm::map_range(range.coor(), [](Attribute a) -> int64_t { 7723ae8e442SValentin Clement return a.cast<IntegerAttr>().getInt(); 7733ae8e442SValentin Clement })); 7743ae8e442SValentin Clement 7753ae8e442SValentin Clement // Unzip the upper and lower bound and convert to a row major format. 7763ae8e442SValentin Clement for (auto i = coordinates.rbegin(), e = coordinates.rend(); i != e; ++i) { 7773ae8e442SValentin Clement uBounds.push_back(*i++); 7783ae8e442SValentin Clement lBounds.push_back(*i); 7793ae8e442SValentin Clement } 7803ae8e442SValentin Clement 7813ae8e442SValentin Clement auto &subscripts = lBounds; 7823ae8e442SValentin Clement auto loc = range.getLoc(); 7833ae8e442SValentin Clement mlir::Value lastOp = adaptor.getOperands()[0]; 7843ae8e442SValentin Clement mlir::Value insertVal = adaptor.getOperands()[1]; 7853ae8e442SValentin Clement 7863ae8e442SValentin Clement auto i64Ty = rewriter.getI64Type(); 7873ae8e442SValentin Clement while (subscripts != uBounds) { 7883ae8e442SValentin Clement // Convert uint64_t's to Attribute's. 7893ae8e442SValentin Clement SmallVector<mlir::Attribute> subscriptAttrs; 7903ae8e442SValentin Clement for (const auto &subscript : subscripts) 7913ae8e442SValentin Clement subscriptAttrs.push_back(IntegerAttr::get(i64Ty, subscript)); 7923ae8e442SValentin Clement lastOp = rewriter.create<mlir::LLVM::InsertValueOp>( 7933ae8e442SValentin Clement loc, ty, lastOp, insertVal, 7943ae8e442SValentin Clement ArrayAttr::get(range.getContext(), subscriptAttrs)); 7953ae8e442SValentin Clement 7963ae8e442SValentin Clement incrementSubscripts(dims, subscripts); 7973ae8e442SValentin Clement } 7983ae8e442SValentin Clement 7993ae8e442SValentin Clement // Convert uint64_t's to Attribute's. 8003ae8e442SValentin Clement SmallVector<mlir::Attribute> subscriptAttrs; 8013ae8e442SValentin Clement for (const auto &subscript : subscripts) 8023ae8e442SValentin Clement subscriptAttrs.push_back( 8033ae8e442SValentin Clement IntegerAttr::get(rewriter.getI64Type(), subscript)); 8043ae8e442SValentin Clement mlir::ArrayRef<mlir::Attribute> arrayRef(subscriptAttrs); 8053ae8e442SValentin Clement 8063ae8e442SValentin Clement rewriter.replaceOpWithNewOp<mlir::LLVM::InsertValueOp>( 8073ae8e442SValentin Clement range, ty, lastOp, insertVal, 8083ae8e442SValentin Clement ArrayAttr::get(range.getContext(), arrayRef)); 8093ae8e442SValentin Clement 8103ae8e442SValentin Clement return success(); 8113ae8e442SValentin Clement } 8123ae8e442SValentin Clement }; 8137b5132daSValentin Clement 8147b5132daSValentin Clement // 8157b5132daSValentin Clement // Primitive operations on Complex types 8167b5132daSValentin Clement // 8177b5132daSValentin Clement 8187b5132daSValentin Clement /// Generate inline code for complex addition/subtraction 8197b5132daSValentin Clement template <typename LLVMOP, typename OPTY> 8207b5132daSValentin Clement mlir::LLVM::InsertValueOp complexSum(OPTY sumop, mlir::ValueRange opnds, 8217b5132daSValentin Clement mlir::ConversionPatternRewriter &rewriter, 8227b5132daSValentin Clement fir::LLVMTypeConverter &lowering) { 8237b5132daSValentin Clement mlir::Value a = opnds[0]; 8247b5132daSValentin Clement mlir::Value b = opnds[1]; 8257b5132daSValentin Clement auto loc = sumop.getLoc(); 8267b5132daSValentin Clement auto ctx = sumop.getContext(); 8277b5132daSValentin Clement auto c0 = mlir::ArrayAttr::get(ctx, rewriter.getI32IntegerAttr(0)); 8287b5132daSValentin Clement auto c1 = mlir::ArrayAttr::get(ctx, rewriter.getI32IntegerAttr(1)); 8297b5132daSValentin Clement mlir::Type eleTy = lowering.convertType(getComplexEleTy(sumop.getType())); 8307b5132daSValentin Clement mlir::Type ty = lowering.convertType(sumop.getType()); 8317b5132daSValentin Clement auto x0 = rewriter.create<mlir::LLVM::ExtractValueOp>(loc, eleTy, a, c0); 8327b5132daSValentin Clement auto y0 = rewriter.create<mlir::LLVM::ExtractValueOp>(loc, eleTy, a, c1); 8337b5132daSValentin Clement auto x1 = rewriter.create<mlir::LLVM::ExtractValueOp>(loc, eleTy, b, c0); 8347b5132daSValentin Clement auto y1 = rewriter.create<mlir::LLVM::ExtractValueOp>(loc, eleTy, b, c1); 8357b5132daSValentin Clement auto rx = rewriter.create<LLVMOP>(loc, eleTy, x0, x1); 8367b5132daSValentin Clement auto ry = rewriter.create<LLVMOP>(loc, eleTy, y0, y1); 8377b5132daSValentin Clement auto r0 = rewriter.create<mlir::LLVM::UndefOp>(loc, ty); 8387b5132daSValentin Clement auto r1 = rewriter.create<mlir::LLVM::InsertValueOp>(loc, ty, r0, rx, c0); 8397b5132daSValentin Clement return rewriter.create<mlir::LLVM::InsertValueOp>(loc, ty, r1, ry, c1); 8407b5132daSValentin Clement } 8417b5132daSValentin Clement 8427b5132daSValentin Clement struct AddcOpConversion : public FIROpConversion<fir::AddcOp> { 8437b5132daSValentin Clement using FIROpConversion::FIROpConversion; 8447b5132daSValentin Clement 8457b5132daSValentin Clement mlir::LogicalResult 8467b5132daSValentin Clement matchAndRewrite(fir::AddcOp addc, OpAdaptor adaptor, 8477b5132daSValentin Clement mlir::ConversionPatternRewriter &rewriter) const override { 8487b5132daSValentin Clement // given: (x + iy) + (x' + iy') 8497b5132daSValentin Clement // result: (x + x') + i(y + y') 8507b5132daSValentin Clement auto r = complexSum<mlir::LLVM::FAddOp>(addc, adaptor.getOperands(), 8517b5132daSValentin Clement rewriter, lowerTy()); 8527b5132daSValentin Clement rewriter.replaceOp(addc, r.getResult()); 8537b5132daSValentin Clement return success(); 8547b5132daSValentin Clement } 8557b5132daSValentin Clement }; 8567b5132daSValentin Clement 8577b5132daSValentin Clement struct SubcOpConversion : public FIROpConversion<fir::SubcOp> { 8587b5132daSValentin Clement using FIROpConversion::FIROpConversion; 8597b5132daSValentin Clement 8607b5132daSValentin Clement mlir::LogicalResult 8617b5132daSValentin Clement matchAndRewrite(fir::SubcOp subc, OpAdaptor adaptor, 8627b5132daSValentin Clement mlir::ConversionPatternRewriter &rewriter) const override { 8637b5132daSValentin Clement // given: (x + iy) - (x' + iy') 8647b5132daSValentin Clement // result: (x - x') + i(y - y') 8657b5132daSValentin Clement auto r = complexSum<mlir::LLVM::FSubOp>(subc, adaptor.getOperands(), 8667b5132daSValentin Clement rewriter, lowerTy()); 8677b5132daSValentin Clement rewriter.replaceOp(subc, r.getResult()); 8687b5132daSValentin Clement return success(); 8697b5132daSValentin Clement } 8707b5132daSValentin Clement }; 8717b5132daSValentin Clement 8727b5132daSValentin Clement /// Inlined complex multiply 8737b5132daSValentin Clement struct MulcOpConversion : public FIROpConversion<fir::MulcOp> { 8747b5132daSValentin Clement using FIROpConversion::FIROpConversion; 8757b5132daSValentin Clement 8767b5132daSValentin Clement mlir::LogicalResult 8777b5132daSValentin Clement matchAndRewrite(fir::MulcOp mulc, OpAdaptor adaptor, 8787b5132daSValentin Clement mlir::ConversionPatternRewriter &rewriter) const override { 8797b5132daSValentin Clement // TODO: Can we use a call to __muldc3 ? 8807b5132daSValentin Clement // given: (x + iy) * (x' + iy') 8817b5132daSValentin Clement // result: (xx'-yy')+i(xy'+yx') 8827b5132daSValentin Clement mlir::Value a = adaptor.getOperands()[0]; 8837b5132daSValentin Clement mlir::Value b = adaptor.getOperands()[1]; 8847b5132daSValentin Clement auto loc = mulc.getLoc(); 8857b5132daSValentin Clement auto *ctx = mulc.getContext(); 8867b5132daSValentin Clement auto c0 = mlir::ArrayAttr::get(ctx, rewriter.getI32IntegerAttr(0)); 8877b5132daSValentin Clement auto c1 = mlir::ArrayAttr::get(ctx, rewriter.getI32IntegerAttr(1)); 8887b5132daSValentin Clement mlir::Type eleTy = convertType(getComplexEleTy(mulc.getType())); 8897b5132daSValentin Clement mlir::Type ty = convertType(mulc.getType()); 8907b5132daSValentin Clement auto x0 = rewriter.create<mlir::LLVM::ExtractValueOp>(loc, eleTy, a, c0); 8917b5132daSValentin Clement auto y0 = rewriter.create<mlir::LLVM::ExtractValueOp>(loc, eleTy, a, c1); 8927b5132daSValentin Clement auto x1 = rewriter.create<mlir::LLVM::ExtractValueOp>(loc, eleTy, b, c0); 8937b5132daSValentin Clement auto y1 = rewriter.create<mlir::LLVM::ExtractValueOp>(loc, eleTy, b, c1); 8947b5132daSValentin Clement auto xx = rewriter.create<mlir::LLVM::FMulOp>(loc, eleTy, x0, x1); 8957b5132daSValentin Clement auto yx = rewriter.create<mlir::LLVM::FMulOp>(loc, eleTy, y0, x1); 8967b5132daSValentin Clement auto xy = rewriter.create<mlir::LLVM::FMulOp>(loc, eleTy, x0, y1); 8977b5132daSValentin Clement auto ri = rewriter.create<mlir::LLVM::FAddOp>(loc, eleTy, xy, yx); 8987b5132daSValentin Clement auto yy = rewriter.create<mlir::LLVM::FMulOp>(loc, eleTy, y0, y1); 8997b5132daSValentin Clement auto rr = rewriter.create<mlir::LLVM::FSubOp>(loc, eleTy, xx, yy); 9007b5132daSValentin Clement auto ra = rewriter.create<mlir::LLVM::UndefOp>(loc, ty); 9017b5132daSValentin Clement auto r1 = rewriter.create<mlir::LLVM::InsertValueOp>(loc, ty, ra, rr, c0); 9027b5132daSValentin Clement auto r0 = rewriter.create<mlir::LLVM::InsertValueOp>(loc, ty, r1, ri, c1); 9037b5132daSValentin Clement rewriter.replaceOp(mulc, r0.getResult()); 9047b5132daSValentin Clement return success(); 9057b5132daSValentin Clement } 9067b5132daSValentin Clement }; 9077b5132daSValentin Clement 9087b5132daSValentin Clement /// Inlined complex division 9097b5132daSValentin Clement struct DivcOpConversion : public FIROpConversion<fir::DivcOp> { 9107b5132daSValentin Clement using FIROpConversion::FIROpConversion; 9117b5132daSValentin Clement 9127b5132daSValentin Clement mlir::LogicalResult 9137b5132daSValentin Clement matchAndRewrite(fir::DivcOp divc, OpAdaptor adaptor, 9147b5132daSValentin Clement mlir::ConversionPatternRewriter &rewriter) const override { 9157b5132daSValentin Clement // TODO: Can we use a call to __divdc3 instead? 9167b5132daSValentin Clement // Just generate inline code for now. 9177b5132daSValentin Clement // given: (x + iy) / (x' + iy') 9187b5132daSValentin Clement // result: ((xx'+yy')/d) + i((yx'-xy')/d) where d = x'x' + y'y' 9197b5132daSValentin Clement mlir::Value a = adaptor.getOperands()[0]; 9207b5132daSValentin Clement mlir::Value b = adaptor.getOperands()[1]; 9217b5132daSValentin Clement auto loc = divc.getLoc(); 9227b5132daSValentin Clement auto *ctx = divc.getContext(); 9237b5132daSValentin Clement auto c0 = mlir::ArrayAttr::get(ctx, rewriter.getI32IntegerAttr(0)); 9247b5132daSValentin Clement auto c1 = mlir::ArrayAttr::get(ctx, rewriter.getI32IntegerAttr(1)); 9257b5132daSValentin Clement mlir::Type eleTy = convertType(getComplexEleTy(divc.getType())); 9267b5132daSValentin Clement mlir::Type ty = convertType(divc.getType()); 9277b5132daSValentin Clement auto x0 = rewriter.create<mlir::LLVM::ExtractValueOp>(loc, eleTy, a, c0); 9287b5132daSValentin Clement auto y0 = rewriter.create<mlir::LLVM::ExtractValueOp>(loc, eleTy, a, c1); 9297b5132daSValentin Clement auto x1 = rewriter.create<mlir::LLVM::ExtractValueOp>(loc, eleTy, b, c0); 9307b5132daSValentin Clement auto y1 = rewriter.create<mlir::LLVM::ExtractValueOp>(loc, eleTy, b, c1); 9317b5132daSValentin Clement auto xx = rewriter.create<mlir::LLVM::FMulOp>(loc, eleTy, x0, x1); 9327b5132daSValentin Clement auto x1x1 = rewriter.create<mlir::LLVM::FMulOp>(loc, eleTy, x1, x1); 9337b5132daSValentin Clement auto yx = rewriter.create<mlir::LLVM::FMulOp>(loc, eleTy, y0, x1); 9347b5132daSValentin Clement auto xy = rewriter.create<mlir::LLVM::FMulOp>(loc, eleTy, x0, y1); 9357b5132daSValentin Clement auto yy = rewriter.create<mlir::LLVM::FMulOp>(loc, eleTy, y0, y1); 9367b5132daSValentin Clement auto y1y1 = rewriter.create<mlir::LLVM::FMulOp>(loc, eleTy, y1, y1); 9377b5132daSValentin Clement auto d = rewriter.create<mlir::LLVM::FAddOp>(loc, eleTy, x1x1, y1y1); 9387b5132daSValentin Clement auto rrn = rewriter.create<mlir::LLVM::FAddOp>(loc, eleTy, xx, yy); 9397b5132daSValentin Clement auto rin = rewriter.create<mlir::LLVM::FSubOp>(loc, eleTy, yx, xy); 9407b5132daSValentin Clement auto rr = rewriter.create<mlir::LLVM::FDivOp>(loc, eleTy, rrn, d); 9417b5132daSValentin Clement auto ri = rewriter.create<mlir::LLVM::FDivOp>(loc, eleTy, rin, d); 9427b5132daSValentin Clement auto ra = rewriter.create<mlir::LLVM::UndefOp>(loc, ty); 9437b5132daSValentin Clement auto r1 = rewriter.create<mlir::LLVM::InsertValueOp>(loc, ty, ra, rr, c0); 9447b5132daSValentin Clement auto r0 = rewriter.create<mlir::LLVM::InsertValueOp>(loc, ty, r1, ri, c1); 9457b5132daSValentin Clement rewriter.replaceOp(divc, r0.getResult()); 9467b5132daSValentin Clement return success(); 9477b5132daSValentin Clement } 9487b5132daSValentin Clement }; 9497b5132daSValentin Clement 9507b5132daSValentin Clement /// Inlined complex negation 9517b5132daSValentin Clement struct NegcOpConversion : public FIROpConversion<fir::NegcOp> { 9527b5132daSValentin Clement using FIROpConversion::FIROpConversion; 9537b5132daSValentin Clement 9547b5132daSValentin Clement mlir::LogicalResult 9557b5132daSValentin Clement matchAndRewrite(fir::NegcOp neg, OpAdaptor adaptor, 9567b5132daSValentin Clement mlir::ConversionPatternRewriter &rewriter) const override { 9577b5132daSValentin Clement // given: -(x + iy) 9587b5132daSValentin Clement // result: -x - iy 9597b5132daSValentin Clement auto *ctxt = neg.getContext(); 9607b5132daSValentin Clement auto eleTy = convertType(getComplexEleTy(neg.getType())); 9617b5132daSValentin Clement auto ty = convertType(neg.getType()); 9627b5132daSValentin Clement auto loc = neg.getLoc(); 9637b5132daSValentin Clement mlir::Value o0 = adaptor.getOperands()[0]; 9647b5132daSValentin Clement auto c0 = mlir::ArrayAttr::get(ctxt, rewriter.getI32IntegerAttr(0)); 9657b5132daSValentin Clement auto c1 = mlir::ArrayAttr::get(ctxt, rewriter.getI32IntegerAttr(1)); 9667b5132daSValentin Clement auto rp = rewriter.create<mlir::LLVM::ExtractValueOp>(loc, eleTy, o0, c0); 9677b5132daSValentin Clement auto ip = rewriter.create<mlir::LLVM::ExtractValueOp>(loc, eleTy, o0, c1); 9687b5132daSValentin Clement auto nrp = rewriter.create<mlir::LLVM::FNegOp>(loc, eleTy, rp); 9697b5132daSValentin Clement auto nip = rewriter.create<mlir::LLVM::FNegOp>(loc, eleTy, ip); 9707b5132daSValentin Clement auto r = rewriter.create<mlir::LLVM::InsertValueOp>(loc, ty, o0, nrp, c0); 9717b5132daSValentin Clement rewriter.replaceOpWithNewOp<mlir::LLVM::InsertValueOp>(neg, ty, r, nip, c1); 9727b5132daSValentin Clement return success(); 9737b5132daSValentin Clement } 9747b5132daSValentin Clement }; 9757b5132daSValentin Clement 976044d5b5dSValentin Clement } // namespace 977044d5b5dSValentin Clement 978044d5b5dSValentin Clement namespace { 979044d5b5dSValentin Clement /// Convert FIR dialect to LLVM dialect 980044d5b5dSValentin Clement /// 981044d5b5dSValentin Clement /// This pass lowers all FIR dialect operations to LLVM IR dialect. An 982044d5b5dSValentin Clement /// MLIR pass is used to lower residual Std dialect to LLVM IR dialect. 983044d5b5dSValentin Clement /// 984044d5b5dSValentin Clement /// This pass is not complete yet. We are upstreaming it in small patches. 985044d5b5dSValentin Clement class FIRToLLVMLowering : public fir::FIRToLLVMLoweringBase<FIRToLLVMLowering> { 986044d5b5dSValentin Clement public: 987044d5b5dSValentin Clement mlir::ModuleOp getModule() { return getOperation(); } 988044d5b5dSValentin Clement 989044d5b5dSValentin Clement void runOnOperation() override final { 9907b5132daSValentin Clement auto mod = getModule(); 9917b5132daSValentin Clement if (!forcedTargetTriple.empty()) { 9927b5132daSValentin Clement fir::setTargetTriple(mod, forcedTargetTriple); 9937b5132daSValentin Clement } 9947b5132daSValentin Clement 995044d5b5dSValentin Clement auto *context = getModule().getContext(); 996044d5b5dSValentin Clement fir::LLVMTypeConverter typeConverter{getModule()}; 997044d5b5dSValentin Clement mlir::OwningRewritePatternList pattern(context); 998*df3b9810SValentin Clement pattern.insert< 999*df3b9810SValentin Clement AddcOpConversion, AddrOfOpConversion, BoxAddrOpConversion, 1000*df3b9810SValentin Clement BoxDimsOpConversion, BoxEleSizeOpConversion, BoxRankOpConversion, 1001*df3b9810SValentin Clement CallOpConversion, ConvertOpConversion, DivcOpConversion, 1002*df3b9810SValentin Clement ExtractValueOpConversion, HasValueOpConversion, GlobalOpConversion, 1003*df3b9810SValentin Clement InsertOnRangeOpConversion, InsertValueOpConversion, LoadOpConversion, 1004*df3b9810SValentin Clement NegcOpConversion, MulcOpConversion, SelectOpConversion, 1005*df3b9810SValentin Clement SelectRankOpConversion, StoreOpConversion, SubcOpConversion, 1006*df3b9810SValentin Clement UndefOpConversion, UnreachableOpConversion, ZeroOpConversion>( 1007*df3b9810SValentin Clement typeConverter); 1008044d5b5dSValentin Clement mlir::populateStdToLLVMConversionPatterns(typeConverter, pattern); 1009044d5b5dSValentin Clement mlir::arith::populateArithmeticToLLVMConversionPatterns(typeConverter, 1010044d5b5dSValentin Clement pattern); 1011044d5b5dSValentin Clement mlir::ConversionTarget target{*context}; 1012044d5b5dSValentin Clement target.addLegalDialect<mlir::LLVM::LLVMDialect>(); 1013044d5b5dSValentin Clement 1014044d5b5dSValentin Clement // required NOPs for applying a full conversion 1015044d5b5dSValentin Clement target.addLegalOp<mlir::ModuleOp>(); 1016044d5b5dSValentin Clement 1017044d5b5dSValentin Clement // apply the patterns 1018044d5b5dSValentin Clement if (mlir::failed(mlir::applyFullConversion(getModule(), target, 1019044d5b5dSValentin Clement std::move(pattern)))) { 1020044d5b5dSValentin Clement signalPassFailure(); 1021044d5b5dSValentin Clement } 1022044d5b5dSValentin Clement } 1023044d5b5dSValentin Clement }; 1024044d5b5dSValentin Clement } // namespace 1025044d5b5dSValentin Clement 1026044d5b5dSValentin Clement std::unique_ptr<mlir::Pass> fir::createFIRToLLVMPass() { 1027044d5b5dSValentin Clement return std::make_unique<FIRToLLVMLowering>(); 1028044d5b5dSValentin Clement } 1029