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" 17044d5b5dSValentin Clement #include "mlir/Conversion/ArithmeticToLLVM/ArithmeticToLLVM.h" 18044d5b5dSValentin Clement #include "mlir/Conversion/LLVMCommon/Pattern.h" 19044d5b5dSValentin Clement #include "mlir/Conversion/LLVMCommon/TypeConverter.h" 20044d5b5dSValentin Clement #include "mlir/Conversion/StandardToLLVM/ConvertStandardToLLVM.h" 21044d5b5dSValentin Clement #include "mlir/IR/BuiltinTypes.h" 223ae8e442SValentin Clement #include "mlir/IR/Matchers.h" 23044d5b5dSValentin Clement #include "mlir/Pass/Pass.h" 24044d5b5dSValentin Clement #include "llvm/ADT/ArrayRef.h" 25044d5b5dSValentin Clement 26044d5b5dSValentin Clement #define DEBUG_TYPE "flang-codegen" 27044d5b5dSValentin Clement 28044d5b5dSValentin Clement // fir::LLVMTypeConverter for converting to LLVM IR dialect types. 29044d5b5dSValentin Clement #include "TypeConverter.h" 30044d5b5dSValentin Clement 31044d5b5dSValentin Clement namespace { 32044d5b5dSValentin Clement /// FIR conversion pattern template 33044d5b5dSValentin Clement template <typename FromOp> 34044d5b5dSValentin Clement class FIROpConversion : public mlir::ConvertOpToLLVMPattern<FromOp> { 35044d5b5dSValentin Clement public: 36044d5b5dSValentin Clement explicit FIROpConversion(fir::LLVMTypeConverter &lowering) 37044d5b5dSValentin Clement : mlir::ConvertOpToLLVMPattern<FromOp>(lowering) {} 38044d5b5dSValentin Clement 39044d5b5dSValentin Clement protected: 40044d5b5dSValentin Clement mlir::Type convertType(mlir::Type ty) const { 41044d5b5dSValentin Clement return lowerTy().convertType(ty); 42044d5b5dSValentin Clement } 43044d5b5dSValentin Clement 44044d5b5dSValentin Clement fir::LLVMTypeConverter &lowerTy() const { 45044d5b5dSValentin Clement return *static_cast<fir::LLVMTypeConverter *>(this->getTypeConverter()); 46044d5b5dSValentin Clement } 47044d5b5dSValentin Clement }; 48044d5b5dSValentin Clement 493ae8e442SValentin Clement /// FIR conversion pattern template 503ae8e442SValentin Clement template <typename FromOp> 513ae8e442SValentin Clement class FIROpAndTypeConversion : public FIROpConversion<FromOp> { 523ae8e442SValentin Clement public: 533ae8e442SValentin Clement using FIROpConversion<FromOp>::FIROpConversion; 543ae8e442SValentin Clement using OpAdaptor = typename FromOp::Adaptor; 553ae8e442SValentin Clement 563ae8e442SValentin Clement mlir::LogicalResult 573ae8e442SValentin Clement matchAndRewrite(FromOp op, OpAdaptor adaptor, 583ae8e442SValentin Clement mlir::ConversionPatternRewriter &rewriter) const final { 593ae8e442SValentin Clement mlir::Type ty = this->convertType(op.getType()); 603ae8e442SValentin Clement return doRewrite(op, ty, adaptor, rewriter); 613ae8e442SValentin Clement } 623ae8e442SValentin Clement 633ae8e442SValentin Clement virtual mlir::LogicalResult 643ae8e442SValentin Clement doRewrite(FromOp addr, mlir::Type ty, OpAdaptor adaptor, 653ae8e442SValentin Clement mlir::ConversionPatternRewriter &rewriter) const = 0; 663ae8e442SValentin Clement }; 673ae8e442SValentin Clement 680c4a7a52SValentin Clement // Lower `fir.address_of` operation to `llvm.address_of` operation. 69044d5b5dSValentin Clement struct AddrOfOpConversion : public FIROpConversion<fir::AddrOfOp> { 70044d5b5dSValentin Clement using FIROpConversion::FIROpConversion; 71044d5b5dSValentin Clement 72044d5b5dSValentin Clement mlir::LogicalResult 73044d5b5dSValentin Clement matchAndRewrite(fir::AddrOfOp addr, OpAdaptor adaptor, 74044d5b5dSValentin Clement mlir::ConversionPatternRewriter &rewriter) const override { 75044d5b5dSValentin Clement auto ty = convertType(addr.getType()); 76044d5b5dSValentin Clement rewriter.replaceOpWithNewOp<mlir::LLVM::AddressOfOp>( 77044d5b5dSValentin Clement addr, ty, addr.symbol().getRootReference().getValue()); 78044d5b5dSValentin Clement return success(); 79044d5b5dSValentin Clement } 80044d5b5dSValentin Clement }; 81044d5b5dSValentin Clement 820c4a7a52SValentin Clement /// Lower `fir.has_value` operation to `llvm.return` operation. 83044d5b5dSValentin Clement struct HasValueOpConversion : public FIROpConversion<fir::HasValueOp> { 84044d5b5dSValentin Clement using FIROpConversion::FIROpConversion; 85044d5b5dSValentin Clement 86044d5b5dSValentin Clement mlir::LogicalResult 87044d5b5dSValentin Clement matchAndRewrite(fir::HasValueOp op, OpAdaptor adaptor, 88044d5b5dSValentin Clement mlir::ConversionPatternRewriter &rewriter) const override { 89044d5b5dSValentin Clement rewriter.replaceOpWithNewOp<LLVM::ReturnOp>(op, adaptor.getOperands()); 90044d5b5dSValentin Clement return success(); 91044d5b5dSValentin Clement } 92044d5b5dSValentin Clement }; 93044d5b5dSValentin Clement 940c4a7a52SValentin Clement /// Lower `fir.global` operation to `llvm.global` operation. 950c4a7a52SValentin Clement /// `fir.insert_on_range` operations are replaced with constant dense attribute 960c4a7a52SValentin Clement /// if they are applied on the full range. 97044d5b5dSValentin Clement struct GlobalOpConversion : public FIROpConversion<fir::GlobalOp> { 98044d5b5dSValentin Clement using FIROpConversion::FIROpConversion; 99044d5b5dSValentin Clement 100044d5b5dSValentin Clement mlir::LogicalResult 101044d5b5dSValentin Clement matchAndRewrite(fir::GlobalOp global, OpAdaptor adaptor, 102044d5b5dSValentin Clement mlir::ConversionPatternRewriter &rewriter) const override { 103044d5b5dSValentin Clement auto tyAttr = convertType(global.getType()); 104044d5b5dSValentin Clement if (global.getType().isa<fir::BoxType>()) 105044d5b5dSValentin Clement tyAttr = tyAttr.cast<mlir::LLVM::LLVMPointerType>().getElementType(); 106044d5b5dSValentin Clement auto loc = global.getLoc(); 107044d5b5dSValentin Clement mlir::Attribute initAttr{}; 108044d5b5dSValentin Clement if (global.initVal()) 109044d5b5dSValentin Clement initAttr = global.initVal().getValue(); 110044d5b5dSValentin Clement auto linkage = convertLinkage(global.linkName()); 111044d5b5dSValentin Clement auto isConst = global.constant().hasValue(); 112044d5b5dSValentin Clement auto g = rewriter.create<mlir::LLVM::GlobalOp>( 113044d5b5dSValentin Clement loc, tyAttr, isConst, linkage, global.sym_name(), initAttr); 114044d5b5dSValentin Clement auto &gr = g.getInitializerRegion(); 115044d5b5dSValentin Clement rewriter.inlineRegionBefore(global.region(), gr, gr.end()); 116044d5b5dSValentin Clement if (!gr.empty()) { 117044d5b5dSValentin Clement // Replace insert_on_range with a constant dense attribute if the 118044d5b5dSValentin Clement // initialization is on the full range. 119044d5b5dSValentin Clement auto insertOnRangeOps = gr.front().getOps<fir::InsertOnRangeOp>(); 120044d5b5dSValentin Clement for (auto insertOp : insertOnRangeOps) { 121044d5b5dSValentin Clement if (isFullRange(insertOp.coor(), insertOp.getType())) { 122044d5b5dSValentin Clement auto seqTyAttr = convertType(insertOp.getType()); 123044d5b5dSValentin Clement auto *op = insertOp.val().getDefiningOp(); 124044d5b5dSValentin Clement auto constant = mlir::dyn_cast<mlir::arith::ConstantOp>(op); 125044d5b5dSValentin Clement if (!constant) { 126044d5b5dSValentin Clement auto convertOp = mlir::dyn_cast<fir::ConvertOp>(op); 127044d5b5dSValentin Clement if (!convertOp) 128044d5b5dSValentin Clement continue; 129044d5b5dSValentin Clement constant = cast<mlir::arith::ConstantOp>( 130044d5b5dSValentin Clement convertOp.value().getDefiningOp()); 131044d5b5dSValentin Clement } 132044d5b5dSValentin Clement mlir::Type vecType = mlir::VectorType::get( 133044d5b5dSValentin Clement insertOp.getType().getShape(), constant.getType()); 134044d5b5dSValentin Clement auto denseAttr = mlir::DenseElementsAttr::get( 135044d5b5dSValentin Clement vecType.cast<ShapedType>(), constant.value()); 136044d5b5dSValentin Clement rewriter.setInsertionPointAfter(insertOp); 137044d5b5dSValentin Clement rewriter.replaceOpWithNewOp<mlir::arith::ConstantOp>( 138044d5b5dSValentin Clement insertOp, seqTyAttr, denseAttr); 139044d5b5dSValentin Clement } 140044d5b5dSValentin Clement } 141044d5b5dSValentin Clement } 142044d5b5dSValentin Clement rewriter.eraseOp(global); 143044d5b5dSValentin Clement return success(); 144044d5b5dSValentin Clement } 145044d5b5dSValentin Clement 146044d5b5dSValentin Clement bool isFullRange(mlir::ArrayAttr indexes, fir::SequenceType seqTy) const { 147044d5b5dSValentin Clement auto extents = seqTy.getShape(); 148044d5b5dSValentin Clement if (indexes.size() / 2 != extents.size()) 149044d5b5dSValentin Clement return false; 150044d5b5dSValentin Clement for (unsigned i = 0; i < indexes.size(); i += 2) { 151044d5b5dSValentin Clement if (indexes[i].cast<IntegerAttr>().getInt() != 0) 152044d5b5dSValentin Clement return false; 153044d5b5dSValentin Clement if (indexes[i + 1].cast<IntegerAttr>().getInt() != extents[i / 2] - 1) 154044d5b5dSValentin Clement return false; 155044d5b5dSValentin Clement } 156044d5b5dSValentin Clement return true; 157044d5b5dSValentin Clement } 158044d5b5dSValentin Clement 1590c4a7a52SValentin Clement // TODO: String comparaison should be avoided. Replace linkName with an 1600c4a7a52SValentin Clement // enumeration. 161044d5b5dSValentin Clement mlir::LLVM::Linkage convertLinkage(Optional<StringRef> optLinkage) const { 162044d5b5dSValentin Clement if (optLinkage.hasValue()) { 163044d5b5dSValentin Clement auto name = optLinkage.getValue(); 164044d5b5dSValentin Clement if (name == "internal") 165044d5b5dSValentin Clement return mlir::LLVM::Linkage::Internal; 166044d5b5dSValentin Clement if (name == "linkonce") 167044d5b5dSValentin Clement return mlir::LLVM::Linkage::Linkonce; 168044d5b5dSValentin Clement if (name == "common") 169044d5b5dSValentin Clement return mlir::LLVM::Linkage::Common; 170044d5b5dSValentin Clement if (name == "weak") 171044d5b5dSValentin Clement return mlir::LLVM::Linkage::Weak; 172044d5b5dSValentin Clement } 173044d5b5dSValentin Clement return mlir::LLVM::Linkage::External; 174044d5b5dSValentin Clement } 175044d5b5dSValentin Clement }; 176044d5b5dSValentin Clement 1778c239909SValentin Clement template <typename OP> 1788c239909SValentin Clement void selectMatchAndRewrite(fir::LLVMTypeConverter &lowering, OP select, 1798c239909SValentin Clement typename OP::Adaptor adaptor, 1808c239909SValentin Clement mlir::ConversionPatternRewriter &rewriter) { 1818c239909SValentin Clement unsigned conds = select.getNumConditions(); 1828c239909SValentin Clement auto cases = select.getCases().getValue(); 1838c239909SValentin Clement mlir::Value selector = adaptor.selector(); 1848c239909SValentin Clement auto loc = select.getLoc(); 1858c239909SValentin Clement assert(conds > 0 && "select must have cases"); 1868c239909SValentin Clement 1878c239909SValentin Clement llvm::SmallVector<mlir::Block *> destinations; 1888c239909SValentin Clement llvm::SmallVector<mlir::ValueRange> destinationsOperands; 1898c239909SValentin Clement mlir::Block *defaultDestination; 1908c239909SValentin Clement mlir::ValueRange defaultOperands; 1918c239909SValentin Clement llvm::SmallVector<int32_t> caseValues; 1928c239909SValentin Clement 1938c239909SValentin Clement for (unsigned t = 0; t != conds; ++t) { 1948c239909SValentin Clement mlir::Block *dest = select.getSuccessor(t); 1958c239909SValentin Clement auto destOps = select.getSuccessorOperands(adaptor.getOperands(), t); 1968c239909SValentin Clement const mlir::Attribute &attr = cases[t]; 1978c239909SValentin Clement if (auto intAttr = attr.template dyn_cast<mlir::IntegerAttr>()) { 1988c239909SValentin Clement destinations.push_back(dest); 1998c239909SValentin Clement destinationsOperands.push_back(destOps.hasValue() ? *destOps 2008c239909SValentin Clement : ValueRange()); 2018c239909SValentin Clement caseValues.push_back(intAttr.getInt()); 2028c239909SValentin Clement continue; 2038c239909SValentin Clement } 2048c239909SValentin Clement assert(attr.template dyn_cast_or_null<mlir::UnitAttr>()); 2058c239909SValentin Clement assert((t + 1 == conds) && "unit must be last"); 2068c239909SValentin Clement defaultDestination = dest; 2078c239909SValentin Clement defaultOperands = destOps.hasValue() ? *destOps : ValueRange(); 2088c239909SValentin Clement } 2098c239909SValentin Clement 2108c239909SValentin Clement // LLVM::SwitchOp takes a i32 type for the selector. 2118c239909SValentin Clement if (select.getSelector().getType() != rewriter.getI32Type()) 2128c239909SValentin Clement selector = 2138c239909SValentin Clement rewriter.create<LLVM::TruncOp>(loc, rewriter.getI32Type(), selector); 2148c239909SValentin Clement 2158c239909SValentin Clement rewriter.replaceOpWithNewOp<mlir::LLVM::SwitchOp>( 2168c239909SValentin Clement select, selector, 2178c239909SValentin Clement /*defaultDestination=*/defaultDestination, 2188c239909SValentin Clement /*defaultOperands=*/defaultOperands, 2198c239909SValentin Clement /*caseValues=*/caseValues, 2208c239909SValentin Clement /*caseDestinations=*/destinations, 2218c239909SValentin Clement /*caseOperands=*/destinationsOperands, 2228c239909SValentin Clement /*branchWeights=*/ArrayRef<int32_t>()); 2238c239909SValentin Clement } 2248c239909SValentin Clement 2258c239909SValentin Clement /// conversion of fir::SelectOp to an if-then-else ladder 2268c239909SValentin Clement struct SelectOpConversion : public FIROpConversion<fir::SelectOp> { 2278c239909SValentin Clement using FIROpConversion::FIROpConversion; 2288c239909SValentin Clement 2298c239909SValentin Clement mlir::LogicalResult 2308c239909SValentin Clement matchAndRewrite(fir::SelectOp op, OpAdaptor adaptor, 2318c239909SValentin Clement mlir::ConversionPatternRewriter &rewriter) const override { 2328c239909SValentin Clement selectMatchAndRewrite<fir::SelectOp>(lowerTy(), op, adaptor, rewriter); 2338c239909SValentin Clement return success(); 2348c239909SValentin Clement } 2358c239909SValentin Clement }; 2368c239909SValentin Clement 2378c239909SValentin Clement /// conversion of fir::SelectRankOp to an if-then-else ladder 2388c239909SValentin Clement struct SelectRankOpConversion : public FIROpConversion<fir::SelectRankOp> { 2398c239909SValentin Clement using FIROpConversion::FIROpConversion; 2408c239909SValentin Clement 2418c239909SValentin Clement mlir::LogicalResult 2428c239909SValentin Clement matchAndRewrite(fir::SelectRankOp op, OpAdaptor adaptor, 2438c239909SValentin Clement mlir::ConversionPatternRewriter &rewriter) const override { 2448c239909SValentin Clement selectMatchAndRewrite<fir::SelectRankOp>(lowerTy(), op, adaptor, rewriter); 2458c239909SValentin Clement return success(); 2468c239909SValentin Clement } 2478c239909SValentin Clement }; 2488c239909SValentin Clement 249044d5b5dSValentin Clement // convert to LLVM IR dialect `undef` 250044d5b5dSValentin Clement struct UndefOpConversion : public FIROpConversion<fir::UndefOp> { 251044d5b5dSValentin Clement using FIROpConversion::FIROpConversion; 252044d5b5dSValentin Clement 253044d5b5dSValentin Clement mlir::LogicalResult 254044d5b5dSValentin Clement matchAndRewrite(fir::UndefOp undef, OpAdaptor, 255044d5b5dSValentin Clement mlir::ConversionPatternRewriter &rewriter) const override { 256044d5b5dSValentin Clement rewriter.replaceOpWithNewOp<mlir::LLVM::UndefOp>( 257044d5b5dSValentin Clement undef, convertType(undef.getType())); 258044d5b5dSValentin Clement return success(); 259044d5b5dSValentin Clement } 260044d5b5dSValentin Clement }; 261a7a61359SValentin Clement 26232e08248SAndrzej Warzynski // convert to LLVM IR dialect `unreachable` 26332e08248SAndrzej Warzynski struct UnreachableOpConversion : public FIROpConversion<fir::UnreachableOp> { 26432e08248SAndrzej Warzynski using FIROpConversion::FIROpConversion; 26532e08248SAndrzej Warzynski 26632e08248SAndrzej Warzynski mlir::LogicalResult 26732e08248SAndrzej Warzynski matchAndRewrite(fir::UnreachableOp unreach, OpAdaptor adaptor, 26832e08248SAndrzej Warzynski mlir::ConversionPatternRewriter &rewriter) const override { 26932e08248SAndrzej Warzynski rewriter.replaceOpWithNewOp<mlir::LLVM::UnreachableOp>(unreach); 27032e08248SAndrzej Warzynski return success(); 27132e08248SAndrzej Warzynski } 27232e08248SAndrzej Warzynski }; 27332e08248SAndrzej Warzynski 274a7a61359SValentin Clement struct ZeroOpConversion : public FIROpConversion<fir::ZeroOp> { 275a7a61359SValentin Clement using FIROpConversion::FIROpConversion; 276a7a61359SValentin Clement 277a7a61359SValentin Clement mlir::LogicalResult 278a7a61359SValentin Clement matchAndRewrite(fir::ZeroOp zero, OpAdaptor, 279a7a61359SValentin Clement mlir::ConversionPatternRewriter &rewriter) const override { 280a7a61359SValentin Clement auto ty = convertType(zero.getType()); 281a7a61359SValentin Clement if (ty.isa<mlir::LLVM::LLVMPointerType>()) { 282a7a61359SValentin Clement rewriter.replaceOpWithNewOp<mlir::LLVM::NullOp>(zero, ty); 283a7a61359SValentin Clement } else if (ty.isa<mlir::IntegerType>()) { 284a7a61359SValentin Clement rewriter.replaceOpWithNewOp<mlir::LLVM::ConstantOp>( 285a7a61359SValentin Clement zero, ty, mlir::IntegerAttr::get(zero.getType(), 0)); 286a7a61359SValentin Clement } else if (mlir::LLVM::isCompatibleFloatingPointType(ty)) { 287a7a61359SValentin Clement rewriter.replaceOpWithNewOp<mlir::LLVM::ConstantOp>( 288a7a61359SValentin Clement zero, ty, mlir::FloatAttr::get(zero.getType(), 0.0)); 289a7a61359SValentin Clement } else { 290a7a61359SValentin Clement // TODO: create ConstantAggregateZero for FIR aggregate/array types. 29152d813edSValentin Clement return rewriter.notifyMatchFailure( 29252d813edSValentin Clement zero, 293a7a61359SValentin Clement "conversion of fir.zero with aggregate type not implemented yet"); 294a7a61359SValentin Clement } 295a7a61359SValentin Clement return success(); 296a7a61359SValentin Clement } 297a7a61359SValentin Clement }; 29832e08248SAndrzej Warzynski 299*54c56347SValentin Clement // Code shared between insert_value and extract_value Ops. 300*54c56347SValentin Clement struct ValueOpCommon { 301*54c56347SValentin Clement // Translate the arguments pertaining to any multidimensional array to 302*54c56347SValentin Clement // row-major order for LLVM-IR. 303*54c56347SValentin Clement static void toRowMajor(SmallVectorImpl<mlir::Attribute> &attrs, 304*54c56347SValentin Clement mlir::Type ty) { 305*54c56347SValentin Clement assert(ty && "type is null"); 306*54c56347SValentin Clement const auto end = attrs.size(); 307*54c56347SValentin Clement for (std::remove_const_t<decltype(end)> i = 0; i < end; ++i) { 308*54c56347SValentin Clement if (auto seq = ty.dyn_cast<mlir::LLVM::LLVMArrayType>()) { 309*54c56347SValentin Clement const auto dim = getDimension(seq); 310*54c56347SValentin Clement if (dim > 1) { 311*54c56347SValentin Clement auto ub = std::min(i + dim, end); 312*54c56347SValentin Clement std::reverse(attrs.begin() + i, attrs.begin() + ub); 313*54c56347SValentin Clement i += dim - 1; 314*54c56347SValentin Clement } 315*54c56347SValentin Clement ty = getArrayElementType(seq); 316*54c56347SValentin Clement } else if (auto st = ty.dyn_cast<mlir::LLVM::LLVMStructType>()) { 317*54c56347SValentin Clement ty = st.getBody()[attrs[i].cast<mlir::IntegerAttr>().getInt()]; 318*54c56347SValentin Clement } else { 319*54c56347SValentin Clement llvm_unreachable("index into invalid type"); 320*54c56347SValentin Clement } 321*54c56347SValentin Clement } 322*54c56347SValentin Clement } 323*54c56347SValentin Clement 324*54c56347SValentin Clement static llvm::SmallVector<mlir::Attribute> 325*54c56347SValentin Clement collectIndices(mlir::ConversionPatternRewriter &rewriter, 326*54c56347SValentin Clement mlir::ArrayAttr arrAttr) { 327*54c56347SValentin Clement llvm::SmallVector<mlir::Attribute> attrs; 328*54c56347SValentin Clement for (auto i = arrAttr.begin(), e = arrAttr.end(); i != e; ++i) { 329*54c56347SValentin Clement if (i->isa<mlir::IntegerAttr>()) { 330*54c56347SValentin Clement attrs.push_back(*i); 331*54c56347SValentin Clement } else { 332*54c56347SValentin Clement auto fieldName = i->cast<mlir::StringAttr>().getValue(); 333*54c56347SValentin Clement ++i; 334*54c56347SValentin Clement auto ty = i->cast<mlir::TypeAttr>().getValue(); 335*54c56347SValentin Clement auto index = ty.cast<fir::RecordType>().getFieldIndex(fieldName); 336*54c56347SValentin Clement attrs.push_back(mlir::IntegerAttr::get(rewriter.getI32Type(), index)); 337*54c56347SValentin Clement } 338*54c56347SValentin Clement } 339*54c56347SValentin Clement return attrs; 340*54c56347SValentin Clement } 341*54c56347SValentin Clement 342*54c56347SValentin Clement private: 343*54c56347SValentin Clement static unsigned getDimension(mlir::LLVM::LLVMArrayType ty) { 344*54c56347SValentin Clement unsigned result = 1; 345*54c56347SValentin Clement for (auto eleTy = ty.getElementType().dyn_cast<mlir::LLVM::LLVMArrayType>(); 346*54c56347SValentin Clement eleTy; 347*54c56347SValentin Clement eleTy = eleTy.getElementType().dyn_cast<mlir::LLVM::LLVMArrayType>()) 348*54c56347SValentin Clement ++result; 349*54c56347SValentin Clement return result; 350*54c56347SValentin Clement } 351*54c56347SValentin Clement 352*54c56347SValentin Clement static mlir::Type getArrayElementType(mlir::LLVM::LLVMArrayType ty) { 353*54c56347SValentin Clement auto eleTy = ty.getElementType(); 354*54c56347SValentin Clement while (auto arrTy = eleTy.dyn_cast<mlir::LLVM::LLVMArrayType>()) 355*54c56347SValentin Clement eleTy = arrTy.getElementType(); 356*54c56347SValentin Clement return eleTy; 357*54c56347SValentin Clement } 358*54c56347SValentin Clement }; 359*54c56347SValentin Clement 360*54c56347SValentin Clement /// Extract a subobject value from an ssa-value of aggregate type 361*54c56347SValentin Clement struct ExtractValueOpConversion 362*54c56347SValentin Clement : public FIROpAndTypeConversion<fir::ExtractValueOp>, 363*54c56347SValentin Clement public ValueOpCommon { 364*54c56347SValentin Clement using FIROpAndTypeConversion::FIROpAndTypeConversion; 365*54c56347SValentin Clement 366*54c56347SValentin Clement mlir::LogicalResult 367*54c56347SValentin Clement doRewrite(fir::ExtractValueOp extractVal, mlir::Type ty, OpAdaptor adaptor, 368*54c56347SValentin Clement mlir::ConversionPatternRewriter &rewriter) const override { 369*54c56347SValentin Clement auto attrs = collectIndices(rewriter, extractVal.coor()); 370*54c56347SValentin Clement toRowMajor(attrs, adaptor.getOperands()[0].getType()); 371*54c56347SValentin Clement auto position = mlir::ArrayAttr::get(extractVal.getContext(), attrs); 372*54c56347SValentin Clement rewriter.replaceOpWithNewOp<mlir::LLVM::ExtractValueOp>( 373*54c56347SValentin Clement extractVal, ty, adaptor.getOperands()[0], position); 374*54c56347SValentin Clement return success(); 375*54c56347SValentin Clement } 376*54c56347SValentin Clement }; 377*54c56347SValentin Clement 378*54c56347SValentin Clement /// InsertValue is the generalized instruction for the composition of new 379*54c56347SValentin Clement /// aggregate type values. 380*54c56347SValentin Clement struct InsertValueOpConversion 381*54c56347SValentin Clement : public FIROpAndTypeConversion<fir::InsertValueOp>, 382*54c56347SValentin Clement public ValueOpCommon { 383*54c56347SValentin Clement using FIROpAndTypeConversion::FIROpAndTypeConversion; 384*54c56347SValentin Clement 385*54c56347SValentin Clement mlir::LogicalResult 386*54c56347SValentin Clement doRewrite(fir::InsertValueOp insertVal, mlir::Type ty, OpAdaptor adaptor, 387*54c56347SValentin Clement mlir::ConversionPatternRewriter &rewriter) const override { 388*54c56347SValentin Clement auto attrs = collectIndices(rewriter, insertVal.coor()); 389*54c56347SValentin Clement toRowMajor(attrs, adaptor.getOperands()[0].getType()); 390*54c56347SValentin Clement auto position = mlir::ArrayAttr::get(insertVal.getContext(), attrs); 391*54c56347SValentin Clement rewriter.replaceOpWithNewOp<mlir::LLVM::InsertValueOp>( 392*54c56347SValentin Clement insertVal, ty, adaptor.getOperands()[0], adaptor.getOperands()[1], 393*54c56347SValentin Clement position); 394*54c56347SValentin Clement return success(); 395*54c56347SValentin Clement } 396*54c56347SValentin Clement }; 397*54c56347SValentin Clement 3983ae8e442SValentin Clement /// InsertOnRange inserts a value into a sequence over a range of offsets. 3993ae8e442SValentin Clement struct InsertOnRangeOpConversion 4003ae8e442SValentin Clement : public FIROpAndTypeConversion<fir::InsertOnRangeOp> { 4013ae8e442SValentin Clement using FIROpAndTypeConversion::FIROpAndTypeConversion; 4023ae8e442SValentin Clement 4033ae8e442SValentin Clement // Increments an array of subscripts in a row major fasion. 4043ae8e442SValentin Clement void incrementSubscripts(const SmallVector<uint64_t> &dims, 4053ae8e442SValentin Clement SmallVector<uint64_t> &subscripts) const { 4063ae8e442SValentin Clement for (size_t i = dims.size(); i > 0; --i) { 4073ae8e442SValentin Clement if (++subscripts[i - 1] < dims[i - 1]) { 4083ae8e442SValentin Clement return; 4093ae8e442SValentin Clement } 4103ae8e442SValentin Clement subscripts[i - 1] = 0; 4113ae8e442SValentin Clement } 4123ae8e442SValentin Clement } 4133ae8e442SValentin Clement 4143ae8e442SValentin Clement mlir::LogicalResult 4153ae8e442SValentin Clement doRewrite(fir::InsertOnRangeOp range, mlir::Type ty, OpAdaptor adaptor, 4163ae8e442SValentin Clement mlir::ConversionPatternRewriter &rewriter) const override { 4173ae8e442SValentin Clement 4183ae8e442SValentin Clement llvm::SmallVector<uint64_t> dims; 4193ae8e442SValentin Clement auto type = adaptor.getOperands()[0].getType(); 4203ae8e442SValentin Clement 4213ae8e442SValentin Clement // Iteratively extract the array dimensions from the type. 4223ae8e442SValentin Clement while (auto t = type.dyn_cast<mlir::LLVM::LLVMArrayType>()) { 4233ae8e442SValentin Clement dims.push_back(t.getNumElements()); 4243ae8e442SValentin Clement type = t.getElementType(); 4253ae8e442SValentin Clement } 4263ae8e442SValentin Clement 4273ae8e442SValentin Clement SmallVector<uint64_t> lBounds; 4283ae8e442SValentin Clement SmallVector<uint64_t> uBounds; 4293ae8e442SValentin Clement 4303ae8e442SValentin Clement // Extract integer value from the attribute 4313ae8e442SValentin Clement SmallVector<int64_t> coordinates = llvm::to_vector<4>( 4323ae8e442SValentin Clement llvm::map_range(range.coor(), [](Attribute a) -> int64_t { 4333ae8e442SValentin Clement return a.cast<IntegerAttr>().getInt(); 4343ae8e442SValentin Clement })); 4353ae8e442SValentin Clement 4363ae8e442SValentin Clement // Unzip the upper and lower bound and convert to a row major format. 4373ae8e442SValentin Clement for (auto i = coordinates.rbegin(), e = coordinates.rend(); i != e; ++i) { 4383ae8e442SValentin Clement uBounds.push_back(*i++); 4393ae8e442SValentin Clement lBounds.push_back(*i); 4403ae8e442SValentin Clement } 4413ae8e442SValentin Clement 4423ae8e442SValentin Clement auto &subscripts = lBounds; 4433ae8e442SValentin Clement auto loc = range.getLoc(); 4443ae8e442SValentin Clement mlir::Value lastOp = adaptor.getOperands()[0]; 4453ae8e442SValentin Clement mlir::Value insertVal = adaptor.getOperands()[1]; 4463ae8e442SValentin Clement 4473ae8e442SValentin Clement auto i64Ty = rewriter.getI64Type(); 4483ae8e442SValentin Clement while (subscripts != uBounds) { 4493ae8e442SValentin Clement // Convert uint64_t's to Attribute's. 4503ae8e442SValentin Clement SmallVector<mlir::Attribute> subscriptAttrs; 4513ae8e442SValentin Clement for (const auto &subscript : subscripts) 4523ae8e442SValentin Clement subscriptAttrs.push_back(IntegerAttr::get(i64Ty, subscript)); 4533ae8e442SValentin Clement lastOp = rewriter.create<mlir::LLVM::InsertValueOp>( 4543ae8e442SValentin Clement loc, ty, lastOp, insertVal, 4553ae8e442SValentin Clement ArrayAttr::get(range.getContext(), subscriptAttrs)); 4563ae8e442SValentin Clement 4573ae8e442SValentin Clement incrementSubscripts(dims, subscripts); 4583ae8e442SValentin Clement } 4593ae8e442SValentin Clement 4603ae8e442SValentin Clement // Convert uint64_t's to Attribute's. 4613ae8e442SValentin Clement SmallVector<mlir::Attribute> subscriptAttrs; 4623ae8e442SValentin Clement for (const auto &subscript : subscripts) 4633ae8e442SValentin Clement subscriptAttrs.push_back( 4643ae8e442SValentin Clement IntegerAttr::get(rewriter.getI64Type(), subscript)); 4653ae8e442SValentin Clement mlir::ArrayRef<mlir::Attribute> arrayRef(subscriptAttrs); 4663ae8e442SValentin Clement 4673ae8e442SValentin Clement rewriter.replaceOpWithNewOp<mlir::LLVM::InsertValueOp>( 4683ae8e442SValentin Clement range, ty, lastOp, insertVal, 4693ae8e442SValentin Clement ArrayAttr::get(range.getContext(), arrayRef)); 4703ae8e442SValentin Clement 4713ae8e442SValentin Clement return success(); 4723ae8e442SValentin Clement } 4733ae8e442SValentin Clement }; 474044d5b5dSValentin Clement } // namespace 475044d5b5dSValentin Clement 476044d5b5dSValentin Clement namespace { 477044d5b5dSValentin Clement /// Convert FIR dialect to LLVM dialect 478044d5b5dSValentin Clement /// 479044d5b5dSValentin Clement /// This pass lowers all FIR dialect operations to LLVM IR dialect. An 480044d5b5dSValentin Clement /// MLIR pass is used to lower residual Std dialect to LLVM IR dialect. 481044d5b5dSValentin Clement /// 482044d5b5dSValentin Clement /// This pass is not complete yet. We are upstreaming it in small patches. 483044d5b5dSValentin Clement class FIRToLLVMLowering : public fir::FIRToLLVMLoweringBase<FIRToLLVMLowering> { 484044d5b5dSValentin Clement public: 485044d5b5dSValentin Clement mlir::ModuleOp getModule() { return getOperation(); } 486044d5b5dSValentin Clement 487044d5b5dSValentin Clement void runOnOperation() override final { 488044d5b5dSValentin Clement auto *context = getModule().getContext(); 489044d5b5dSValentin Clement fir::LLVMTypeConverter typeConverter{getModule()}; 490044d5b5dSValentin Clement mlir::OwningRewritePatternList pattern(context); 491*54c56347SValentin Clement pattern.insert< 492*54c56347SValentin Clement AddrOfOpConversion, ExtractValueOpConversion, HasValueOpConversion, 493*54c56347SValentin Clement GlobalOpConversion, InsertOnRangeOpConversion, InsertValueOpConversion, 494*54c56347SValentin Clement SelectOpConversion, SelectRankOpConversion, UndefOpConversion, 495*54c56347SValentin Clement UnreachableOpConversion, ZeroOpConversion>(typeConverter); 496044d5b5dSValentin Clement mlir::populateStdToLLVMConversionPatterns(typeConverter, pattern); 497044d5b5dSValentin Clement mlir::arith::populateArithmeticToLLVMConversionPatterns(typeConverter, 498044d5b5dSValentin Clement pattern); 499044d5b5dSValentin Clement mlir::ConversionTarget target{*context}; 500044d5b5dSValentin Clement target.addLegalDialect<mlir::LLVM::LLVMDialect>(); 501044d5b5dSValentin Clement 502044d5b5dSValentin Clement // required NOPs for applying a full conversion 503044d5b5dSValentin Clement target.addLegalOp<mlir::ModuleOp>(); 504044d5b5dSValentin Clement 505044d5b5dSValentin Clement // apply the patterns 506044d5b5dSValentin Clement if (mlir::failed(mlir::applyFullConversion(getModule(), target, 507044d5b5dSValentin Clement std::move(pattern)))) { 508044d5b5dSValentin Clement signalPassFailure(); 509044d5b5dSValentin Clement } 510044d5b5dSValentin Clement } 511044d5b5dSValentin Clement }; 512044d5b5dSValentin Clement } // namespace 513044d5b5dSValentin Clement 514044d5b5dSValentin Clement std::unique_ptr<mlir::Pass> fir::createFIRToLLVMPass() { 515044d5b5dSValentin Clement return std::make_unique<FIRToLLVMLowering>(); 516044d5b5dSValentin Clement } 517