FLANG
Utils.h
1//===-- Optimizer/Support/Utils.h -------------------------------*- C++ -*-===//
2//
3// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4// See https://llvm.org/LICENSE.txt for license information.
5// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6//
7//===----------------------------------------------------------------------===//
8//
9// Coding style: https://mlir.llvm.org/getting_started/DeveloperGuide/
10//
11//===----------------------------------------------------------------------===//
12
13#ifndef FORTRAN_OPTIMIZER_SUPPORT_UTILS_H
14#define FORTRAN_OPTIMIZER_SUPPORT_UTILS_H
15
16#include "flang/Optimizer/Builder/FIRBuilder.h"
17#include "flang/Optimizer/Builder/Todo.h"
18#include "flang/Optimizer/Dialect/CUF/Attributes/CUFAttr.h"
19#include "flang/Optimizer/Dialect/FIROps.h"
20#include "flang/Optimizer/Dialect/FIRType.h"
21#include "flang/Optimizer/Support/FatalError.h"
22#include "flang/Support/default-kinds.h"
23#include "mlir/Dialect/Arith/IR/Arith.h"
24#include "mlir/Dialect/Func/IR/FuncOps.h"
25#include "mlir/IR/BuiltinAttributes.h"
26#include "mlir/IR/BuiltinOps.h"
27#include "mlir/Interfaces/FunctionInterfaces.h"
28#include "llvm/ADT/DenseMap.h"
29#include "llvm/ADT/StringRef.h"
30#include <string>
31
32#include "flang/Optimizer/CodeGen/TypeConverter.h"
33
34namespace fir {
36inline std::int64_t toInt(mlir::arith::ConstantOp cop) {
37 return mlir::cast<mlir::IntegerAttr>(cop.getValue())
38 .getValue()
39 .getSExtValue();
40}
41
42// Translate front-end KINDs for use in the IR and code gen.
43inline std::vector<fir::KindTy>
44fromDefaultKinds(const Fortran::common::IntrinsicTypeDefaultKinds &defKinds) {
45 return {static_cast<fir::KindTy>(defKinds.GetDefaultKind(
46 Fortran::common::TypeCategory::Character)),
47 static_cast<fir::KindTy>(
48 defKinds.GetDefaultKind(Fortran::common::TypeCategory::Complex)),
49 static_cast<fir::KindTy>(defKinds.doublePrecisionKind()),
50 static_cast<fir::KindTy>(
51 defKinds.GetDefaultKind(Fortran::common::TypeCategory::Integer)),
52 static_cast<fir::KindTy>(
53 defKinds.GetDefaultKind(Fortran::common::TypeCategory::Logical)),
54 static_cast<fir::KindTy>(
55 defKinds.GetDefaultKind(Fortran::common::TypeCategory::Real))};
56}
57
58inline std::string mlirTypeToString(mlir::Type type) {
59 std::string result{};
60 llvm::raw_string_ostream sstream(result);
61 sstream << type;
62 return result;
63}
64
65inline std::optional<int> mlirFloatTypeToKind(mlir::Type type) {
66 if (type.isF16())
67 return 2;
68 else if (type.isBF16())
69 return 3;
70 else if (type.isF32())
71 return 4;
72 else if (type.isF64())
73 return 8;
74 else if (type.isF80())
75 return 10;
76 else if (type.isF128())
77 return 16;
78 return std::nullopt;
79}
80
81inline std::string mlirTypeToIntrinsicFortran(fir::FirOpBuilder &builder,
82 mlir::Type type,
83 mlir::Location loc,
84 const llvm::Twine &name) {
85 if (auto floatTy = mlir::dyn_cast<mlir::FloatType>(type)) {
86 if (std::optional<int> kind = mlirFloatTypeToKind(type))
87 return "REAL(KIND="s + std::to_string(*kind) + ")";
88 } else if (auto cplxTy = mlir::dyn_cast<mlir::ComplexType>(type)) {
89 if (std::optional<int> kind = mlirFloatTypeToKind(cplxTy.getElementType()))
90 return "COMPLEX(KIND="s + std::to_string(*kind) + ")";
91 } else if (type.isUnsignedInteger()) {
92 if (type.isInteger(8))
93 return "UNSIGNED(KIND=1)";
94 else if (type.isInteger(16))
95 return "UNSIGNED(KIND=2)";
96 else if (type.isInteger(32))
97 return "UNSIGNED(KIND=4)";
98 else if (type.isInteger(64))
99 return "UNSIGNED(KIND=8)";
100 else if (type.isInteger(128))
101 return "UNSIGNED(KIND=16)";
102 } else if (type.isInteger(8))
103 return "INTEGER(KIND=1)";
104 else if (type.isInteger(16))
105 return "INTEGER(KIND=2)";
106 else if (type.isInteger(32))
107 return "INTEGER(KIND=4)";
108 else if (type.isInteger(64))
109 return "INTEGER(KIND=8)";
110 else if (type.isInteger(128))
111 return "INTEGER(KIND=16)";
112 else if (type == fir::LogicalType::get(builder.getContext(), 1))
113 return "LOGICAL(KIND=1)";
114 else if (type == fir::LogicalType::get(builder.getContext(), 2))
115 return "LOGICAL(KIND=2)";
116 else if (type == fir::LogicalType::get(builder.getContext(), 4))
117 return "LOGICAL(KIND=4)";
118 else if (type == fir::LogicalType::get(builder.getContext(), 8))
119 return "LOGICAL(KIND=8)";
120
121 fir::emitFatalError(loc, "unsupported type in " + name + ": " +
122 fir::mlirTypeToString(type));
123}
124
125inline void intrinsicTypeTODO(fir::FirOpBuilder &builder, mlir::Type type,
126 mlir::Location loc,
127 const llvm::Twine &intrinsicName) {
128 TODO(loc,
129 "intrinsic: " +
130 fir::mlirTypeToIntrinsicFortran(builder, type, loc, intrinsicName) +
131 " in " + intrinsicName);
132}
133
134inline void intrinsicTypeTODO2(fir::FirOpBuilder &builder, mlir::Type type1,
135 mlir::Type type2, mlir::Location loc,
136 const llvm::Twine &intrinsicName) {
137 TODO(loc,
138 "intrinsic: {" +
139 fir::mlirTypeToIntrinsicFortran(builder, type2, loc, intrinsicName) +
140 ", " +
141 fir::mlirTypeToIntrinsicFortran(builder, type2, loc, intrinsicName) +
142 "} in " + intrinsicName);
143}
144
145inline std::pair<Fortran::common::TypeCategory, KindMapping::KindTy>
146mlirTypeToCategoryKind(mlir::Location loc, mlir::Type type) {
147 if (auto floatTy = mlir::dyn_cast<mlir::FloatType>(type)) {
148 if (std::optional<int> kind = mlirFloatTypeToKind(type))
149 return {Fortran::common::TypeCategory::Real, *kind};
150 } else if (auto cplxTy = mlir::dyn_cast<mlir::ComplexType>(type)) {
151 if (std::optional<int> kind = mlirFloatTypeToKind(cplxTy.getElementType()))
152 return {Fortran::common::TypeCategory::Complex, *kind};
153 } else if (type.isInteger(8))
154 return {type.isUnsignedInteger() ? Fortran::common::TypeCategory::Unsigned
155 : Fortran::common::TypeCategory::Integer,
156 1};
157 else if (type.isInteger(16))
158 return {type.isUnsignedInteger() ? Fortran::common::TypeCategory::Unsigned
159 : Fortran::common::TypeCategory::Integer,
160 2};
161 else if (type.isInteger(32))
162 return {type.isUnsignedInteger() ? Fortran::common::TypeCategory::Unsigned
163 : Fortran::common::TypeCategory::Integer,
164 4};
165 else if (type.isInteger(64))
166 return {type.isUnsignedInteger() ? Fortran::common::TypeCategory::Unsigned
167 : Fortran::common::TypeCategory::Integer,
168 8};
169 else if (type.isInteger(128))
170 return {type.isUnsignedInteger() ? Fortran::common::TypeCategory::Unsigned
171 : Fortran::common::TypeCategory::Integer,
172 16};
173 else if (auto logicalType = mlir::dyn_cast<fir::LogicalType>(type))
174 return {Fortran::common::TypeCategory::Logical, logicalType.getFKind()};
175 else if (auto charType = mlir::dyn_cast<fir::CharacterType>(type))
176 return {Fortran::common::TypeCategory::Character, charType.getFKind()};
177 else if (mlir::isa<fir::RecordType>(type))
178 return {Fortran::common::TypeCategory::Derived, 0};
179 fir::emitFatalError(loc, "unsupported type: " + fir::mlirTypeToString(type));
180}
181
186fir::TypeInfoOp
187lookupTypeInfoOp(fir::RecordType recordType, mlir::ModuleOp module,
188 const mlir::SymbolTable *symbolTable = nullptr);
189
194fir::TypeInfoOp
195lookupTypeInfoOp(llvm::StringRef name, mlir::ModuleOp module,
196 const mlir::SymbolTable *symbolTable = nullptr);
197
201std::optional<llvm::ArrayRef<int64_t>> getComponentLowerBoundsIfNonDefault(
202 fir::RecordType recordType, llvm::StringRef component,
203 mlir::ModuleOp module, const mlir::SymbolTable *symbolTable = nullptr);
204
207std::optional<bool>
208isRecordWithFinalRoutine(fir::RecordType recordType, mlir::ModuleOp module,
209 const mlir::SymbolTable *symbolTable = nullptr);
210
212mlir::LLVM::ConstantOp
213genConstantIndex(mlir::Location loc, mlir::Type ity,
214 mlir::ConversionPatternRewriter &rewriter,
215 std::int64_t offset);
216
221mlir::Value computeElementDistance(mlir::Location loc,
222 mlir::Type llvmObjectType, mlir::Type idxTy,
223 mlir::ConversionPatternRewriter &rewriter,
224 const mlir::DataLayout &dataLayout);
225
226// Compute the alloc scale size (constant factors encoded in the array type).
227// We do this for arrays without a constant interior or arrays of character with
228// dynamic length arrays, since those are the only ones that get decayed to a
229// pointer to the element type.
230mlir::Value genAllocationScaleSize(mlir::Location loc, mlir::Type dataTy,
231 mlir::Type ity,
232 mlir::ConversionPatternRewriter &rewriter);
233
238mlir::Value integerCast(const fir::LLVMTypeConverter &converter,
239 mlir::Location loc,
240 mlir::ConversionPatternRewriter &rewriter,
241 mlir::Type ty, mlir::Value val, bool fold = false);
242
248std::optional<bool> isNewAllocationResult(mlir::OpResult result);
249
252std::string getPresentableFunctionName(mlir::FunctionOpInterface func);
253} // namespace fir
254
255#endif // FORTRAN_OPTIMIZER_SUPPORT_UTILS_H
Definition default-kinds.h:26
Definition AbstractConverter.h:37
mlir::Value integerCast(const fir::LLVMTypeConverter &converter, mlir::Location loc, mlir::ConversionPatternRewriter &rewriter, mlir::Type ty, mlir::Value val, bool fold=false)
Definition Utils.cpp:110
std::string getPresentableFunctionName(mlir::FunctionOpInterface func)
Definition Utils.cpp:152
std::optional< bool > isNewAllocationResult(mlir::OpResult result)
Definition Utils.cpp:135
std::optional< bool > isRecordWithFinalRoutine(fir::RecordType recordType, mlir::ModuleOp module, const mlir::SymbolTable *symbolTable=nullptr)
Definition Utils.cpp:55
std::optional< llvm::ArrayRef< int64_t > > getComponentLowerBoundsIfNonDefault(fir::RecordType recordType, llvm::StringRef component, mlir::ModuleOp module, const mlir::SymbolTable *symbolTable=nullptr)
Definition Utils.cpp:40
std::int64_t toInt(mlir::arith::ConstantOp cop)
Return the integer value of a arith::ConstantOp.
Definition Utils.h:36
fir::TypeInfoOp lookupTypeInfoOp(fir::RecordType recordType, mlir::ModuleOp module, const mlir::SymbolTable *symbolTable=nullptr)
Definition Utils.cpp:18
mlir::LLVM::ConstantOp genConstantIndex(mlir::Location loc, mlir::Type ity, mlir::ConversionPatternRewriter &rewriter, std::int64_t offset)
Generate a LLVM constant value of type ity, using the provided offset.
Definition Utils.cpp:65
void emitFatalError(mlir::Location loc, const llvm::Twine &message, bool genCrashDiag=true)
Definition FatalError.h:25
mlir::Value computeElementDistance(mlir::Location loc, mlir::Type llvmObjectType, mlir::Type idxTy, mlir::ConversionPatternRewriter &rewriter, const mlir::DataLayout &dataLayout)
Definition Utils.cpp:73