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"
22*3ae8e442SValentin 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 
49*3ae8e442SValentin Clement /// FIR conversion pattern template
50*3ae8e442SValentin Clement template <typename FromOp>
51*3ae8e442SValentin Clement class FIROpAndTypeConversion : public FIROpConversion<FromOp> {
52*3ae8e442SValentin Clement public:
53*3ae8e442SValentin Clement   using FIROpConversion<FromOp>::FIROpConversion;
54*3ae8e442SValentin Clement   using OpAdaptor = typename FromOp::Adaptor;
55*3ae8e442SValentin Clement 
56*3ae8e442SValentin Clement   mlir::LogicalResult
57*3ae8e442SValentin Clement   matchAndRewrite(FromOp op, OpAdaptor adaptor,
58*3ae8e442SValentin Clement                   mlir::ConversionPatternRewriter &rewriter) const final {
59*3ae8e442SValentin Clement     mlir::Type ty = this->convertType(op.getType());
60*3ae8e442SValentin Clement     return doRewrite(op, ty, adaptor, rewriter);
61*3ae8e442SValentin Clement   }
62*3ae8e442SValentin Clement 
63*3ae8e442SValentin Clement   virtual mlir::LogicalResult
64*3ae8e442SValentin Clement   doRewrite(FromOp addr, mlir::Type ty, OpAdaptor adaptor,
65*3ae8e442SValentin Clement             mlir::ConversionPatternRewriter &rewriter) const = 0;
66*3ae8e442SValentin Clement };
67*3ae8e442SValentin 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 
177044d5b5dSValentin Clement // convert to LLVM IR dialect `undef`
178044d5b5dSValentin Clement struct UndefOpConversion : public FIROpConversion<fir::UndefOp> {
179044d5b5dSValentin Clement   using FIROpConversion::FIROpConversion;
180044d5b5dSValentin Clement 
181044d5b5dSValentin Clement   mlir::LogicalResult
182044d5b5dSValentin Clement   matchAndRewrite(fir::UndefOp undef, OpAdaptor,
183044d5b5dSValentin Clement                   mlir::ConversionPatternRewriter &rewriter) const override {
184044d5b5dSValentin Clement     rewriter.replaceOpWithNewOp<mlir::LLVM::UndefOp>(
185044d5b5dSValentin Clement         undef, convertType(undef.getType()));
186044d5b5dSValentin Clement     return success();
187044d5b5dSValentin Clement   }
188044d5b5dSValentin Clement };
189a7a61359SValentin Clement 
19032e08248SAndrzej Warzynski // convert to LLVM IR dialect `unreachable`
19132e08248SAndrzej Warzynski struct UnreachableOpConversion : public FIROpConversion<fir::UnreachableOp> {
19232e08248SAndrzej Warzynski   using FIROpConversion::FIROpConversion;
19332e08248SAndrzej Warzynski 
19432e08248SAndrzej Warzynski   mlir::LogicalResult
19532e08248SAndrzej Warzynski   matchAndRewrite(fir::UnreachableOp unreach, OpAdaptor adaptor,
19632e08248SAndrzej Warzynski                   mlir::ConversionPatternRewriter &rewriter) const override {
19732e08248SAndrzej Warzynski     rewriter.replaceOpWithNewOp<mlir::LLVM::UnreachableOp>(unreach);
19832e08248SAndrzej Warzynski     return success();
19932e08248SAndrzej Warzynski   }
20032e08248SAndrzej Warzynski };
20132e08248SAndrzej Warzynski 
202a7a61359SValentin Clement struct ZeroOpConversion : public FIROpConversion<fir::ZeroOp> {
203a7a61359SValentin Clement   using FIROpConversion::FIROpConversion;
204a7a61359SValentin Clement 
205a7a61359SValentin Clement   mlir::LogicalResult
206a7a61359SValentin Clement   matchAndRewrite(fir::ZeroOp zero, OpAdaptor,
207a7a61359SValentin Clement                   mlir::ConversionPatternRewriter &rewriter) const override {
208a7a61359SValentin Clement     auto ty = convertType(zero.getType());
209a7a61359SValentin Clement     if (ty.isa<mlir::LLVM::LLVMPointerType>()) {
210a7a61359SValentin Clement       rewriter.replaceOpWithNewOp<mlir::LLVM::NullOp>(zero, ty);
211a7a61359SValentin Clement     } else if (ty.isa<mlir::IntegerType>()) {
212a7a61359SValentin Clement       rewriter.replaceOpWithNewOp<mlir::LLVM::ConstantOp>(
213a7a61359SValentin Clement           zero, ty, mlir::IntegerAttr::get(zero.getType(), 0));
214a7a61359SValentin Clement     } else if (mlir::LLVM::isCompatibleFloatingPointType(ty)) {
215a7a61359SValentin Clement       rewriter.replaceOpWithNewOp<mlir::LLVM::ConstantOp>(
216a7a61359SValentin Clement           zero, ty, mlir::FloatAttr::get(zero.getType(), 0.0));
217a7a61359SValentin Clement     } else {
218a7a61359SValentin Clement       // TODO: create ConstantAggregateZero for FIR aggregate/array types.
21952d813edSValentin Clement       return rewriter.notifyMatchFailure(
22052d813edSValentin Clement           zero,
221a7a61359SValentin Clement           "conversion of fir.zero with aggregate type not implemented yet");
222a7a61359SValentin Clement     }
223a7a61359SValentin Clement     return success();
224a7a61359SValentin Clement   }
225a7a61359SValentin Clement };
22632e08248SAndrzej Warzynski 
227*3ae8e442SValentin Clement /// InsertOnRange inserts a value into a sequence over a range of offsets.
228*3ae8e442SValentin Clement struct InsertOnRangeOpConversion
229*3ae8e442SValentin Clement     : public FIROpAndTypeConversion<fir::InsertOnRangeOp> {
230*3ae8e442SValentin Clement   using FIROpAndTypeConversion::FIROpAndTypeConversion;
231*3ae8e442SValentin Clement 
232*3ae8e442SValentin Clement   // Increments an array of subscripts in a row major fasion.
233*3ae8e442SValentin Clement   void incrementSubscripts(const SmallVector<uint64_t> &dims,
234*3ae8e442SValentin Clement                            SmallVector<uint64_t> &subscripts) const {
235*3ae8e442SValentin Clement     for (size_t i = dims.size(); i > 0; --i) {
236*3ae8e442SValentin Clement       if (++subscripts[i - 1] < dims[i - 1]) {
237*3ae8e442SValentin Clement         return;
238*3ae8e442SValentin Clement       }
239*3ae8e442SValentin Clement       subscripts[i - 1] = 0;
240*3ae8e442SValentin Clement     }
241*3ae8e442SValentin Clement   }
242*3ae8e442SValentin Clement 
243*3ae8e442SValentin Clement   mlir::LogicalResult
244*3ae8e442SValentin Clement   doRewrite(fir::InsertOnRangeOp range, mlir::Type ty, OpAdaptor adaptor,
245*3ae8e442SValentin Clement             mlir::ConversionPatternRewriter &rewriter) const override {
246*3ae8e442SValentin Clement 
247*3ae8e442SValentin Clement     llvm::SmallVector<uint64_t> dims;
248*3ae8e442SValentin Clement     auto type = adaptor.getOperands()[0].getType();
249*3ae8e442SValentin Clement 
250*3ae8e442SValentin Clement     // Iteratively extract the array dimensions from the type.
251*3ae8e442SValentin Clement     while (auto t = type.dyn_cast<mlir::LLVM::LLVMArrayType>()) {
252*3ae8e442SValentin Clement       dims.push_back(t.getNumElements());
253*3ae8e442SValentin Clement       type = t.getElementType();
254*3ae8e442SValentin Clement     }
255*3ae8e442SValentin Clement 
256*3ae8e442SValentin Clement     SmallVector<uint64_t> lBounds;
257*3ae8e442SValentin Clement     SmallVector<uint64_t> uBounds;
258*3ae8e442SValentin Clement 
259*3ae8e442SValentin Clement     // Extract integer value from the attribute
260*3ae8e442SValentin Clement     SmallVector<int64_t> coordinates = llvm::to_vector<4>(
261*3ae8e442SValentin Clement         llvm::map_range(range.coor(), [](Attribute a) -> int64_t {
262*3ae8e442SValentin Clement           return a.cast<IntegerAttr>().getInt();
263*3ae8e442SValentin Clement         }));
264*3ae8e442SValentin Clement 
265*3ae8e442SValentin Clement     // Unzip the upper and lower bound and convert to a row major format.
266*3ae8e442SValentin Clement     for (auto i = coordinates.rbegin(), e = coordinates.rend(); i != e; ++i) {
267*3ae8e442SValentin Clement       uBounds.push_back(*i++);
268*3ae8e442SValentin Clement       lBounds.push_back(*i);
269*3ae8e442SValentin Clement     }
270*3ae8e442SValentin Clement 
271*3ae8e442SValentin Clement     auto &subscripts = lBounds;
272*3ae8e442SValentin Clement     auto loc = range.getLoc();
273*3ae8e442SValentin Clement     mlir::Value lastOp = adaptor.getOperands()[0];
274*3ae8e442SValentin Clement     mlir::Value insertVal = adaptor.getOperands()[1];
275*3ae8e442SValentin Clement 
276*3ae8e442SValentin Clement     auto i64Ty = rewriter.getI64Type();
277*3ae8e442SValentin Clement     while (subscripts != uBounds) {
278*3ae8e442SValentin Clement       // Convert uint64_t's to Attribute's.
279*3ae8e442SValentin Clement       SmallVector<mlir::Attribute> subscriptAttrs;
280*3ae8e442SValentin Clement       for (const auto &subscript : subscripts)
281*3ae8e442SValentin Clement         subscriptAttrs.push_back(IntegerAttr::get(i64Ty, subscript));
282*3ae8e442SValentin Clement       lastOp = rewriter.create<mlir::LLVM::InsertValueOp>(
283*3ae8e442SValentin Clement           loc, ty, lastOp, insertVal,
284*3ae8e442SValentin Clement           ArrayAttr::get(range.getContext(), subscriptAttrs));
285*3ae8e442SValentin Clement 
286*3ae8e442SValentin Clement       incrementSubscripts(dims, subscripts);
287*3ae8e442SValentin Clement     }
288*3ae8e442SValentin Clement 
289*3ae8e442SValentin Clement     // Convert uint64_t's to Attribute's.
290*3ae8e442SValentin Clement     SmallVector<mlir::Attribute> subscriptAttrs;
291*3ae8e442SValentin Clement     for (const auto &subscript : subscripts)
292*3ae8e442SValentin Clement       subscriptAttrs.push_back(
293*3ae8e442SValentin Clement           IntegerAttr::get(rewriter.getI64Type(), subscript));
294*3ae8e442SValentin Clement     mlir::ArrayRef<mlir::Attribute> arrayRef(subscriptAttrs);
295*3ae8e442SValentin Clement 
296*3ae8e442SValentin Clement     rewriter.replaceOpWithNewOp<mlir::LLVM::InsertValueOp>(
297*3ae8e442SValentin Clement         range, ty, lastOp, insertVal,
298*3ae8e442SValentin Clement         ArrayAttr::get(range.getContext(), arrayRef));
299*3ae8e442SValentin Clement 
300*3ae8e442SValentin Clement     return success();
301*3ae8e442SValentin Clement   }
302*3ae8e442SValentin Clement };
303044d5b5dSValentin Clement } // namespace
304044d5b5dSValentin Clement 
305044d5b5dSValentin Clement namespace {
306044d5b5dSValentin Clement /// Convert FIR dialect to LLVM dialect
307044d5b5dSValentin Clement ///
308044d5b5dSValentin Clement /// This pass lowers all FIR dialect operations to LLVM IR dialect. An
309044d5b5dSValentin Clement /// MLIR pass is used to lower residual Std dialect to LLVM IR dialect.
310044d5b5dSValentin Clement ///
311044d5b5dSValentin Clement /// This pass is not complete yet. We are upstreaming it in small patches.
312044d5b5dSValentin Clement class FIRToLLVMLowering : public fir::FIRToLLVMLoweringBase<FIRToLLVMLowering> {
313044d5b5dSValentin Clement public:
314044d5b5dSValentin Clement   mlir::ModuleOp getModule() { return getOperation(); }
315044d5b5dSValentin Clement 
316044d5b5dSValentin Clement   void runOnOperation() override final {
317044d5b5dSValentin Clement     auto *context = getModule().getContext();
318044d5b5dSValentin Clement     fir::LLVMTypeConverter typeConverter{getModule()};
319044d5b5dSValentin Clement     mlir::OwningRewritePatternList pattern(context);
320*3ae8e442SValentin Clement     pattern.insert<AddrOfOpConversion, HasValueOpConversion, GlobalOpConversion,
321*3ae8e442SValentin Clement                    InsertOnRangeOpConversion, UndefOpConversion,
322*3ae8e442SValentin Clement                    UnreachableOpConversion, ZeroOpConversion>(typeConverter);
323044d5b5dSValentin Clement     mlir::populateStdToLLVMConversionPatterns(typeConverter, pattern);
324044d5b5dSValentin Clement     mlir::arith::populateArithmeticToLLVMConversionPatterns(typeConverter,
325044d5b5dSValentin Clement                                                             pattern);
326044d5b5dSValentin Clement     mlir::ConversionTarget target{*context};
327044d5b5dSValentin Clement     target.addLegalDialect<mlir::LLVM::LLVMDialect>();
328044d5b5dSValentin Clement 
329044d5b5dSValentin Clement     // required NOPs for applying a full conversion
330044d5b5dSValentin Clement     target.addLegalOp<mlir::ModuleOp>();
331044d5b5dSValentin Clement 
332044d5b5dSValentin Clement     // apply the patterns
333044d5b5dSValentin Clement     if (mlir::failed(mlir::applyFullConversion(getModule(), target,
334044d5b5dSValentin Clement                                                std::move(pattern)))) {
335044d5b5dSValentin Clement       signalPassFailure();
336044d5b5dSValentin Clement     }
337044d5b5dSValentin Clement   }
338044d5b5dSValentin Clement };
339044d5b5dSValentin Clement } // namespace
340044d5b5dSValentin Clement 
341044d5b5dSValentin Clement std::unique_ptr<mlir::Pass> fir::createFIRToLLVMPass() {
342044d5b5dSValentin Clement   return std::make_unique<FIRToLLVMLowering>();
343044d5b5dSValentin Clement }
344