35#include "llvm/Analysis/ValueTracking.h"
36#include "llvm/IR/DataLayout.h"
37#include "llvm/IR/GlobalVariable.h"
38#include "llvm/IR/Intrinsics.h"
39#include "llvm/IR/Type.h"
43using namespace CodeGen;
46 "Clang max alignment greater than what LLVM supports?");
48void CodeGenFunction::EmitDecl(
const Decl &D) {
50 case Decl::BuiltinTemplate:
51 case Decl::TranslationUnit:
52 case Decl::ExternCContext:
54 case Decl::UnresolvedUsingTypename:
55 case Decl::ClassTemplateSpecialization:
56 case Decl::ClassTemplatePartialSpecialization:
57 case Decl::VarTemplateSpecialization:
58 case Decl::VarTemplatePartialSpecialization:
59 case Decl::TemplateTypeParm:
60 case Decl::UnresolvedUsingValue:
61 case Decl::NonTypeTemplateParm:
62 case Decl::CXXDeductionGuide:
64 case Decl::CXXConstructor:
65 case Decl::CXXDestructor:
66 case Decl::CXXConversion:
68 case Decl::MSProperty:
69 case Decl::IndirectField:
71 case Decl::ObjCAtDefsField:
73 case Decl::ImplicitParam:
74 case Decl::ClassTemplate:
75 case Decl::VarTemplate:
76 case Decl::FunctionTemplate:
77 case Decl::TypeAliasTemplate:
78 case Decl::TemplateTemplateParm:
79 case Decl::ObjCMethod:
80 case Decl::ObjCCategory:
81 case Decl::ObjCProtocol:
82 case Decl::ObjCInterface:
83 case Decl::ObjCCategoryImpl:
84 case Decl::ObjCImplementation:
85 case Decl::ObjCProperty:
86 case Decl::ObjCCompatibleAlias:
87 case Decl::PragmaComment:
88 case Decl::PragmaDetectMismatch:
89 case Decl::AccessSpec:
90 case Decl::LinkageSpec:
92 case Decl::ObjCPropertyImpl:
93 case Decl::FileScopeAsm:
94 case Decl::TopLevelStmt:
96 case Decl::FriendTemplate:
99 case Decl::UsingShadow:
100 case Decl::ConstructorUsingShadow:
101 case Decl::ObjCTypeParam:
103 case Decl::UnresolvedUsingIfExists:
104 case Decl::HLSLBuffer:
105 llvm_unreachable(
"Declaration should not be in declstmts!");
107 case Decl::CXXRecord:
115 DI->EmitAndRetainType(
getContext().getEnumType(cast<EnumDecl>(&D)));
118 case Decl::EnumConstant:
119 case Decl::StaticAssert:
123 case Decl::UnnamedGlobalConstant:
124 case Decl::TemplateParamObject:
125 case Decl::OMPThreadPrivate:
126 case Decl::OMPAllocate:
127 case Decl::OMPCapturedExpr:
128 case Decl::OMPRequires:
131 case Decl::ImplicitConceptSpecialization:
132 case Decl::LifetimeExtendedTemporary:
133 case Decl::RequiresExprBody:
137 case Decl::NamespaceAlias:
139 DI->EmitNamespaceAlias(cast<NamespaceAliasDecl>(D));
143 DI->EmitUsingDecl(cast<UsingDecl>(D));
145 case Decl::UsingEnum:
147 DI->EmitUsingEnumDecl(cast<UsingEnumDecl>(D));
149 case Decl::UsingPack:
150 for (
auto *Using : cast<UsingPackDecl>(D).expansions())
153 case Decl::UsingDirective:
155 DI->EmitUsingDirective(cast<UsingDirectiveDecl>(D));
158 case Decl::Decomposition: {
159 const VarDecl &VD = cast<VarDecl>(D);
161 "Should not see file-scope variables inside a function!");
163 if (
auto *DD = dyn_cast<DecompositionDecl>(&VD))
164 for (
auto *B : DD->bindings())
165 if (
auto *HD = B->getHoldingVar())
170 case Decl::OMPDeclareReduction:
173 case Decl::OMPDeclareMapper:
177 case Decl::TypeAlias: {
178 QualType Ty = cast<TypedefNameDecl>(D).getUnderlyingType();
180 DI->EmitAndRetainType(Ty);
203 llvm::GlobalValue::LinkageTypes
Linkage =
226 std::string ContextName;
228 if (
auto *CD = dyn_cast<CapturedDecl>(DC))
229 DC = cast<DeclContext>(CD->getNonClosureContext());
230 if (
const auto *FD = dyn_cast<FunctionDecl>(DC))
232 else if (
const auto *BD = dyn_cast<BlockDecl>(DC))
234 else if (
const auto *OMD = dyn_cast<ObjCMethodDecl>(DC))
235 ContextName = OMD->getSelector().getAsString();
237 llvm_unreachable(
"Unknown context for static var decl");
249 if (llvm::Constant *ExistingGV = StaticLocalDeclMap[&D])
268 llvm::Constant *
Init =
nullptr;
270 D.
hasAttr<CUDASharedAttr>() || D.
hasAttr<LoaderUninitializedAttr>())
271 Init = llvm::UndefValue::get(LTy);
275 llvm::GlobalVariable *GV =
new llvm::GlobalVariable(
277 nullptr, llvm::GlobalVariable::NotThreadLocal, TargetAS);
278 GV->setAlignment(
getContext().getDeclAlign(&D).getAsAlign());
281 GV->setComdat(TheModule.getOrInsertComdat(GV->getName()));
290 llvm::Constant *Addr = GV;
291 if (AS != ExpectedAS) {
293 *
this, GV, AS, ExpectedAS,
295 getContext().getTargetAddressSpace(ExpectedAS)));
306 if (isa<BlockDecl>(DC) || isa<CapturedDecl>(DC)) {
314 if (
const auto *CD = dyn_cast<CXXConstructorDecl>(DC))
316 else if (
const auto *DD = dyn_cast<CXXDestructorDecl>(DC))
318 else if (
const auto *FD = dyn_cast<FunctionDecl>(DC))
323 assert(isa<ObjCMethodDecl>(DC) &&
"unexpected parent code decl");
338llvm::GlobalVariable *
340 llvm::GlobalVariable *GV) {
342 llvm::Constant *
Init = emitter.tryEmitForInitializer(D);
354 GV->setConstant(
false);
366 assert(VarSize == CstSize &&
"Emitted constant has unexpected size");
373 if (GV->getValueType() !=
Init->getType()) {
374 llvm::GlobalVariable *OldGV = GV;
376 GV =
new llvm::GlobalVariable(
378 OldGV->getLinkage(),
Init,
"",
379 OldGV, OldGV->getThreadLocalMode(),
380 OldGV->getType()->getPointerAddressSpace());
381 GV->setVisibility(OldGV->getVisibility());
382 GV->setDSOLocal(OldGV->isDSOLocal());
383 GV->setComdat(OldGV->getComdat());
389 OldGV->replaceAllUsesWith(GV);
392 OldGV->eraseFromParent();
400 GV->setInitializer(
Init);
402 emitter.finalize(GV);
415 llvm::GlobalValue::LinkageTypes
Linkage) {
425 setAddrOfLocalVar(&D,
Address(addr, elemTy, alignment));
434 llvm::Type *expectedType = addr->getType();
436 llvm::GlobalVariable *var =
437 cast<llvm::GlobalVariable>(addr->stripPointerCasts());
446 if (D.
getInit() && !isCudaSharedVar)
454 if (
auto *SA = D.
getAttr<PragmaClangBSSSectionAttr>())
455 var->addAttribute(
"bss-section", SA->getName());
456 if (
auto *SA = D.
getAttr<PragmaClangDataSectionAttr>())
457 var->addAttribute(
"data-section", SA->getName());
458 if (
auto *SA = D.
getAttr<PragmaClangRodataSectionAttr>())
459 var->addAttribute(
"rodata-section", SA->getName());
460 if (
auto *SA = D.
getAttr<PragmaClangRelroSectionAttr>())
461 var->addAttribute(
"relro-section", SA->getName());
463 if (
const SectionAttr *SA = D.
getAttr<SectionAttr>())
464 var->setSection(SA->getName());
468 else if (D.
hasAttr<UsedAttr>())
479 llvm::Constant *castedAddr =
480 llvm::ConstantExpr::getPointerBitCastOrAddrSpaceCast(var, expectedType);
481 LocalDeclMap.find(&D)->second =
Address(castedAddr, elemTy, alignment);
497 CodeGenFunction::Destroyer *destroyer,
498 bool useEHCleanupForArray)
499 : addr(addr),
type(
type), destroyer(destroyer),
500 useEHCleanupForArray(useEHCleanupForArray) {}
504 CodeGenFunction::Destroyer *destroyer;
505 bool useEHCleanupForArray;
509 bool useEHCleanupForArray =
510 flags.isForNormalCleanup() && this->useEHCleanupForArray;
516 template <
class Derived>
519 : NRVOFlag(NRVOFlag), Loc(addr), Ty(
type) {}
521 llvm::Value *NRVOFlag;
527 bool NRVO = flags.isForNormalCleanup() && NRVOFlag;
529 llvm::BasicBlock *SkipDtorBB =
nullptr;
534 llvm::Value *DidNRVO =
536 CGF.
Builder.CreateCondBr(DidNRVO, SkipDtorBB, RunDtorBB);
540 static_cast<Derived *
>(
this)->emitDestructorCall(CGF);
545 virtual ~DestroyNRVOVariable() =
default;
548 struct DestroyNRVOVariableCXX final
549 : DestroyNRVOVariable<DestroyNRVOVariableCXX> {
552 : DestroyNRVOVariable<DestroyNRVOVariableCXX>(addr,
type, NRVOFlag),
564 struct DestroyNRVOVariableC final
565 : DestroyNRVOVariable<DestroyNRVOVariableC> {
567 : DestroyNRVOVariable<DestroyNRVOVariableC>(addr, Ty, NRVOFlag) {}
576 CallStackRestore(
Address Stack) : Stack(Stack) {}
577 bool isRedundantBeforeReturn()
override {
return true; }
585 std::pair<llvm::Value *, llvm::Value *> AddrSizePair;
586 KmpcAllocFree(
const std::pair<llvm::Value *, llvm::Value *> &AddrSizePair)
587 : AddrSizePair(AddrSizePair) {}
596 ExtendGCLifetime(
const VarDecl *var) : Var(*var) {}
610 llvm::Constant *CleanupFn;
614 CallCleanupFunction(llvm::Constant *CleanupFn,
const CGFunctionInfo *Info,
616 : CleanupFn(CleanupFn), FnInfo(*Info), Var(*Var) {}
651 llvm_unreachable(
"present but none");
659 (var.
hasAttr<ObjCPreciseLifetimeAttr>()
683 if (
const Expr *e = dyn_cast<Expr>(
s)) {
686 s = e = e->IgnoreParenCasts();
688 if (
const DeclRefExpr *ref = dyn_cast<DeclRefExpr>(e))
689 return (ref->getDecl() == &var);
690 if (
const BlockExpr *be = dyn_cast<BlockExpr>(e)) {
691 const BlockDecl *block = be->getBlockDecl();
692 for (
const auto &I : block->
captures()) {
693 if (I.getVariable() == &var)
699 for (
const Stmt *SubStmt :
s->children())
708 if (!
decl)
return false;
709 if (!isa<VarDecl>(
decl))
return false;
716 bool needsCast =
false;
723 case CK_BlockPointerToObjCPointerCast:
729 case CK_LValueToRValue: {
772 if (!
SanOpts.
has(SanitizerKind::NullabilityAssign))
781 SanitizerScope SanScope(
this);
783 llvm::Constant *StaticData[] = {
785 llvm::ConstantInt::get(
Int8Ty, 0),
788 SanitizerHandler::TypeMismatch, StaticData, RHS);
792 LValue lvalue,
bool capturedByInit) {
804 init = DIE->getExpr();
808 if (
auto *EWC = dyn_cast<ExprWithCleanups>(init)) {
809 CodeGenFunction::RunCleanupsScope
Scope(*
this);
810 return EmitScalarInit(EWC->getSubExpr(), D, lvalue, capturedByInit);
817 bool accessedByInit =
false;
819 accessedByInit = (capturedByInit ||
isAccessedBy(D, init));
820 if (accessedByInit) {
823 if (capturedByInit) {
845 llvm::Value *value =
nullptr;
849 llvm_unreachable(
"present but none");
852 if (!D || !isa<VarDecl>(D) || !cast<VarDecl>(D)->isARCPseudoStrong()) {
911 unsigned &NumStores) {
913 if (isa<llvm::ConstantAggregateZero>(
Init) ||
914 isa<llvm::ConstantPointerNull>(
Init) ||
915 isa<llvm::UndefValue>(
Init))
917 if (isa<llvm::ConstantInt>(
Init) || isa<llvm::ConstantFP>(
Init) ||
918 isa<llvm::ConstantVector>(
Init) || isa<llvm::BlockAddress>(
Init) ||
919 isa<llvm::ConstantExpr>(
Init))
920 return Init->isNullValue() || NumStores--;
923 if (isa<llvm::ConstantArray>(
Init) || isa<llvm::ConstantStruct>(
Init)) {
924 for (
unsigned i = 0, e =
Init->getNumOperands(); i != e; ++i) {
925 llvm::Constant *Elt = cast<llvm::Constant>(
Init->getOperand(i));
932 if (llvm::ConstantDataSequential *CDS =
933 dyn_cast<llvm::ConstantDataSequential>(
Init)) {
934 for (
unsigned i = 0, e = CDS->getNumElements(); i != e; ++i) {
935 llvm::Constant *Elt = CDS->getElementAsConstant(i);
952 assert(!
Init->isNullValue() && !isa<llvm::UndefValue>(
Init) &&
953 "called emitStoresForInitAfterBZero for zero or undef value.");
955 if (isa<llvm::ConstantInt>(
Init) || isa<llvm::ConstantFP>(
Init) ||
956 isa<llvm::ConstantVector>(
Init) || isa<llvm::BlockAddress>(
Init) ||
957 isa<llvm::ConstantExpr>(
Init)) {
958 auto *I = Builder.CreateStore(
Init, Loc, isVolatile);
960 I->addAnnotationMetadata(
"auto-init");
964 if (llvm::ConstantDataSequential *CDS =
965 dyn_cast<llvm::ConstantDataSequential>(
Init)) {
966 for (
unsigned i = 0, e = CDS->getNumElements(); i != e; ++i) {
967 llvm::Constant *Elt = CDS->getElementAsConstant(i);
970 if (!Elt->isNullValue() && !isa<llvm::UndefValue>(Elt))
972 CGM, Elt, Builder.CreateConstInBoundsGEP2_32(Loc, 0, i), isVolatile,
973 Builder, IsAutoInit);
978 assert((isa<llvm::ConstantStruct>(
Init) || isa<llvm::ConstantArray>(
Init)) &&
979 "Unknown value type!");
981 for (
unsigned i = 0, e =
Init->getNumOperands(); i != e; ++i) {
982 llvm::Constant *Elt = cast<llvm::Constant>(
Init->getOperand(i));
985 if (!Elt->isNullValue() && !isa<llvm::UndefValue>(Elt))
987 Builder.CreateConstInBoundsGEP2_32(Loc, 0, i),
988 isVolatile, Builder, IsAutoInit);
996 uint64_t GlobalSize) {
998 if (isa<llvm::ConstantAggregateZero>(
Init))
return true;
1004 unsigned StoreBudget = 6;
1005 uint64_t SizeLimit = 32;
1007 return GlobalSize > SizeLimit &&
1017 uint64_t GlobalSize,
1018 const llvm::DataLayout &DL) {
1019 uint64_t SizeLimit = 32;
1020 if (GlobalSize <= SizeLimit)
1022 return llvm::isBytewiseValue(
Init, DL);
1029 uint64_t GlobalByteSize) {
1031 uint64_t ByteSizeLimit = 64;
1034 if (GlobalByteSize <= ByteSizeLimit)
1044 if (isPattern == IsPattern::Yes)
1047 return llvm::Constant::getNullValue(Ty);
1051 llvm::Constant *constant);
1056 llvm::StructType *STy,
1057 llvm::Constant *constant) {
1059 const llvm::StructLayout *Layout = DL.getStructLayout(STy);
1060 llvm::Type *Int8Ty = llvm::IntegerType::getInt8Ty(CGM.
getLLVMContext());
1061 unsigned SizeSoFar = 0;
1063 bool NestedIntact =
true;
1064 for (
unsigned i = 0, e = STy->getNumElements(); i != e; i++) {
1065 unsigned CurOff = Layout->getElementOffset(i);
1066 if (SizeSoFar < CurOff) {
1067 assert(!STy->isPacked());
1068 auto *PadTy = llvm::ArrayType::get(Int8Ty, CurOff - SizeSoFar);
1071 llvm::Constant *CurOp;
1072 if (constant->isZeroValue())
1073 CurOp = llvm::Constant::getNullValue(STy->getElementType(i));
1075 CurOp = cast<llvm::Constant>(constant->getAggregateElement(i));
1078 NestedIntact =
false;
1079 Values.push_back(NewOp);
1080 SizeSoFar = CurOff + DL.getTypeAllocSize(CurOp->getType());
1082 unsigned TotalSize = Layout->getSizeInBytes();
1083 if (SizeSoFar < TotalSize) {
1084 auto *PadTy = llvm::ArrayType::get(Int8Ty, TotalSize - SizeSoFar);
1087 if (NestedIntact && Values.size() == STy->getNumElements())
1089 return llvm::ConstantStruct::getAnon(Values, STy->isPacked());
1095 llvm::Constant *constant) {
1096 llvm::Type *OrigTy = constant->getType();
1097 if (
const auto STy = dyn_cast<llvm::StructType>(OrigTy))
1099 if (
auto *ArrayTy = dyn_cast<llvm::ArrayType>(OrigTy)) {
1101 uint64_t Size = ArrayTy->getNumElements();
1104 llvm::Type *ElemTy = ArrayTy->getElementType();
1105 bool ZeroInitializer = constant->isNullValue();
1106 llvm::Constant *OpValue, *PaddedOp;
1107 if (ZeroInitializer) {
1108 OpValue = llvm::Constant::getNullValue(ElemTy);
1111 for (
unsigned Op = 0; Op != Size; ++Op) {
1112 if (!ZeroInitializer) {
1113 OpValue = constant->getAggregateElement(Op);
1116 Values.push_back(PaddedOp);
1118 auto *NewElemTy = Values[0]->getType();
1119 if (NewElemTy == ElemTy)
1121 auto *NewArrayTy = llvm::ArrayType::get(NewElemTy, Size);
1122 return llvm::ConstantArray::get(NewArrayTy, Values);
1131 llvm::Constant *Constant,
1133 auto FunctionName = [&](
const DeclContext *DC) -> std::string {
1134 if (
const auto *FD = dyn_cast<FunctionDecl>(DC)) {
1135 if (
const auto *CC = dyn_cast<CXXConstructorDecl>(FD))
1136 return CC->getNameAsString();
1137 if (
const auto *CD = dyn_cast<CXXDestructorDecl>(FD))
1138 return CD->getNameAsString();
1140 }
else if (
const auto *OM = dyn_cast<ObjCMethodDecl>(DC)) {
1141 return OM->getNameAsString();
1142 }
else if (isa<BlockDecl>(DC)) {
1144 }
else if (isa<CapturedDecl>(DC)) {
1145 return "<captured>";
1147 llvm_unreachable(
"expected a function or method");
1153 llvm::GlobalVariable *&CacheEntry = InitializerConstants[&D];
1154 if (!CacheEntry || CacheEntry->getInitializer() != Constant) {
1155 auto *Ty = Constant->getType();
1156 bool isConstant =
true;
1157 llvm::GlobalVariable *InsertBefore =
nullptr;
1164 Name = (
"__const." + FunctionName(DC) +
"." + D.
getName()).str();
1166 llvm_unreachable(
"local variable has no parent function or method");
1167 llvm::GlobalVariable *GV =
new llvm::GlobalVariable(
1168 getModule(), Ty, isConstant, llvm::GlobalValue::PrivateLinkage,
1169 Constant, Name, InsertBefore, llvm::GlobalValue::NotThreadLocal, AS);
1171 GV->setUnnamedAddr(llvm::GlobalValue::UnnamedAddr::Global);
1173 }
else if (CacheEntry->getAlignment() < uint64_t(Align.
getQuantity())) {
1174 CacheEntry->setAlignment(Align.
getAsAlign());
1177 return Address(CacheEntry, CacheEntry->getValueType(), Align);
1183 llvm::Constant *Constant,
1192 llvm::Constant *constant,
bool IsAutoInit) {
1193 auto *Ty = constant->getType();
1194 uint64_t ConstantSize = CGM.
getDataLayout().getTypeAllocSize(Ty);
1198 bool canDoSingleStore = Ty->isIntOrIntVectorTy() ||
1199 Ty->isPtrOrPtrVectorTy() || Ty->isFPOrFPVectorTy();
1200 if (canDoSingleStore) {
1201 auto *I = Builder.CreateStore(constant, Loc, isVolatile);
1203 I->addAnnotationMetadata(
"auto-init");
1207 auto *SizeVal = llvm::ConstantInt::get(CGM.
IntPtrTy, ConstantSize);
1212 auto *I = Builder.CreateMemSet(Loc, llvm::ConstantInt::get(CGM.
Int8Ty, 0),
1213 SizeVal, isVolatile);
1215 I->addAnnotationMetadata(
"auto-init");
1217 bool valueAlreadyCorrect =
1218 constant->isNullValue() || isa<llvm::UndefValue>(constant);
1219 if (!valueAlreadyCorrect) {
1228 llvm::Value *Pattern =
1231 uint64_t
Value = 0x00;
1232 if (!isa<llvm::UndefValue>(Pattern)) {
1233 const llvm::APInt &AP = cast<llvm::ConstantInt>(Pattern)->getValue();
1234 assert(AP.getBitWidth() <= 8);
1235 Value = AP.getLimitedValue();
1237 auto *I = Builder.CreateMemSet(
1238 Loc, llvm::ConstantInt::get(CGM.
Int8Ty,
Value), SizeVal, isVolatile);
1240 I->addAnnotationMetadata(
"auto-init");
1246 if (
auto *STy = dyn_cast<llvm::StructType>(Ty)) {
1247 const llvm::StructLayout *Layout =
1249 for (
unsigned i = 0; i != constant->getNumOperands(); i++) {
1251 Address EltPtr = Builder.CreateConstInBoundsByteGEP(
1254 constant->getAggregateElement(i), IsAutoInit);
1257 }
else if (
auto *ATy = dyn_cast<llvm::ArrayType>(Ty)) {
1258 for (
unsigned i = 0; i != ATy->getNumElements(); i++) {
1259 Address EltPtr = Builder.CreateConstGEP(
1262 constant->getAggregateElement(i), IsAutoInit);
1270 Builder.CreateMemCpy(Loc,
1273 SizeVal, isVolatile);
1275 I->addAnnotationMetadata(
"auto-init");
1282 llvm::Constant *constant =
1294 assert(!isa<llvm::UndefValue>(constant));
1300 auto *Ty = constant->getType();
1301 if (isa<llvm::UndefValue>(constant))
1303 if (Ty->isStructTy() || Ty->isArrayTy() || Ty->isVectorTy())
1304 for (llvm::Use &Op : constant->operands())
1311 llvm::Constant *constant) {
1312 auto *Ty = constant->getType();
1313 if (isa<llvm::UndefValue>(constant))
1315 if (!(Ty->isStructTy() || Ty->isArrayTy() || Ty->isVectorTy()))
1320 for (
unsigned Op = 0, NumOp = constant->getNumOperands(); Op != NumOp; ++Op) {
1321 auto *OpValue = cast<llvm::Constant>(constant->getOperand(Op));
1324 if (Ty->isStructTy())
1325 return llvm::ConstantStruct::get(cast<llvm::StructType>(Ty), Values);
1326 if (Ty->isArrayTy())
1327 return llvm::ConstantArray::get(cast<llvm::ArrayType>(Ty), Values);
1328 assert(Ty->isVectorTy());
1329 return llvm::ConstantVector::get(Values);
1345 llvm::Value *Addr) {
1346 if (!ShouldEmitLifetimeMarkers)
1349 assert(Addr->getType()->getPointerAddressSpace() ==
1351 "Pointer should be in alloca address space");
1352 llvm::Value *SizeV = llvm::ConstantInt::get(
1356 C->setDoesNotThrow();
1361 assert(Addr->getType()->getPointerAddressSpace() ==
1363 "Pointer should be in alloca address space");
1366 C->setDoesNotThrow();
1378 while (
getContext().getAsVariableArrayType(Type1D)) {
1380 if (
auto *
C = dyn_cast<llvm::ConstantInt>(VlaSize.NumElts))
1384 Twine Name = Twine(
"__vla_expr") + Twine(VLAExprCounter++);
1386 StringRef NameRef = Name.toStringRef(Buffer);
1388 VLAExprNames.push_back(&Ident);
1392 Dimensions.emplace_back(SizeExprAddr.getPointer(),
1395 Type1D = VlaSize.Type;
1404 unsigned NameIdx = 0;
1405 for (
auto &VlaSize : Dimensions) {
1407 if (
auto *
C = dyn_cast<llvm::ConstantInt>(VlaSize.NumElts))
1408 MD = llvm::ConstantAsMetadata::get(
C);
1413 SizeTy->getScalarSizeInBits(),
false);
1418 ArtificialDecl->setImplicit();
1423 assert(MD &&
"No Size expression debug node created");
1430CodeGenFunction::AutoVarEmission
1437 AutoVarEmission emission(D);
1440 emission.IsEscapingByRef = isEscapingByRef;
1465 address = OpenMPLocalAddr;
1466 AllocaAddr = OpenMPLocalAddr;
1479 getContext().getBaseElementType(Ty)->isObjCObjectPointerType()) &&
1498 assert(emission.wasEmittedAsGlobal());
1503 emission.IsConstantAggregate =
true;
1518 const auto *RD = RecordTy->getDecl();
1519 const auto *CXXRD = dyn_cast<CXXRecordDecl>(RD);
1520 if ((CXXRD && !CXXRD->hasTrivialDestructor()) ||
1521 RD->isNonTrivialToPrimitiveDestroy()) {
1538 llvm::Type *allocaTy;
1539 if (isEscapingByRef) {
1541 allocaTy = byrefInfo.Type;
1542 allocaAlignment = byrefInfo.ByrefAlignment;
1545 allocaAlignment = alignment;
1552 nullptr, &AllocaAddr);
1557 bool IsMSCatchParam =
1579 emission.SizeForLifetimeMarkers =
1583 assert(!emission.useLifetimeMarkers());
1594 bool VarAllocated =
false;
1597 if (RT.isDelayedVariableLengthDecl(*
this, &D)) {
1599 std::pair<llvm::Value *, llvm::Value *> AddrSizePair =
1606 address =
Base.getAddress(*
this);
1614 VarAllocated =
true;
1618 if (!VarAllocated) {
1619 if (!DidCallStackSave) {
1624 llvm::Value *
V =
Builder.CreateStackSave();
1628 DidCallStackSave =
true;
1649 setAddrOfLocalVar(&D, address);
1650 emission.Addr = address;
1651 emission.AllocaAddr = AllocaAddr;
1660 if (UsePointerValue) {
1672 if (emission.useLifetimeMarkers())
1674 emission.getOriginalAllocatedAddress(),
1675 emission.getSizeForLifetimeMarkers());
1685 if (
const Expr *E = dyn_cast<Expr>(S))
1687 for (
const Stmt *SubStmt : S->children())
1700 if (
const BlockExpr *BE = dyn_cast<BlockExpr>(E)) {
1702 for (
const auto &I :
Block->captures()) {
1703 if (I.getVariable() == &Var)
1711 if (
const StmtExpr *SE = dyn_cast<StmtExpr>(E)) {
1713 for (
const auto *BI : CS->
body())
1714 if (
const auto *BIE = dyn_cast<Expr>(BI)) {
1718 else if (
const auto *DS = dyn_cast<DeclStmt>(BI)) {
1720 for (
const auto *I : DS->decls()) {
1721 if (
const auto *VD = dyn_cast<VarDecl>((I))) {
1752 !Construct->requiresZeroInitialization())
1758void CodeGenFunction::emitZeroOrPatternForAutoVarInit(
QualType type,
1763 bool isVolatile =
type.isVolatileQualified();
1764 if (!
Size.isZero()) {
1765 switch (trivialAutoVarInit) {
1767 llvm_unreachable(
"Uninitialized handled by caller");
1791 auto SizeVal = VlaSize.NumElts;
1793 switch (trivialAutoVarInit) {
1795 llvm_unreachable(
"Uninitialized handled by caller");
1800 if (!EltSize.
isOne())
1803 SizeVal, isVolatile);
1804 I->addAnnotationMetadata(
"auto-init");
1818 llvm::Value *IsZeroSizedVLA =
Builder.CreateICmpEQ(
1819 SizeVal, llvm::ConstantInt::get(SizeVal->getType(), 0),
1821 Builder.CreateCondBr(IsZeroSizedVLA, ContBB, SetupBB);
1823 if (!EltSize.
isOne())
1825 llvm::Value *BaseSizeInChars =
1828 llvm::Value *End =
Builder.CreateInBoundsGEP(
1829 Begin.getElementType(),
Begin.getPointer(), SizeVal,
"vla.end");
1830 llvm::BasicBlock *OriginBB =
Builder.GetInsertBlock();
1832 llvm::PHINode *Cur =
Builder.CreatePHI(
Begin.getType(), 2,
"vla.cur");
1833 Cur->addIncoming(
Begin.getPointer(), OriginBB);
1839 BaseSizeInChars, isVolatile);
1840 I->addAnnotationMetadata(
"auto-init");
1842 Builder.CreateInBoundsGEP(
Int8Ty, Cur, BaseSizeInChars,
"vla.next");
1843 llvm::Value *Done =
Builder.CreateICmpEQ(Next, End,
"vla-init.isdone");
1844 Builder.CreateCondBr(Done, ContBB, LoopBB);
1845 Cur->addIncoming(Next, LoopBB);
1852 assert(emission.Variable &&
"emission was not valid!");
1855 if (emission.wasEmittedAsGlobal())
return;
1857 const VarDecl &D = *emission.Variable;
1872 if (emission.IsEscapingByRef)
1879 type.isNonTrivialToPrimitiveDefaultInitialize() ==
1882 if (emission.IsEscapingByRef)
1891 bool capturedByInit =
1894 bool locIsByrefHeader = !capturedByInit;
1896 locIsByrefHeader ? emission.getObjectAddress(*
this) : emission.Addr;
1902 : (D.
getAttr<UninitializedAttr>()
1906 auto initializeWhatIsTechnicallyUninitialized = [&](
Address Loc) {
1907 if (trivialAutoVarInit ==
1912 if (emission.IsEscapingByRef && !locIsByrefHeader)
1915 return emitZeroOrPatternForAutoVarInit(
type, D, Loc);
1919 return initializeWhatIsTechnicallyUninitialized(Loc);
1921 llvm::Constant *constant =
nullptr;
1922 if (emission.IsConstantAggregate ||
1924 assert(!capturedByInit &&
"constant init contains a capturing block?");
1926 if (constant && !constant->isZeroValue() &&
1927 (trivialAutoVarInit !=
1945 initializeWhatIsTechnicallyUninitialized(Loc);
1951 if (!emission.IsConstantAggregate) {
1974 LValue lvalue,
bool capturedByInit) {
1977 if (
type->isReferenceType()) {
1996 if (
type->isAtomicType()) {
2000 if (isa<VarDecl>(D))
2002 else if (
auto *FD = dyn_cast<FieldDecl>(D))
2012 llvm_unreachable(
"bad evaluation kind");
2017 const CodeGenFunction::AutoVarEmission &emission,
2023 Address addr = emission.getObjectAddress(*
this);
2025 const VarDecl *var = emission.Variable;
2033 llvm_unreachable(
"no cleanup for trivially-destructible variable");
2038 if (emission.NRVOFlag) {
2039 assert(!
type->isArrayType());
2041 EHStack.pushCleanup<DestroyNRVOVariableCXX>(cleanupKind, addr,
type, dtor,
2055 if (!var->
hasAttr<ObjCPreciseLifetimeAttr>())
2064 if (emission.NRVOFlag) {
2065 assert(!
type->isArrayType());
2066 EHStack.pushCleanup<DestroyNRVOVariableC>(cleanupKind, addr,
2067 emission.NRVOFlag,
type);
2078 bool useEHCleanup = (cleanupKind &
EHCleanup);
2079 EHStack.pushCleanup<DestroyObject>(cleanupKind, addr,
type, destroyer,
2084 assert(emission.Variable &&
"emission was not valid!");
2087 if (emission.wasEmittedAsGlobal())
return;
2093 const VarDecl &D = *emission.Variable;
2101 D.
hasAttr<ObjCPreciseLifetimeAttr>()) {
2106 if (
const CleanupAttr *CA = D.
getAttr<CleanupAttr>()) {
2110 assert(F &&
"Could not find function!");
2119 if (emission.IsEscapingByRef &&
2122 if (emission.Variable->getType().isObjCGCWeak())
2143 llvm_unreachable(
"Unknown DestructionKind");
2150 assert(dtorKind &&
"cannot push destructor for trivial type");
2160 assert(dtorKind &&
"cannot push destructor for trivial type");
2169 bool useEHCleanupForArray) {
2170 pushFullExprCleanup<DestroyObject>(cleanupKind, addr,
type,
2171 destroyer, useEHCleanupForArray);
2175 EHStack.pushCleanup<CallStackRestore>(
Kind, SPMem);
2179 CleanupKind Kind, std::pair<llvm::Value *, llvm::Value *> AddrSizePair) {
2180 EHStack.pushCleanup<KmpcAllocFree>(
Kind, AddrSizePair);
2185 Destroyer *destroyer,
2186 bool useEHCleanupForArray) {
2194 EHStack.pushCleanup<DestroyObject>(
2196 destroyer, useEHCleanupForArray);
2198 return pushCleanupAfterFullExprWithActiveFlag<DestroyObject>(
2207 using ConditionalCleanupType =
2215 EHStack.pushCleanup<ConditionalCleanupType>(
2216 static_cast<CleanupKind>(cleanupKind & ~NormalCleanup), SavedAddr,
type,
2217 destroyer, useEHCleanupForArray);
2221 pushCleanupAfterFullExprWithActiveFlag<ConditionalCleanupType>(
2222 cleanupKind, ActiveFlag, SavedAddr,
type, destroyer,
2223 useEHCleanupForArray);
2238 Destroyer *destroyer,
2239 bool useEHCleanupForArray) {
2242 return destroyer(*
this, addr,
type);
2251 bool checkZeroLength =
true;
2254 if (llvm::ConstantInt *constLength = dyn_cast<llvm::ConstantInt>(
length)) {
2256 if (constLength->isZero())
return;
2257 checkZeroLength =
false;
2264 checkZeroLength, useEHCleanupForArray);
2281 Destroyer *destroyer,
2282 bool checkZeroLength,
2283 bool useEHCleanup) {
2291 if (checkZeroLength) {
2292 llvm::Value *isEmpty =
Builder.CreateICmpEQ(begin, end,
2293 "arraydestroy.isempty");
2294 Builder.CreateCondBr(isEmpty, doneBB, bodyBB);
2298 llvm::BasicBlock *entryBB =
Builder.GetInsertBlock();
2300 llvm::PHINode *elementPast =
2301 Builder.CreatePHI(begin->getType(), 2,
"arraydestroy.elementPast");
2302 elementPast->addIncoming(end, entryBB);
2305 llvm::Value *negativeOne = llvm::ConstantInt::get(
SizeTy, -1,
true);
2307 llvm::Value *element =
Builder.CreateInBoundsGEP(
2308 llvmElementType, elementPast, negativeOne,
"arraydestroy.element");
2315 destroyer(*
this,
Address(element, llvmElementType, elementAlign),
2322 llvm::Value *done =
Builder.CreateICmpEQ(element, begin,
"arraydestroy.done");
2323 Builder.CreateCondBr(done, doneBB, bodyBB);
2324 elementPast->addIncoming(element,
Builder.GetInsertBlock());
2333 llvm::Value *begin, llvm::Value *end,
2339 unsigned arrayDepth = 0;
2348 llvm::Value *zero = llvm::ConstantInt::get(CGF.
SizeTy, 0);
2351 begin = CGF.
Builder.CreateInBoundsGEP(
2352 elemTy, begin, gepIndices,
"pad.arraybegin");
2353 end = CGF.
Builder.CreateInBoundsGEP(
2354 elemTy, end, gepIndices,
"pad.arrayend");
2369 llvm::Value *ArrayBegin;
2370 llvm::Value *ArrayEnd;
2372 CodeGenFunction::Destroyer *Destroyer;
2375 RegularPartialArrayDestroy(llvm::Value *arrayBegin, llvm::Value *arrayEnd,
2377 CodeGenFunction::Destroyer *destroyer)
2378 : ArrayBegin(arrayBegin), ArrayEnd(arrayEnd),
2379 ElementType(elementType), Destroyer(destroyer),
2380 ElementAlign(elementAlign) {}
2384 ElementType, ElementAlign, Destroyer);
2392 llvm::Value *ArrayBegin;
2395 CodeGenFunction::Destroyer *Destroyer;
2398 IrregularPartialArrayDestroy(llvm::Value *arrayBegin,
2402 CodeGenFunction::Destroyer *destroyer)
2403 : ArrayBegin(arrayBegin), ArrayEndPointer(arrayEndPointer),
2404 ElementType(elementType), Destroyer(destroyer),
2405 ElementAlign(elementAlign) {}
2410 ElementType, ElementAlign, Destroyer);
2425 Destroyer *destroyer) {
2426 pushFullExprCleanup<IrregularPartialArrayDestroy>(
EHCleanup,
2427 arrayBegin, arrayEndPointer,
2428 elementType, elementAlign,
2439 llvm::Value *arrayEnd,
2442 Destroyer *destroyer) {
2443 pushFullExprCleanup<RegularPartialArrayDestroy>(
EHCleanup,
2444 arrayBegin, arrayEnd,
2445 elementType, elementAlign,
2451 if (LifetimeStartFn)
2452 return LifetimeStartFn;
2453 LifetimeStartFn = llvm::Intrinsic::getDeclaration(&
getModule(),
2455 return LifetimeStartFn;
2461 return LifetimeEndFn;
2462 LifetimeEndFn = llvm::Intrinsic::getDeclaration(&
getModule(),
2464 return LifetimeEndFn;
2473 ConsumeARCParameter(llvm::Value *param,
2475 : Param(param), Precise(precise) {}
2490 bool NoDebugInfo =
false;
2492 assert((isa<ParmVarDecl>(D) || isa<ImplicitParamDecl>(D)) &&
2493 "Invalid argument to EmitParmDecl");
2497 if (!isa<llvm::GlobalValue>(Arg.getAnyValue()))
2498 Arg.getAnyValue()->setName(D.
getName());
2503 if (
auto IPD = dyn_cast<ImplicitParamDecl>(&D)) {
2507 llvm::Value *
V = Arg.isIndirect()
2509 : Arg.getDirectValue();
2521 bool DoStore =
false;
2523 bool UseIndirectDebugAddress =
false;
2526 if (Arg.isIndirect()) {
2527 DeclPtr = Arg.getIndirectAddress();
2533 AllocaPtr = DeclPtr;
2540 if (UseIndirectDebugAddress) {
2543 D.
getName() +
".indirect_addr");
2550 if (SrcLangAS != DestLangAS) {
2551 assert(
getContext().getTargetAddressSpace(SrcLangAS) ==
2557 *
this,
V, SrcLangAS, DestLangAS, T,
true),
2570 "unexpected destructor type");
2572 CalleeDestructedParamCleanups[cast<ParmVarDecl>(&D)] =
2583 DeclPtr = OpenMPLocalAddr;
2584 AllocaPtr = DeclPtr;
2588 D.
getName() +
".addr", &AllocaPtr);
2593 llvm::Value *ArgVal = (DoStore ? Arg.getDirectValue() :
nullptr);
2603 bool isConsumed = D.
hasAttr<NSConsumedAttr>();
2608 "pseudo-strong variable isn't strong?");
2609 assert(qs.
hasConst() &&
"pseudo-strong variable should be const!");
2614 if (Arg.isIndirect() && !ArgVal)
2658 setAddrOfLocalVar(&D, DeclPtr);
2666 if (
const auto *Var = dyn_cast_or_null<ParmVarDecl>(&D))
2671 if (D.
hasAttr<AnnotateAttr>())
2677 if (requiresReturnValueNullabilityCheck()) {
2680 SanitizerScope SanScope(
this);
2681 RetValNullabilityPrecondition =
2682 Builder.CreateAnd(RetValNullabilityPrecondition,
2683 Builder.CreateIsNotNull(Arg.getAnyValue()));
2690 if (!LangOpts.OpenMP || (!LangOpts.EmitAllDecls && !D->
isUsed()))
2697 if (!LangOpts.OpenMP || LangOpts.OpenMPSimd ||
2698 (!LangOpts.EmitAllDecls && !D->
isUsed()))
2709 const auto *DE = cast<DeclRefExpr>(E);
2710 const auto *VD = cast<VarDecl>(DE->getDecl());
2733 if (Entry->getType()->getAddressSpace() == TargetAS)
2738 llvm::PointerType *PTy = llvm::PointerType::get(Ty, TargetAS);
2743 llvm::GlobalVariable *DummyGV =
new llvm::GlobalVariable(
2744 getModule(), Entry->getValueType(),
false,
2745 llvm::GlobalValue::CommonLinkage,
nullptr,
"dummy",
nullptr,
2746 llvm::GlobalVariable::NotThreadLocal, Entry->getAddressSpace());
2747 Entry->replaceAllUsesWith(DummyGV);
2749 Entry->mutateType(PTy);
2750 llvm::Constant *NewPtrForOldDecl =
2751 llvm::ConstantExpr::getPointerBitCastOrAddrSpaceCast(
2752 Entry, DummyGV->getType());
2756 DummyGV->replaceAllUsesWith(NewPtrForOldDecl);
2757 DummyGV->eraseFromParent();
2761std::optional<CharUnits>
2763 if (
const auto *AA = VD->
getAttr<OMPAllocateDeclAttr>()) {
2764 if (
Expr *Alignment = AA->getAlignment()) {
2765 unsigned UserAlign =
2766 Alignment->EvaluateKnownConstInt(
getContext()).getExtValue();
2774 std::max<unsigned>(UserAlign, NaturalAlign.
getQuantity()));
2777 return std::nullopt;
Defines the clang::ASTContext interface.
static void emitStoresForInitAfterBZero(CodeGenModule &CGM, llvm::Constant *Init, Address Loc, bool isVolatile, CGBuilderTy &Builder, bool IsAutoInit)
For inits that canEmitInitWithFewStoresAfterBZero returned true for, emit the scalar stores that woul...
static bool isCapturedBy(const VarDecl &, const Expr *)
Determines whether the given __block variable is potentially captured by the given expression.
static void emitPartialArrayDestroy(CodeGenFunction &CGF, llvm::Value *begin, llvm::Value *end, QualType type, CharUnits elementAlign, CodeGenFunction::Destroyer *destroyer)
Perform partial array destruction as if in an EH cleanup.
static void emitStoresForPatternInit(CodeGenModule &CGM, const VarDecl &D, Address Loc, bool isVolatile, CGBuilderTy &Builder)
static bool canEmitInitWithFewStoresAfterBZero(llvm::Constant *Init, unsigned &NumStores)
Decide whether we can emit the non-zero parts of the specified initializer with equal or fewer than N...
static llvm::Constant * patternOrZeroFor(CodeGenModule &CGM, IsPattern isPattern, llvm::Type *Ty)
Generate a constant filled with either a pattern or zeroes.
static llvm::Constant * constWithPadding(CodeGenModule &CGM, IsPattern isPattern, llvm::Constant *constant)
Replace all padding bytes in a given constant with either a pattern byte or 0x00.
static llvm::Value * shouldUseMemSetToInitialize(llvm::Constant *Init, uint64_t GlobalSize, const llvm::DataLayout &DL)
Decide whether we should use memset to initialize a local variable instead of using a memcpy from a c...
static std::string getStaticDeclName(CodeGenModule &CGM, const VarDecl &D)
static void emitStoresForConstant(CodeGenModule &CGM, const VarDecl &D, Address Loc, bool isVolatile, CGBuilderTy &Builder, llvm::Constant *constant, bool IsAutoInit)
static bool shouldSplitConstantStore(CodeGenModule &CGM, uint64_t GlobalByteSize)
Decide whether we want to split a constant structure or array store into a sequence of its fields' st...
static llvm::Constant * replaceUndef(CodeGenModule &CGM, IsPattern isPattern, llvm::Constant *constant)
static bool tryEmitARCCopyWeakInit(CodeGenFunction &CGF, const LValue &destLV, const Expr *init)
static bool shouldUseBZeroPlusStoresToInitialize(llvm::Constant *Init, uint64_t GlobalSize)
Decide whether we should use bzero plus some stores to initialize a local variable instead of using a...
static llvm::Constant * constStructWithPadding(CodeGenModule &CGM, IsPattern isPattern, llvm::StructType *STy, llvm::Constant *constant)
Helper function for constWithPadding() to deal with padding in structures.
static bool containsUndef(llvm::Constant *constant)
static bool isAccessedBy(const VarDecl &var, const Stmt *s)
static void EmitAutoVarWithLifetime(CodeGenFunction &CGF, const VarDecl &var, Address addr, Qualifiers::ObjCLifetime lifetime)
EmitAutoVarWithLifetime - Does the setup required for an automatic variable with lifetime.
static Address createUnnamedGlobalForMemcpyFrom(CodeGenModule &CGM, const VarDecl &D, CGBuilderTy &Builder, llvm::Constant *Constant, CharUnits Align)
static void emitStoresForZeroInit(CodeGenModule &CGM, const VarDecl &D, Address Loc, bool isVolatile, CGBuilderTy &Builder)
static void drillIntoBlockVariable(CodeGenFunction &CGF, LValue &lvalue, const VarDecl *var)
CodeGenFunction::ComplexPairTy ComplexPairTy
This file defines OpenMP nodes for declarative directives.
static const RecordType * getRecordType(QualType QT)
Checks that the passed in QualType either is of RecordType or points to RecordType.
static const NamedDecl * getDefinition(const Decl *D)
Defines the SourceManager interface.
__device__ __2f16 float __ockl_bool s
CharUnits getTypeAlignInChars(QualType T) const
Return the ABI-specified alignment of a (complete) type T, in characters.
QualType getPointerType(QualType T) const
Return the uniqued reference to the type for a pointer to the specified type.
const LangOptions & getLangOpts() const
QualType getIntTypeForBitwidth(unsigned DestWidth, unsigned Signed) const
getIntTypeForBitwidth - sets integer QualTy according to specified details: bitwidth,...
CharUnits getDeclAlign(const Decl *D, bool ForAlignof=false) const
Return a conservative estimate of the alignment of the specified decl D.
const ArrayType * getAsArrayType(QualType T) const
Type Query functions.
CharUnits getTypeSizeInChars(QualType T) const
Return the size of the specified (complete) type T, in characters.
const VariableArrayType * getAsVariableArrayType(QualType T) const
unsigned getTargetAddressSpace(LangAS AS) const
Represents an array type, per C99 6.7.5.2 - Array Declarators.
Represents a block literal declaration, which is like an unnamed FunctionDecl.
ArrayRef< Capture > captures() const
BlockExpr - Adaptor class for mixing a BlockDecl with expressions.
Represents a call to a C++ constructor.
Represents a C++ constructor within a class.
A use of a default initializer in a constructor or in aggregate initialization.
Represents a C++ destructor within a class.
CharUnits - This is an opaque type for sizes expressed in character units.
llvm::Align getAsAlign() const
getAsAlign - Returns Quantity as a valid llvm::Align, Beware llvm::Align assumes power of two 8-bit b...
QuantityType getQuantity() const
getQuantity - Get the raw integer representation of this quantity.
static CharUnits One()
One - Construct a CharUnits quantity of one.
CharUnits alignmentOfArrayElement(CharUnits elementSize) const
Given that this is the alignment of the first element of an array, return the minimum alignment of an...
bool isOne() const
isOne - Test whether the quantity equals one.
static CharUnits fromQuantity(QuantityType Quantity)
fromQuantity - Construct a CharUnits quantity from a raw integer type.
bool hasReducedDebugInfo() const
Check if type and variable info should be emitted.
ABIArgInfo - Helper class to encapsulate information about how a specific C type should be passed to ...
bool getIndirectByVal() const
CharUnits getAlignment() const
Return the alignment of this pointer.
llvm::Type * getElementType() const
Return the type of the values stored in this address.
Address withPointer(llvm::Value *NewPointer, KnownNonNull_t IsKnownNonNull) const
Return address with different pointer, but same element type and alignment.
Address withElementType(llvm::Type *ElemTy) const
Return address with different element type, but same pointer and alignment.
KnownNonNull_t isKnownNonNull() const
Whether the pointer is known not to be null.
llvm::Value * getPointer() const
static AggValueSlot forLValue(const LValue &LV, CodeGenFunction &CGF, IsDestructed_t isDestructed, NeedsGCBarriers_t needsGC, IsAliased_t isAliased, Overlap_t mayOverlap, IsZeroed_t isZeroed=IsNotZeroed, IsSanitizerChecked_t isChecked=IsNotSanitizerChecked)
static ApplyDebugLocation CreateDefaultArtificial(CodeGenFunction &CGF, SourceLocation TemporaryLocation)
Apply TemporaryLocation if it is valid.
llvm::StoreInst * CreateStore(llvm::Value *Val, Address Addr, bool IsVolatile=false)
llvm::CallInst * CreateMemSet(Address Dest, llvm::Value *Value, llvm::Value *Size, bool IsVolatile=false)
llvm::LoadInst * CreateLoad(Address Addr, const llvm::Twine &Name="")
llvm::LoadInst * CreateFlagLoad(llvm::Value *Addr, const llvm::Twine &Name="")
Emit a load from an i1 flag variable.
llvm::CallInst * CreateMemCpy(Address Dest, Address Src, llvm::Value *Size, bool IsVolatile=false)
static CGCallee forDirect(llvm::Constant *functionPtr, const CGCalleeInfo &abstractInfo=CGCalleeInfo())
This class gathers all debug information during compilation and is responsible for emitting to llvm g...
void EmitGlobalVariable(llvm::GlobalVariable *GV, const VarDecl *Decl)
Emit information about a global variable.
Param2DILocTy & getParamDbgMappings()
llvm::DILocalVariable * EmitDeclareOfArgVariable(const VarDecl *Decl, llvm::Value *AI, unsigned ArgNo, CGBuilderTy &Builder, bool UsePointerValue=false)
Emit call to llvm.dbg.declare for an argument variable declaration.
llvm::DILocalVariable * EmitDeclareOfAutoVariable(const VarDecl *Decl, llvm::Value *AI, CGBuilderTy &Builder, const bool UsePointerValue=false)
Emit call to llvm.dbg.declare for an automatic variable declaration.
void setLocation(SourceLocation Loc)
Update the current source location.
void registerVLASizeExpression(QualType Ty, llvm::Metadata *SizeExpr)
Register VLA size expression debug node with the qualified type.
CGFunctionInfo - Class to encapsulate the information about a function definition.
const_arg_iterator arg_begin() const
MutableArrayRef< ArgInfo > arguments()
virtual void EmitWorkGroupLocalVarDecl(CodeGenFunction &CGF, const VarDecl &D)
Emit the IR required for a work-group-local variable declaration, and add an entry to CGF's LocalDecl...
Allows to disable automatic handling of functions used in target regions as those marked as omp decla...
virtual void getKmpcFreeShared(CodeGenFunction &CGF, const std::pair< llvm::Value *, llvm::Value * > &AddrSizePair)
Get call to __kmpc_free_shared.
void emitUserDefinedMapper(const OMPDeclareMapperDecl *D, CodeGenFunction *CGF=nullptr)
Emit the function for the user defined mapper construct.
virtual void processRequiresDirective(const OMPRequiresDecl *D)
Perform check on requires decl to ensure that target architecture supports unified addressing.
virtual std::pair< llvm::Value *, llvm::Value * > getKmpcAllocShared(CodeGenFunction &CGF, const VarDecl *VD)
Get call to __kmpc_alloc_shared.
virtual void emitUserDefinedReduction(CodeGenFunction *CGF, const OMPDeclareReductionDecl *D)
Emit code for the specified user defined reduction construct.
virtual Address getAddressOfLocalVariable(CodeGenFunction &CGF, const VarDecl *VD)
Gets the OpenMP-specific address of the local variable.
CallArgList - Type for representing both the value and type of arguments in a call.
void add(RValue rvalue, QualType type)
CodeGenFunction - This class organizes the per-function state that is used while generating LLVM code...
void emitAutoVarTypeCleanup(const AutoVarEmission &emission, QualType::DestructionKind dtorKind)
void enterByrefCleanup(CleanupKind Kind, Address Addr, BlockFieldFlags Flags, bool LoadBlockVarAddr, bool CanThrow)
Enter a cleanup to destroy a __block variable.
llvm::Value * EmitLifetimeStart(llvm::TypeSize Size, llvm::Value *Addr)
static TypeEvaluationKind getEvaluationKind(QualType T)
getEvaluationKind - Return the TypeEvaluationKind of QualType T.
static bool ContainsLabel(const Stmt *S, bool IgnoreCaseStmts=false)
ContainsLabel - Return true if the statement contains a label in it.
void PopCleanupBlock(bool FallThroughIsBranchThrough=false)
PopCleanupBlock - Will pop the cleanup entry on the stack and process all branch fixups.
static Destroyer destroyNonTrivialCStruct
static bool cxxDestructorCanThrow(QualType T)
Check if T is a C++ class that has a destructor that can throw.
SanitizerSet SanOpts
Sanitizers enabled for this function.
llvm::DenseMap< const VarDecl *, llvm::Value * > NRVOFlags
A mapping from NRVO variables to the flags used to indicate when the NRVO has been applied to this va...
void EmitARCMoveWeak(Address dst, Address src)
void EmitAutoVarDecl(const VarDecl &D)
EmitAutoVarDecl - Emit an auto variable declaration.
void EmitLifetimeEnd(llvm::Value *Size, llvm::Value *Addr)
void EmitStoreThroughLValue(RValue Src, LValue Dst, bool isInit=false)
EmitStoreThroughLValue - Store the specified rvalue into the specified lvalue, where both are guarant...
void pushLifetimeExtendedDestroy(CleanupKind kind, Address addr, QualType type, Destroyer *destroyer, bool useEHCleanupForArray)
static bool hasScalarEvaluationKind(QualType T)
const BlockByrefInfo & getBlockByrefInfo(const VarDecl *var)
void EmitDecl(const Decl &D)
EmitDecl - Emit a declaration.
void pushIrregularPartialArrayCleanup(llvm::Value *arrayBegin, Address arrayEndPointer, QualType elementType, CharUnits elementAlignment, Destroyer *destroyer)
llvm::Value * emitArrayLength(const ArrayType *arrayType, QualType &baseType, Address &addr)
emitArrayLength - Compute the length of an array, even if it's a VLA, and drill down to the base elem...
VlaSizePair getVLASize(const VariableArrayType *vla)
Returns an LLVM value that corresponds to the size, in non-variably-sized elements,...
CleanupKind getARCCleanupKind()
Retrieves the default cleanup kind for an ARC cleanup.
bool CurFuncIsThunk
In C++, whether we are code generating a thunk.
void EmitCXXDestructorCall(const CXXDestructorDecl *D, CXXDtorType Type, bool ForVirtualBase, bool Delegating, Address This, QualType ThisTy)
LValue EmitLValue(const Expr *E, KnownNonNull_t IsKnownNonNull=NotKnownNonNull)
EmitLValue - Emit code to compute a designator that specifies the location of the expression.
void EmitExtendGCLifetime(llvm::Value *object)
EmitExtendGCLifetime - Given a pointer to an Objective-C object, make sure it survives garbage collec...
llvm::Value * EmitARCStoreWeak(Address addr, llvm::Value *value, bool ignored)
void EmitVariablyModifiedType(QualType Ty)
EmitVLASize - Capture all the sizes for the VLA expressions in the given variably-modified type and s...
llvm::BasicBlock * createBasicBlock(const Twine &name="", llvm::Function *parent=nullptr, llvm::BasicBlock *before=nullptr)
createBasicBlock - Create an LLVM basic block.
const LangOptions & getLangOpts() const
llvm::Constant * EmitCheckTypeDescriptor(QualType T)
Emit a description of a type in a format suitable for passing to a runtime sanitizer handler.
void pushEHDestroy(QualType::DestructionKind dtorKind, Address addr, QualType type)
void emitArrayDestroy(llvm::Value *begin, llvm::Value *end, QualType elementType, CharUnits elementAlign, Destroyer *destroyer, bool checkZeroLength, bool useEHCleanup)
void EmitBlock(llvm::BasicBlock *BB, bool IsFinished=false)
EmitBlock - Emit the given block.
llvm::AllocaInst * CreateTempAlloca(llvm::Type *Ty, const Twine &Name="tmp", llvm::Value *ArraySize=nullptr)
CreateTempAlloca - This creates an alloca and inserts it into the entry block if ArraySize is nullptr...
const CodeGen::CGBlockInfo * BlockInfo
void EmitExprAsInit(const Expr *init, const ValueDecl *D, LValue lvalue, bool capturedByInit)
EmitExprAsInit - Emits the code necessary to initialize a location in memory with the given initializ...
void emitByrefStructureInit(const AutoVarEmission &emission)
ComplexPairTy EmitComplexExpr(const Expr *E, bool IgnoreReal=false, bool IgnoreImag=false)
EmitComplexExpr - Emit the computation of the specified expression of complex type,...
@ TCK_NonnullAssign
Checking the value assigned to a _Nonnull pointer. Must not be null.
llvm::Value * EmitARCStoreStrongCall(Address addr, llvm::Value *value, bool resultIgnored)
RValue EmitCall(const CGFunctionInfo &CallInfo, const CGCallee &Callee, ReturnValueSlot ReturnValue, const CallArgList &Args, llvm::CallBase **callOrInvoke, bool IsMustTail, SourceLocation Loc)
EmitCall - Generate a call of the given function, expecting the given result type,...
llvm::Type * ConvertTypeForMem(QualType T)
llvm::Value * EmitARCUnsafeUnretainedScalarExpr(const Expr *expr)
void EmitAutoVarInit(const AutoVarEmission &emission)
void EmitScalarInit(const Expr *init, const ValueDecl *D, LValue lvalue, bool capturedByInit)
DominatingValue< T >::saved_type saveValueInCond(T value)
void EmitStaticVarDecl(const VarDecl &D, llvm::GlobalValue::LinkageTypes Linkage)
static Destroyer destroyCXXObject
void EmitVarAnnotations(const VarDecl *D, llvm::Value *V)
Emit local annotations for the local variable V, declared by D.
Destroyer * getDestroyer(QualType::DestructionKind destructionKind)
void EmitAtomicInit(Expr *E, LValue lvalue)
const TargetInfo & getTarget() const
bool isInConditionalBranch() const
isInConditionalBranch - Return true if we're currently emitting one branch or the other of a conditio...
void emitDestroy(Address addr, QualType type, Destroyer *destroyer, bool useEHCleanupForArray)
void EmitCXXGuardedInit(const VarDecl &D, llvm::GlobalVariable *DeclPtr, bool PerformInit)
Emit code in this function to perform a guarded variable initialization.
void EmitARCRelease(llvm::Value *value, ARCPreciseLifetime_t precise)
void EmitCheck(ArrayRef< std::pair< llvm::Value *, SanitizerMask > > Checked, SanitizerHandler Check, ArrayRef< llvm::Constant * > StaticArgs, ArrayRef< llvm::Value * > DynamicArgs)
Create a basic block that will either trap or call a handler function in the UBSan runtime with the p...
void EmitARCCopyWeak(Address dst, Address src)
void setBlockContextParameter(const ImplicitParamDecl *D, unsigned argNum, llvm::Value *ptr)
void defaultInitNonTrivialCStructVar(LValue Dst)
bool HaveInsertPoint() const
HaveInsertPoint - True if an insertion point is defined.
llvm::Constant * EmitCheckSourceLocation(SourceLocation Loc)
Emit a description of a source location in a format suitable for passing to a runtime sanitizer handl...
bool isTrivialInitializer(const Expr *Init)
Determine whether the given initializer is trivial in the sense that it requires no code to be genera...
CGDebugInfo * getDebugInfo()
Address emitBlockByrefAddress(Address baseAddr, const VarDecl *V, bool followForward=true)
BuildBlockByrefAddress - Computes the location of the data in a variable which is declared as __block...
LValue EmitDeclRefLValue(const DeclRefExpr *E)
AggValueSlot::Overlap_t getOverlapForFieldInit(const FieldDecl *FD)
Determine whether a field initialization may overlap some other object.
Address createCleanupActiveFlag()
Address CreateDefaultAlignTempAlloca(llvm::Type *Ty, const Twine &Name="tmp")
CreateDefaultAlignedTempAlloca - This creates an alloca with the default ABI alignment of the given L...
llvm::Value * EmitARCRetainAutoreleaseScalarExpr(const Expr *expr)
const TargetCodeGenInfo & getTargetHooks() const
RValue EmitReferenceBindingToExpr(const Expr *E)
Emits a reference binding to the passed in expression.
void EmitAggExpr(const Expr *E, AggValueSlot AS)
EmitAggExpr - Emit the computation of the specified expression of aggregate type.
VlaSizePair getVLAElements1D(const VariableArrayType *vla)
Return the number of elements for a single dimension for the given array type.
void EmitVarDecl(const VarDecl &D)
EmitVarDecl - Emit a local variable declaration.
ASTContext & getContext() const
llvm::Value * EmitLoadOfScalar(Address Addr, bool Volatile, QualType Ty, SourceLocation Loc, AlignmentSource Source=AlignmentSource::Type, bool isNontemporal=false)
EmitLoadOfScalar - Load a scalar value from an address, taking care to appropriately convert from the...
Address CreateMemTemp(QualType T, const Twine &Name="tmp", Address *Alloca=nullptr)
CreateMemTemp - Create a temporary memory object of the given type, with appropriate alignmen and cas...
void EmitNullabilityCheck(LValue LHS, llvm::Value *RHS, SourceLocation Loc)
Given an assignment *LHS = RHS, emit a test that checks if RHS is nonnull, if LHS is marked _Nonnull.
void Destroyer(CodeGenFunction &CGF, Address addr, QualType ty)
AutoVarEmission EmitAutoVarAlloca(const VarDecl &var)
void pushDestroy(QualType::DestructionKind dtorKind, Address addr, QualType type)
Address ReturnValuePointer
ReturnValuePointer - The temporary alloca to hold a pointer to sret.
void EmitAutoVarCleanups(const AutoVarEmission &emission)
llvm::GlobalVariable * AddInitializerToStaticVarDecl(const VarDecl &D, llvm::GlobalVariable *GV)
AddInitializerToStaticVarDecl - Add the initializer for 'D' to the global variable that has already b...
static Destroyer destroyARCWeak
bool needsEHCleanup(QualType::DestructionKind kind)
Determines whether an EH cleanup is required to destroy a type with the given destruction kind.
CleanupKind getCleanupKind(QualType::DestructionKind kind)
llvm::Type * ConvertType(QualType T)
void EmitARCInitWeak(Address addr, llvm::Value *value)
static Destroyer destroyARCStrongPrecise
void initFullExprCleanupWithFlag(Address ActiveFlag)
VarBypassDetector Bypasses
llvm::Value * EmitARCRetainNonBlock(llvm::Value *value)
void pushStackRestore(CleanupKind kind, Address SPMem)
LValue MakeAddrLValue(Address Addr, QualType T, AlignmentSource Source=AlignmentSource::Type)
void EmitStoreOfComplex(ComplexPairTy V, LValue dest, bool isInit)
EmitStoreOfComplex - Store a complex number into the specified l-value.
const CGFunctionInfo * CurFnInfo
void pushKmpcAllocFree(CleanupKind Kind, std::pair< llvm::Value *, llvm::Value * > AddrSizePair)
void EmitParmDecl(const VarDecl &D, ParamValue Arg, unsigned ArgNo)
EmitParmDecl - Emit a ParmVarDecl or an ImplicitParamDecl.
Address ReturnValue
ReturnValue - The temporary alloca to hold the return value.
static Destroyer destroyARCStrongImprecise
void EnsureInsertPoint()
EnsureInsertPoint - Ensure that an insertion point is defined so that emitted IR has a place to go.
llvm::LLVMContext & getLLVMContext()
llvm::Value * EmitScalarExpr(const Expr *E, bool IgnoreResultAssign=false)
EmitScalarExpr - Emit the computation of the specified expression of LLVM scalar type,...
void EmitAndRegisterVariableArrayDimensions(CGDebugInfo *DI, const VarDecl &D, bool EmitDebugInfo)
Emits the alloca and debug information for the size expressions for each dimension of an array.
llvm::Value * EmitARCRetainScalarExpr(const Expr *expr)
void pushRegularPartialArrayCleanup(llvm::Value *arrayBegin, llvm::Value *arrayEnd, QualType elementType, CharUnits elementAlignment, Destroyer *destroyer)
void EmitStoreOfScalar(llvm::Value *Value, Address Addr, bool Volatile, QualType Ty, AlignmentSource Source=AlignmentSource::Type, bool isInit=false, bool isNontemporal=false)
EmitStoreOfScalar - Store a scalar value to an address, taking care to appropriately convert from the...
bool hasLabelBeenSeenInCurrentScope() const
Return true if a label was seen in the current scope.
This class organizes the cross-function state that is used while generating LLVM code.
StringRef getBlockMangledName(GlobalDecl GD, const BlockDecl *BD)
void setGVProperties(llvm::GlobalValue *GV, GlobalDecl GD) const
Set visibility, dllimport/dllexport and dso_local.
llvm::Module & getModule() const
void setStaticLocalDeclAddress(const VarDecl *D, llvm::Constant *C)
llvm::Function * getLLVMLifetimeStartFn()
Lazily declare the @llvm.lifetime.start intrinsic.
llvm::Constant * GetAddrOfFunction(GlobalDecl GD, llvm::Type *Ty=nullptr, bool ForVTable=false, bool DontDefer=false, ForDefinition_t IsForDefinition=NotForDefinition)
Return the address of the given function.
Address createUnnamedGlobalFrom(const VarDecl &D, llvm::Constant *Constant, CharUnits Align)
llvm::Constant * getNullPointer(llvm::PointerType *T, QualType QT)
Get target specific null pointer.
void ErrorUnsupported(const Stmt *S, const char *Type)
Print out an error that codegen doesn't support the specified stmt yet.
const LangOptions & getLangOpts() const
CharUnits getNaturalTypeAlignment(QualType T, LValueBaseInfo *BaseInfo=nullptr, TBAAAccessInfo *TBAAInfo=nullptr, bool forPointeeType=false)
CodeGenTypes & getTypes()
CGOpenCLRuntime & getOpenCLRuntime()
Return a reference to the configured OpenCL runtime.
void addUsedGlobal(llvm::GlobalValue *GV)
Add a global to a list to be added to the llvm.used metadata.
void EmitOMPAllocateDecl(const OMPAllocateDecl *D)
Emit a code for the allocate directive.
llvm::GlobalValue::LinkageTypes getLLVMLinkageVarDefinition(const VarDecl *VD)
Returns LLVM linkage for a declarator.
const llvm::DataLayout & getDataLayout() const
void addUsedOrCompilerUsedGlobal(llvm::GlobalValue *GV)
Add a global to a list to be added to the llvm.compiler.used metadata.
CGOpenMPRuntime & getOpenMPRuntime()
Return a reference to the configured OpenMP runtime.
SanitizerMetadata * getSanitizerMetadata()
llvm::Constant * getOrCreateStaticVarDecl(const VarDecl &D, llvm::GlobalValue::LinkageTypes Linkage)
llvm::Constant * GetAddrOfGlobal(GlobalDecl GD, ForDefinition_t IsForDefinition=NotForDefinition)
void AddGlobalAnnotations(const ValueDecl *D, llvm::GlobalValue *GV)
Add global annotations that are set on D, for the global GV.
void setTLSMode(llvm::GlobalValue *GV, const VarDecl &D) const
Set the TLS mode for the given LLVM GlobalValue for the thread-local variable declaration D.
ASTContext & getContext() const
void EmitOMPDeclareMapper(const OMPDeclareMapperDecl *D, CodeGenFunction *CGF=nullptr)
Emit a code for declare mapper construct.
llvm::Function * getLLVMLifetimeEndFn()
Lazily declare the @llvm.lifetime.end intrinsic.
bool supportsCOMDAT() const
void EmitOMPRequiresDecl(const OMPRequiresDecl *D)
Emit a code for requires directive.
const TargetCodeGenInfo & getTargetCodeGenInfo()
const CodeGenOptions & getCodeGenOpts() const
StringRef getMangledName(GlobalDecl GD)
std::optional< CharUnits > getOMPAllocateAlignment(const VarDecl *VD)
Return the alignment specified in an allocate directive, if present.
llvm::LLVMContext & getLLVMContext()
llvm::GlobalValue * GetGlobalValue(StringRef Ref)
void EmitOMPDeclareReduction(const OMPDeclareReductionDecl *D, CodeGenFunction *CGF=nullptr)
Emit a code for declare reduction construct.
llvm::Constant * EmitNullConstant(QualType T)
Return the result of value-initializing the given type, i.e.
LangAS GetGlobalConstantAddressSpace() const
Return the AST address space of constant literal, which is used to emit the constant literal as globa...
LangAS GetGlobalVarAddressSpace(const VarDecl *D)
Return the AST address space of the underlying global variable for D, as determined by its declaratio...
llvm::ConstantInt * getSize(CharUnits numChars)
Emit the given number of characters as a value of type size_t.
const CGFunctionInfo & arrangeFunctionDeclaration(const FunctionDecl *FD)
Free functions are functions that are compatible with an ordinary C function pointer type.
llvm::Type * ConvertTypeForMem(QualType T, bool ForBitField=false)
ConvertTypeForMem - Convert type T into a llvm::Type.
llvm::Constant * tryEmitAbstractForInitializer(const VarDecl &D)
Try to emit the initializer of the given declaration as an abstract constant.
Information for lazily generating a cleanup.
ConditionalCleanup stores the saved form of its parameters, then restores them and performs the clean...
stable_iterator stable_begin() const
Create a stable reference to the top of the EH stack.
LValue - This represents an lvalue references.
Address getAddress(CodeGenFunction &CGF) const
llvm::Value * getPointer(CodeGenFunction &CGF) const
void setNonGC(bool Value)
void setAddress(Address address)
Qualifiers::ObjCLifetime getObjCLifetime() const
RValue - This trivial value class is used to represent the result of an expression that is evaluated.
static RValue get(llvm::Value *V)
ReturnValueSlot - Contains the address where the return value of a function can be stored,...
virtual llvm::Value * performAddrSpaceCast(CodeGen::CodeGenFunction &CGF, llvm::Value *V, LangAS SrcAddr, LangAS DestAddr, llvm::Type *DestTy, bool IsNonNull=false) const
Perform address space cast of an expression of pointer type.
bool IsBypassed(const VarDecl *D) const
Returns true if the variable declaration was by bypassed by any goto or switch statement.
CompoundStmt - This represents a group of statements like { stmt stmt }.
DeclContext - This is used only as base class of specific decl types that can act as declaration cont...
A reference to a declared variable, function, enum, etc.
Decl - This represents one declaration (or definition), e.g.
const DeclContext * getParentFunctionOrMethod(bool LexicalParent=false) const
If this decl is defined inside a function/method/block it returns the corresponding DeclContext,...
Decl * getNonClosureContext()
Find the innermost non-closure ancestor of this declaration, walking up through blocks,...
SourceLocation getLocation() const
bool isUsed(bool CheckUsedAttr=true) const
Whether any (re-)declaration of the entity was used, meaning that a definition is required.
DeclContext * getDeclContext()
This represents one expression.
Expr * IgnoreParenCasts() LLVM_READONLY
Skip past any parentheses and casts which might surround this expression until reaching a fixed point...
Expr * IgnoreParens() LLVM_READONLY
Skip past any parentheses which might surround this expression until reaching a fixed point.
bool isLValue() const
isLValue - True if this expression is an "l-value" according to the rules of the current language.
bool isConstantInitializer(ASTContext &Ctx, bool ForRef, const Expr **Culprit=nullptr) const
isConstantInitializer - Returns true if this expression can be emitted to IR as a constant,...
SourceLocation getExprLoc() const LLVM_READONLY
getExprLoc - Return the preferred location for the arrow when diagnosing a problem with a generic exp...
Represents a function declaration or definition.
GlobalDecl - represents a global declaration.
const Decl * getDecl() const
One of these records is kept for each identifier that is lexed.
IdentifierInfo & getOwn(StringRef Name)
Gets an IdentifierInfo for the given name without consulting external sources.
StringRef getName() const
Get the name of identifier for this declaration as a StringRef.
std::string getNameAsString() const
Get a human-readable name for the declaration, even if it is one of the special kinds of names (C++ c...
bool isExternallyVisible() const
This represents '#pragma omp allocate ...' directive.
This represents '#pragma omp declare mapper ...' directive.
This represents '#pragma omp declare reduction ...' directive.
This represents '#pragma omp requires...' directive.
A (possibly-)qualified type.
@ DK_objc_strong_lifetime
@ PDIK_Struct
The type is a struct containing a field whose type is not PCK_Trivial.
LangAS getAddressSpace() const
Return the address space of this type.
bool isConstant(const ASTContext &Ctx) const
Qualifiers getQualifiers() const
Retrieve the set of qualifiers applied to this type.
Qualifiers::ObjCLifetime getObjCLifetime() const
Returns lifetime attribute of this type.
QualType getNonReferenceType() const
If Type is a reference type (e.g., const int&), returns the type that the reference refers to ("const...
QualType getUnqualifiedType() const
Retrieve the unqualified variant of the given type, removing as little sugar as possible.
bool isConstantStorage(const ASTContext &Ctx, bool ExcludeCtor, bool ExcludeDtor)
bool isPODType(const ASTContext &Context) const
Determine whether this is a Plain Old Data (POD) type (C++ 3.9p10).
The collection of all-type qualifiers we support.
@ OCL_Strong
Assigning into this object requires the old value to be released and the new value to be retained.
@ OCL_ExplicitNone
This object can be modified without requiring retains or releases.
@ OCL_None
There is no lifetime qualification on this type.
@ OCL_Weak
Reading or writing from this object requires a barrier call.
@ OCL_Autoreleasing
Assigning into this object requires a lifetime extension.
ObjCLifetime getObjCLifetime() const
bool isParamDestroyedInCallee() const
A helper class that allows the use of isa/cast/dyncast to detect TagType objects of structs/unions/cl...
RecordDecl * getDecl() const
Scope - A scope is a transient data structure that is used while parsing the program.
static const uint64_t MaximumAlignment
Encodes a location in the source.
StmtExpr - This is the GNU Statement Expression extension: ({int X=4; X;}).
Stmt - This represents one statement.
bool isMicrosoft() const
Is this ABI an MSVC-compatible ABI?
TargetCXXABI getCXXABI() const
Get the C++ ABI currently in use.
bool isConstantSizeType() const
Return true if this is not a variable sized type, according to the rules of C99 6....
const T * castAs() const
Member-template castAs<specific type>.
bool isVariablyModifiedType() const
Whether this type is a variably-modified type (C99 6.7.5).
const T * getAs() const
Member-template getAs<specific type>'.
bool isRecordType() const
std::optional< NullabilityKind > getNullability() const
Determine the nullability of the given type.
Represent the declaration of a variable (in which case it is an lvalue) a function (in which case it ...
Represents a variable declaration or definition.
static VarDecl * Create(ASTContext &C, DeclContext *DC, SourceLocation StartLoc, SourceLocation IdLoc, const IdentifierInfo *Id, QualType T, TypeSourceInfo *TInfo, StorageClass S)
bool isConstexpr() const
Whether this variable is (C++11) constexpr.
TLSKind getTLSKind() const
bool hasFlexibleArrayInit(const ASTContext &Ctx) const
Whether this variable has a flexible array member initialized with one or more elements.
bool hasGlobalStorage() const
Returns true for all variables that do not have local storage.
CharUnits getFlexibleArrayInitChars(const ASTContext &Ctx) const
If hasFlexibleArrayInit is true, compute the number of additional bytes necessary to store those elem...
bool mightBeUsableInConstantExpressions(const ASTContext &C) const
Determine whether this variable's value might be usable in a constant expression, according to the re...
bool isNRVOVariable() const
Determine whether this local variable can be used with the named return value optimization (NRVO).
bool isExceptionVariable() const
Determine whether this variable is the exception variable in a C++ catch statememt or an Objective-C ...
QualType::DestructionKind needsDestruction(const ASTContext &Ctx) const
Would the destruction of this variable have any effect, and if so, what kind?
const Expr * getInit() const
bool hasExternalStorage() const
Returns true if a variable has extern or private_extern storage.
bool isARCPseudoStrong() const
Determine whether this variable is an ARC pseudo-__strong variable.
bool hasLocalStorage() const
Returns true if a variable with function scope is a non-static local variable.
bool isLocalVarDecl() const
Returns true for local variable declarations other than parameters.
StorageDuration getStorageDuration() const
Get the storage duration of this variable, per C++ [basic.stc].
bool isEscapingByref() const
Indicates the capture is a __block variable that is captured by a block that can potentially escape (...
Defines the clang::TargetInfo interface.
@ Decl
The l-value was an access to a declared entity or something equivalently strong, like the address of ...
llvm::Constant * initializationPatternFor(CodeGenModule &, llvm::Type *)
@ NormalCleanup
Denotes a cleanup that should run when a scope is exited using normal control flow (falling off the e...
@ EHCleanup
Denotes a cleanup that should run when a scope is exited using exceptional control flow (a throw stat...
ARCPreciseLifetime_t
Does an ARC strong l-value have precise lifetime?
const internal::VariadicAllOfMatcher< Type > type
Matches Types in the clang AST.
const AstTypeMatcher< ArrayType > arrayType
Matches all kinds of arrays.
const internal::VariadicAllOfMatcher< Decl > decl
Matches declarations.
const internal::VariadicDynCastAllOfMatcher< Stmt, CastExpr > castExpr
Matches any cast nodes of Clang's AST.
bool Zero(InterpState &S, CodePtr OpPC)
bool Null(InterpState &S, CodePtr OpPC)
@ Ctor_Base
Base object ctor.
@ NonNull
Values of this type can never be null.
Linkage
Describes the different kinds of linkage (C++ [basic.link], C99 6.2.2) that an entity may have.
@ SD_Automatic
Automatic storage duration (most local variables).
@ Dtor_Base
Base object dtor.
@ Dtor_Complete
Complete object dtor.
LangAS
Defines the address space values used by the address space qualifier of QualType.
@ VK_LValue
An l-value expression is a reference to an object with independent storage.
@ ThreadPrivateVar
Parameter for Thread private variable.
float __ovld __cnfn length(float)
Return the length of vector p, i.e., sqrt(p.x2 + p.y 2 + ...)
static Address getAddressOfLocalVariable(CodeGenFunction &CGF, const VarDecl *VD)
Gets the OpenMP-specific address of the local variable /p VD.
llvm::IntegerType * Int64Ty
llvm::IntegerType * Int8Ty
i8, i16, i32, and i64
llvm::IntegerType * SizeTy
llvm::IntegerType * IntPtrTy
llvm::PointerType * AllocaInt8PtrTy
LangAS getASTAllocaAddressSpace() const
A metaprogramming class for ensuring that a value will dominate an arbitrary position in a function.
bool has(SanitizerMask K) const
Check if a certain (single) sanitizer is enabled.