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