clang 23.0.0git
CIRGenModule.h
Go to the documentation of this file.
1//===--- CIRGenModule.h - Per-Module state for CIR gen ----------*- 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// This is the internal per-translation-unit state used for CIR translation.
10//
11//===----------------------------------------------------------------------===//
12
13#ifndef LLVM_CLANG_LIB_CIR_CODEGEN_CIRGENMODULE_H
14#define LLVM_CLANG_LIB_CIR_CODEGEN_CIRGENMODULE_H
15
16#include "CIRGenBuilder.h"
17#include "CIRGenCUDARuntime.h"
18#include "CIRGenCall.h"
19#include "CIRGenOpenMPRuntime.h"
20#include "CIRGenTypeCache.h"
21#include "CIRGenTypes.h"
22#include "CIRGenVTables.h"
23#include "CIRGenValue.h"
24
25#include "clang/AST/CharUnits.h"
28
29#include "TargetInfo.h"
30#include "mlir/Dialect/Ptr/IR/MemorySpaceInterfaces.h"
31#include "mlir/IR/Builders.h"
32#include "mlir/IR/BuiltinOps.h"
33#include "mlir/IR/MLIRContext.h"
34#include "clang/AST/Decl.h"
38#include "llvm/ADT/StringMap.h"
39#include "llvm/ADT/StringRef.h"
40#include "llvm/TargetParser/Triple.h"
41
42namespace clang {
43class ASTContext;
44class CodeGenOptions;
45class Decl;
46class GlobalDecl;
47class LangOptions;
48class TargetInfo;
49class VarDecl;
50
51namespace CIRGen {
52
53class CIRGenFunction;
54class CIRGenCXXABI;
55
56enum ForDefinition_t : bool { NotForDefinition = false, ForDefinition = true };
57
58/// This class organizes the cross-function state that is used while generating
59/// CIR code.
60class CIRGenModule : public CIRGenTypeCache {
61 CIRGenModule(CIRGenModule &) = delete;
62 CIRGenModule &operator=(CIRGenModule &) = delete;
63
64public:
65 CIRGenModule(mlir::MLIRContext &mlirContext, clang::ASTContext &astContext,
66 const clang::CodeGenOptions &cgo,
68
70
71private:
72 mutable std::unique_ptr<TargetCIRGenInfo> theTargetCIRGenInfo;
73
74 CIRGenBuilderTy builder;
75
76 /// Hold Clang AST information.
77 clang::ASTContext &astContext;
78
79 const clang::LangOptions &langOpts;
80
81 const clang::CodeGenOptions &codeGenOpts;
82
83 /// A "module" matches a c/cpp source file: containing a list of functions.
84 mlir::ModuleOp theModule;
85
87
88 const clang::TargetInfo &target;
89
90 std::unique_ptr<CIRGenCXXABI> abi;
91
92 CIRGenTypes genTypes;
93
94 /// Holds information about C++ vtables.
95 CIRGenVTables vtables;
96
97 /// Holds the CUDA runtime
98 std::unique_ptr<CIRGenCUDARuntime> cudaRuntime;
99
100 /// Holds the OpenMP runtime
101 std::unique_ptr<CIRGenOpenMPRuntime> openMPRuntime;
102
103 /// Per-function codegen information. Updated everytime emitCIR is called
104 /// for FunctionDecls's.
105 CIRGenFunction *curCGF = nullptr;
106
108
109 /// Accumulated record layout entries, materialized in release().
110 llvm::SmallVector<mlir::NamedAttribute> recordLayoutEntries;
111
112 llvm::DenseSet<clang::GlobalDecl> diagnosedConflictingDefinitions;
113
114 /// -------
115 /// Annotations
116 /// -------
117
118 /// We store each annotation as an attribute of GlobalOp and FuncOp rather
119 /// than collecting them into a single module-level list. The deferred map
120 /// lets us attach annotations at the end of codegen so the most up-to-date
121 /// ValueDecl (which carries all inherited annotations) is used.
122
123 /// Used for uniquing of annotation arguments.
124 llvm::DenseMap<unsigned, mlir::ArrayAttr> annotationArgs;
125
126 /// Store deferred function annotations so they can be emitted at the end
127 /// with the most up to date ValueDecl that will have all the inherited
128 /// annotations.
129 llvm::DenseMap<llvm::StringRef, const clang::ValueDecl *> deferredAnnotations;
130
131 /// A queue of (optional) vtables to consider emitting.
132 std::vector<const CXXRecordDecl *> deferredVTables;
133
134 /// A queue of (optional) vtables that may be emitted opportunistically.
135 std::vector<const CXXRecordDecl *> opportunisticVTables;
136
137 void createCUDARuntime();
138 void createOpenMPRuntime();
139
140 /// A helper for constructAttributeList that handles return attributes.
141 void constructFunctionReturnAttributes(const CIRGenFunctionInfo &info,
142 const Decl *targetDecl, bool isThunk,
143 mlir::NamedAttrList &retAttrs);
144 /// A helper for constructAttributeList that handles argument attributes.
145 void constructFunctionArgumentAttributes(
146 const CIRGenFunctionInfo &info, const clang::Decl *targetDecl,
147 bool isThunk, bool attrOnCallSite,
149 /// A helper function for constructAttributeList that determines whether a
150 /// return value might have been discarded.
151 bool mayDropFunctionReturn(const ASTContext &context, QualType retTy);
152 /// A helper function for constructAttributeList that determines whether
153 /// `noundef` on a return is possible.
154 bool hasStrictReturn(QualType retTy, const Decl *targetDecl);
155
156 llvm::DenseMap<const Expr *, mlir::Operation *>
157 materializedGlobalTemporaryMap;
158
159public:
160 mlir::ModuleOp getModule() const { return theModule; }
161 CIRGenBuilderTy &getBuilder() { return builder; }
162
163 /// Queue a record layout entry for materialization in release().
164 void addRecordLayout(mlir::StringAttr name, cir::RecordLayoutAttr attr) {
165 recordLayoutEntries.push_back(mlir::NamedAttribute(name, attr));
166 }
167 clang::ASTContext &getASTContext() const { return astContext; }
168 const clang::TargetInfo &getTarget() const { return target; }
169 const clang::CodeGenOptions &getCodeGenOpts() const { return codeGenOpts; }
170 clang::DiagnosticsEngine &getDiags() const { return diags; }
171 CIRGenTypes &getTypes() { return genTypes; }
172 const clang::LangOptions &getLangOpts() const { return langOpts; }
173
174 CIRGenCXXABI &getCXXABI() const { return *abi; }
175 mlir::MLIRContext &getMLIRContext() { return *builder.getContext(); }
176
178 // FIXME(cir): instead of creating a CIRDataLayout every time, set it as an
179 // attribute for the CIRModule class.
180 return cir::CIRDataLayout(theModule);
181 }
182
183 /// -------
184 /// Handling globals
185 /// -------
186
187 mlir::Operation *lastGlobalOp = nullptr;
188
189 /// Keep a map between lambda fields and names, this needs to be per module
190 /// since lambdas might get generated later as part of defered work, and since
191 /// the pointers are supposed to be uniqued, should be fine. Revisit this if
192 /// it ends up taking too much memory.
193 llvm::DenseMap<const clang::FieldDecl *, llvm::StringRef> lambdaFieldToName;
194 /// Map BlockAddrInfoAttr (function name, label name) to the corresponding CIR
195 /// LabelOp. This provides the main lookup table used to resolve block
196 /// addresses into their label operations.
197 llvm::DenseMap<cir::BlockAddrInfoAttr, cir::LabelOp> blockAddressInfoToLabel;
198 cir::LabelOp lookupBlockAddressInfo(cir::BlockAddrInfoAttr blockInfo);
199 void mapBlockAddress(cir::BlockAddrInfoAttr blockInfo, cir::LabelOp label);
200
201 /// Add a global value to the llvmUsed list.
202 void addUsedGlobal(cir::CIRGlobalValueInterface gv);
203
204 /// Add a global value to the llvmCompilerUsed list.
205 void addCompilerUsedGlobal(cir::CIRGlobalValueInterface gv);
206
207 /// Add a global to a list to be added to the llvm.compiler.used metadata.
208 void addUsedOrCompilerUsedGlobal(cir::CIRGlobalValueInterface gv);
209
210 /// Emit llvm.used and llvm.compiler.used globals.
211 void emitLLVMUsed();
212
213 /// Tell the consumer that this variable has been instantiated.
215
216 llvm::DenseMap<const Decl *, cir::GlobalOp> staticLocalDeclMap;
217 llvm::DenseMap<const VarDecl *, cir::GlobalOp> initializerConstants;
218
219 /// Cache for O(1) symbol lookups by name, replacing the O(N) linear scan
220 /// in SymbolTable::lookupSymbolIn that getGlobalValue used previously.
221 llvm::StringMap<mlir::Operation *> symbolLookupCache;
222
223 mlir::Operation *getGlobalValue(llvm::StringRef ref);
224
225 /// O(1) lookup of a FuncOp by name in the symbol cache.
226 /// Returns nullptr if the name is not found or is not a FuncOp.
227 cir::FuncOp lookupFuncOp(llvm::StringRef name) {
228 auto *op = getGlobalValue(name);
229 return op ? mlir::dyn_cast<cir::FuncOp>(op) : cir::FuncOp{};
230 }
231
232 void insertGlobalSymbol(mlir::Operation *op) {
233 if (auto sym = mlir::dyn_cast<mlir::SymbolOpInterface>(op))
234 symbolLookupCache[sym.getName()] = op;
235 }
236 void eraseGlobalSymbol(mlir::Operation *op) {
237 if (auto sym = mlir::dyn_cast<mlir::SymbolOpInterface>(op)) {
238 auto it = symbolLookupCache.find(sym.getName());
239 if (it != symbolLookupCache.end() && it->second == op)
240 symbolLookupCache.erase(it);
241 }
242 }
243
244 cir::GlobalOp getStaticLocalDeclAddress(const VarDecl *d) {
245 return staticLocalDeclMap[d];
246 }
247
248 void setStaticLocalDeclAddress(const VarDecl *d, cir::GlobalOp c) {
249 staticLocalDeclMap[d] = c;
250 }
251
252 cir::GlobalOp getOrCreateStaticVarDecl(const VarDecl &d,
253 cir::GlobalLinkageKind linkage);
254
255 Address createUnnamedGlobalFrom(const VarDecl &d, mlir::Attribute constAttr,
256 CharUnits align);
257
258 /// If the specified mangled name is not in the module, create and return an
259 /// mlir::GlobalOp value
260 cir::GlobalOp getOrCreateCIRGlobal(llvm::StringRef mangledName, mlir::Type ty,
261 LangAS langAS, const VarDecl *d,
262 ForDefinition_t isForDefinition);
263
264 cir::GlobalOp getOrCreateCIRGlobal(const VarDecl *d, mlir::Type ty,
265 ForDefinition_t isForDefinition);
266
267 cir::GlobalOp
268 createGlobalOp(mlir::Location loc, llvm::StringRef name, mlir::Type t,
269 bool isConstant = false,
270 mlir::ptr::MemorySpaceAttrInterface addrSpace = {},
271 mlir::Operation *insertPoint = nullptr);
272
273 /// Add a global constructor or destructor to the module.
274 /// The priority is optional, if not specified, the default priority is used.
275 void addGlobalCtor(cir::FuncOp ctor,
276 std::optional<int> priority = std::nullopt);
277 void addGlobalDtor(cir::FuncOp dtor,
278 std::optional<int> priority = std::nullopt);
279
281 // In C23 (N3096) $6.7.10:
282 // """
283 // If any object is initialized with an empty initializer, then it is
284 // subject to default initialization:
285 // - if it is an aggregate, every member is initialized (recursively)
286 // according to these rules, and any padding is initialized to zero bits;
287 // - if it is a union, the first named member is initialized (recursively)
288 // according to these rules, and any padding is initialized to zero bits.
289 //
290 // If the aggregate or union contains elements or members that are
291 // aggregates or unions, these rules apply recursively to the subaggregates
292 // or contained unions.
293 //
294 // If there are fewer initializers in a brace-enclosed list than there are
295 // elements or members of an aggregate, or fewer characters in a string
296 // literal used to initialize an array of known size than there are elements
297 // in the array, the remainder of the aggregate is subject to default
298 // initialization.
299 // """
300 //
301 // The standard seems ambiguous in the following two areas:
302 // 1. For a union type with empty initializer, if the first named member is
303 // not the largest member, then the bytes comes after the first named member
304 // but before padding are left unspecified. An example is:
305 // union U { int a; long long b;};
306 // union U u = {}; // The first 4 bytes are 0, but 4-8 bytes are left
307 // unspecified.
308 //
309 // 2. It only mentions padding for empty initializer, but doesn't mention
310 // padding for a non empty initialization list. And if the aggregation or
311 // union contains elements or members that are aggregates or unions, and
312 // some are non empty initializers, while others are empty initializers,
313 // the padding initialization is unclear. An example is:
314 // struct S1 { int a; long long b; };
315 // struct S2 { char c; struct S1 s1; };
316 // // The values for paddings between s2.c and s2.s1.a, between s2.s1.a
317 // and s2.s1.b are unclear.
318 // struct S2 s2 = { 'c' };
319 //
320 // Here we choose to zero initiailize left bytes of a union type because
321 // projects like the Linux kernel are relying on this behavior. If we don't
322 // explicitly zero initialize them, the undef values can be optimized to
323 // return garbage data. We also choose to zero initialize paddings for
324 // aggregates and unions, no matter they are initialized by empty
325 // initializers or non empty initializers. This can provide a consistent
326 // behavior. So projects like the Linux kernel can rely on it.
327 return !getLangOpts().CPlusPlus;
328 }
329
330 llvm::StringMap<unsigned> cgGlobalNames;
331 std::string getUniqueGlobalName(const std::string &baseName);
332
333 /// Return the mlir::Value for the address of the given global variable.
334 /// If Ty is non-null and if the global doesn't exist, then it will be created
335 /// with the specified type instead of whatever the normal requested type
336 /// would be. If IsForDefinition is true, it is guaranteed that an actual
337 /// global with type Ty will be returned, not conversion of a variable with
338 /// the same mangled name but some other type.
339 mlir::Value
340 getAddrOfGlobalVar(const VarDecl *d, mlir::Type ty = {},
341 ForDefinition_t isForDefinition = NotForDefinition);
342
343 /// Get or create a thunk function with the given name and type.
344 cir::FuncOp getAddrOfThunk(StringRef name, mlir::Type fnTy, GlobalDecl gd);
345
346 /// Return the mlir::GlobalViewAttr for the address of the given global.
347 cir::GlobalViewAttr getAddrOfGlobalVarAttr(const VarDecl *d);
348
349 /// Get the GlobalOp of a template parameter object.
350 cir::GlobalOp
352 // Get the GlobalOp of a source_location object.
353 cir::GlobalOp
355
357 const CXXRecordDecl *derivedClass,
358 llvm::iterator_range<CastExpr::path_const_iterator> path);
359
360 /// Get the CIR attributes and calling convention to use for a particular
361 /// function type.
362 ///
363 /// \param name - The function name.
364 /// \param info - The function type information.
365 /// \param calleeInfo - The callee information these attributes are being
366 /// constructed for. If valid, the attributes applied to this decl may
367 /// contribute to the function attributes and calling convention.
368 /// \param attrs [out] - On return, the attribute list to use.
369 /// \param callingConv [out] - On return, the calling convention to use.
370 /// \param sideEffect [out] - On return, the side effect type of the
371 /// attributes.
372 /// \param attrOnCallSite - Whether or not the attributes are on a call site.
373 /// \param isThunk - Whether the function is a thunk.
375 llvm::StringRef name, const CIRGenFunctionInfo &info,
376 CIRGenCalleeInfo calleeInfo, mlir::NamedAttrList &attrs,
378 mlir::NamedAttrList &retAttrs, cir::CallingConv &callingConv,
379 cir::SideEffect &sideEffect, bool attrOnCallSite, bool isThunk);
380 /// Helper function for constructAttributeList/others. Builds a set of
381 /// function attributes to add to a function based on language opts, codegen
382 /// opts, and some small properties.
383 void addDefaultFunctionAttributes(StringRef name, bool hasOptNoneAttr,
384 bool attrOnCallSite,
385 mlir::NamedAttrList &attrs);
386
387 /// Will return a global variable of the given type. If a variable with a
388 /// different type already exists then a new variable with the right type
389 /// will be created and all uses of the old variable will be replaced with a
390 /// bitcast to the new variable.
392 mlir::Location loc, llvm::StringRef name, mlir::Type ty,
393 cir::GlobalLinkageKind linkage, clang::CharUnits alignment);
394
395 void emitVTable(const CXXRecordDecl *rd);
396
397 /// Return the appropriate linkage for the vtable, VTT, and type information
398 /// of the given class.
399 cir::GlobalLinkageKind getVTableLinkage(const CXXRecordDecl *rd);
400
401 /// Get the address of the RTTI descriptor for the given type.
402 mlir::Attribute getAddrOfRTTIDescriptor(mlir::Location loc, QualType ty,
403 bool forEH = false);
404
405 static mlir::SymbolTable::Visibility getMLIRVisibility(Visibility v) {
406 switch (v) {
408 return mlir::SymbolTable::Visibility::Public;
409 case HiddenVisibility:
410 return mlir::SymbolTable::Visibility::Private;
412 // The distinction between ProtectedVisibility and DefaultVisibility is
413 // that symbols with ProtectedVisibility, while visible to the dynamic
414 // linker like DefaultVisibility, are guaranteed to always dynamically
415 // resolve to a symbol in the current shared object. There is currently no
416 // equivalent MLIR visibility, so we fall back on the fact that the symbol
417 // is visible.
418 return mlir::SymbolTable::Visibility::Public;
419 }
420 llvm_unreachable("unknown visibility!");
421 }
422
423 static cir::VisibilityKind getCIRVisibilityKind(Visibility v) {
424 switch (v) {
426 return cir::VisibilityKind::Default;
427 case HiddenVisibility:
428 return cir::VisibilityKind::Hidden;
430 return cir::VisibilityKind::Protected;
431 }
432
433 llvm_unreachable("unknown visibility!");
434 }
435
436 llvm::DenseMap<mlir::Attribute, cir::GlobalOp> constantStringMap;
437 llvm::DenseMap<const UnnamedGlobalConstantDecl *, cir::GlobalOp>
439 llvm::DenseMap<const CompoundLiteralExpr *, cir::GlobalOp>
441
442 cir::GlobalOp
447 cir::GlobalOp gv) {
448 [[maybe_unused]] bool ok = emittedCompoundLiterals.insert({e, gv}).second;
449 assert(ok && "compound literal global already emitted");
450 }
451
452 /// Return a constant array for the given string.
453 mlir::Attribute getConstantArrayFromStringLiteral(const StringLiteral *e);
454
455 /// Return a global symbol reference to a constant array for the given string
456 /// literal.
457 cir::GlobalOp getGlobalForStringLiteral(const StringLiteral *s,
458 llvm::StringRef name = ".str");
459
460 /// Return a global symbol reference to a constant array for the given string
461 /// literal.
462 cir::GlobalViewAttr
464 llvm::StringRef name = ".str");
465
466 /// Returns the address space for temporary allocations in the language. This
467 /// ensures that the allocated variable's address space matches the
468 /// expectations of the AST, rather than using the target's allocation address
469 /// space, which may lead to type mismatches in other parts of the IR.
471
472 /// Set attributes which are common to any form of a global definition (alias,
473 /// Objective-C method, function, global variable).
474 ///
475 /// NOTE: This should only be called for definitions.
476 void setCommonAttributes(GlobalDecl gd, mlir::Operation *op);
477
479
480 /// Helpers to convert the presumed location of Clang's SourceLocation to an
481 /// MLIR Location.
482 mlir::Location getLoc(clang::SourceLocation cLoc);
483 mlir::Location getLoc(clang::SourceRange cRange);
484
485 /// Return the best known alignment for an unknown pointer to a
486 /// particular class.
488
489 /// FIXME: this could likely be a common helper and not necessarily related
490 /// with codegen.
492 LValueBaseInfo *baseInfo = nullptr,
493 bool forPointeeType = false);
496 LValueBaseInfo *baseInfo = nullptr);
497
498 /// Returns the minimum object size for an object of the given class type
499 /// (or a class derived from it).
501
502 /// Returns the minimum object size for an object of the given type.
508
509 /// TODO: Add TBAAAccessInfo
511 const CXXRecordDecl *baseDecl,
512 CharUnits expectedTargetAlign);
513
514 /// Returns the assumed alignment of a virtual base of a class.
516 const CXXRecordDecl *derived,
517 const CXXRecordDecl *vbase);
518
519 cir::FuncOp
521 const CIRGenFunctionInfo *fnInfo = nullptr,
522 cir::FuncType fnType = nullptr, bool dontDefer = false,
523 ForDefinition_t isForDefinition = NotForDefinition) {
524 return getAddrAndTypeOfCXXStructor(gd, fnInfo, fnType, dontDefer,
525 isForDefinition)
526 .second;
527 }
528
529 std::pair<cir::FuncType, cir::FuncOp> getAddrAndTypeOfCXXStructor(
530 clang::GlobalDecl gd, const CIRGenFunctionInfo *fnInfo = nullptr,
531 cir::FuncType fnType = nullptr, bool dontDefer = false,
532 ForDefinition_t isForDefinition = NotForDefinition);
533
534 /// List of global values which are required to be present in the object file;
535 /// This is used for forcing visibility of symbols which may otherwise be
536 /// optimized out.
537 std::vector<cir::CIRGlobalValueInterface> llvmUsed;
538 std::vector<cir::CIRGlobalValueInterface> llvmCompilerUsed;
539
540 mlir::Type getVTableComponentType();
541 CIRGenVTables &getVTables() { return vtables; }
542
544 return vtables.getItaniumVTableContext();
545 }
547 return vtables.getItaniumVTableContext();
548 }
549
550 /// This contains all the decls which have definitions but which are deferred
551 /// for emission and therefore should only be output if they are actually
552 /// used. If a decl is in this, then it is known to have not been referenced
553 /// yet.
554 std::map<llvm::StringRef, clang::GlobalDecl> deferredDecls;
555
556 // This is a list of deferred decls which we have seen that *are* actually
557 // referenced. These get code generated when the module is done.
558 std::vector<clang::GlobalDecl> deferredDeclsToEmit;
560 deferredDeclsToEmit.emplace_back(GD);
561 }
562
564
565 /// Determine whether the definition must be emitted; if this returns \c
566 /// false, the definition can be emitted lazily if it's used.
567 bool mustBeEmitted(const clang::ValueDecl *d);
568
569 /// Check if `fd` ends up calling itself directly through asm label or
570 /// builtin-pointer-to-self trickery (e.g., glibc's `extern inline` libc
571 /// wrappers that call `__builtin_strrchr`, which the codegen lowers to a
572 /// call on the same asm-named symbol). Emitting an
573 /// `available_externally` body for such a function feeds the LLVM
574 /// Decide whether to emit the body of `gd` to CIR. Returns false for
575 /// available_externally functions that are trivially recursive (PR9614).
576 /// Mirrors classic CodeGen's `CodeGenModule::shouldEmitFunction`.
578
579 /// Determine whether the definition can be emitted eagerly, or should be
580 /// delayed until the end of the translation unit. This is relevant for
581 /// definitions whose linkage can change, e.g. implicit function
582 /// instantiations which may later be explicitly instantiated.
584
585 bool verifyModule() const;
586
587 /// Return the address of the given function. If funcType is non-null, then
588 /// this function will use the specified type if it has to create it.
589 // TODO: this is a bit weird as `GetAddr` given we give back a FuncOp?
590 cir::FuncOp
591 getAddrOfFunction(clang::GlobalDecl gd, mlir::Type funcType = nullptr,
592 bool forVTable = false, bool dontDefer = false,
593 ForDefinition_t isForDefinition = NotForDefinition);
594
595 mlir::Operation *
597 ForDefinition_t isForDefinition = NotForDefinition);
598
599 // Return whether RTTI information should be emitted for this target.
600 bool shouldEmitRTTI(bool forEH = false) {
601 return (forEH || getLangOpts().RTTI) && !getLangOpts().CUDAIsDevice &&
602 !(getLangOpts().OpenMP && getLangOpts().OpenMPIsTargetDevice &&
603 getTriple().isNVPTX());
604 }
605
606 /// Emit type info if type of an expression is a variably modified
607 /// type. Also emit proper debug info for cast types.
609 CIRGenFunction *cgf = nullptr);
610
612 deferredVTables.push_back(rd);
613 }
614
615 /// Emit code for a single global function or variable declaration. Forward
616 /// declarations are emitted lazily.
618
619 void emitAliasForGlobal(llvm::StringRef mangledName, mlir::Operation *op,
620 GlobalDecl aliasGD, cir::FuncOp aliasee,
621 cir::GlobalLinkageKind linkage);
622
623 /// Emit a definition for an `__attribute__((alias))` declaration.
625
626 mlir::Type convertType(clang::QualType type);
627
628 /// Set the visibility for the given global.
629 void setGlobalVisibility(cir::CIRGlobalValueInterface gv,
630 const NamedDecl *d) const;
631 void setDSOLocal(mlir::Operation *op) const;
632 void setDSOLocal(cir::CIRGlobalValueInterface gv) const;
633
634 /// Set visibility, dllimport/dllexport and dso_local.
635 /// This must be called after dllimport/dllexport is set.
636 void setGVProperties(mlir::Operation *op, const NamedDecl *d) const;
637 void setGVPropertiesAux(mlir::Operation *op, const NamedDecl *d) const;
638
639 /// Set TLS mode for the given operation based on the given variable
640 /// declaration. If `isExtendingDecl` is true, then the operation is a
641 /// temporary whose lifetime is extended by the variable declared by `d`.
642 void setTLSMode(mlir::Operation *op, const VarDecl &d,
643 bool isExtendingDecl = false);
644
645 /// Get TLS mode from CodeGenOptions.
646 cir::TLS_Model getDefaultCIRTLSModel() const;
647
648 /// Set function attributes for a function declaration.
649 void setFunctionAttributes(GlobalDecl gd, cir::FuncOp f,
650 bool isIncompleteFunction, bool isThunk);
651
652 /// Set the CIR function attributes (Sext, zext, etc).
654 cir::FuncOp func, bool isThunk);
655
656 /// Set extra attributes (inline, etc.) for a function.
658 cir::FuncOp f);
659
661 mlir::Operation *op = nullptr);
662 void emitGlobalFunctionDefinition(clang::GlobalDecl gd, mlir::Operation *op);
664 bool isTentative = false);
665
666 /// Helper function for the below two that will create the
667 /// constructor/destructor in specified regions, rather than in the GlobalOp.
668 void emitCXXSpecialVarDeclInit(const VarDecl *varDecl, cir::GlobalOp addr,
669 bool performInit, mlir::Region &ctorRegion,
670 mlir::Region &dtorRegion);
671 /// Emit the function that initializes the specified static-local variable.
672 void emitCXXStaticLocalVarDeclInit(const VarDecl *varDecl, cir::GlobalOp addr,
673 bool performInit);
674 /// Emit the function that initializes the specified global
675 void emitCXXGlobalVarDeclInit(const VarDecl *varDecl, cir::GlobalOp addr,
676 bool performInit);
677
678 void setGlobalTlsReferences(const VarDecl &vd, cir::GlobalOp globalOp);
679 void emitCXXGlobalVarDeclInitFunc(const VarDecl *vd, cir::GlobalOp addr,
680 bool performInit);
681
685 template <typename BeforeOpTy, typename DataClauseTy>
686 void emitGlobalOpenACCDeclareDataOperands(const Expr *varOperand,
687 DataClauseTy dataClause,
688 OpenACCModifierKind modifiers,
689 bool structured, bool implicit,
690 bool requiresDtor);
691 // Each of the acc.routine operations must have a unique name, so we just use
692 // an integer counter. This is how Flang does it, so it seems reasonable.
693 unsigned routineCounter = 0;
694 void emitOpenACCRoutineDecl(const clang::FunctionDecl *funcDecl,
695 cir::FuncOp func, SourceLocation pragmaLoc,
697
705
706 // C++ related functions.
707 void emitDeclContext(const DeclContext *dc);
708
709 /// Return the result of value-initializing the given type, i.e. a null
710 /// expression of the given type.
711 mlir::Value emitNullConstant(QualType t, mlir::Location loc);
712
713 mlir::TypedAttr emitNullConstantAttr(QualType t);
714
715 /// Return a null constant appropriate for zero-initializing a base class with
716 /// the given type. This is usually, but not always, an LLVM null constant.
717 mlir::TypedAttr emitNullConstantForBase(const CXXRecordDecl *record);
718
719 mlir::Value emitMemberPointerConstant(const UnaryOperator *e);
720 /// Returns a null attribute to represent either a null method or null data
721 /// member, depending on the type of mpt.
722 mlir::TypedAttr emitNullMemberAttr(QualType t, const MemberPointerType *mpt);
723
724 /// Build a GEP-style field-index path from \p destClass to \p field.
725 /// Returns std::nullopt and emits errorNYI for virtual-base paths.
726 std::optional<llvm::SmallVector<int32_t>>
727 buildMemberPath(const CXXRecordDecl *destClass, const FieldDecl *field);
728
729 llvm::StringRef getMangledName(clang::GlobalDecl gd);
730 // This function is to support the OpenACC 'bind' clause, which names an
731 // alternate name for the function to be called by. This function mangles
732 // `attachedFunction` as-if its name was actually `bindName` (that is, with
733 // the same signature). It has some additional complications, as the 'bind'
734 // target is always going to be a global function, so member functions need an
735 // explicit instead of implicit 'this' parameter, and thus gets mangled
736 // differently.
737 std::string getOpenACCBindMangledName(const IdentifierInfo *bindName,
738 const FunctionDecl *attachedFunction);
739
740 void emitTentativeDefinition(const VarDecl *d);
741
742 // Make sure that this type is translated.
743 void updateCompletedType(const clang::TagDecl *td);
744
745 // Produce code for this constructor/destructor. This method doesn't try to
746 // apply any ABI rules about which other constructors/destructors are needed
747 // or if they are alias to each other.
749
750 bool lookupRepresentativeDecl(llvm::StringRef mangledName,
751 clang::GlobalDecl &gd) const;
752
753 bool supportsCOMDAT() const;
754 void maybeSetTrivialComdat(const clang::Decl &d, mlir::Operation *op);
755
756 static void setInitializer(cir::GlobalOp &op, mlir::Attribute value);
757
758 // Whether a global variable should be emitted by CUDA/HIP host/device
759 // related attributes.
760 bool shouldEmitCUDAGlobalVar(const VarDecl *global) const;
761
762 /// Print the postfix for externalized static variable or kernels for single
763 /// source offloading languages CUDA and HIP. The unique postfix is created
764 /// using either the CUID argument, or the file's UniqueID and active macros.
765 /// The fallback method without a CUID requires that the offloading toolchain
766 /// does not define separate macros via the -cc1 options.
767 void printPostfixForExternalizedDecl(llvm::raw_ostream &os, const Decl *d);
768
769 /// Replace all uses of the old global with the new global, updating types
770 /// and references as needed. Erases the old global when done.
771 void replaceGlobal(cir::GlobalOp oldGV, cir::GlobalOp newGV);
772
773 void replaceUsesOfNonProtoTypeWithRealFunction(mlir::Operation *old,
774 cir::FuncOp newFn);
775
776 cir::FuncOp
777 getOrCreateCIRFunction(llvm::StringRef mangledName, mlir::Type funcType,
778 clang::GlobalDecl gd, bool forVTable,
779 bool dontDefer = false, bool isThunk = false,
780 ForDefinition_t isForDefinition = NotForDefinition,
781 mlir::NamedAttrList extraAttrs = {});
782
783 cir::FuncOp getOrCreateCIRFunction(llvm::StringRef mangledName,
784 mlir::Type funcType, clang::GlobalDecl gd,
785 bool forVTable,
786 mlir::NamedAttrList extraAttrs) {
787 return getOrCreateCIRFunction(mangledName, funcType, gd, forVTable,
788 /*dontDefer=*/false, /*isThunk=*/false,
789 NotForDefinition, extraAttrs);
790 }
791
792 cir::FuncOp createCIRFunction(mlir::Location loc, llvm::StringRef name,
793 cir::FuncType funcType,
794 const clang::FunctionDecl *funcDecl);
795
796 /// Create a CIR function with builtin attribute set.
797 cir::FuncOp createCIRBuiltinFunction(mlir::Location loc, llvm::StringRef name,
798 cir::FuncType ty,
799 const clang::FunctionDecl *fd);
800
801 /// Mark the function as a special member (e.g. constructor, destructor)
802 void setCXXSpecialMemberAttr(cir::FuncOp funcOp,
803 const clang::FunctionDecl *funcDecl);
804
805 cir::FuncOp createRuntimeFunction(cir::FuncType ty, llvm::StringRef name,
806 mlir::NamedAttrList extraAttrs = {},
807 bool isLocal = false,
808 bool assumeConvergent = false);
809
810 static constexpr const char *builtinCoroId = "__builtin_coro_id";
811 static constexpr const char *builtinCoroAlloc = "__builtin_coro_alloc";
812 static constexpr const char *builtinCoroBegin = "__builtin_coro_begin";
813 static constexpr const char *builtinCoroEnd = "__builtin_coro_end";
814 static constexpr const char *builtinCoroFree = "__builtin_coro_free";
815
816 /// Given a builtin id for a function like "__builtin_fabsf", return a
817 /// Function* for "fabsf".
818 cir::FuncOp getBuiltinLibFunction(const FunctionDecl *fd, unsigned builtinID);
819
821 assert(cudaRuntime != nullptr);
822 return *cudaRuntime;
823 }
824
826 assert(openMPRuntime != nullptr);
827 return *openMPRuntime;
828 }
829
832
833 mlir::IntegerAttr getSize(CharUnits size) {
834 return builder.getSizeFromCharUnits(size);
835 }
836
837 /// Emit any needed decls for which code generation was deferred.
838 void emitDeferred();
839
841 /// Emit any vtables which we deferred and still have a use for.
842 void emitDeferredVTables();
843
844 /// Try to emit external vtables as available_externally if they have emitted
845 /// all inlined virtual functions. It runs after EmitDeferred() and therefore
846 /// is not allowed to create new references to things that need to be emitted
847 /// lazily.
849
850 /// Helper for `emitDeferred` to apply actual codegen.
851 void emitGlobalDecl(const clang::GlobalDecl &d);
852
853 const llvm::Triple &getTriple() const { return target.getTriple(); }
854
855 // Finalize CIR code generation.
856 void release();
857
858 /// Returns a pointer to a global variable representing a temporary with
859 /// static or thread storage duration.
860 mlir::Operation *getAddrOfGlobalTemporary(const MaterializeTemporaryExpr *mte,
861 const Expr *init);
862
863 /// -------
864 /// Visibility and Linkage
865 /// -------
866
867 static mlir::SymbolTable::Visibility
868 getMLIRVisibilityFromCIRLinkage(cir::GlobalLinkageKind GLK);
869 static cir::VisibilityKind getGlobalVisibilityKindFromClangVisibility(
870 clang::VisibilityAttr::VisibilityType visibility);
871 cir::VisibilityAttr getGlobalVisibilityAttrFromDecl(const Decl *decl);
872 cir::GlobalLinkageKind getFunctionLinkage(GlobalDecl gd);
873 static mlir::SymbolTable::Visibility getMLIRVisibility(cir::GlobalOp op);
874 cir::GlobalLinkageKind getCIRLinkageForDeclarator(const DeclaratorDecl *dd,
875 GVALinkage linkage);
876 void setFunctionLinkage(GlobalDecl gd, cir::FuncOp f) {
877 cir::GlobalLinkageKind l = getFunctionLinkage(gd);
878 f.setLinkageAttr(cir::GlobalLinkageKindAttr::get(&getMLIRContext(), l));
879 mlir::SymbolTable::setSymbolVisibility(f,
881 }
882
883 cir::GlobalLinkageKind getCIRLinkageVarDefinition(const VarDecl *vd);
884
885 void addReplacement(llvm::StringRef name, mlir::Operation *op);
886
887 /// Helpers to emit "not yet implemented" error diagnostics
888 DiagnosticBuilder errorNYI(SourceLocation, llvm::StringRef);
889
890 template <typename T>
891 DiagnosticBuilder errorNYI(SourceLocation loc, llvm::StringRef feature,
892 const T &name) {
893 unsigned diagID =
894 diags.getCustomDiagID(DiagnosticsEngine::Error,
895 "ClangIR code gen Not Yet Implemented: %0: %1");
896 return diags.Report(loc, diagID) << feature << name;
897 }
898
899 DiagnosticBuilder errorNYI(mlir::Location loc, llvm::StringRef feature) {
900 // TODO: Convert the location to a SourceLocation
901 unsigned diagID = diags.getCustomDiagID(
902 DiagnosticsEngine::Error, "ClangIR code gen Not Yet Implemented: %0");
903 return diags.Report(diagID) << feature;
904 }
905
906 DiagnosticBuilder errorNYI(llvm::StringRef feature) const {
907 // TODO: Make a default location? currSrcLoc?
908 unsigned diagID = diags.getCustomDiagID(
909 DiagnosticsEngine::Error, "ClangIR code gen Not Yet Implemented: %0");
910 return diags.Report(diagID) << feature;
911 }
912
913 DiagnosticBuilder errorNYI(SourceRange, llvm::StringRef);
914
915 template <typename T>
916 DiagnosticBuilder errorNYI(SourceRange loc, llvm::StringRef feature,
917 const T &name) {
918 return errorNYI(loc.getBegin(), feature, name) << loc;
919 }
920
921 /// Emit a general error that something can't be done.
922 void error(SourceLocation loc, llvm::StringRef error);
923
924 /// Print out an error that codegen doesn't support the specified stmt yet.
925 void errorUnsupported(const Stmt *s, llvm::StringRef type);
926
927 /// Print out an error that codegen doesn't support the specified decl yet.
928 void errorUnsupported(const Decl *d, llvm::StringRef type);
929
930 /// Emits AMDGPU specific Metadata.
931 void emitAMDGPUMetadata();
932
933 /// Add global annotations for a global value (GlobalOp or FuncOp).
934 void addGlobalAnnotations(const clang::ValueDecl *d, mlir::Operation *gv);
935
936private:
937 /// Search \p currentClass and its non-virtual base subobjects for \p field,
938 /// appending CIR field indices along the path from \p currentClass.
939 bool findFieldMemberPath(const CXXRecordDecl *currentClass,
940 const FieldDecl *field,
942
943 // An ordered map of canonical GlobalDecls to their mangled names.
944 llvm::MapVector<clang::GlobalDecl, llvm::StringRef> mangledDeclNames;
945 llvm::StringMap<clang::GlobalDecl, llvm::BumpPtrAllocator> manglings;
946
947 // FIXME: should we use llvm::TrackingVH<mlir::Operation> here?
948 llvm::MapVector<StringRef, mlir::Operation *> replacements;
949 /// Call replaceAllUsesWith on all pairs in replacements.
950 void applyReplacements();
951
952 bool getCPUAndFeaturesAttributes(GlobalDecl gd,
953 llvm::StringMap<std::string> &attrs,
954 bool setTargetFeatures = true);
955 void setNonAliasAttributes(GlobalDecl gd, mlir::Operation *op);
956
957 /// Map source language used to a CIR attribute.
958 std::optional<cir::SourceLanguage> getCIRSourceLanguage() const;
959
960 /// Emit all the global annotations.
961 void emitGlobalAnnotations();
962
963 /// Build (or fetch from the dedup cache) the args ArrayAttr for an
964 /// annotation. Returns the empty ArrayAttr when the annotation has none.
965 mlir::ArrayAttr getOrCreateAnnotationArgs(const clang::AnnotateAttr *attr);
966
967 /// Create cir::AnnotationAttr for a single AnnotateAttr on a global.
968 cir::AnnotationAttr emitAnnotateAttr(const clang::AnnotateAttr *aa);
969
970 /// Return the AST address space of the underlying global variable for D, as
971 /// determined by its declaration. Normally this is the same as the address
972 /// space of D's type, but in CUDA, address spaces are associated with
973 /// declarations, not types. If D is nullptr, return the default address
974 /// space for global variable.
975 ///
976 /// For languages without explicit address spaces, if D has default address
977 /// space, target-specific global or constant address space may be returned.
978 LangAS getGlobalVarAddressSpace(const VarDecl *decl);
979};
980} // namespace CIRGen
981
982} // namespace clang
983
984#endif // LLVM_CLANG_LIB_CIR_CODEGEN_CIRGENMODULE_H
Defines the SourceManager interface.
Holds long-lived AST nodes (such as types and decls) that can be referred to throughout the semantic ...
Definition ASTContext.h:223
CharUnits getTypeSizeInChars(QualType T) const
Return the size of the specified (complete) type T, in characters.
Implements C++ ABI-specific code generation functions.
Abstract information about a function or function prototype.
Definition CIRGenCall.h:27
cir::GlobalOp getAddrOfUnnamedGlobalConstantDecl(const UnnamedGlobalConstantDecl *gcd)
void setGlobalVisibility(cir::CIRGlobalValueInterface gv, const NamedDecl *d) const
Set the visibility for the given global.
void addUsedOrCompilerUsedGlobal(cir::CIRGlobalValueInterface gv)
Add a global to a list to be added to the llvm.compiler.used metadata.
void emitCXXGlobalVarDeclInit(const VarDecl *varDecl, cir::GlobalOp addr, bool performInit)
Emit the function that initializes the specified global.
void replaceUsesOfNonProtoTypeWithRealFunction(mlir::Operation *old, cir::FuncOp newFn)
This function is called when we implement a function with no prototype, e.g.
bool shouldEmitFunction(clang::GlobalDecl gd)
Check if fd ends up calling itself directly through asm label or builtin-pointer-to-self trickery (e....
llvm::StringRef getMangledName(clang::GlobalDecl gd)
cir::GlobalOp getOrCreateStaticVarDecl(const VarDecl &d, cir::GlobalLinkageKind linkage)
CharUnits computeNonVirtualBaseClassOffset(const CXXRecordDecl *derivedClass, llvm::iterator_range< CastExpr::path_const_iterator > path)
DiagnosticBuilder errorNYI(SourceLocation, llvm::StringRef)
Helpers to emit "not yet implemented" error diagnostics.
void emitDeferred()
Emit any needed decls for which code generation was deferred.
cir::GlobalLinkageKind getCIRLinkageVarDefinition(const VarDecl *vd)
clang::ASTContext & getASTContext() const
bool isPaddedAtomicType(QualType type)
void insertGlobalSymbol(mlir::Operation *op)
cir::FuncOp getAddrOfCXXStructor(clang::GlobalDecl gd, const CIRGenFunctionInfo *fnInfo=nullptr, cir::FuncType fnType=nullptr, bool dontDefer=false, ForDefinition_t isForDefinition=NotForDefinition)
CIRGenCUDARuntime & getCUDARuntime()
llvm::DenseMap< cir::BlockAddrInfoAttr, cir::LabelOp > blockAddressInfoToLabel
Map BlockAddrInfoAttr (function name, label name) to the corresponding CIR LabelOp.
void emitTopLevelDecl(clang::Decl *decl)
CharUnits getDynamicOffsetAlignment(CharUnits actualBaseAlign, const CXXRecordDecl *baseDecl, CharUnits expectedTargetAlign)
TODO: Add TBAAAccessInfo.
void emitGlobalOpenACCDeclareDataOperands(const Expr *varOperand, DataClauseTy dataClause, OpenACCModifierKind modifiers, bool structured, bool implicit, bool requiresDtor)
void emitOMPDeclareMapper(const OMPDeclareMapperDecl *d)
void addReplacement(llvm::StringRef name, mlir::Operation *op)
mlir::Type convertType(clang::QualType type)
bool shouldEmitRTTI(bool forEH=false)
cir::GlobalOp getGlobalForStringLiteral(const StringLiteral *s, llvm::StringRef name=".str")
Return a global symbol reference to a constant array for the given string literal.
std::vector< cir::CIRGlobalValueInterface > llvmUsed
List of global values which are required to be present in the object file; This is used for forcing v...
void emitOMPCapturedExpr(const OMPCapturedExprDecl *d)
llvm::DenseMap< const VarDecl *, cir::GlobalOp > initializerConstants
llvm::DenseMap< const CompoundLiteralExpr *, cir::GlobalOp > emittedCompoundLiterals
bool mustBeEmitted(const clang::ValueDecl *d)
Determine whether the definition must be emitted; if this returns false, the definition can be emitte...
void emitGlobalOpenACCDeclareDecl(const clang::OpenACCDeclareDecl *cd)
mlir::IntegerAttr getSize(CharUnits size)
void addDefaultFunctionAttributes(StringRef name, bool hasOptNoneAttr, bool attrOnCallSite, mlir::NamedAttrList &attrs)
Helper function for constructAttributeList/others.
void setGlobalTlsReferences(const VarDecl &vd, cir::GlobalOp globalOp)
DiagnosticBuilder errorNYI(SourceRange loc, llvm::StringRef feature, const T &name)
CIRGenBuilderTy & getBuilder()
void setDSOLocal(mlir::Operation *op) const
void emitCXXStaticLocalVarDeclInit(const VarDecl *varDecl, cir::GlobalOp addr, bool performInit)
Emit the function that initializes the specified static-local variable.
std::string getUniqueGlobalName(const std::string &baseName)
std::pair< cir::FuncType, cir::FuncOp > getAddrAndTypeOfCXXStructor(clang::GlobalDecl gd, const CIRGenFunctionInfo *fnInfo=nullptr, cir::FuncType fnType=nullptr, bool dontDefer=false, ForDefinition_t isForDefinition=NotForDefinition)
ItaniumVTableContext & getItaniumVTableContext()
void setGVProperties(mlir::Operation *op, const NamedDecl *d) const
Set visibility, dllimport/dllexport and dso_local.
cir::GlobalOp getOrCreateCIRGlobal(llvm::StringRef mangledName, mlir::Type ty, LangAS langAS, const VarDecl *d, ForDefinition_t isForDefinition)
If the specified mangled name is not in the module, create and return an mlir::GlobalOp value.
cir::FuncOp createCIRBuiltinFunction(mlir::Location loc, llvm::StringRef name, cir::FuncType ty, const clang::FunctionDecl *fd)
Create a CIR function with builtin attribute set.
cir::GlobalOp getAddrOfTemplateParamObject(const TemplateParamObjectDecl *tpo)
Get the GlobalOp of a template parameter object.
llvm::DenseMap< const Decl *, cir::GlobalOp > staticLocalDeclMap
cir::FuncOp lookupFuncOp(llvm::StringRef name)
O(1) lookup of a FuncOp by name in the symbol cache.
void emitGlobalOpenACCRoutineDecl(const clang::OpenACCRoutineDecl *cd)
clang::CharUnits getClassPointerAlignment(const clang::CXXRecordDecl *rd)
Return the best known alignment for an unknown pointer to a particular class.
void handleCXXStaticMemberVarInstantiation(VarDecl *vd)
Tell the consumer that this variable has been instantiated.
llvm::DenseMap< const UnnamedGlobalConstantDecl *, cir::GlobalOp > unnamedGlobalConstantDeclMap
std::vector< cir::CIRGlobalValueInterface > llvmCompilerUsed
CharUnits getMinimumClassObjectSize(const CXXRecordDecl *cd)
Returns the minimum object size for an object of the given class type (or a class derived from it).
void emitOMPRequiresDecl(const OMPRequiresDecl *d)
void emitGlobalDefinition(clang::GlobalDecl gd, mlir::Operation *op=nullptr)
clang::DiagnosticsEngine & getDiags() const
CharUnits getVBaseAlignment(CharUnits derivedAlign, const CXXRecordDecl *derived, const CXXRecordDecl *vbase)
Returns the assumed alignment of a virtual base of a class.
cir::GlobalLinkageKind getCIRLinkageForDeclarator(const DeclaratorDecl *dd, GVALinkage linkage)
mlir::Attribute getAddrOfRTTIDescriptor(mlir::Location loc, QualType ty, bool forEH=false)
Get the address of the RTTI descriptor for the given type.
void setFunctionAttributes(GlobalDecl gd, cir::FuncOp f, bool isIncompleteFunction, bool isThunk)
Set function attributes for a function declaration.
static mlir::SymbolTable::Visibility getMLIRVisibilityFromCIRLinkage(cir::GlobalLinkageKind GLK)
const ItaniumVTableContext & getItaniumVTableContext() const
const clang::TargetInfo & getTarget() const
mlir::TypedAttr emitNullConstantForBase(const CXXRecordDecl *record)
Return a null constant appropriate for zero-initializing a base class with the given type.
void setCIRFunctionAttributes(GlobalDecl gd, const CIRGenFunctionInfo &info, cir::FuncOp func, bool isThunk)
Set the CIR function attributes (Sext, zext, etc).
cir::FuncOp getBuiltinLibFunction(const FunctionDecl *fd, unsigned builtinID)
Given a builtin id for a function like "__builtin_fabsf", return a Function* for "fabsf".
const llvm::Triple & getTriple() const
static mlir::SymbolTable::Visibility getMLIRVisibility(Visibility v)
void emitTentativeDefinition(const VarDecl *d)
CharUnits getMinimumObjectSize(QualType ty)
Returns the minimum object size for an object of the given type.
void emitAliasDefinition(GlobalDecl gd)
Emit a definition for an __attribute__((alias)) declaration.
void addUsedGlobal(cir::CIRGlobalValueInterface gv)
Add a global value to the llvmUsed list.
cir::GlobalOp createOrReplaceCXXRuntimeVariable(mlir::Location loc, llvm::StringRef name, mlir::Type ty, cir::GlobalLinkageKind linkage, clang::CharUnits alignment)
Will return a global variable of the given type.
void emitOMPAllocateDecl(const OMPAllocateDecl *d)
void error(SourceLocation loc, llvm::StringRef error)
Emit a general error that something can't be done.
void emitGlobalDecl(const clang::GlobalDecl &d)
Helper for emitDeferred to apply actual codegen.
void emitGlobalVarDefinition(const clang::VarDecl *vd, bool isTentative=false)
cir::FuncOp createRuntimeFunction(cir::FuncType ty, llvm::StringRef name, mlir::NamedAttrList extraAttrs={}, bool isLocal=false, bool assumeConvergent=false)
cir::FuncOp getAddrOfThunk(StringRef name, mlir::Type fnTy, GlobalDecl gd)
Get or create a thunk function with the given name and type.
DiagnosticBuilder errorNYI(mlir::Location loc, llvm::StringRef feature)
void addRecordLayout(mlir::StringAttr name, cir::RecordLayoutAttr attr)
Queue a record layout entry for materialization in release().
cir::FuncOp getAddrOfFunction(clang::GlobalDecl gd, mlir::Type funcType=nullptr, bool forVTable=false, bool dontDefer=false, ForDefinition_t isForDefinition=NotForDefinition)
Return the address of the given function.
mlir::TypedAttr emitNullConstantAttr(QualType t)
void emitAliasForGlobal(llvm::StringRef mangledName, mlir::Operation *op, GlobalDecl aliasGD, cir::FuncOp aliasee, cir::GlobalLinkageKind linkage)
std::optional< llvm::SmallVector< int32_t > > buildMemberPath(const CXXRecordDecl *destClass, const FieldDecl *field)
Build a GEP-style field-index path from destClass to field.
void emitLLVMUsed()
Emit llvm.used and llvm.compiler.used globals.
mlir::Value emitMemberPointerConstant(const UnaryOperator *e)
void emitGlobalOpenACCDecl(const clang::OpenACCConstructDecl *cd)
void setTLSMode(mlir::Operation *op, const VarDecl &d, bool isExtendingDecl=false)
Set TLS mode for the given operation based on the given variable declaration.
void emitExplicitCastExprType(const ExplicitCastExpr *e, CIRGenFunction *cgf=nullptr)
Emit type info if type of an expression is a variably modified type.
const cir::CIRDataLayout getDataLayout() const
void eraseGlobalSymbol(mlir::Operation *op)
mlir::Operation * getAddrOfGlobalTemporary(const MaterializeTemporaryExpr *mte, const Expr *init)
Returns a pointer to a global variable representing a temporary with static or thread storage duratio...
std::map< llvm::StringRef, clang::GlobalDecl > deferredDecls
This contains all the decls which have definitions but which are deferred for emission and therefore ...
void errorUnsupported(const Stmt *s, llvm::StringRef type)
Print out an error that codegen doesn't support the specified stmt yet.
mlir::Value getAddrOfGlobalVar(const VarDecl *d, mlir::Type ty={}, ForDefinition_t isForDefinition=NotForDefinition)
Return the mlir::Value for the address of the given global variable.
llvm::StringMap< mlir::Operation * > symbolLookupCache
Cache for O(1) symbol lookups by name, replacing the O(N) linear scan in SymbolTable::lookupSymbolIn ...
static void setInitializer(cir::GlobalOp &op, mlir::Attribute value)
cir::GlobalViewAttr getAddrOfGlobalVarAttr(const VarDecl *d)
Return the mlir::GlobalViewAttr for the address of the given global.
void addGlobalCtor(cir::FuncOp ctor, std::optional< int > priority=std::nullopt)
Add a global constructor or destructor to the module.
cir::GlobalLinkageKind getFunctionLinkage(GlobalDecl gd)
void updateCompletedType(const clang::TagDecl *td)
const clang::CodeGenOptions & getCodeGenOpts() const
void emitDeferredVTables()
Emit any vtables which we deferred and still have a use for.
const clang::LangOptions & getLangOpts() const
void printPostfixForExternalizedDecl(llvm::raw_ostream &os, const Decl *d)
Print the postfix for externalized static variable or kernels for single source offloading languages ...
void constructAttributeList(llvm::StringRef name, const CIRGenFunctionInfo &info, CIRGenCalleeInfo calleeInfo, mlir::NamedAttrList &attrs, llvm::MutableArrayRef< mlir::NamedAttrList > argAttrs, mlir::NamedAttrList &retAttrs, cir::CallingConv &callingConv, cir::SideEffect &sideEffect, bool attrOnCallSite, bool isThunk)
Get the CIR attributes and calling convention to use for a particular function type.
cir::FuncOp getOrCreateCIRFunction(llvm::StringRef mangledName, mlir::Type funcType, clang::GlobalDecl gd, bool forVTable, bool dontDefer=false, bool isThunk=false, ForDefinition_t isForDefinition=NotForDefinition, mlir::NamedAttrList extraAttrs={})
void emitOpenACCRoutineDecl(const clang::FunctionDecl *funcDecl, cir::FuncOp func, SourceLocation pragmaLoc, ArrayRef< const OpenACCClause * > clauses)
static constexpr const char * builtinCoroAlloc
void emitVTablesOpportunistically()
Try to emit external vtables as available_externally if they have emitted all inlined virtual functio...
cir::GlobalOp createGlobalOp(mlir::Location loc, llvm::StringRef name, mlir::Type t, bool isConstant=false, mlir::ptr::MemorySpaceAttrInterface addrSpace={}, mlir::Operation *insertPoint=nullptr)
cir::TLS_Model getDefaultCIRTLSModel() const
Get TLS mode from CodeGenOptions.
void addGlobalDtor(cir::FuncOp dtor, std::optional< int > priority=std::nullopt)
Add a function to the list that will be called when the module is unloaded.
void addDeferredDeclToEmit(clang::GlobalDecl GD)
bool shouldEmitCUDAGlobalVar(const VarDecl *global) const
cir::FuncOp createCIRFunction(mlir::Location loc, llvm::StringRef name, cir::FuncType funcType, const clang::FunctionDecl *funcDecl)
const TargetCIRGenInfo & getTargetCIRGenInfo()
void emitCXXGlobalVarDeclInitFunc(const VarDecl *vd, cir::GlobalOp addr, bool performInit)
static cir::VisibilityKind getCIRVisibilityKind(Visibility v)
void addDeferredVTable(const CXXRecordDecl *rd)
void setStaticLocalDeclAddress(const VarDecl *d, cir::GlobalOp c)
void setGVPropertiesAux(mlir::Operation *op, const NamedDecl *d) const
LangAS getLangTempAllocaAddressSpace() const
Returns the address space for temporary allocations in the language.
cir::FuncOp codegenCXXStructor(clang::GlobalDecl gd)
mlir::Location getLoc(clang::SourceLocation cLoc)
Helpers to convert the presumed location of Clang's SourceLocation to an MLIR Location.
llvm::DenseMap< mlir::Attribute, cir::GlobalOp > constantStringMap
mlir::Operation * lastGlobalOp
void replaceGlobal(cir::GlobalOp oldGV, cir::GlobalOp newGV)
Replace all uses of the old global with the new global, updating types and references as needed.
static cir::VisibilityKind getGlobalVisibilityKindFromClangVisibility(clang::VisibilityAttr::VisibilityType visibility)
llvm::StringMap< unsigned > cgGlobalNames
void setCXXSpecialMemberAttr(cir::FuncOp funcOp, const clang::FunctionDecl *funcDecl)
Mark the function as a special member (e.g. constructor, destructor)
mlir::TypedAttr emitNullMemberAttr(QualType t, const MemberPointerType *mpt)
Returns a null attribute to represent either a null method or null data member, depending on the type...
mlir::Operation * getGlobalValue(llvm::StringRef ref)
cir::GlobalOp getAddrOfConstantCompoundLiteralIfEmitted(const CompoundLiteralExpr *e)
mlir::Value emitNullConstant(QualType t, mlir::Location loc)
Return the result of value-initializing the given type, i.e.
void emitOMPDeclareReduction(const OMPDeclareReductionDecl *d)
mlir::ModuleOp getModule() const
void addCompilerUsedGlobal(cir::CIRGlobalValueInterface gv)
Add a global value to the llvmCompilerUsed list.
clang::CharUnits getNaturalTypeAlignment(clang::QualType t, LValueBaseInfo *baseInfo=nullptr, bool forPointeeType=false)
FIXME: this could likely be a common helper and not necessarily related with codegen.
mlir::MLIRContext & getMLIRContext()
mlir::Operation * getAddrOfGlobal(clang::GlobalDecl gd, ForDefinition_t isForDefinition=NotForDefinition)
DiagnosticBuilder errorNYI(llvm::StringRef feature) const
void maybeSetTrivialComdat(const clang::Decl &d, mlir::Operation *op)
cir::GlobalOp getStaticLocalDeclAddress(const VarDecl *d)
CIRGenCXXABI & getCXXABI() const
cir::GlobalViewAttr getAddrOfConstantStringFromLiteral(const StringLiteral *s, llvm::StringRef name=".str")
Return a global symbol reference to a constant array for the given string literal.
void setAddrOfConstantCompoundLiteral(const CompoundLiteralExpr *e, cir::GlobalOp gv)
bool lookupRepresentativeDecl(llvm::StringRef mangledName, clang::GlobalDecl &gd) const
void emitDeclContext(const DeclContext *dc)
static constexpr const char * builtinCoroBegin
static constexpr const char * builtinCoroId
cir::FuncOp getOrCreateCIRFunction(llvm::StringRef mangledName, mlir::Type funcType, clang::GlobalDecl gd, bool forVTable, mlir::NamedAttrList extraAttrs)
clang::CharUnits getNaturalPointeeTypeAlignment(clang::QualType t, LValueBaseInfo *baseInfo=nullptr)
static constexpr const char * builtinCoroFree
void emitGlobal(clang::GlobalDecl gd)
Emit code for a single global function or variable declaration.
cir::LabelOp lookupBlockAddressInfo(cir::BlockAddrInfoAttr blockInfo)
bool mayBeEmittedEagerly(const clang::ValueDecl *d)
Determine whether the definition can be emitted eagerly, or should be delayed until the end of the tr...
llvm::DenseMap< const clang::FieldDecl *, llvm::StringRef > lambdaFieldToName
Keep a map between lambda fields and names, this needs to be per module since lambdas might get gener...
DiagnosticBuilder errorNYI(SourceLocation loc, llvm::StringRef feature, const T &name)
void mapBlockAddress(cir::BlockAddrInfoAttr blockInfo, cir::LabelOp label)
static constexpr const char * builtinCoroEnd
void addGlobalAnnotations(const clang::ValueDecl *d, mlir::Operation *gv)
Add global annotations for a global value (GlobalOp or FuncOp).
void setCIRFunctionAttributesForDefinition(const clang::FunctionDecl *fd, cir::FuncOp f)
Set extra attributes (inline, etc.) for a function.
std::string getOpenACCBindMangledName(const IdentifierInfo *bindName, const FunctionDecl *attachedFunction)
void emitGlobalFunctionDefinition(clang::GlobalDecl gd, mlir::Operation *op)
CIRGenVTables & getVTables()
void setFunctionLinkage(GlobalDecl gd, cir::FuncOp f)
std::vector< clang::GlobalDecl > deferredDeclsToEmit
void emitVTable(const CXXRecordDecl *rd)
This is a callback from Sema to tell us that a particular vtable is required to be emitted in this tr...
void emitOMPThreadPrivateDecl(const OMPThreadPrivateDecl *d)
CIRGenOpenMPRuntime & getOpenMPRuntime()
void emitAMDGPUMetadata()
Emits AMDGPU specific Metadata.
void emitOMPGroupPrivateDecl(const OMPGroupPrivateDecl *d)
cir::GlobalLinkageKind getVTableLinkage(const CXXRecordDecl *rd)
Return the appropriate linkage for the vtable, VTT, and type information of the given class.
Address createUnnamedGlobalFrom(const VarDecl &d, mlir::Attribute constAttr, CharUnits align)
void emitCXXSpecialVarDeclInit(const VarDecl *varDecl, cir::GlobalOp addr, bool performInit, mlir::Region &ctorRegion, mlir::Region &dtorRegion)
Helper function for the below two that will create the constructor/destructor in specified regions,...
mlir::Attribute getConstantArrayFromStringLiteral(const StringLiteral *e)
Return a constant array for the given string.
cir::VisibilityAttr getGlobalVisibilityAttrFromDecl(const Decl *decl)
void setCommonAttributes(GlobalDecl gd, mlir::Operation *op)
Set attributes which are common to any form of a global definition (alias, Objective-C method,...
This class organizes the cross-module state that is used while lowering AST types to CIR types.
Definition CIRGenTypes.h:50
Represents a C++ struct/union/class.
Definition DeclCXX.h:258
CharUnits - This is an opaque type for sizes expressed in character units.
Definition CharUnits.h:38
CodeGenOptions - Track various options which control how the code is optimized and passed to the back...
CompoundLiteralExpr - [C99 6.5.2.5].
Definition Expr.h:3611
DeclContext - This is used only as base class of specific decl types that can act as declaration cont...
Definition DeclBase.h:1466
Decl - This represents one declaration (or definition), e.g.
Definition DeclBase.h:86
Represents a ValueDecl that came out of a declarator.
Definition Decl.h:780
A little helper class used to produce diagnostics.
Concrete class used by the front-end to report problems and issues.
Definition Diagnostic.h:233
ExplicitCastExpr - An explicit cast written in the source code.
Definition Expr.h:3934
This represents one expression.
Definition Expr.h:112
Represents a member of a struct/union/class.
Definition Decl.h:3191
Represents a function declaration or definition.
Definition Decl.h:2027
GlobalDecl - represents a global declaration.
Definition GlobalDecl.h:57
One of these records is kept for each identifier that is lexed.
Keeps track of the various options that can be enabled, which controls the dialect of C or C++ that i...
Represents a prvalue temporary that is written into memory so that a reference can bind to it.
Definition ExprCXX.h:4920
A pointer to member type per C++ 8.3.3 - Pointers to members.
Definition TypeBase.h:3717
This represents a decl that may have a name.
Definition Decl.h:274
This represents 'pragma omp allocate ...' directive.
Definition DeclOpenMP.h:536
Pseudo declaration for capturing expressions.
Definition DeclOpenMP.h:445
This represents 'pragma omp declare mapper ...' directive.
Definition DeclOpenMP.h:349
This represents 'pragma omp declare reduction ...' directive.
Definition DeclOpenMP.h:239
This represents 'pragma omp groupprivate ...' directive.
Definition DeclOpenMP.h:173
This represents 'pragma omp requires...' directive.
Definition DeclOpenMP.h:479
This represents 'pragma omp threadprivate ...' directive.
Definition DeclOpenMP.h:110
A (possibly-)qualified type.
Definition TypeBase.h:937
Encodes a location in the source.
A trivial tuple used to represent a source range.
SourceLocation getBegin() const
Stmt - This represents one statement.
Definition Stmt.h:86
StringLiteral - This represents a string literal expression, e.g.
Definition Expr.h:1805
Represents the declaration of a struct/union/class/enum.
Definition Decl.h:3748
Exposes information about the current target.
Definition TargetInfo.h:227
A template parameter object.
CXXRecordDecl * getAsCXXRecordDecl() const
Retrieves the CXXRecordDecl that this type refers to, either because the type is a RecordType or beca...
Definition Type.h:26
UnaryOperator - This represents the unary-expression's (except sizeof and alignof),...
Definition Expr.h:2250
An artificial decl, representing a global anonymous constant value which is uniquified by value withi...
Definition DeclCXX.h:4473
Represent the declaration of a variable (in which case it is an lvalue) a function (in which case it ...
Definition Decl.h:712
Represents a variable declaration or definition.
Definition Decl.h:932
Defines the clang::TargetInfo interface.
@ Decl
The l-value was an access to a declared entity or something equivalently strong, like the address of ...
const internal::VariadicDynCastAllOfMatcher< Decl, VarDecl > varDecl
Matches variable declarations.
const internal::VariadicAllOfMatcher< Type > type
Matches Types in the clang AST.
const internal::VariadicAllOfMatcher< Decl > decl
Matches declarations.
The JSON file list parser is used to communicate input to InstallAPI.
GVALinkage
A more specific kind of linkage than enum Linkage.
Definition Linkage.h:72
OpenACCModifierKind
LangAS
Defines the address space values used by the address space qualifier of QualType.
Visibility
Describes the different kinds of visibility that a declaration may have.
Definition Visibility.h:34
@ HiddenVisibility
Objects with "hidden" visibility are not seen by the dynamic linker.
Definition Visibility.h:37
@ ProtectedVisibility
Objects with "protected" visibility are seen by the dynamic linker but always dynamically resolve to ...
Definition Visibility.h:42
@ DefaultVisibility
Objects with "default" visibility are seen by the dynamic linker and act like normal objects.
Definition Visibility.h:46