23#include "llvm/IR/IRBuilder.h"
24#include "llvm/IR/IntrinsicInst.h"
25#include "llvm/IR/Intrinsics.h"
26#include "llvm/Transforms/Utils/Cloning.h"
35 : CGM(CGM), VTContext(CGM.getContext().getVTableContext()) {}
39 return GetOrCreateLLVMFunction(Name, FnTy, GD,
true,
44 llvm::GlobalVariable *VTable =
45 CGM.getCXXABI().getAddrOfVTable(RD,
CharUnits());
50 llvm::Function *ThunkFn,
bool ForVTable,
60 ThunkFn->setDLLStorageClass(llvm::GlobalValue::DefaultStorageClass);
61 ThunkFn->setDSOLocal(
true);
65 ThunkFn->setComdat(CGM.
getModule().getOrInsertComdat(ThunkFn->getName()));
84 llvm::BasicBlock *AdjustNull =
nullptr;
85 llvm::BasicBlock *AdjustNotNull =
nullptr;
86 llvm::BasicBlock *AdjustEnd =
nullptr;
108 if (NullCheckValue) {
109 CGF.
Builder.CreateBr(AdjustEnd);
111 CGF.
Builder.CreateBr(AdjustEnd);
114 llvm::PHINode *PHI = CGF.
Builder.CreatePHI(ReturnValue->getType(), 2);
115 PHI->addIncoming(ReturnValue, AdjustNotNull);
116 PHI->addIncoming(llvm::Constant::getNullValue(ReturnValue->getType()),
130 llvm::ValueToValueMapTy &VMap) {
132 auto *DIS = Fn->getSubprogram();
135 auto *NewDIS = llvm::MDNode::replaceWithDistinct(DIS->clone());
140 NewDIS->replaceRetainedNodes(llvm::MDTuple::get(Fn->getContext(), {}));
141 VMap.MD()[DIS].reset(NewDIS);
153 auto PrepareVariableMapping = [&VMap](llvm::DILocalVariable *DILocal) {
154 if (DILocal->isResolved())
157 if (llvm::DIType *Ty = DILocal->getType(); Ty && !Ty->isResolved())
158 VMap.MD()[Ty].reset(Ty);
163 for (
auto &BB : *Fn) {
165 for (llvm::DbgVariableRecord &DVR :
166 llvm::filterDbgVars(I.getDbgRecordRange()))
167 PrepareVariableMapping(DVR.getVariable());
169 if (
auto *DII = dyn_cast<llvm::DbgVariableIntrinsic>(&I))
170 PrepareVariableMapping(DII->getVariable());
201 llvm::Type *Ty =
CGM.getTypes().GetFunctionType(FnInfo);
202 llvm::Value *Callee =
CGM.GetAddrOfFunction(GD, Ty,
true);
209 CGM.ErrorUnsupported(MD,
"return-adjusting thunk with variadic arguments");
212 assert(!BaseFn->isDeclaration() &&
"cannot clone undefined variadic method");
215 llvm::ValueToValueMapTy VMap;
220 llvm::Function *NewFn = llvm::CloneFunction(BaseFn, VMap);
221 Fn->replaceAllUsesWith(NewFn);
223 Fn->eraseFromParent();
230 llvm::Function::arg_iterator AI = Fn->arg_begin();
231 if (
CGM.ReturnTypeUsesSRet(FnInfo))
239 llvm::BasicBlock *EntryBB = &Fn->front();
240 llvm::BasicBlock::iterator ThisStore =
241 llvm::find_if(*EntryBB, [&](llvm::Instruction &I) {
244 assert(ThisStore != EntryBB->end() &&
245 "Store of this should be in entry block?");
247 Builder.SetInsertPoint(&*ThisStore);
250 llvm::Value *AdjustedThisPtr =
CGM.getCXXABI().performThisAdjustment(
251 *
this, ThisPtr, ThisValueClass, Thunk);
252 AdjustedThisPtr =
Builder.CreateBitCast(AdjustedThisPtr,
253 ThisStore->getOperand(0)->getType());
254 ThisStore->setOperand(0, AdjustedThisPtr);
258 for (llvm::BasicBlock &BB : *Fn) {
259 llvm::Instruction *T = BB.getTerminator();
262 T->eraseFromParent();
276 bool IsUnprototyped) {
277 assert(!
CurGD.getDecl() &&
"CurGD was already set!");
286 ResultType =
CGM.getContext().VoidTy;
287 else if (
CGM.getCXXABI().HasThisReturn(GD))
288 ResultType = ThisType;
289 else if (
CGM.getCXXABI().hasMostDerivedReturn(GD))
290 ResultType =
CGM.getContext().VoidPtrTy;
296 CGM.getCXXABI().buildThisParam(*
this, FunctionArgs);
299 if (!IsUnprototyped) {
303 CGM.getCXXABI().addImplicitStructorParams(*
this, ResultType,
315 CGM.getCXXABI().EmitInstanceFunctionProlog(*
this);
316 CXXThisValue = CXXABIThisValue;
332 bool IsUnprototyped) {
334 "Please use a new CGF for this thunk");
343 llvm::Value *AdjustedThisPtr =
345 ThisValueClass, *Thunk)
354 CGM.ErrorUnsupported(
355 MD,
"return-adjusting thunk with incomplete parameter type");
357 llvm_unreachable(
"shouldn't try to emit musttail return-adjusting "
358 "thunks for variadic functions");
360 CGM.ErrorUnsupported(
361 MD,
"non-trivial argument copy for return-adjusting thunk");
373 CGM.getCXXABI().adjustCallArgsForDestructorThunk(*
this,
CurGD, CallArgs);
376 unsigned PrefixArgs = CallArgs.size() - 1;
387 assert(CallFnInfo.getRegParm() ==
CurFnInfo->getRegParm() &&
388 CallFnInfo.isNoReturn() ==
CurFnInfo->isNoReturn() &&
389 CallFnInfo.getCallingConvention() ==
CurFnInfo->getCallingConvention());
391 similar(CallFnInfo.getReturnInfo(), CallFnInfo.getReturnType(),
393 assert(CallFnInfo.arg_size() ==
CurFnInfo->arg_size());
394 for (
unsigned i = 0, e =
CurFnInfo->arg_size(); i != e; ++i)
395 assert(
similar(CallFnInfo.arg_begin()[i].info,
396 CallFnInfo.arg_begin()[i].type,
404 :
CGM.getCXXABI().hasMostDerivedReturn(
CurGD)
405 ?
CGM.getContext().VoidPtrTy
415 llvm::CallBase *CallOrInvoke;
417 CallArgs, &CallOrInvoke);
422 else if (llvm::CallInst*
Call = dyn_cast<llvm::CallInst>(CallOrInvoke))
423 Call->setTailCallKind(llvm::CallInst::TCK_Tail);
427 CGM.getCXXABI().EmitReturnFromThunk(*
this, RV, ResultType);
436 llvm::Value *AdjustedThisPtr,
437 llvm::FunctionCallee Callee) {
449 llvm::Type *ThisType = Args[ThisArgNo]->getType();
450 if (ThisType != AdjustedThisPtr->getType())
451 AdjustedThisPtr =
Builder.CreateBitCast(AdjustedThisPtr, ThisType);
452 Args[ThisArgNo] = AdjustedThisPtr;
454 assert(ThisAI.
isInAlloca() &&
"this is passed directly or inalloca");
457 if (ThisType != AdjustedThisPtr->getType())
458 AdjustedThisPtr =
Builder.CreateBitCast(AdjustedThisPtr, ThisType);
459 Builder.CreateStore(AdjustedThisPtr, ThisAddr);
464 llvm::CallInst *
Call =
Builder.CreateCall(Callee, Args);
465 Call->setTailCallKind(llvm::CallInst::TCK_MustTail);
469 llvm::AttributeList Attrs;
470 CGM.ConstructAttributeList(Callee.getCallee()->getName(), *
CurFnInfo, GD,
473 Call->setAttributes(Attrs);
476 if (
Call->getType()->isVoidTy())
491 bool IsUnprototyped) {
502 Ty =
CGM.getTypes().GetFunctionType(FnInfo);
504 llvm::Constant *Callee =
CGM.GetAddrOfFunction(GD, Ty,
true);
508 &Thunk, IsUnprototyped);
512 bool IsUnprototyped,
bool ForVTable) {
529llvm::Constant *CodeGenVTables::maybeEmitThunk(GlobalDecl GD,
537 SmallString<256> Name;
538 MangleContext &MCtx = CGM.getCXXABI().getMangleContext();
539 llvm::raw_svector_ostream
Out(Name);
541 if (
const CXXDestructorDecl *DD = dyn_cast<CXXDestructorDecl>(MD)) {
547 if (CGM.getContext().useAbbreviatedThunkName(GD, Name.str())) {
549 if (
const CXXDestructorDecl *DD = dyn_cast<CXXDestructorDecl>(MD))
556 llvm::Type *ThunkVTableTy = CGM.getTypes().GetFunctionTypeForVTable(GD);
557 llvm::Constant *Thunk = CGM.GetAddrOfThunk(Name, ThunkVTableTy, GD);
560 bool IsUnprototyped = !CGM.getTypes().isFuncTypeConvertible(
567 const CGFunctionInfo &FnInfo =
568 IsUnprototyped ? CGM.getTypes().arrangeUnprototypedMustTailThunk(MD)
569 : CGM.getTypes().arrangeGlobalDeclaration(GD);
570 llvm::FunctionType *ThunkFnTy = CGM.getTypes().GetFunctionType(FnInfo);
575 if (ThunkFn->getFunctionType() != ThunkFnTy) {
576 llvm::GlobalValue *OldThunkFn = ThunkFn;
578 assert(OldThunkFn->isDeclaration() &&
"Shouldn't replace non-declaration");
581 OldThunkFn->setName(StringRef());
582 ThunkFn = llvm::Function::Create(ThunkFnTy, llvm::Function::ExternalLinkage,
583 Name.str(), &CGM.getModule());
584 CGM.SetLLVMFunctionAttributes(MD, FnInfo, ThunkFn,
false);
586 if (!OldThunkFn->use_empty()) {
587 OldThunkFn->replaceAllUsesWith(ThunkFn);
591 OldThunkFn->eraseFromParent();
594 bool ABIHasKeyFunctions = CGM.getTarget().getCXXABI().hasKeyFunctions();
595 bool UseAvailableExternallyLinkage = ForVTable && ABIHasKeyFunctions;
597 if (!ThunkFn->isDeclaration()) {
598 if (!ABIHasKeyFunctions || UseAvailableExternallyLinkage) {
612 ThunkFn->addFnAttr(
"thunk");
614 CGM.SetLLVMFunctionAttributesForDefinition(GD.
getDecl(), ThunkFn);
621 bool ShouldCloneVarArgs =
false;
622 if (!IsUnprototyped && ThunkFn->isVarArg()) {
623 ShouldCloneVarArgs =
true;
625 switch (CGM.getTriple().getArch()) {
626 case llvm::Triple::x86_64:
627 case llvm::Triple::x86:
628 case llvm::Triple::aarch64:
629 ShouldCloneVarArgs =
false;
637 if (ShouldCloneVarArgs) {
638 if (UseAvailableExternallyLinkage)
641 CodeGenFunction(CGM).GenerateVarArgsThunk(ThunkFn, FnInfo, GD, TI);
644 CodeGenFunction(CGM).generateThunk(ThunkFn, FnInfo, GD, TI, IsUnprototyped);
660 VTContext->getThunkInfo(GD);
662 if (!ThunkInfoVector)
665 for (
const ThunkInfo& Thunk : *ThunkInfoVector)
666 maybeEmitThunk(GD, Thunk,
false);
670 llvm::Constant *component,
671 unsigned vtableAddressPoint,
672 bool vtableHasLocalLinkage,
673 bool isCompleteDtor)
const {
675 if (component->isNullValue())
676 return builder.
add(llvm::ConstantInt::get(CGM.
Int32Ty, 0));
693 auto stubLinkage = vtableHasLocalLinkage
694 ? llvm::GlobalValue::InternalLinkage
695 : llvm::GlobalValue::LinkOnceODRLinkage;
697 llvm::Constant *target;
698 if (
auto *func = dyn_cast<llvm::Function>(globalVal)) {
699 target = llvm::DSOLocalEquivalent::get(func);
702 rttiProxyName.append(
".rtti_proxy");
708 llvm::GlobalVariable *proxy =
module.getNamedGlobal(rttiProxyName);
710 proxy =
new llvm::GlobalVariable(module, globalVal->getType(),
712 globalVal, rttiProxyName);
713 proxy->setDSOLocal(
true);
714 proxy->setUnnamedAddr(llvm::GlobalValue::UnnamedAddr::Global);
715 if (!proxy->hasLocalLinkage()) {
716 proxy->setVisibility(llvm::GlobalValue::HiddenVisibility);
717 proxy->setComdat(module.getOrInsertComdat(rttiProxyName));
740 return CGM.getVTableComponentType();
746 builder.
add(llvm::ConstantExpr::getIntToPtr(
758 const VTableLayout &layout,
759 unsigned componentIndex,
760 llvm::Constant *rtti,
761 unsigned &nextVTableThunkIndex,
762 unsigned vtableAddressPoint,
763 bool vtableHasLocalLinkage) {
766 bool RelativeCXXABIVTables = CGM.getLangOpts().RelativeCXXABIVTables;
767 auto addOffsetConstant =
770 switch (component.getKind()) {
772 return addOffsetConstant(CGM, builder, component.getVCallOffset());
775 return addOffsetConstant(CGM, builder, component.getVBaseOffset());
778 return addOffsetConstant(CGM, builder, component.getOffsetToTop());
781 if (RelativeCXXABIVTables)
782 return addRelativeComponent(builder, rtti, vtableAddressPoint,
783 vtableHasLocalLinkage,
786 return builder.
add(rtti);
791 GlobalDecl GD = component.getGlobalDecl(
792 CGM.getContext().getTargetInfo().emitVectorDeletingDtors(
793 CGM.getContext().getLangOpts()));
797 layout.
vtable_thunks()[nextVTableThunkIndex].first == componentIndex;
799 if (CGM.getLangOpts().CUDA) {
807 CGM.getLangOpts().CUDAIsDevice
808 ? MD->
hasAttr<CUDADeviceAttr>()
809 : (MD->
hasAttr<CUDAHostAttr>() || !MD->
hasAttr<CUDADeviceAttr>());
810 if (!CanEmitMethod) {
812 nextVTableThunkIndex++;
814 llvm::ConstantExpr::getNullValue(CGM.GlobalsInt8PtrTy));
819 auto getSpecialVirtualFn = [&](StringRef
name) -> llvm::Constant * {
827 if (RelativeCXXABIVTables)
828 return llvm::ConstantPointerNull::get(CGM.GlobalsInt8PtrTy);
832 if (CGM.getLangOpts().OpenMP && CGM.getLangOpts().OpenMPIsTargetDevice &&
833 CGM.getTriple().isNVPTX())
834 return llvm::ConstantPointerNull::get(CGM.GlobalsInt8PtrTy);
835 llvm::FunctionType *fnTy =
836 llvm::FunctionType::get(CGM.VoidTy,
false);
838 CGM.CreateRuntimeFunction(fnTy, name).getCallee());
839 if (
auto f = dyn_cast<llvm::Function>(fn))
840 f->setUnnamedAddr(llvm::GlobalValue::UnnamedAddr::Global);
844 llvm::Constant *fnPtr;
850 getSpecialVirtualFn(CGM.getCXXABI().GetPureVirtualCallName());
851 fnPtr = PureVirtualFn;
855 if (!DeletedVirtualFn)
857 getSpecialVirtualFn(CGM.getCXXABI().GetDeletedVirtualCallName());
858 fnPtr = DeletedVirtualFn;
861 }
else if (IsThunk) {
862 auto &thunkInfo = layout.
vtable_thunks()[nextVTableThunkIndex].second;
864 nextVTableThunkIndex++;
865 fnPtr = maybeEmitThunk(GD, thunkInfo,
true);
866 if (CGM.getCodeGenOpts().PointerAuth.CXXVirtualFunctionPointers) {
867 assert(thunkInfo.Method &&
"Method not set");
873 llvm::Type *fnTy = CGM.getTypes().GetFunctionTypeForVTable(GD);
874 fnPtr = CGM.GetAddrOfFunction(GD, fnTy,
true);
875 if (CGM.getCodeGenOpts().PointerAuth.CXXVirtualFunctionPointers)
879 if (RelativeCXXABIVTables) {
880 return addRelativeComponent(
881 builder, fnPtr, vtableAddressPoint, vtableHasLocalLinkage,
888 unsigned FnAS = fnPtr->getType()->getPointerAddressSpace();
889 unsigned GVAS = CGM.GlobalsInt8PtrTy->getPointerAddressSpace();
893 llvm::ConstantExpr::getAddrSpaceCast(fnPtr, CGM.GlobalsInt8PtrTy);
894 if (
const auto &Schema =
895 CGM.getCodeGenOpts().PointerAuth.CXXVirtualFunctionPointers)
897 return builder.
add(fnPtr);
902 if (RelativeCXXABIVTables)
903 return builder.
add(llvm::ConstantExpr::getNullValue(CGM.Int32Ty));
908 llvm_unreachable(
"Unexpected vtable component kind");
914 for (
unsigned i = 0, e = layout.
getNumVTables(); i != e; ++i)
915 tys.push_back(llvm::ArrayType::get(componentType, layout.
getVTableSize(i)));
917 return llvm::StructType::get(CGM.getLLVMContext(), tys);
922 llvm::Constant *rtti,
923 bool vtableHasLocalLinkage) {
927 unsigned nextVTableThunkIndex = 0;
928 for (
unsigned vtableIndex = 0, endIndex = layout.
getNumVTables();
929 vtableIndex != endIndex; ++vtableIndex) {
930 auto vtableElem = builder.
beginArray(componentType);
933 size_t vtableEnd = vtableStart + layout.
getVTableSize(vtableIndex);
934 for (
size_t componentIndex = vtableStart; componentIndex < vtableEnd;
936 addVTableComponent(vtableElem, layout, componentIndex, rtti,
937 nextVTableThunkIndex, addressPoints[vtableIndex],
938 vtableHasLocalLinkage);
940 vtableElem.finishAndAddTo(builder);
946 llvm::GlobalVariable::LinkageTypes
Linkage,
947 VTableAddressPointsMapTy &AddressPoints) {
949 DI->completeClassData(
Base.getBase());
951 std::unique_ptr<VTableLayout> VTLayout(
953 Base.getBase(),
Base.getBaseOffset(), BaseIsVirtual, RD));
956 AddressPoints = VTLayout->getAddressPoints();
960 llvm::raw_svector_ostream Out(OutName);
962 .mangleCXXCtorVTable(RD,
Base.getBaseOffset().getQuantity(),
963 Base.getBase(), Out);
966 bool UsingRelativeLayout = CGM.getLangOpts().RelativeCXXABIVTables;
967 bool VTableAliasExists =
968 UsingRelativeLayout && CGM.getModule().getNamedAlias(Name);
969 if (VTableAliasExists) {
971 Name.append(
".local");
981 if (
Linkage == llvm::GlobalVariable::AvailableExternallyLinkage)
982 Linkage = llvm::GlobalVariable::InternalLinkage;
984 llvm::Align Align = CGM.getDataLayout().getABITypeAlign(VTType);
987 llvm::GlobalVariable *VTable =
988 CGM.CreateOrReplaceCXXRuntimeVariable(Name, VTType,
Linkage, Align);
990 llvm::Constant *RTTI = CGM.GetAddrOfRTTIDescriptor(
991 CGM.getContext().getCanonicalTagType(
Base.getBase()));
997 VTable->hasLocalLinkage());
998 components.finishAndSetAsInitializer(VTable);
1002 assert(!VTable->isDeclaration() &&
"Shouldn't set properties on declaration");
1003 CGM.setGVProperties(VTable, RD);
1005 CGM.EmitVTableTypeMetadata(RD, VTable, *VTLayout);
1007 if (UsingRelativeLayout) {
1009 if (!VTable->isDSOLocal())
1025 if (CGM.getLangOpts().Sanitize.has(SanitizerKind::HWAddress)) {
1026 llvm::GlobalValue::SanitizerMetadata Meta;
1027 if (GV->hasSanitizerMetadata())
1028 Meta = GV->getSanitizerMetadata();
1029 Meta.NoHWAddress =
true;
1030 GV->setSanitizerMetadata(Meta);
1041 llvm::StringRef AliasNameRef) {
1042 assert(CGM.getLangOpts().RelativeCXXABIVTables &&
1043 "Can only use this if the relative vtable ABI is used");
1044 assert(!VTable->isDSOLocal() &&
"This should be called only if the vtable is "
1045 "not guaranteed to be dso_local");
1050 if (VTable->hasAvailableExternallyLinkage())
1057 VTable->setName(AliasName +
".local");
1059 auto Linkage = VTable->getLinkage();
1060 assert(llvm::GlobalAlias::isValidLinkage(
Linkage) &&
1061 "Invalid vtable alias linkage");
1063 llvm::GlobalAlias *VTableAlias = CGM.getModule().getNamedAlias(AliasName);
1065 VTableAlias = llvm::GlobalAlias::create(VTable->getValueType(),
1066 VTable->getAddressSpace(),
Linkage,
1067 AliasName, &CGM.getModule());
1069 assert(VTableAlias->getValueType() == VTable->getValueType());
1070 assert(VTableAlias->getLinkage() ==
Linkage);
1072 VTableAlias->setVisibility(VTable->getVisibility());
1075 if (!VTable->hasComdat()) {
1076 VTable->setLinkage(llvm::GlobalValue::InternalLinkage);
1086 VTable->setVisibility(llvm::GlobalValue::HiddenVisibility);
1089 VTableAlias->setAliasee(VTable);
1101llvm::GlobalVariable::LinkageTypes
1104 return llvm::GlobalVariable::InternalLinkage;
1113 IsInNamedModule ?
nullptr : Context.getCurrentKeyFunction(RD);
1114 if (IsInNamedModule || (keyFunction && !RD->
hasAttr<DLLImportAttr>())) {
1118 if (keyFunction && keyFunction->
hasBody(def))
1121 bool IsExternalDefinition =
1132 (IsInNamedModule || def || CodeGenOpts.OptimizationLevel > 0 ||
1133 CodeGenOpts.getDebugInfo() != llvm::codegenoptions::NoDebugInfo) &&
1134 "Shouldn't query vtable linkage without the class in module units, "
1135 "key function, optimizations, or debug info");
1136 if (IsExternalDefinition && CodeGenOpts.OptimizationLevel > 0)
1137 return llvm::GlobalVariable::AvailableExternallyLinkage;
1139 if (keyFunction && keyFunction->
isInlined())
1140 return !Context.getLangOpts().AppleKext
1141 ? llvm::GlobalVariable::LinkOnceODRLinkage
1142 : llvm::Function::InternalLinkage;
1144 return llvm::GlobalVariable::ExternalLinkage;
1147 return !Context.getLangOpts().AppleKext ?
1148 llvm::GlobalVariable::LinkOnceODRLinkage :
1149 llvm::Function::InternalLinkage;
1152 return !Context.getLangOpts().AppleKext ?
1153 llvm::GlobalVariable::WeakODRLinkage :
1154 llvm::Function::InternalLinkage;
1157 return IsExternalDefinition
1158 ? llvm::GlobalVariable::AvailableExternallyLinkage
1159 : llvm::GlobalVariable::ExternalLinkage;
1165 if (Context.getLangOpts().AppleKext)
1166 return llvm::Function::InternalLinkage;
1168 llvm::GlobalVariable::LinkageTypes DiscardableODRLinkage =
1169 llvm::GlobalValue::LinkOnceODRLinkage;
1170 llvm::GlobalVariable::LinkageTypes NonDiscardableODRLinkage =
1171 llvm::GlobalValue::WeakODRLinkage;
1172 if (RD->
hasAttr<DLLExportAttr>()) {
1174 DiscardableODRLinkage = NonDiscardableODRLinkage;
1175 }
else if (RD->
hasAttr<DLLImportAttr>()) {
1177 DiscardableODRLinkage = llvm::GlobalVariable::AvailableExternallyLinkage;
1178 NonDiscardableODRLinkage = llvm::GlobalVariable::AvailableExternallyLinkage;
1185 return DiscardableODRLinkage;
1191 return DiscardableODRLinkage;
1193 ? llvm::GlobalVariable::AvailableExternallyLinkage
1194 : llvm::GlobalVariable::ExternalLinkage;
1197 return NonDiscardableODRLinkage;
1200 llvm_unreachable(
"Invalid TemplateSpecializationKind!");
1210 VTables.GenerateClassData(theClass);
1211 EmittedVTables.insert(theClass);
1217 DI->completeClassData(RD);
1220 CGM.getCXXABI().emitVirtualInheritanceTables(RD);
1222 CGM.getCXXABI().emitVTableDefinitions(*
this, RD);
1236 assert(RD->
isDynamicClass() &&
"Non-dynamic classes have no VTable.");
1240 if (CGM.getTarget().getCXXABI().isMicrosoft())
1262 const CXXMethodDecl *keyFunction = CGM.getContext().getCurrentKeyFunction(RD);
1268 return !keyFunction->
hasBody();
1287void CodeGenModule::EmitDeferredVTables() {
1291 size_t savedSize = DeferredVTables.size();
1294 for (
const CXXRecordDecl *RD : DeferredVTables) {
1297 if (EmittedVTables.count(RD) &&
1302 VTables.GenerateClassData(RD);
1303 else if (shouldOpportunisticallyEmitVTables())
1304 OpportunisticVTables.push_back(RD);
1307 assert(savedSize == DeferredVTables.size() &&
1308 "deferred extra vtables during vtable emission?");
1309 DeferredVTables.clear();
1313 if (RD->
hasAttr<LTOVisibilityPublicAttr>() || RD->
hasAttr<UuidAttr>() ||
1325 if (
auto *ND = dyn_cast<NamespaceDecl>(D))
1327 if (II->isStr(
"std") || II->isStr(
"stdext"))
1349 const CXXRecordDecl *RD, llvm::DenseSet<const CXXRecordDecl *> &Visited) {
1355 if (!Visited.insert(RD).second)
1356 return llvm::GlobalObject::VCallVisibilityTranslationUnit;
1359 llvm::GlobalObject::VCallVisibility TypeVis;
1361 TypeVis = llvm::GlobalObject::VCallVisibilityTranslationUnit;
1363 TypeVis = llvm::GlobalObject::VCallVisibilityLinkageUnit;
1365 TypeVis = llvm::GlobalObject::VCallVisibilityPublic;
1367 for (
const auto &B : RD->
bases())
1368 if (B.getType()->getAsCXXRecordDecl()->isDynamicClass())
1373 for (
const auto &B : RD->
vbases())
1374 if (B.getType()->getAsCXXRecordDecl()->isDynamicClass())
1383 llvm::GlobalVariable *VTable,
1395 struct AddressPoint {
1399 bool operator<(
const AddressPoint &RHS)
const {
1400 int D =
TypeName.compare(RHS.TypeName);
1401 return D < 0 || (D == 0 && Offset < RHS.Offset);
1404 std::vector<AddressPoint> AddressPoints;
1406 AddressPoint N{AP.first.getBase(),
1408 AP.second.AddressPointIndex,
1410 llvm::raw_string_ostream Stream(N.TypeName);
1413 AddressPoints.push_back(std::move(N));
1417 llvm::sort(AddressPoints);
1420 for (
auto AP : AddressPoints) {
1428 for (
unsigned I = 0; I != Comps.size(); ++I) {
1432 Context.getMemberPointerType(Comps[I].getFunctionDecl()->getType(),
1433 std::nullopt, AP.Base));
1434 VTable->addTypeMetadata((ComponentWidth * I).getQuantity(), MD);
1440 llvm::DenseSet<const CXXRecordDecl *> Visited;
1441 llvm::GlobalObject::VCallVisibility TypeVis =
1443 if (TypeVis != llvm::GlobalObject::VCallVisibilityPublic)
1444 VTable->setVCallVisibilityMetadata(TypeVis);
static RValue PerformReturnAdjustment(CodeGenFunction &CGF, QualType ResultType, RValue RV, const ThunkInfo &Thunk)
static void resolveTopLevelMetadata(llvm::Function *Fn, llvm::ValueToValueMapTy &VMap)
This function clones a function's DISubprogram node and enters it into a value map with the intent th...
static void AddRelativeLayoutOffset(const CodeGenModule &CGM, ConstantArrayBuilder &builder, CharUnits offset)
static void AddPointerLayoutOffset(const CodeGenModule &CGM, ConstantArrayBuilder &builder, CharUnits offset)
static bool similar(const ABIArgInfo &infoL, CanQualType typeL, const ABIArgInfo &infoR, CanQualType typeR)
static bool shouldEmitAvailableExternallyVTable(const CIRGenModule &cgm, const CXXRecordDecl *rd)
static bool shouldEmitVTableAtEndOfTranslationUnit(CIRGenModule &cgm, const CXXRecordDecl *rd)
Given that we're currently at the end of the translation unit, and we've emitted a reference to the v...
static void setThunkProperties(CIRGenModule &cgm, const ThunkInfo &thunk, cir::FuncOp thunkFn, bool forVTable, GlobalDecl gd)
static bool shouldEmitVTableThunk(CIRGenModule &cgm, const CXXMethodDecl *md, bool isUnprototyped, bool forVTable)
static Decl::Kind getKind(const Decl *D)
static bool hasAttr(const Decl *D, bool IgnoreImplicitAttr)
CanQualType getCanonicalTagType(const TagDecl *TD) const
Represents a static or instance method of a struct/union/class.
const CXXRecordDecl * getParent() const
Return the parent of this method declaration, which is the class in which this method is defined.
QualType getThisType() const
Return the type of the this pointer.
QualType getFunctionObjectParameterType() const
Represents a C++ struct/union/class.
TemplateSpecializationKind getTemplateSpecializationKind() const
Determine whether this particular class is a specialization or instantiation of a class template or m...
base_class_range vbases()
bool isDynamicClass() const
unsigned getNumVBases() const
Retrieves the number of virtual base classes of this class.
CharUnits - This is an opaque type for sizes expressed in character units.
QuantityType getQuantity() const
getQuantity - Get the raw integer representation of this quantity.
ABIArgInfo - Helper class to encapsulate information about how a specific C type should be passed to ...
@ Indirect
Indirect - Pass the argument indirectly via a hidden pointer with the specified alignment (0 indicate...
bool isSRetAfterThis() const
Like RawAddress, an abstract representation of an aligned address, but the pointer contained in this ...
llvm::Type * getElementType() const
Return the type of the values stored in this address.
static ApplyDebugLocation CreateArtificial(CodeGenFunction &CGF)
Apply TemporaryLocation if it is valid.
static ApplyDebugLocation CreateEmpty(CodeGenFunction &CGF)
Set the IRBuilder to not attach debug locations.
llvm::Value * CreateIsNull(Address Addr, const Twine &Name="")
virtual bool canSpeculativelyEmitVTable(const CXXRecordDecl *RD) const =0
Determine whether it's possible to emit a vtable for RD, even though we do not know that the vtable h...
virtual llvm::Value * performReturnAdjustment(CodeGenFunction &CGF, Address Ret, const CXXRecordDecl *UnadjustedClass, const ReturnAdjustment &RA)=0
virtual void setThunkLinkage(llvm::Function *Thunk, bool ForVTable, GlobalDecl GD, bool ReturnAdjustment)=0
virtual bool exportThunk()=0
MangleContext & getMangleContext()
Gets the mangle context.
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...
CGFunctionInfo - Class to encapsulate the information about a function definition.
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...
GlobalDecl CurGD
CurGD - The GlobalDecl for the current function being compiled.
void EmitMustTailThunk(GlobalDecl GD, llvm::Value *AdjustedThisPtr, llvm::FunctionCallee Callee)
Emit a musttail call for a thunk with a potentially adjusted this pointer.
Address LoadCXXThisAddress()
bool CurFuncIsThunk
In C++, whether we are code generating a thunk.
void EmitDelegateCallArg(CallArgList &args, const VarDecl *param, SourceLocation loc)
EmitDelegateCallArg - We are performing a delegate call; that is, the current function is delegating ...
llvm::BasicBlock * createBasicBlock(const Twine &name="", llvm::Function *parent=nullptr, llvm::BasicBlock *before=nullptr)
createBasicBlock - Create an LLVM basic block.
Address makeNaturalAddressForPointer(llvm::Value *Ptr, QualType T, CharUnits Alignment=CharUnits::Zero(), bool ForPointeeType=false, LValueBaseInfo *BaseInfo=nullptr, TBAAAccessInfo *TBAAInfo=nullptr, KnownNonNull_t IsKnownNonNull=NotKnownNonNull)
Construct an address with the natural alignment of T.
const Decl * CurCodeDecl
CurCodeDecl - This is the inner-most code context, which includes blocks.
void EmitCallAndReturnForThunk(llvm::FunctionCallee Callee, const ThunkInfo *Thunk, bool IsUnprototyped)
void StartFunction(GlobalDecl GD, QualType RetTy, llvm::Function *Fn, const CGFunctionInfo &FnInfo, const FunctionArgList &Args, SourceLocation Loc=SourceLocation(), SourceLocation StartLoc=SourceLocation())
Emit code for the start of a function.
llvm::Function * GenerateVarArgsThunk(llvm::Function *Fn, const CGFunctionInfo &FnInfo, GlobalDecl GD, const ThunkInfo &Thunk)
RValue EmitCall(const CGFunctionInfo &CallInfo, const CGCallee &Callee, ReturnValueSlot ReturnValue, const CallArgList &Args, llvm::CallBase **CallOrInvoke, bool IsMustTail, SourceLocation Loc, bool IsVirtualFunctionPointerThunk=false)
EmitCall - Generate a call of the given function, expecting the given result type,...
const Decl * CurFuncDecl
CurFuncDecl - Holds the Decl for the current outermost non-closure context.
bool AutoreleaseResult
In ARC, whether we should autorelease the return value.
llvm::Type * ConvertTypeForMem(QualType T)
void generateThunk(llvm::Function *Fn, const CGFunctionInfo &FnInfo, GlobalDecl GD, const ThunkInfo &Thunk, bool IsUnprototyped)
Generate a thunk for the given method.
void StartThunk(llvm::Function *Fn, GlobalDecl GD, const CGFunctionInfo &FnInfo, bool IsUnprototyped)
static bool hasAggregateEvaluationKind(QualType T)
void FinishFunction(SourceLocation EndLoc=SourceLocation())
FinishFunction - Complete IR generation of the current function.
llvm::Value * LoadCXXThis()
LoadCXXThis - Load the value of 'this'.
const CGFunctionInfo * CurFnInfo
Address GetAddrOfLocalVar(const VarDecl *VD)
GetAddrOfLocalVar - Return the address of a local variable.
Address ReturnValue
ReturnValue - The temporary alloca to hold the return value.
llvm::LLVMContext & getLLVMContext()
void EmitBlock(llvm::BasicBlock *BB, bool IsFinished=false)
EmitBlock - Emit the given block.
This class organizes the cross-function state that is used while generating LLVM code.
void setGVProperties(llvm::GlobalValue *GV, GlobalDecl GD) const
Set visibility, dllimport/dllexport and dso_local.
void AddVTableTypeMetadata(llvm::GlobalVariable *VTable, CharUnits Offset, const CXXRecordDecl *RD)
Create and attach type metadata for the given vtable.
llvm::GlobalObject::VCallVisibility GetVCallVisibilityLevel(const CXXRecordDecl *RD, llvm::DenseSet< const CXXRecordDecl * > &Visited)
Returns the vcall visibility of the given type.
llvm::Module & getModule() const
CodeGenVTables & getVTables()
CharUnits GetTargetTypeStoreSize(llvm::Type *Ty) const
Return the store size, in character units, of the given LLVM type.
const LangOptions & getLangOpts() const
const TargetInfo & getTarget() const
void EmitVTableTypeMetadata(const CXXRecordDecl *RD, llvm::GlobalVariable *VTable, const VTableLayout &VTLayout)
Emit type metadata for the given vtable using the given layout.
bool HasHiddenLTOVisibility(const CXXRecordDecl *RD)
Returns whether the given record has hidden LTO visibility and therefore may participate in (single-m...
CGCXXABI & getCXXABI() const
CharUnits getClassPointerAlignment(const CXXRecordDecl *CD)
Returns the assumed alignment of an opaque pointer to the given class.
const llvm::Triple & getTriple() const
bool AlwaysHasLTOVisibilityPublic(const CXXRecordDecl *RD)
Returns whether the given record has public LTO visibility (regardless of -lto-whole-program-visibili...
void EmitVTable(CXXRecordDecl *Class)
This is a callback from Sema to tell us that a particular vtable is required to be emitted in this tr...
void setFunctionLinkage(GlobalDecl GD, llvm::Function *F)
ASTContext & getContext() const
llvm::Type * getVTableComponentType() const
bool supportsCOMDAT() const
const CodeGenOptions & getCodeGenOpts() const
llvm::GlobalVariable::LinkageTypes getVTableLinkage(const CXXRecordDecl *RD)
Return the appropriate linkage for the vtable, VTT, and type information of the given class.
llvm::Metadata * CreateMetadataIdentifierForVirtualMemPtrType(QualType T)
Create a metadata identifier that is intended to be used to check virtual calls via a member function...
llvm::Constant * GetAddrOfThunk(StringRef Name, llvm::Type *FnTy, GlobalDecl GD)
Get the address of the thunk for the given global decl.
llvm::GlobalVariable * GetAddrOfVTable(const CXXRecordDecl *RD)
GetAddrOfVTable - Get the address of the VTable for the given record decl.
void createVTableInitializer(ConstantStructBuilder &builder, const VTableLayout &layout, llvm::Constant *rtti, bool vtableHasLocalLinkage)
Add vtable components for the given vtable layout to the given global initializer.
void GenerateClassData(const CXXRecordDecl *RD)
GenerateClassData - Generate all the class data required to be generated upon definition of a KeyFunc...
void GenerateRelativeVTableAlias(llvm::GlobalVariable *VTable, llvm::StringRef AliasNameRef)
Generate a public facing alias for the vtable and make the vtable either hidden or private.
ItaniumVTableContext & getItaniumVTableContext()
CodeGenVTables(CodeGenModule &CGM)
llvm::GlobalVariable * GenerateConstructionVTable(const CXXRecordDecl *RD, const BaseSubobject &Base, bool BaseIsVirtual, llvm::GlobalVariable::LinkageTypes Linkage, VTableAddressPointsMapTy &AddressPoints)
GenerateConstructionVTable - Generate a construction vtable for the given base subobject.
llvm::Type * getVTableType(const VTableLayout &layout)
Returns the type of a vtable with the given layout.
llvm::Type * getVTableComponentType() const
Return the type used as components for a vtable.
bool isVTableExternal(const CXXRecordDecl *RD)
At this point in the translation unit, does it appear that can we rely on the vtable being defined el...
void RemoveHwasanMetadata(llvm::GlobalValue *GV) const
Specify a global should not be instrumented with hwasan.
void EmitThunks(GlobalDecl GD)
EmitThunks - Emit the associated thunks for the given global decl.
void addRelativeOffsetToPosition(llvm::IntegerType *type, llvm::Constant *target, size_t position)
Same as addRelativeOffset(), but instead relative to an element in this aggregate,...
void add(llvm::Constant *value)
Add a new value to this initializer.
void addNullPointer(llvm::PointerType *ptrTy)
Add a null pointer of a specific type.
void addSignedPointer(llvm::Constant *Pointer, const PointerAuthSchema &Schema, GlobalDecl CalleeDecl, QualType CalleeType)
Add a signed pointer using the given pointer authentication schema.
ArrayBuilder beginArray(llvm::Type *eltTy=nullptr)
A helper class of ConstantInitBuilder, used for building constant array initializers.
StructBuilder beginStruct(llvm::StructType *structTy=nullptr)
The standard implementation of ConstantInitBuilder used in Clang.
A helper class of ConstantInitBuilder, used for building constant struct initializers.
FunctionArgList - Type for representing both the decl and type of parameters to a function.
RValue - This trivial value class is used to represent the result of an expression that is evaluated.
static RValue get(llvm::Value *V)
llvm::Value * getScalarVal() const
getScalarVal() - Return the Value* of this scalar value.
static RequiredArgs forPrototypePlus(const FunctionProtoType *prototype, unsigned additional)
Compute the arguments required by the given formal prototype, given that there may be some additional...
ReturnValueSlot - Contains the address where the return value of a function can be stored,...
DeclContext - This is used only as base class of specific decl types that can act as declaration cont...
DeclContext * getParent()
getParent - Returns the containing DeclContext.
DeclContext * getRedeclContext()
getRedeclContext - Retrieve the context in which an entity conflicts with other entities of the same ...
bool isInNamedModule() const
Whether this declaration comes from a named module.
SourceLocation getLocation() const
bool shouldEmitInExternalSource() const
Whether the definition of the declaration should be emitted in external sources.
Represents a function declaration or definition.
param_iterator param_end()
bool isInlined() const
Determine whether this function should be inlined, because it is either marked "inline" or "constexpr...
ArrayRef< ParmVarDecl * > parameters() const
param_iterator param_begin()
TemplateSpecializationKind getTemplateSpecializationKind() const
Determine what kind of template instantiation this function represents.
bool hasBody(const FunctionDecl *&Definition) const
Returns true if the function has a body.
bool isDefined(const FunctionDecl *&Definition, bool CheckForPendingFriendDefinition=false) const
Returns true if the function has a definition that does not need to be instantiated.
Represents a prototype with parameter type info, e.g.
QualType getReturnType() const
GlobalDecl - represents a global declaration.
GlobalDecl getWithDecl(const Decl *D)
CXXDtorType getDtorType() const
const Decl * getDecl() const
One of these records is kept for each identifier that is lexed.
GlobalDecl findOriginalMethod(GlobalDecl GD)
Return the method that added the v-table slot that will be used to call the given method.
Visibility getVisibility() const
Linkage getLinkage() const
virtual void mangleCXXDtorThunk(const CXXDestructorDecl *DD, CXXDtorType Type, const ThunkInfo &Thunk, bool ElideOverrideInfo, raw_ostream &)=0
virtual void mangleThunk(const CXXMethodDecl *MD, const ThunkInfo &Thunk, bool ElideOverrideInfo, raw_ostream &)=0
virtual void mangleCanonicalTypeName(QualType T, raw_ostream &, bool NormalizeIntegers=false)=0
Generates a unique string for an externally visible type for use with TBAA or type uniquing.
LinkageInfo getLinkageAndVisibility() const
Determines the linkage and visibility of this entity.
bool isExternallyVisible() const
Represents a parameter to a function.
A (possibly-)qualified type.
bool isVolatileQualified() const
Determine whether this type is volatile-qualified.
Encodes a location in the source.
bool isMicrosoft() const
Is this ABI an MSVC-compatible ABI?
TargetCXXABI getCXXABI() const
Get the C++ ABI currently in use.
CXXRecordDecl * getAsCXXRecordDecl() const
Retrieves the CXXRecordDecl that this type refers to, either because the type is a RecordType or beca...
const T * castAs() const
Member-template castAs<specific type>.
bool isReferenceType() const
const CXXRecordDecl * getPointeeCXXRecordDecl() const
If this is a pointer or reference to a RecordType, return the CXXRecordDecl that the type refers to.
QualType getPointeeType() const
If this is a pointer, ObjC object pointer, or block pointer, this returns the respective pointee.
@ CK_DeletingDtorPointer
A pointer to the deleting destructor.
@ CK_UnusedFunctionPointer
An entry that is never used.
@ CK_CompleteDtorPointer
A pointer to the complete destructor.
SmallVector< ThunkInfo, 1 > ThunkInfoVectorTy
const AddressPointsIndexMapTy & getAddressPointIndices() const
size_t getVTableOffset(size_t i) const
ArrayRef< VTableComponent > vtable_components() const
size_t getNumVTables() const
ArrayRef< VTableThunkTy > vtable_thunks() const
const AddressPointsMapTy & getAddressPoints() const
size_t getVTableSize(size_t i) const
The JSON file list parser is used to communicate input to InstallAPI.
CanQual< Type > CanQualType
Represents a canonical, potentially-qualified type.
bool isa(CodeGen::Address addr)
nullptr
This class represents a compute construct, representing a 'Kind' of ‘parallel’, 'serial',...
bool operator<(DeclarationName LHS, DeclarationName RHS)
Ordering on two declaration names.
Linkage
Describes the different kinds of linkage (C++ [basic.link], C99 6.2.2) that an entity may have.
@ Dtor_Base
Base object dtor.
TemplateSpecializationKind
Describes the kind of template specialization that a particular template specialization declaration r...
@ TSK_ExplicitInstantiationDefinition
This template specialization was instantiated from a template due to an explicit instantiation defini...
@ TSK_ExplicitInstantiationDeclaration
This template specialization was instantiated from a template due to an explicit instantiation declar...
@ TSK_ExplicitSpecialization
This template specialization was declared or defined by an explicit specialization (C++ [temp....
@ TSK_ImplicitInstantiation
This template specialization was implicitly instantiated from a template.
@ TSK_Undeclared
This template specialization was formed from a template-id but has not yet been declared,...
CallingConv
CallingConv - Specifies the calling convention that a function uses.
U cast(CodeGen::Address addr)
bool isExternallyVisible(Linkage L)
@ HiddenVisibility
Objects with "hidden" visibility are not seen by the dynamic linker.
llvm::PointerType * GlobalsInt8PtrTy
llvm::IntegerType * Int32Ty
llvm::IntegerType * PtrDiffTy
The this pointer adjustment as well as an optional return adjustment for a thunk.
ReturnAdjustment Return
The return adjustment.