30#include "llvm/ADT/StringExtras.h"
31#include "llvm/ADT/StringSet.h"
32#include "llvm/IR/Intrinsics.h"
35using namespace CodeGen;
40struct VBTableGlobals {
45class MicrosoftCXXABI :
public CGCXXABI {
48 :
CGCXXABI(CGM), BaseClassDescriptorType(nullptr),
49 ClassHierarchyDescriptorType(nullptr),
50 CompleteObjectLocatorType(nullptr), CatchableTypeType(nullptr),
51 ThrowInfoType(nullptr) {
54 "visibility export mapping option unimplemented in this ABI");
69 if (isa<CXXDestructorDecl>(GD.
getDecl())) {
78 case Dtor_Comdat: llvm_unreachable(
"emitting dtor comdat as function?");
80 llvm_unreachable(
"bad dtor kind");
89 assert(Args.size() >= 2 &&
90 "expected the arglist to have at least two args!");
100 std::vector<CharUnits> VBPtrOffsets;
104 const VBTableGlobals &VBGlobals = enumerateVBTables(RD);
105 for (
const std::unique_ptr<VPtrInfo> &VBT : *VBGlobals.VBTables) {
110 if (VBT->getVBaseWithVPtr())
112 VBPtrOffsets.push_back(Offs);
114 llvm::array_pod_sort(VBPtrOffsets.begin(), VBPtrOffsets.end());
130 llvm::GlobalVariable *getMSCompleteObjectLocator(
const CXXRecordDecl *RD,
151 llvm::Type *StdTypeInfoPtrTy)
override;
163 llvm::BasicBlock *CastSuccess,
164 llvm::BasicBlock *CastFail)
override {
165 llvm_unreachable(
"unsupported");
171 llvm::BasicBlock *CastEnd)
override;
230 AddedStructorArgCounts
244 llvm::GlobalValue::LinkageTypes
251 auto *MD = cast<CXXMethodDecl>(GD.
getDecl());
253 if (MD->isVirtual()) {
255 if (
const auto *DD = dyn_cast<CXXDestructorDecl>(MD)) {
259 return MD->getParent();
277 return MD->getParent();
283 bool VirtualCall)
override;
308 llvm::GlobalVariable *VTable);
314 CodeGenFunction::VPtr Vptr)
override;
319 return !VTableClass->
hasAttr<MSNoVTableAttr>();
340 DeleteOrMemberCallExpr
E)
override;
345 "Only deleting destructor thunks are available in this ABI");
352 llvm::GlobalVariable *
354 llvm::GlobalVariable::LinkageTypes
Linkage);
356 llvm::GlobalVariable *
360 llvm::raw_svector_ostream Out(OutName);
362 StringRef MangledName = OutName.str();
364 if (
auto *VDispMap = CGM.
getModule().getNamedGlobal(MangledName))
370 llvm::UndefValue::get(CGM.
IntTy));
371 Map[0] = llvm::ConstantInt::get(CGM.
IntTy, 0);
372 bool AnyDifferent =
false;
373 for (
const auto &I : SrcRD->
vbases()) {
374 const CXXRecordDecl *VBase = I.getType()->getAsCXXRecordDecl();
380 Map[SrcVBIndex] = llvm::ConstantInt::get(CGM.
IntTy, DstVBIndex * 4);
381 AnyDifferent |= SrcVBIndex != DstVBIndex;
387 llvm::ArrayType *VDispMapTy = llvm::ArrayType::get(CGM.
IntTy, Map.size());
388 llvm::Constant *
Init = llvm::ConstantArray::get(VDispMapTy, Map);
389 llvm::GlobalValue::LinkageTypes
Linkage =
391 ? llvm::GlobalValue::LinkOnceODRLinkage
392 : llvm::GlobalValue::InternalLinkage;
393 auto *VDispMap =
new llvm::GlobalVariable(
400 llvm::GlobalVariable *GV)
const;
408 Thunk->setLinkage(llvm::GlobalValue::InternalLinkage);
410 Thunk->setLinkage(llvm::GlobalValue::WeakODRLinkage);
412 Thunk->setLinkage(llvm::GlobalValue::LinkOnceODRLinkage);
432 LangOptions::MSVC2019_5) &&
439 llvm::GlobalVariable *DeclPtr,
440 bool PerformInit)
override;
442 llvm::FunctionCallee Dtor,
443 llvm::Constant *Addr)
override;
474 llvm::Value *NumElements,
481 friend struct MSRTTIBuilder;
483 bool isImageRelative()
const {
488 llvm::StructType *getTypeDescriptorType(StringRef TypeInfoString) {
490 TDTypeName += llvm::utostr(TypeInfoString.size());
491 llvm::StructType *&TypeDescriptorType =
492 TypeDescriptorTypeMap[TypeInfoString.size()];
493 if (TypeDescriptorType)
494 return TypeDescriptorType;
495 llvm::Type *FieldTypes[] = {
498 llvm::ArrayType::get(CGM.
Int8Ty, TypeInfoString.size() + 1)};
500 llvm::StructType::create(CGM.
getLLVMContext(), FieldTypes, TDTypeName);
501 return TypeDescriptorType;
504 llvm::Type *getImageRelativeType(llvm::Type *PtrType) {
505 if (!isImageRelative())
510 llvm::StructType *getBaseClassDescriptorType() {
511 if (BaseClassDescriptorType)
512 return BaseClassDescriptorType;
513 llvm::Type *FieldTypes[] = {
520 getImageRelativeType(getClassHierarchyDescriptorType()->getPointerTo()),
522 BaseClassDescriptorType = llvm::StructType::create(
524 return BaseClassDescriptorType;
527 llvm::StructType *getClassHierarchyDescriptorType() {
528 if (ClassHierarchyDescriptorType)
529 return ClassHierarchyDescriptorType;
531 ClassHierarchyDescriptorType = llvm::StructType::create(
533 llvm::Type *FieldTypes[] = {
537 getImageRelativeType(
538 getBaseClassDescriptorType()->getPointerTo()->getPointerTo()),
540 ClassHierarchyDescriptorType->setBody(FieldTypes);
541 return ClassHierarchyDescriptorType;
544 llvm::StructType *getCompleteObjectLocatorType() {
545 if (CompleteObjectLocatorType)
546 return CompleteObjectLocatorType;
547 CompleteObjectLocatorType = llvm::StructType::create(
549 llvm::Type *FieldTypes[] = {
554 getImageRelativeType(getClassHierarchyDescriptorType()->getPointerTo()),
555 getImageRelativeType(CompleteObjectLocatorType),
558 if (!isImageRelative())
559 FieldTypesRef = FieldTypesRef.drop_back();
560 CompleteObjectLocatorType->setBody(FieldTypesRef);
561 return CompleteObjectLocatorType;
564 llvm::GlobalVariable *getImageBase() {
565 StringRef Name =
"__ImageBase";
566 if (llvm::GlobalVariable *GV = CGM.
getModule().getNamedGlobal(Name))
571 llvm::GlobalValue::ExternalLinkage,
577 llvm::Constant *getImageRelativeConstant(llvm::Constant *PtrVal) {
578 if (!isImageRelative())
581 if (PtrVal->isNullValue())
582 return llvm::Constant::getNullValue(CGM.
IntTy);
584 llvm::Constant *ImageBaseAsInt =
585 llvm::ConstantExpr::getPtrToInt(getImageBase(), CGM.
IntPtrTy);
586 llvm::Constant *PtrValAsInt =
587 llvm::ConstantExpr::getPtrToInt(PtrVal, CGM.
IntPtrTy);
588 llvm::Constant *Diff =
589 llvm::ConstantExpr::getSub(PtrValAsInt, ImageBaseAsInt,
591 return llvm::ConstantExpr::getTrunc(Diff, CGM.
IntTy);
599 llvm::Constant *getZeroInt() {
600 return llvm::ConstantInt::get(CGM.
IntTy, 0);
603 llvm::Constant *getAllOnesInt() {
604 return llvm::Constant::getAllOnesValue(CGM.
IntTy);
618 llvm::Value *VBPtrOffset,
619 llvm::Value *VBTableOffset,
620 llvm::Value **VBPtr =
nullptr);
625 int32_t VBTableOffset,
626 llvm::Value **VBPtr =
nullptr) {
627 assert(VBTableOffset % 4 == 0 &&
"should be byte offset into table of i32s");
628 llvm::Value *VBPOffset = llvm::ConstantInt::get(CGM.
IntTy, VBPtrOffset),
629 *VBTOffset = llvm::ConstantInt::get(CGM.
IntTy, VBTableOffset);
630 return GetVBaseOffsetFromVBPtr(CGF,
Base, VBPOffset, VBTOffset, VBPtr);
633 std::tuple<Address, llvm::Value *, const CXXRecordDecl *>
641 llvm::Value *VirtualBaseAdjustmentOffset,
642 llvm::Value *VBPtrOffset );
646 llvm::Constant *EmitFullMemberPointer(llvm::Constant *FirstField,
647 bool IsMemberFunction,
650 unsigned VBTableIndex);
659 const VBTableGlobals &enumerateVBTables(
const CXXRecordDecl *RD);
662 llvm::Function *EmitVirtualMemPtrThunk(
const CXXMethodDecl *MD,
675 return RD->
hasAttr<MSInheritanceAttr>();
689 bool Inequality)
override;
700 llvm::Value *EmitNonNullMemberPointerConversion(
708 llvm::Value *Src)
override;
711 llvm::Constant *Src)
override;
720 Address This, llvm::Value *&ThisPtrForCall,
726 llvm::StructType *getCatchableTypeType() {
727 if (CatchableTypeType)
728 return CatchableTypeType;
729 llvm::Type *FieldTypes[] = {
738 CatchableTypeType = llvm::StructType::create(
740 return CatchableTypeType;
743 llvm::StructType *getCatchableTypeArrayType(uint32_t NumEntries) {
744 llvm::StructType *&CatchableTypeArrayType =
745 CatchableTypeArrayTypeMap[NumEntries];
746 if (CatchableTypeArrayType)
747 return CatchableTypeArrayType;
750 CTATypeName += llvm::utostr(NumEntries);
752 getImageRelativeType(getCatchableTypeType()->getPointerTo());
753 llvm::Type *FieldTypes[] = {
755 llvm::ArrayType::get(CTType, NumEntries)
757 CatchableTypeArrayType =
758 llvm::StructType::create(CGM.
getLLVMContext(), FieldTypes, CTATypeName);
759 return CatchableTypeArrayType;
762 llvm::StructType *getThrowInfoType() {
764 return ThrowInfoType;
765 llvm::Type *FieldTypes[] = {
771 ThrowInfoType = llvm::StructType::create(CGM.
getLLVMContext(), FieldTypes,
773 return ThrowInfoType;
779 llvm::Type *Args[] = {CGM.
Int8PtrTy, getThrowInfoType()->getPointerTo()};
780 llvm::FunctionType *FTy =
781 llvm::FunctionType::get(CGM.
VoidTy, Args,
false);
782 llvm::FunctionCallee Throw =
786 if (
auto *Fn = dyn_cast<llvm::Function>(Throw.getCallee()))
787 Fn->setCallingConv(llvm::CallingConv::X86_StdCall);
795 llvm::Constant *getCatchableType(
QualType T,
796 uint32_t NVOffset = 0,
797 int32_t VBPtrOffset = -1,
798 uint32_t VBIndex = 0);
800 llvm::GlobalVariable *getCatchableTypeArray(
QualType T);
804 std::pair<llvm::Value *, const CXXRecordDecl *>
812 typedef std::pair<const CXXRecordDecl *, CharUnits> VFTableIdTy;
813 typedef llvm::DenseMap<VFTableIdTy, llvm::GlobalVariable *> VTablesMapTy;
814 typedef llvm::DenseMap<VFTableIdTy, llvm::GlobalValue *> VFTablesMapTy;
816 VFTablesMapTy VFTablesMap;
817 VTablesMapTy VTablesMap;
824 llvm::DenseMap<const CXXRecordDecl *, VBTableGlobals> VBTablesMap;
829 GuardInfo() =
default;
830 llvm::GlobalVariable *Guard =
nullptr;
831 unsigned BitIndex = 0;
836 llvm::DenseMap<const DeclContext *, GuardInfo> GuardVariableMap;
837 llvm::DenseMap<const DeclContext *, GuardInfo> ThreadLocalGuardVariableMap;
838 llvm::DenseMap<const DeclContext *, unsigned> ThreadSafeGuardNumMap;
840 llvm::DenseMap<size_t, llvm::StructType *> TypeDescriptorTypeMap;
841 llvm::StructType *BaseClassDescriptorType;
842 llvm::StructType *ClassHierarchyDescriptorType;
843 llvm::StructType *CompleteObjectLocatorType;
845 llvm::DenseMap<QualType, llvm::GlobalVariable *> CatchableTypeArrays;
847 llvm::StructType *CatchableTypeType;
848 llvm::DenseMap<uint32_t, llvm::StructType *> CatchableTypeArrayTypeMap;
849 llvm::StructType *ThrowInfoType;
855MicrosoftCXXABI::getRecordArgABI(
const CXXRecordDecl *RD)
const {
867 case llvm::Triple::thumb:
873 case llvm::Triple::x86: {
884 return RAA_DirectInMemory;
887 case llvm::Triple::x86_64:
888 case llvm::Triple::aarch64:
892 llvm_unreachable(
"invalid enum");
904 llvm::Value *MDThis = EmitVirtualDestructorCall(CGF, Dtor, DtorType, Ptr, DE);
909void MicrosoftCXXABI::emitRethrow(
CodeGenFunction &CGF,
bool isNoReturn) {
910 llvm::Value *Args[] = {
911 llvm::ConstantPointerNull::get(CGM.
Int8PtrTy),
912 llvm::ConstantPointerNull::get(getThrowInfoType()->getPointerTo())};
924 VarDecl *CatchParam = S->getExceptionDecl();
925 llvm::BasicBlock *CatchPadBB = CGF.
Builder.GetInsertBlock();
926 llvm::CatchPadInst *CPI =
927 cast<llvm::CatchPadInst>(CatchPadBB->getFirstNonPHI());
938 CPI->setArgOperand(2,
var.getObjectAddress(CGF).emitRawPointer(CGF));
946std::tuple<Address, llvm::Value *, const CXXRecordDecl *>
957 return std::make_tuple(
Value, llvm::ConstantInt::get(CGF.
Int32Ty, 0),
966 PolymorphicBase = BaseDecl;
970 assert(PolymorphicBase &&
"polymorphic class has no apparent vfptr?");
972 llvm::Value *Offset =
973 GetVirtualBaseClassOffset(CGF,
Value, SrcDecl, PolymorphicBase);
975 Value.getElementType(),
Value.emitRawPointer(CGF), Offset);
978 return std::make_tuple(
Address(Ptr, CGF.
Int8Ty, VBaseAlign), Offset,
982bool MicrosoftCXXABI::shouldTypeidBeNullChecked(
QualType SrcRecordTy) {
984 return !getContext().getASTRecordLayout(SrcDecl).hasExtendableVFPtr();
988 llvm::Value *Argument) {
989 llvm::Type *ArgTypes[] = {CGF.
Int8PtrTy};
990 llvm::FunctionType *FTy =
991 llvm::FunctionType::get(CGF.
Int8PtrTy, ArgTypes,
false);
992 llvm::Value *Args[] = {Argument};
998 llvm::CallBase *
Call =
1000 Call->setDoesNotReturn();
1001 CGF.
Builder.CreateUnreachable();
1007 llvm::Type *StdTypeInfoPtrTy) {
1008 std::tie(ThisPtr, std::ignore, std::ignore) =
1009 performBaseAdjustment(CGF, ThisPtr, SrcRecordTy);
1011 return CGF.
Builder.CreateBitCast(Typeid, StdTypeInfoPtrTy);
1014bool MicrosoftCXXABI::shouldDynamicCastCallBeNullChecked(
bool SrcIsPtr,
1018 !getContext().getASTRecordLayout(SrcDecl).hasExtendableVFPtr();
1021llvm::Value *MicrosoftCXXABI::emitDynamicCastCall(
1023 QualType DestRecordTy, llvm::BasicBlock *CastEnd) {
1024 llvm::Value *SrcRTTI =
1026 llvm::Value *DestRTTI =
1029 llvm::Value *Offset;
1030 std::tie(This, Offset, std::ignore) =
1031 performBaseAdjustment(CGF, This, SrcRecordTy);
1032 llvm::Value *ThisPtr =
This.emitRawPointer(CGF);
1044 llvm::FunctionType::get(CGF.
Int8PtrTy, ArgTypes,
false),
1046 llvm::Value *Args[] = {
1047 ThisPtr, Offset, SrcRTTI, DestRTTI,
1052llvm::Value *MicrosoftCXXABI::emitDynamicCastToVoid(
CodeGenFunction &CGF,
1055 std::tie(
Value, std::ignore, std::ignore) =
1056 performBaseAdjustment(CGF,
Value, SrcRecordTy);
1060 llvm::Type *ArgTypes[] = {CGF.
Int8PtrTy};
1062 llvm::FunctionType::get(CGF.
Int8PtrTy, ArgTypes,
false),
1064 llvm::Value *Args[] = {
Value.emitRawPointer(CGF)};
1072llvm::Value *MicrosoftCXXABI::GetVirtualBaseClassOffset(
1078 llvm::Value *VBPtrOffset = llvm::ConstantInt::get(CGM.
PtrDiffTy, VBPtrChars);
1083 llvm::Value *VBTableOffset =
1086 llvm::Value *VBPtrToNewBase =
1087 GetVBaseOffsetFromVBPtr(CGF, This, VBPtrOffset, VBTableOffset);
1090 return CGF.
Builder.CreateNSWAdd(VBPtrOffset, VBPtrToNewBase);
1093bool MicrosoftCXXABI::HasThisReturn(
GlobalDecl GD)
const {
1094 return isa<CXXConstructorDecl>(GD.
getDecl());
1098 return isa<CXXDestructorDecl>(GD.
getDecl()) &&
1102bool MicrosoftCXXABI::hasMostDerivedReturn(
GlobalDecl GD)
const {
1112 uint64_t NumElts = 0;
1115 isa<VectorType>(
Base)) {
1152 if (
auto *Ctor = dyn_cast<CXXConstructorDecl>(
D)) {
1153 if (Ctor->isUserProvided())
1155 }
else if (
auto *Template = dyn_cast<FunctionTemplateDecl>(
D)) {
1156 if (isa<CXXConstructorDecl>(Template->getTemplatedDecl()))
1158 }
else if (
auto *MethodDecl = dyn_cast<CXXMethodDecl>(
D)) {
1159 if (MethodDecl->isCopyAssignmentOperator() && MethodDecl->isDeleted())
1168bool MicrosoftCXXABI::classifyReturnType(
CGFunctionInfo &FI)
const {
1179 if (isIndirectReturn) {
1200 llvm::Value *IsMostDerivedClass = getStructorImplicitParamValue(CGF);
1201 assert(IsMostDerivedClass &&
1202 "ctor for a class with virtual bases must have an implicit parameter");
1203 llvm::Value *IsCompleteObject =
1204 CGF.
Builder.CreateIsNotNull(IsMostDerivedClass,
"is_complete_object");
1206 llvm::BasicBlock *CallVbaseCtorsBB = CGF.
createBasicBlock(
"ctor.init_vbases");
1207 llvm::BasicBlock *SkipVbaseCtorsBB = CGF.
createBasicBlock(
"ctor.skip_vbases");
1208 CGF.
Builder.CreateCondBr(IsCompleteObject,
1209 CallVbaseCtorsBB, SkipVbaseCtorsBB);
1214 EmitVBPtrStores(CGF, RD);
1218 return SkipVbaseCtorsBB;
1223 llvm::Value *IsMostDerivedClass = getStructorImplicitParamValue(CGF);
1224 assert(IsMostDerivedClass &&
1225 "ctor for a class with virtual bases must have an implicit parameter");
1226 llvm::Value *IsCompleteObject =
1227 CGF.
Builder.CreateIsNotNull(IsMostDerivedClass,
"is_complete_object");
1229 llvm::BasicBlock *CallVbaseDtorsBB = CGF.
createBasicBlock(
"Dtor.dtor_vbases");
1230 llvm::BasicBlock *SkipVbaseDtorsBB = CGF.
createBasicBlock(
"Dtor.skip_vbases");
1231 CGF.
Builder.CreateCondBr(IsCompleteObject,
1232 CallVbaseDtorsBB, SkipVbaseDtorsBB);
1237 return SkipVbaseDtorsBB;
1240void MicrosoftCXXABI::initializeHiddenVirtualInheritanceMembers(
1260 llvm::Value *Int8This =
nullptr;
1263 const CXXRecordDecl *VBase = S.getType()->getAsCXXRecordDecl();
1264 auto I = VBaseMap.find(VBase);
1265 assert(I != VBaseMap.end());
1266 if (!I->second.hasVtorDisp())
1269 llvm::Value *VBaseOffset =
1270 GetVirtualBaseClassOffset(CGF, getThisAddress(CGF), RD, VBase);
1271 uint64_t ConstantVBaseOffset = I->second.VBaseOffset.getQuantity();
1274 llvm::Value *VtorDispValue = Builder.CreateSub(
1275 VBaseOffset, llvm::ConstantInt::get(CGM.
PtrDiffTy, ConstantVBaseOffset),
1277 VtorDispValue = Builder.CreateTruncOrBitCast(VtorDispValue, CGF.
Int32Ty);
1280 Int8This = getThisValue(CGF);
1282 llvm::Value *VtorDispPtr =
1283 Builder.CreateInBoundsGEP(CGF.
Int8Ty, Int8This, VBaseOffset);
1285 VtorDispPtr = Builder.CreateConstGEP1_32(CGF.
Int8Ty, VtorDispPtr, -4);
1287 Builder.CreateAlignedStore(VtorDispValue, VtorDispPtr,
1298 return ExpectedCallingConv == ActualCallingConv;
1309 if (
D->
hasAttr<DLLExportAttr>() &&
D->isDefaultConstructor() &&
1313 Fn->setLinkage(llvm::GlobalValue::WeakODRLinkage);
1326 const VBTableGlobals &VBGlobals = enumerateVBTables(RD);
1327 for (
unsigned I = 0,
E = VBGlobals.VBTables->size(); I !=
E; ++I) {
1328 const std::unique_ptr<VPtrInfo> &VBT = (*VBGlobals.VBTables)[I];
1329 llvm::GlobalVariable *GV = VBGlobals.Globals[I];
1334 if (VBT->getVBaseWithVPtr())
1337 llvm::Value *GVPtr =
1345MicrosoftCXXABI::buildStructorSignature(
GlobalDecl GD,
1347 AddedStructorArgCounts Added;
1349 if (isa<CXXDestructorDecl>(GD.
getDecl()) &&
1352 ArgTys.push_back(getContext().IntTy);
1355 auto *CD = dyn_cast<CXXConstructorDecl>(GD.
getDecl());
1364 if (
Class->getNumVBases()) {
1366 ArgTys.insert(ArgTys.begin() + 1, getContext().IntTy);
1369 ArgTys.push_back(getContext().IntTy);
1377void MicrosoftCXXABI::setCXXDestructorDLLStorage(llvm::GlobalValue *GV,
1383 GV->setDLLStorageClass(llvm::GlobalValue::DefaultStorageClass);
1390llvm::GlobalValue::LinkageTypes MicrosoftCXXABI::getCXXDestructorLinkage(
1395 return llvm::GlobalValue::InternalLinkage;
1406 if (Dtor->
hasAttr<DLLExportAttr>())
1407 return llvm::GlobalValue::WeakODRLinkage;
1408 if (Dtor->
hasAttr<DLLImportAttr>())
1409 return llvm::GlobalValue::AvailableExternallyLinkage;
1410 return llvm::GlobalValue::LinkOnceODRLinkage;
1415 return llvm::GlobalValue::LinkOnceODRLinkage;
1417 llvm_unreachable(
"MS C++ ABI does not support comdat dtors");
1419 llvm_unreachable(
"invalid dtor type");
1431 if (
D->getParent()->getNumVBases() > 0 &&
D->
hasAttr<DLLExportAttr>())
1436MicrosoftCXXABI::getVirtualFunctionPrologueThisAdjustment(
GlobalDecl GD) {
1458 if (isa<CXXDestructorDecl>(MD))
1463 getContext().getASTRecordLayout(MD->
getParent());
1470Address MicrosoftCXXABI::adjustThisArgumentForVirtualFunctionCall(
1476 CharUnits Adjustment = getVirtualFunctionPrologueThisAdjustment(GD);
1511 Result = Result.withElementType(CGF.
Int8Ty);
1515 llvm::Value *VBaseOffset =
1516 GetVirtualBaseClassOffset(CGF, Result, Derived, VBase);
1518 Result.getElementType(), Result.emitRawPointer(CGF), VBaseOffset);
1523 if (!StaticOffset.
isZero()) {
1525 Result = Result.withElementType(CGF.
Int8Ty);
1544 assert(isa<CXXConstructorDecl>(MD) || isa<CXXDestructorDecl>(MD));
1549 ImplicitParamKind::Other);
1554 Params.insert(Params.begin() + 1, IsMostDerived);
1556 Params.push_back(IsMostDerived);
1557 getStructorImplicitParamDecl(CGF) = IsMostDerived;
1562 ImplicitParamKind::Other);
1563 Params.push_back(ShouldDelete);
1564 getStructorImplicitParamDecl(CGF) = ShouldDelete;
1568void MicrosoftCXXABI::EmitInstanceFunctionProlog(
CodeGenFunction &CGF) {
1585 llvm::Value *
This = loadIncomingCXXThis(CGF);
1588 CharUnits Adjustment = getVirtualFunctionPrologueThisAdjustment(CGF.
CurGD);
1589 if (!Adjustment.
isZero()) {
1595 setCXXABIThisValue(CGF, This);
1605 if (HasThisReturn(CGF.
CurGD) || hasMostDerivedReturn(CGF.
CurGD))
1609 assert(getStructorImplicitParamDecl(CGF) &&
1610 "no implicit parameter for a constructor with virtual bases?");
1611 getStructorImplicitParamValue(CGF)
1618 assert(getStructorImplicitParamDecl(CGF) &&
1619 "no implicit parameter for a deleting destructor?");
1620 getStructorImplicitParamValue(CGF)
1623 "should_call_delete");
1633 if (!
D->getParent()->getNumVBases())
1634 return AddedStructorArgs{};
1638 llvm::Value *MostDerivedArg;
1640 MostDerivedArg = getStructorImplicitParamValue(CGF);
1645 return AddedStructorArgs::prefix({{MostDerivedArg, getContext().IntTy}});
1647 return AddedStructorArgs::suffix({{MostDerivedArg, getContext().IntTy}});
1650llvm::Value *MicrosoftCXXABI::getCXXDestructorImplicitParam(
1671 assert(
Type != CXXDtorType::Dtor_Deleting &&
1672 "The deleting destructor should only be called via a virtual call");
1677 llvm::BasicBlock *BaseDtorEndBB =
nullptr;
1678 if (ForVirtualBase && isa<CXXConstructorDecl>(CGF.
CurCodeDecl)) {
1679 BaseDtorEndBB = EmitDtorCompleteObjectHandler(CGF);
1689 if (BaseDtorEndBB) {
1691 CGF.
Builder.CreateBr(BaseDtorEndBB);
1696void MicrosoftCXXABI::emitVTableTypeMetadata(
const VPtrInfo &Info,
1698 llvm::GlobalVariable *VTable) {
1710 llvm::GlobalObject::VCallVisibility TypeVis =
1712 if (TypeVis != llvm::GlobalObject::VCallVisibilityPublic)
1713 VTable->setVCallVisibilityMetadata(TypeVis);
1720 getContext().getLangOpts().RTTIData
1721 ? getContext().toCharUnitsFromBits(
1722 getContext().getTargetInfo().getPointerWidth(LangAS::Default))
1741 getContext().getASTRecordLayout(DerivedRD);
1747 Offset = VBI->second.VBaseOffset;
1748 if (!Offset.isZero())
1763 for (
const std::unique_ptr<VPtrInfo>& Info : VFPtrs) {
1764 llvm::GlobalVariable *VTable = getAddrOfVTable(RD, Info->
FullOffsetInMDC);
1765 if (VTable->hasInitializer())
1771 llvm::Constant *RTTI =
nullptr;
1774 RTTI = getMSCompleteObjectLocator(RD, *Info);
1777 auto components = builder.beginStruct();
1779 VTable->hasLocalLinkage());
1780 components.finishAndSetAsInitializer(VTable);
1782 emitVTableTypeMetadata(*Info, RD, VTable);
1786bool MicrosoftCXXABI::isVirtualOffsetNeededForVTableField(
1788 return Vptr.NearestVBase !=
nullptr;
1791llvm::Value *MicrosoftCXXABI::getVTableAddressPointInStructor(
1794 llvm::Constant *VTableAddressPoint = getVTableAddressPoint(
Base, VTableClass);
1795 if (!VTableAddressPoint) {
1796 assert(
Base.getBase()->getNumVBases() &&
1797 !getContext().getASTRecordLayout(
Base.getBase()).hasOwnVFPtr());
1799 return VTableAddressPoint;
1805 llvm::raw_svector_ostream Out(Name);
1812 (void)getAddrOfVTable(VTableClass,
Base.getBaseOffset());
1813 VFTableIdTy
ID(VTableClass,
Base.getBaseOffset());
1814 return VFTablesMap[
ID];
1817llvm::GlobalVariable *MicrosoftCXXABI::getAddrOfVTable(
const CXXRecordDecl *RD,
1823 VFTableIdTy
ID(RD, VPtrOffset);
1824 VTablesMapTy::iterator I;
1826 std::tie(I, Inserted) = VTablesMap.insert(std::make_pair(ID,
nullptr));
1830 llvm::GlobalVariable *&VTable = I->second;
1835 if (DeferredVFTables.insert(RD).second) {
1843 llvm::StringSet<> ObservedMangledNames;
1844 for (
size_t J = 0, F = VFPtrs.size(); J != F; ++J) {
1847 if (!ObservedMangledNames.insert(Name.str()).second)
1848 llvm_unreachable(
"Already saw this mangling before?");
1853 const std::unique_ptr<VPtrInfo> *VFPtrI =
1854 llvm::find_if(VFPtrs, [&](
const std::unique_ptr<VPtrInfo> &VPI) {
1855 return VPI->FullOffsetInMDC == VPtrOffset;
1857 if (VFPtrI == VFPtrs.end()) {
1858 VFTablesMap[
ID] =
nullptr;
1861 const std::unique_ptr<VPtrInfo> &VFPtr = *VFPtrI;
1873 llvm::GlobalValue::LinkageTypes VFTableLinkage =
1874 RD->
hasAttr<DLLImportAttr>() ? llvm::GlobalValue::LinkOnceODRLinkage
1876 bool VFTableComesFromAnotherTU =
1877 llvm::GlobalValue::isAvailableExternallyLinkage(VFTableLinkage) ||
1878 llvm::GlobalValue::isExternalLinkage(VFTableLinkage);
1879 bool VTableAliasIsRequred =
1880 !VFTableComesFromAnotherTU && getContext().getLangOpts().RTTIData;
1882 if (llvm::GlobalValue *VFTable =
1883 CGM.
getModule().getNamedGlobal(VFTableName)) {
1884 VFTablesMap[
ID] = VFTable;
1885 VTable = VTableAliasIsRequred
1886 ? cast<llvm::GlobalVariable>(
1887 cast<llvm::GlobalAlias>(VFTable)->getAliaseeObject())
1888 :
cast<
llvm::GlobalVariable>(VFTable);
1894 llvm::GlobalValue::LinkageTypes VTableLinkage =
1895 VTableAliasIsRequred ? llvm::GlobalValue::PrivateLinkage : VFTableLinkage;
1897 StringRef VTableName = VTableAliasIsRequred ? StringRef() : VFTableName.str();
1903 llvm::GlobalValue *VFTable;
1904 VTable =
new llvm::GlobalVariable(CGM.
getModule(), VTableType,
1905 true, VTableLinkage,
1906 nullptr, VTableName);
1907 VTable->setUnnamedAddr(llvm::GlobalValue::UnnamedAddr::Global);
1909 llvm::Comdat *
C =
nullptr;
1910 if (!VFTableComesFromAnotherTU &&
1911 llvm::GlobalValue::isWeakForLinker(VFTableLinkage))
1912 C = CGM.
getModule().getOrInsertComdat(VFTableName.str());
1917 if (VTableAliasIsRequred) {
1918 llvm::Value *GEPIndices[] = {llvm::ConstantInt::get(CGM.
Int32Ty, 0),
1919 llvm::ConstantInt::get(CGM.
Int32Ty, 0),
1920 llvm::ConstantInt::get(CGM.
Int32Ty, 1)};
1923 llvm::Constant *VTableGEP = llvm::ConstantExpr::getInBoundsGetElementPtr(
1924 VTable->getValueType(), VTable, GEPIndices);
1925 if (llvm::GlobalValue::isWeakForLinker(VFTableLinkage)) {
1926 VFTableLinkage = llvm::GlobalValue::ExternalLinkage;
1928 C->setSelectionKind(llvm::Comdat::Largest);
1930 VFTable = llvm::GlobalAlias::create(CGM.
Int8PtrTy,
1932 VFTableName.str(), VTableGEP,
1934 VFTable->setUnnamedAddr(llvm::GlobalValue::UnnamedAddr::Global);
1943 VTable->setComdat(
C);
1945 if (RD->
hasAttr<DLLExportAttr>())
1946 VFTable->setDLLStorageClass(llvm::GlobalValue::DLLExportStorageClass);
1948 VFTablesMap[
ID] = VFTable;
1959 Ty = Ty->getPointerTo();
1961 adjustThisArgumentForVirtualFunctionCall(CGF, GD, This,
true);
1963 auto *MethodDecl = cast<CXXMethodDecl>(GD.
getDecl());
1964 llvm::Value *VTable = CGF.
GetVTablePtr(VPtr, Ty->getPointerTo(),
1965 MethodDecl->getParent());
1972 auto getObjectWithVPtr = [&] {
1975 [&](
const std::unique_ptr<VPtrInfo> &Info) {
1976 return Info->FullOffsetInMDC == ML.VFPtrOffset;
1985 getObjectWithVPtr(), VTable, Ty,
1993 llvm::Value *VFuncPtr =
1994 Builder.CreateConstInBoundsGEP1_64(Ty, VTable, ML.
Index,
"vfn");
1995 VFunc = Builder.CreateAlignedLoad(Ty, VFuncPtr, CGF.
getPointerAlign());
2002llvm::Value *MicrosoftCXXABI::EmitVirtualDestructorCall(
2004 Address This, DeleteOrMemberCallExpr
E) {
2007 assert((CE !=
nullptr) ^ (
D !=
nullptr));
2008 assert(CE ==
nullptr || CE->arg_begin() == CE->arg_end());
2020 llvm::Value *ImplicitParam = llvm::ConstantInt::get(
2026 ThisTy = CE->getObjectType();
2028 ThisTy =
D->getDestroyedType();
2031 This = adjustThisArgumentForVirtualFunctionCall(CGF, GD, This,
true);
2034 ImplicitParam, Context.
IntTy, CE);
2038const VBTableGlobals &
2039MicrosoftCXXABI::enumerateVBTables(
const CXXRecordDecl *RD) {
2042 llvm::DenseMap<const CXXRecordDecl*, VBTableGlobals>::iterator Entry;
2044 std::tie(Entry, Added) =
2045 VBTablesMap.insert(std::make_pair(RD, VBTableGlobals()));
2046 VBTableGlobals &VBGlobals = Entry->second;
2051 VBGlobals.VBTables = &Context.enumerateVBTables(RD);
2056 for (VPtrInfoVector::const_iterator I = VBGlobals.VBTables->begin(),
2057 E = VBGlobals.VBTables->end();
2059 VBGlobals.Globals.push_back(getAddrOfVBTable(**I, RD,
Linkage));
2066MicrosoftCXXABI::EmitVirtualMemPtrThunk(
const CXXMethodDecl *MD,
2068 assert(!isa<CXXConstructorDecl>(MD) && !isa<CXXDestructorDecl>(MD) &&
2069 "can't form pointers to ctors or virtual dtors");
2073 llvm::raw_svector_ostream Out(ThunkName);
2074 getMangleContext().mangleVirtualMemPtrThunk(MD, ML, Out);
2077 if (llvm::GlobalValue *GV = CGM.
getModule().getNamedValue(ThunkName))
2078 return cast<llvm::Function>(GV);
2084 llvm::Function *ThunkFn =
2085 llvm::Function::Create(ThunkTy, llvm::Function::ExternalLinkage,
2087 assert(ThunkFn->getName() == ThunkName &&
"name was uniqued!");
2090 ? llvm::GlobalValue::LinkOnceODRLinkage
2091 : llvm::GlobalValue::InternalLinkage);
2093 ThunkFn->setComdat(CGM.
getModule().getOrInsertComdat(ThunkFn->getName()));
2102 ThunkFn->addFnAttr(
"thunk");
2105 ThunkFn->setUnnamedAddr(llvm::GlobalValue::UnnamedAddr::None);
2115 buildThisParam(CGF, FunctionArgs);
2122 setCXXABIThisValue(CGF, loadIncomingCXXThis(CGF));
2126 llvm::Type *ThunkPtrTy = ThunkTy->getPointerTo();
2128 getThisAddress(CGF), ThunkPtrTy->getPointerTo(), MD->
getParent());
2130 llvm::Value *VFuncPtr = CGF.
Builder.CreateConstInBoundsGEP1_64(
2131 ThunkPtrTy, VTable, ML.
Index,
"vfn");
2140void MicrosoftCXXABI::emitVirtualInheritanceTables(
const CXXRecordDecl *RD) {
2141 const VBTableGlobals &VBGlobals = enumerateVBTables(RD);
2142 for (
unsigned I = 0,
E = VBGlobals.VBTables->size(); I !=
E; ++I) {
2143 const std::unique_ptr<VPtrInfo>& VBT = (*VBGlobals.VBTables)[I];
2144 llvm::GlobalVariable *GV = VBGlobals.Globals[I];
2145 if (GV->isDeclaration())
2146 emitVBTableDefinition(*VBT, RD, GV);
2150llvm::GlobalVariable *
2152 llvm::GlobalVariable::LinkageTypes
Linkage) {
2154 llvm::raw_svector_ostream Out(OutName);
2155 getMangleContext().mangleCXXVBTable(RD, VBT.
MangledPath, Out);
2156 StringRef Name = OutName.str();
2158 llvm::ArrayType *VBTableType =
2161 assert(!CGM.
getModule().getNamedGlobal(Name) &&
2162 "vbtable with this name already exists: mangling bug?");
2167 GV->setUnnamedAddr(llvm::GlobalValue::UnnamedAddr::Global);
2169 if (RD->
hasAttr<DLLImportAttr>())
2170 GV->setDLLStorageClass(llvm::GlobalValue::DLLImportStorageClass);
2171 else if (RD->
hasAttr<DLLExportAttr>())
2172 GV->setDLLStorageClass(llvm::GlobalValue::DLLExportStorageClass);
2174 if (!GV->hasExternalLinkage())
2175 emitVBTableDefinition(VBT, RD, GV);
2180void MicrosoftCXXABI::emitVBTableDefinition(
const VPtrInfo &VBT,
2182 llvm::GlobalVariable *GV)
const {
2186 "should only emit vbtables for classes with vbtables");
2190 const ASTRecordLayout &DerivedLayout = getContext().getASTRecordLayout(RD);
2197 Offsets[0] = llvm::ConstantInt::get(CGM.
IntTy, -VBPtrOffset.
getQuantity());
2200 for (
const auto &I : ObjectWithVPtr->
vbases()) {
2201 const CXXRecordDecl *VBase = I.getType()->getAsCXXRecordDecl();
2203 assert(!Offset.isNegative());
2208 CompleteVBPtrOffset +=
2210 Offset -= CompleteVBPtrOffset;
2212 unsigned VBIndex = Context.getVBTableIndex(ObjectWithVPtr, VBase);
2213 assert(Offsets[VBIndex] ==
nullptr &&
"The same vbindex seen twice?");
2214 Offsets[VBIndex] = llvm::ConstantInt::get(CGM.
IntTy, Offset.getQuantity());
2217 assert(Offsets.size() ==
2218 cast<llvm::ArrayType>(GV->getValueType())->getNumElements());
2219 llvm::ArrayType *VBTableType =
2220 llvm::ArrayType::get(CGM.
IntTy, Offsets.size());
2221 llvm::Constant *
Init = llvm::ConstantArray::get(VBTableType, Offsets);
2222 GV->setInitializer(
Init);
2224 if (RD->
hasAttr<DLLImportAttr>())
2225 GV->setLinkage(llvm::GlobalVariable::AvailableExternallyLinkage);
2228llvm::Value *MicrosoftCXXABI::performThisAdjustment(
2233 return This.emitRawPointer(CGF);
2239 V =
This.emitRawPointer(CGF);
2249 CGF.
Builder.CreateNeg(VtorDisp));
2262 llvm::Value *VBaseOffset = GetVBaseOffsetFromVBPtr(
2281llvm::Value *MicrosoftCXXABI::performReturnAdjustment(
2286 return Ret.emitRawPointer(CGF);
2290 llvm::Value *
V =
Ret.emitRawPointer(CGF);
2295 llvm::Value *VBaseOffset =
2314bool MicrosoftCXXABI::requiresArrayCookie(
const CXXNewExpr *
expr) {
2317 return expr->getAllocatedType().isDestructedType();
2328llvm::Value *MicrosoftCXXABI::readArrayCookieImpl(
CodeGenFunction &CGF,
2337 llvm::Value *numElements,
2340 assert(requiresArrayCookie(
expr));
2343 CharUnits cookieSize = getArrayCookieSizeImpl(elementType);
2358 llvm::FunctionCallee Dtor,
2359 llvm::Constant *Addr) {
2364 llvm::FunctionType *TLRegDtorTy = llvm::FunctionType::get(
2365 CGF.
IntTy, DtorStub->getType(),
false);
2368 TLRegDtorTy,
"__tlregdtor", llvm::AttributeList(),
true);
2369 if (llvm::Function *TLRegDtorFn =
2370 dyn_cast<llvm::Function>(TLRegDtor.getCallee()))
2371 TLRegDtorFn->setDoesNotThrow();
2377 llvm::FunctionCallee Dtor,
2378 llvm::Constant *Addr) {
2393void MicrosoftCXXABI::EmitThreadLocalInitFuncs(
2397 if (CXXThreadLocalInits.empty())
2402 ?
"/include:___dyn_tls_init@12"
2403 :
"/include:__dyn_tls_init");
2408 auto AddToXDU = [&CGM](llvm::Function *InitFunc) {
2409 llvm::GlobalVariable *InitFuncPtr =
new llvm::GlobalVariable(
2410 CGM.
getModule(), InitFunc->getType(),
true,
2411 llvm::GlobalVariable::InternalLinkage, InitFunc,
2412 Twine(InitFunc->getName(),
"$initializer$"));
2413 InitFuncPtr->setSection(
".CRT$XDU");
2420 std::vector<llvm::Function *> NonComdatInits;
2421 for (
size_t I = 0,
E = CXXThreadLocalInitVars.size(); I !=
E; ++I) {
2422 llvm::GlobalVariable *GV = cast<llvm::GlobalVariable>(
2424 llvm::Function *F = CXXThreadLocalInits[I];
2427 if (llvm::Comdat *
C = GV->getComdat())
2428 AddToXDU(F)->setComdat(
C);
2430 NonComdatInits.push_back(F);
2433 if (!NonComdatInits.empty()) {
2434 llvm::FunctionType *FTy =
2435 llvm::FunctionType::get(CGM.
VoidTy,
false);
2451 llvm::Constant *TlsGuardConstant =
2453 llvm::GlobalValue *TlsGuard = cast<llvm::GlobalValue>(TlsGuardConstant);
2455 TlsGuard->setThreadLocal(
true);
2463 llvm::FunctionType *FTy =
2464 llvm::FunctionType::get(llvm::Type::getVoidTy(CGM.
getLLVMContext()), {},
2467 FTy,
"__dyn_tls_on_demand_init",
2469 llvm::AttributeList::FunctionIndex,
2470 llvm::Attribute::NoUnwind),
2475 llvm::BasicBlock *DynInitBB,
2476 llvm::BasicBlock *ContinueBB) {
2477 llvm::LoadInst *TlsGuardValue =
2479 llvm::Value *CmpResult =
2481 CGF.
Builder.CreateCondBr(CmpResult, DynInitBB, ContinueBB);
2485 llvm::GlobalValue *TlsGuard,
2486 llvm::BasicBlock *ContinueBB) {
2488 llvm::Function *InitializerFunction =
2490 llvm::CallInst *CallVal = CGF.
Builder.CreateCall(InitializerFunction);
2491 CallVal->setCallingConv(InitializerFunction->getCallingConv());
2493 CGF.
Builder.CreateBr(ContinueBB);
2497 llvm::BasicBlock *DynInitBB =
2499 llvm::BasicBlock *ContinueBB =
2505 CGF.
Builder.SetInsertPoint(DynInitBB);
2507 CGF.
Builder.SetInsertPoint(ContinueBB);
2526 Address Addr(
V, RealVarTy, Alignment);
2530 AlignmentSource::Decl)
2536 StringRef VarName(
"_Init_thread_epoch");
2538 if (
auto *GV = CGM.
getModule().getNamedGlobal(VarName))
2540 auto *GV =
new llvm::GlobalVariable(
2542 false, llvm::GlobalVariable::ExternalLinkage,
2544 nullptr, llvm::GlobalVariable::GeneralDynamicTLSModel);
2550 llvm::FunctionType *FTy =
2551 llvm::FunctionType::get(llvm::Type::getVoidTy(CGM.
getLLVMContext()),
2552 CGM.
IntTy->getPointerTo(),
false);
2554 FTy,
"_Init_thread_header",
2556 llvm::AttributeList::FunctionIndex,
2557 llvm::Attribute::NoUnwind),
2562 llvm::FunctionType *FTy =
2563 llvm::FunctionType::get(llvm::Type::getVoidTy(CGM.
getLLVMContext()),
2564 CGM.
IntTy->getPointerTo(),
false);
2566 FTy,
"_Init_thread_footer",
2568 llvm::AttributeList::FunctionIndex,
2569 llvm::Attribute::NoUnwind),
2574 llvm::FunctionType *FTy =
2575 llvm::FunctionType::get(llvm::Type::getVoidTy(CGM.
getLLVMContext()),
2576 CGM.
IntTy->getPointerTo(),
false);
2578 FTy,
"_Init_thread_abort",
2580 llvm::AttributeList::FunctionIndex,
2581 llvm::Attribute::NoUnwind),
2589 ResetGuardBit(
Address Guard,
unsigned GuardNum)
2590 : Guard(Guard), GuardNum(GuardNum) {}
2596 llvm::LoadInst *LI = Builder.CreateLoad(Guard);
2597 llvm::ConstantInt *Mask =
2598 llvm::ConstantInt::get(CGF.
IntTy, ~(1ULL << GuardNum));
2599 Builder.CreateStore(Builder.CreateAnd(LI, Mask), Guard);
2605 CallInitThreadAbort(
RawAddress Guard) : Guard(Guard.getPointer()) {}
2615 llvm::GlobalVariable *GV,
2618 if (!
D.isStaticLocal()) {
2619 assert(GV->hasWeakLinkage() || GV->hasLinkOnceLinkage());
2621 llvm::Function *F = CGF.
CurFn;
2622 F->setLinkage(llvm::GlobalValue::LinkOnceODRLinkage);
2623 F->setComdat(CGM.
getModule().getOrInsertComdat(F->getName()));
2628 bool ThreadlocalStatic =
D.getTLSKind();
2629 bool ThreadsafeStatic = getContext().
getLangOpts().ThreadsafeStatics;
2633 bool HasPerVariableGuard = ThreadsafeStatic && !ThreadlocalStatic;
2636 llvm::IntegerType *GuardTy = CGF.
Int32Ty;
2637 llvm::ConstantInt *
Zero = llvm::ConstantInt::get(GuardTy, 0);
2641 GuardInfo *GI =
nullptr;
2642 if (ThreadlocalStatic)
2644 else if (!ThreadsafeStatic)
2647 llvm::GlobalVariable *GuardVar = GI ? GI->Guard :
nullptr;
2649 if (
D.isExternallyVisible()) {
2652 GuardNum = getContext().getStaticLocalNumber(&
D);
2653 assert(GuardNum > 0);
2655 }
else if (HasPerVariableGuard) {
2659 GuardNum = GI->BitIndex++;
2662 if (!HasPerVariableGuard && GuardNum >= 32) {
2663 if (
D.isExternallyVisible())
2664 ErrorUnsupportedABI(CGF,
"more than 32 guarded initializations");
2673 llvm::raw_svector_ostream Out(GuardName);
2674 if (HasPerVariableGuard)
2675 getMangleContext().mangleThreadSafeStaticGuardVariable(&
D, GuardNum,
2678 getMangleContext().mangleStaticGuardVariable(&
D, Out);
2684 new llvm::GlobalVariable(CGM.
getModule(), GuardTy,
false,
2685 GV->getLinkage(), Zero, GuardName.str());
2686 GuardVar->setVisibility(GV->getVisibility());
2687 GuardVar->setDLLStorageClass(GV->getDLLStorageClass());
2688 GuardVar->setAlignment(GuardAlign.
getAsAlign());
2689 if (GuardVar->isWeakForLinker())
2690 GuardVar->setComdat(
2691 CGM.
getModule().getOrInsertComdat(GuardVar->getName()));
2694 if (GI && !HasPerVariableGuard)
2695 GI->Guard = GuardVar;
2700 assert(GuardVar->getLinkage() == GV->getLinkage() &&
2701 "static local from the same function had different linkage");
2703 if (!HasPerVariableGuard) {
2711 llvm::ConstantInt *Bit = llvm::ConstantInt::get(GuardTy, 1ULL << GuardNum);
2712 llvm::LoadInst *LI = Builder.CreateLoad(GuardAddr);
2713 llvm::Value *NeedsInit =
2714 Builder.CreateICmpEQ(Builder.CreateAnd(LI, Bit), Zero);
2718 CodeGenFunction::GuardKind::VariableGuard, &
D);
2723 Builder.CreateStore(Builder.CreateOr(LI, Bit), GuardAddr);
2727 Builder.CreateBr(EndBlock);
2745 llvm::LoadInst *FirstGuardLoad = Builder.CreateLoad(GuardAddr);
2746 FirstGuardLoad->setOrdering(llvm::AtomicOrdering::Unordered);
2747 llvm::LoadInst *InitThreadEpoch =
2749 llvm::Value *IsUninitialized =
2750 Builder.CreateICmpSGT(FirstGuardLoad, InitThreadEpoch);
2754 CodeGenFunction::GuardKind::VariableGuard, &
D);
2760 GuardAddr.getPointer());
2761 llvm::LoadInst *SecondGuardLoad = Builder.CreateLoad(GuardAddr);
2762 SecondGuardLoad->setOrdering(llvm::AtomicOrdering::Unordered);
2763 llvm::Value *ShouldDoInit =
2764 Builder.CreateICmpEQ(SecondGuardLoad, getAllOnesInt());
2766 Builder.CreateCondBr(ShouldDoInit, InitBlock, EndBlock);
2774 GuardAddr.getPointer());
2775 Builder.CreateBr(EndBlock);
2804 fields.push_back(CGM.
IntTy);
2808 fields.push_back(CGM.
IntTy);
2810 fields.push_back(CGM.
IntTy);
2812 fields.push_back(CGM.
IntTy);
2814 if (fields.size() == 1)
2819void MicrosoftCXXABI::
2822 assert(fields.empty());
2827 fields.push_back(llvm::Constant::getNullValue(CGM.
VoidPtrTy));
2830 fields.push_back(getZeroInt());
2832 fields.push_back(getAllOnesInt());
2837 fields.push_back(getZeroInt());
2839 fields.push_back(getZeroInt());
2841 fields.push_back(getAllOnesInt());
2847 GetNullMemberPointerFields(MPT, fields);
2848 if (fields.size() == 1)
2850 llvm::Constant *Res = llvm::ConstantStruct::getAnon(fields);
2851 assert(Res->getType() == ConvertMemberPointerType(MPT));
2856MicrosoftCXXABI::EmitFullMemberPointer(llvm::Constant *FirstField,
2857 bool IsMemberFunction,
2860 unsigned VBTableIndex) {
2869 fields.push_back(FirstField);
2872 fields.push_back(llvm::ConstantInt::get(
2878 Offs = getContext().getASTRecordLayout(RD).getVBPtrOffset();
2884 fields.push_back(llvm::ConstantInt::get(CGM.
IntTy, VBTableIndex));
2886 return llvm::ConstantStruct::getAnon(fields);
2895llvm::Constant *MicrosoftCXXABI::EmitMemberDataPointer(
const CXXRecordDecl *RD,
2898 MSInheritanceModel::Virtual)
2899 offset -= getContext().getOffsetOfBaseWithVBPtr(RD);
2900 llvm::Constant *FirstField =
2902 return EmitFullMemberPointer(FirstField,
false, RD,
2906llvm::Constant *MicrosoftCXXABI::EmitMemberPointer(
const APValue &MP,
2911 return EmitNullMemberPointer(DstTy);
2917 if (
const CXXMethodDecl *MD = dyn_cast<CXXMethodDecl>(MPD)) {
2918 C = EmitMemberFunctionPointer(MD);
2925 const FieldDecl *FD = dyn_cast<FieldDecl>(MPD);
2927 FD = cast<FieldDecl>(*cast<IndirectFieldDecl>(MPD)->chain_begin());
2930 C = EmitMemberDataPointer(RD, FieldOffset);
2933 if (!MemberPointerPath.empty()) {
2934 const CXXRecordDecl *SrcRD = cast<CXXRecordDecl>(MPD->getDeclContext());
2946 if (DerivedMember) {
2954 if (BS.getType()->getAsCXXRecordDecl()->getCanonicalDecl() ==
2955 Base->getCanonicalDecl())
2956 DerivedToBasePath.push_back(&BS);
2959 assert(DerivedToBasePath.size() == MemberPointerPath.size());
2961 CastKind CK = DerivedMember ? CK_DerivedToBaseMemberPointer
2962 : CK_BaseToDerivedMemberPointer;
2963 C = EmitMemberPointerConversion(SrcTy, DstTy, CK, DerivedToBasePath.begin(),
2964 DerivedToBasePath.end(),
C);
2970MicrosoftCXXABI::EmitMemberFunctionPointer(
const CXXMethodDecl *MD) {
2971 assert(MD->
isInstance() &&
"Member function must not be static!");
2977 unsigned VBTableIndex = 0;
2978 llvm::Constant *FirstField;
2983 if (Types.isFuncTypeConvertible(FPT)) {
2985 Ty = Types.GetFunctionType(Types.arrangeCXXMethodDeclaration(MD));
2995 FirstField = EmitVirtualMemPtrThunk(MD, ML);
2999 VBTableIndex = VTableContext.getVBTableIndex(RD, ML.
VBase) * 4;
3002 if (VBTableIndex == 0 &&
3004 MSInheritanceModel::Virtual)
3005 NonVirtualBaseAdjustment -= getContext().getOffsetOfBaseWithVBPtr(RD);
3008 return EmitFullMemberPointer(FirstField,
true, RD,
3009 NonVirtualBaseAdjustment, VBTableIndex);
3024 llvm::ICmpInst::Predicate
Eq;
3025 llvm::Instruction::BinaryOps
And,
Or;
3027 Eq = llvm::ICmpInst::ICMP_NE;
3028 And = llvm::Instruction::Or;
3029 Or = llvm::Instruction::And;
3031 Eq = llvm::ICmpInst::ICMP_EQ;
3032 And = llvm::Instruction::And;
3033 Or = llvm::Instruction::Or;
3042 return Builder.CreateICmp(
Eq, L, R);
3045 llvm::Value *L0 = Builder.CreateExtractValue(L, 0,
"lhs.0");
3046 llvm::Value *R0 = Builder.CreateExtractValue(R, 0,
"rhs.0");
3047 llvm::Value *Cmp0 = Builder.CreateICmp(
Eq, L0, R0,
"memptr.cmp.first");
3050 llvm::Value *Res =
nullptr;
3051 llvm::StructType *LType = cast<llvm::StructType>(L->getType());
3052 for (
unsigned I = 1,
E = LType->getNumElements(); I !=
E; ++I) {
3053 llvm::Value *LF = Builder.CreateExtractValue(L, I);
3054 llvm::Value *RF = Builder.CreateExtractValue(R, I);
3055 llvm::Value *Cmp = Builder.CreateICmp(
Eq, LF, RF,
"memptr.cmp.rest");
3057 Res = Builder.CreateBinOp(
And, Res, Cmp);
3065 llvm::Value *
Zero = llvm::Constant::getNullValue(L0->getType());
3066 llvm::Value *IsZero = Builder.CreateICmp(
Eq, L0, Zero,
"memptr.cmp.iszero");
3067 Res = Builder.CreateBinOp(
Or, Res, IsZero);
3072 return Builder.CreateBinOp(
And, Res, Cmp0,
"memptr.cmp");
3077 llvm::Value *MemPtr,
3083 fields.push_back(llvm::Constant::getNullValue(CGM.
VoidPtrTy));
3085 GetNullMemberPointerFields(MPT, fields);
3086 assert(!fields.empty());
3087 llvm::Value *FirstField = MemPtr;
3088 if (MemPtr->getType()->isStructTy())
3089 FirstField = Builder.CreateExtractValue(MemPtr, 0);
3090 llvm::Value *Res = Builder.CreateICmpNE(FirstField, fields[0],
"memptr.cmp0");
3098 for (
int I = 1,
E = fields.size(); I <
E; ++I) {
3099 llvm::Value *
Field = Builder.CreateExtractValue(MemPtr, I);
3100 llvm::Value *Next = Builder.CreateICmpNE(Field, fields[I],
"memptr.cmp");
3101 Res = Builder.CreateOr(Res, Next,
"memptr.tobool");
3107 llvm::Constant *Val) {
3110 llvm::Constant *FirstField = Val->getType()->isStructTy() ?
3111 Val->getAggregateElement(0
U) : Val;
3112 return FirstField->isNullValue();
3117 if (isZeroInitializable(MPT) && Val->isNullValue())
3123 GetNullMemberPointerFields(MPT, Fields);
3124 if (Fields.size() == 1) {
3125 assert(Val->getType()->isIntegerTy());
3126 return Val == Fields[0];
3130 for (I = 0,
E = Fields.size(); I !=
E; ++I) {
3131 if (Val->getAggregateElement(I) != Fields[I])
3140 llvm::Value *VBPtrOffset,
3141 llvm::Value *VBTableOffset,
3142 llvm::Value **VBPtrOut) {
3145 llvm::Value *VBPtr = Builder.CreateInBoundsGEP(
3146 CGM.
Int8Ty,
This.emitRawPointer(CGF), VBPtrOffset,
"vbptr");
3151 if (
auto CI = dyn_cast<llvm::ConstantInt>(VBPtrOffset)) {
3152 VBPtrAlign =
This.getAlignment().alignmentAtOffset(
3158 llvm::Value *VBTable = Builder.CreateAlignedLoad(
3159 CGM.
Int32Ty->getPointerTo(0), VBPtr, VBPtrAlign,
"vbtable");
3162 llvm::Value *VBTableIndex = Builder.CreateAShr(
3163 VBTableOffset, llvm::ConstantInt::get(VBTableOffset->getType(), 2),
3167 llvm::Value *VBaseOffs =
3168 Builder.CreateInBoundsGEP(CGM.
Int32Ty, VBTable, VBTableIndex);
3169 return Builder.CreateAlignedLoad(CGM.
Int32Ty, VBaseOffs,
3175llvm::Value *MicrosoftCXXABI::AdjustVirtualBase(
3177 Address Base, llvm::Value *VBTableOffset, llvm::Value *VBPtrOffset) {
3180 llvm::BasicBlock *OriginalBB =
nullptr;
3181 llvm::BasicBlock *SkipAdjustBB =
nullptr;
3182 llvm::BasicBlock *VBaseAdjustBB =
nullptr;
3189 OriginalBB = Builder.GetInsertBlock();
3192 llvm::Value *IsVirtual =
3193 Builder.CreateICmpNE(VBTableOffset, getZeroInt(),
3195 Builder.CreateCondBr(IsVirtual, VBaseAdjustBB, SkipAdjustBB);
3207 "member pointer representation requires a "
3208 "complete class type for %0 to perform this expression");
3211 offs = getContext().getASTRecordLayout(RD).getVBPtrOffset();
3214 llvm::Value *VBPtr =
nullptr;
3215 llvm::Value *VBaseOffs =
3216 GetVBaseOffsetFromVBPtr(CGF,
Base, VBPtrOffset, VBTableOffset, &VBPtr);
3217 llvm::Value *AdjustedBase =
3218 Builder.CreateInBoundsGEP(CGM.
Int8Ty, VBPtr, VBaseOffs);
3221 if (VBaseAdjustBB) {
3222 Builder.CreateBr(SkipAdjustBB);
3224 llvm::PHINode *Phi = Builder.CreatePHI(CGM.
Int8PtrTy, 2,
"memptr.base");
3225 Phi->addIncoming(
Base.emitRawPointer(CGF), OriginalBB);
3226 Phi->addIncoming(AdjustedBase, VBaseAdjustBB);
3229 return AdjustedBase;
3232llvm::Value *MicrosoftCXXABI::EmitMemberDataPointerAddress(
3242 llvm::Value *FieldOffset = MemPtr;
3243 llvm::Value *VirtualBaseAdjustmentOffset =
nullptr;
3244 llvm::Value *VBPtrOffset =
nullptr;
3245 if (MemPtr->getType()->isStructTy()) {
3248 FieldOffset = Builder.CreateExtractValue(MemPtr, I++);
3250 VBPtrOffset = Builder.CreateExtractValue(MemPtr, I++);
3252 VirtualBaseAdjustmentOffset = Builder.CreateExtractValue(MemPtr, I++);
3256 if (VirtualBaseAdjustmentOffset) {
3257 Addr = AdjustVirtualBase(CGF,
E, RD,
Base, VirtualBaseAdjustmentOffset,
3260 Addr =
Base.emitRawPointer(CGF);
3264 return Builder.CreateInBoundsGEP(CGF.
Int8Ty, Addr, FieldOffset,
3272 assert(
E->getCastKind() == CK_DerivedToBaseMemberPointer ||
3273 E->getCastKind() == CK_BaseToDerivedMemberPointer ||
3274 E->getCastKind() == CK_ReinterpretMemberPointer);
3277 if (isa<llvm::Constant>(Src))
3278 return EmitMemberPointerConversion(
E, cast<llvm::Constant>(Src));
3288 bool IsReinterpret =
E->getCastKind() == CK_ReinterpretMemberPointer;
3289 if (IsReinterpret && IsFunc)
3294 if (IsReinterpret &&
3301 llvm::Value *
IsNotNull = EmitMemberPointerIsNotNull(CGF, Src, SrcTy);
3302 llvm::Constant *DstNull = EmitNullMemberPointer(DstTy);
3306 if (IsReinterpret) {
3309 assert(Src->getType() == DstNull->getType());
3310 return Builder.CreateSelect(
IsNotNull, Src, DstNull);
3313 llvm::BasicBlock *OriginalBB = Builder.GetInsertBlock();
3316 Builder.CreateCondBr(
IsNotNull, ConvertBB, ContinueBB);
3319 llvm::Value *Dst = EmitNonNullMemberPointerConversion(
3320 SrcTy, DstTy,
E->getCastKind(),
E->path_begin(),
E->path_end(), Src,
3323 Builder.CreateBr(ContinueBB);
3327 llvm::PHINode *Phi = Builder.CreatePHI(DstNull->getType(), 2,
"memptr.converted");
3328 Phi->addIncoming(DstNull, OriginalBB);
3329 Phi->addIncoming(Dst, ConvertBB);
3333llvm::Value *MicrosoftCXXABI::EmitNonNullMemberPointerConversion(
3343 bool IsConstant = isa<llvm::Constant>(Src);
3346 llvm::Value *FirstField = Src;
3347 llvm::Value *NonVirtualBaseAdjustment = getZeroInt();
3348 llvm::Value *VirtualBaseAdjustmentOffset = getZeroInt();
3349 llvm::Value *VBPtrOffset = getZeroInt();
3353 FirstField = Builder.CreateExtractValue(Src, I++);
3355 NonVirtualBaseAdjustment = Builder.CreateExtractValue(Src, I++);
3357 VBPtrOffset = Builder.CreateExtractValue(Src, I++);
3359 VirtualBaseAdjustmentOffset = Builder.CreateExtractValue(Src, I++);
3362 bool IsDerivedToBase = (CK == CK_DerivedToBaseMemberPointer);
3368 llvm::Value *&NVAdjustField = IsFunc ? NonVirtualBaseAdjustment : FirstField;
3375 llvm::Value *SrcVBIndexEqZero =
3376 Builder.CreateICmpEQ(VirtualBaseAdjustmentOffset, getZeroInt());
3377 if (SrcInheritance == MSInheritanceModel::Virtual) {
3378 if (int64_t SrcOffsetToFirstVBase =
3379 getContext().getOffsetOfBaseWithVBPtr(SrcRD).getQuantity()) {
3380 llvm::Value *UndoSrcAdjustment = Builder.CreateSelect(
3382 llvm::ConstantInt::get(CGM.
IntTy, SrcOffsetToFirstVBase),
3384 NVAdjustField = Builder.CreateNSWAdd(NVAdjustField, UndoSrcAdjustment);
3395 llvm::Constant *BaseClassOffset = llvm::ConstantInt::get(
3400 llvm::Value *NVDisp;
3401 if (IsDerivedToBase)
3402 NVDisp = Builder.CreateNSWSub(NVAdjustField, BaseClassOffset,
"adj");
3404 NVDisp = Builder.CreateNSWAdd(NVAdjustField, BaseClassOffset,
"adj");
3406 NVAdjustField = Builder.CreateSelect(SrcVBIndexEqZero, NVDisp, getZeroInt());
3410 llvm::Value *DstVBIndexEqZero = SrcVBIndexEqZero;
3413 if (llvm::GlobalVariable *VDispMap =
3414 getAddrOfVirtualDisplacementMap(SrcRD, DstRD)) {
3415 llvm::Value *VBIndex = Builder.CreateExactUDiv(
3416 VirtualBaseAdjustmentOffset, llvm::ConstantInt::get(CGM.
IntTy, 4));
3418 llvm::Constant *Mapping = VDispMap->getInitializer();
3419 VirtualBaseAdjustmentOffset =
3420 Mapping->getAggregateElement(cast<llvm::Constant>(VBIndex));
3422 llvm::Value *Idxs[] = {getZeroInt(), VBIndex};
3423 VirtualBaseAdjustmentOffset = Builder.CreateAlignedLoad(
3424 CGM.
IntTy, Builder.CreateInBoundsGEP(VDispMap->getValueType(),
3430 Builder.CreateICmpEQ(VirtualBaseAdjustmentOffset, getZeroInt());
3437 llvm::Value *DstVBPtrOffset = llvm::ConstantInt::get(
3439 getContext().getASTRecordLayout(DstRD).getVBPtrOffset().getQuantity());
3441 Builder.CreateSelect(DstVBIndexEqZero, getZeroInt(), DstVBPtrOffset);
3447 if (DstInheritance == MSInheritanceModel::Virtual) {
3448 if (int64_t DstOffsetToFirstVBase =
3449 getContext().getOffsetOfBaseWithVBPtr(DstRD).getQuantity()) {
3450 llvm::Value *DoDstAdjustment = Builder.CreateSelect(
3452 llvm::ConstantInt::get(CGM.
IntTy, DstOffsetToFirstVBase),
3454 NVAdjustField = Builder.CreateNSWSub(NVAdjustField, DoDstAdjustment);
3463 Dst = llvm::UndefValue::get(ConvertMemberPointerType(DstTy));
3465 Dst = Builder.CreateInsertValue(Dst, FirstField, Idx++);
3467 Dst = Builder.CreateInsertValue(Dst, NonVirtualBaseAdjustment, Idx++);
3469 Dst = Builder.CreateInsertValue(Dst, VBPtrOffset, Idx++);
3471 Dst = Builder.CreateInsertValue(Dst, VirtualBaseAdjustmentOffset, Idx++);
3477MicrosoftCXXABI::EmitMemberPointerConversion(
const CastExpr *
E,
3478 llvm::Constant *Src) {
3485 return EmitMemberPointerConversion(SrcTy, DstTy, CK,
E->path_begin(),
3486 E->path_end(), Src);
3489llvm::Constant *MicrosoftCXXABI::EmitMemberPointerConversion(
3493 assert(CK == CK_DerivedToBaseMemberPointer ||
3494 CK == CK_BaseToDerivedMemberPointer ||
3495 CK == CK_ReinterpretMemberPointer);
3498 if (MemberPointerConstantIsNull(SrcTy, Src))
3499 return EmitNullMemberPointer(DstTy);
3504 if (CK == CK_ReinterpretMemberPointer)
3508 auto *Dst = cast<llvm::Constant>(EmitNonNullMemberPointerConversion(
3509 SrcTy, DstTy, CK, PathBegin, PathEnd, Src, Builder));
3514CGCallee MicrosoftCXXABI::EmitLoadOfMemberFunctionPointer(
3516 llvm::Value *&ThisPtrForCall, llvm::Value *MemPtr,
3528 llvm::Value *FunctionPointer = MemPtr;
3529 llvm::Value *NonVirtualBaseAdjustment =
nullptr;
3530 llvm::Value *VirtualBaseAdjustmentOffset =
nullptr;
3531 llvm::Value *VBPtrOffset =
nullptr;
3532 if (MemPtr->getType()->isStructTy()) {
3535 FunctionPointer = Builder.CreateExtractValue(MemPtr, I++);
3537 NonVirtualBaseAdjustment = Builder.CreateExtractValue(MemPtr, I++);
3539 VBPtrOffset = Builder.CreateExtractValue(MemPtr, I++);
3541 VirtualBaseAdjustmentOffset = Builder.CreateExtractValue(MemPtr, I++);
3544 if (VirtualBaseAdjustmentOffset) {
3545 ThisPtrForCall = AdjustVirtualBase(CGF,
E, RD, This,
3546 VirtualBaseAdjustmentOffset, VBPtrOffset);
3548 ThisPtrForCall =
This.emitRawPointer(CGF);
3551 if (NonVirtualBaseAdjustment)
3552 ThisPtrForCall = Builder.CreateInBoundsGEP(CGF.
Int8Ty, ThisPtrForCall,
3553 NonVirtualBaseAdjustment);
3560 return new MicrosoftCXXABI(CGM);
3594 StringRef MangledName(
"??_7type_info@@6B@");
3595 if (
auto VTable = CGM.
getModule().getNamedGlobal(MangledName))
3599 llvm::GlobalVariable::ExternalLinkage,
3600 nullptr, MangledName);
3613 IsPrivateOnPath = 1 | 8,
3617 HasHierarchyDescriptor = 64
3623 MSRTTIClass *getFirstChild() {
return this + 1; }
3624 static MSRTTIClass *getNextChild(MSRTTIClass *Child) {
3625 return Child + 1 + Child->NumBases;
3629 uint32_t Flags, NumBases, OffsetInVBase;
3633uint32_t MSRTTIClass::initialize(
const MSRTTIClass *
Parent,
3635 Flags = HasHierarchyDescriptor;
3637 VirtualRoot =
nullptr;
3641 Flags |= IsPrivate | IsPrivateOnPath;
3647 if (
Parent->Flags & IsPrivateOnPath)
3648 Flags |= IsPrivateOnPath;
3649 VirtualRoot =
Parent->VirtualRoot;
3650 OffsetInVBase =
Parent->OffsetInVBase + RD->getASTContext()
3651 .getASTRecordLayout(
Parent->RD).getBaseClassOffset(RD).getQuantity();
3655 MSRTTIClass *Child = getFirstChild();
3657 NumBases += Child->initialize(
this, &
Base) + 1;
3658 Child = getNextChild(Child);
3663static llvm::GlobalValue::LinkageTypes getLinkageForRTTI(
QualType Ty) {
3665 case Linkage::Invalid:
3666 llvm_unreachable(
"Linkage hasn't been computed!");
3669 case Linkage::Internal:
3670 case Linkage::UniqueExternal:
3671 return llvm::GlobalValue::InternalLinkage;
3673 case Linkage::VisibleNone:
3674 case Linkage::Module:
3675 case Linkage::External:
3676 return llvm::GlobalValue::LinkOnceODRLinkage;
3678 llvm_unreachable(
"Invalid linkage!");
3684struct MSRTTIBuilder {
3686 HasBranchingHierarchy = 1,
3687 HasVirtualBranchingHierarchy = 2,
3688 HasAmbiguousBases = 4
3691 MSRTTIBuilder(MicrosoftCXXABI &ABI,
const CXXRecordDecl *RD)
3692 : CGM(ABI.CGM), Context(CGM.getContext()),
3693 VMContext(CGM.getLLVMContext()),
Module(CGM.getModule()), RD(RD),
3694 Linkage(getLinkageForRTTI(CGM.getContext().getTagDeclType(RD))),
3697 llvm::GlobalVariable *getBaseClassDescriptor(
const MSRTTIClass &Classes);
3698 llvm::GlobalVariable *
3700 llvm::GlobalVariable *getClassHierarchyDescriptor();
3701 llvm::GlobalVariable *getCompleteObjectLocator(
const VPtrInfo &Info);
3705 llvm::LLVMContext &VMContext;
3708 llvm::GlobalVariable::LinkageTypes
Linkage;
3709 MicrosoftCXXABI &ABI;
3718 Classes.push_back(MSRTTIClass(RD));
3729 for (MSRTTIClass *
Class = &Classes.front();
Class <= &Classes.back();) {
3730 if ((
Class->Flags & MSRTTIClass::IsVirtual) &&
3731 !VirtualBases.insert(
Class->RD).second) {
3735 if (!UniqueBases.insert(
Class->RD).second)
3736 AmbiguousBases.insert(
Class->RD);
3739 if (AmbiguousBases.empty())
3741 for (MSRTTIClass &
Class : Classes)
3742 if (AmbiguousBases.count(
Class.RD))
3743 Class.Flags |= MSRTTIClass::IsAmbiguous;
3746llvm::GlobalVariable *MSRTTIBuilder::getClassHierarchyDescriptor() {
3749 llvm::raw_svector_ostream Out(MangledName);
3750 ABI.getMangleContext().mangleCXXRTTIClassHierarchyDescriptor(RD, Out);
3754 if (
auto CHD =
Module.getNamedGlobal(MangledName))
3760 Classes.front().initialize(
nullptr,
nullptr);
3763 for (
const MSRTTIClass &Class : Classes) {
3764 if (
Class.RD->getNumBases() > 1)
3765 Flags |= HasBranchingHierarchy;
3768 if (
Class.Flags & MSRTTIClass::IsAmbiguous)
3769 Flags |= HasAmbiguousBases;
3771 if ((Flags & HasBranchingHierarchy) && RD->getNumVBases() != 0)
3772 Flags |= HasVirtualBranchingHierarchy;
3775 llvm::Value *GEPIndices[] = {llvm::ConstantInt::get(CGM.
IntTy, 0),
3776 llvm::ConstantInt::get(CGM.
IntTy, 0)};
3779 auto Type = ABI.getClassHierarchyDescriptorType();
3783 if (CHD->isWeakForLinker())
3784 CHD->setComdat(CGM.
getModule().getOrInsertComdat(CHD->getName()));
3786 auto *Bases = getBaseClassArray(Classes);
3789 llvm::Constant *Fields[] = {
3790 llvm::ConstantInt::get(CGM.
IntTy, 0),
3791 llvm::ConstantInt::get(CGM.
IntTy, Flags),
3792 llvm::ConstantInt::get(CGM.
IntTy, Classes.size()),
3793 ABI.getImageRelativeConstant(llvm::ConstantExpr::getInBoundsGetElementPtr(
3794 Bases->getValueType(), Bases,
3797 CHD->setInitializer(llvm::ConstantStruct::get(
Type, Fields));
3801llvm::GlobalVariable *
3805 llvm::raw_svector_ostream Out(MangledName);
3806 ABI.getMangleContext().mangleCXXRTTIBaseClassArray(RD, Out);
3814 llvm::Type *PtrType = ABI.getImageRelativeType(
3815 ABI.getBaseClassDescriptorType()->getPointerTo());
3816 auto *ArrType = llvm::ArrayType::get(PtrType, Classes.size() + 1);
3818 new llvm::GlobalVariable(
Module, ArrType,
3820 nullptr, MangledName);
3821 if (BCA->isWeakForLinker())
3822 BCA->setComdat(CGM.
getModule().getOrInsertComdat(BCA->getName()));
3826 for (MSRTTIClass &Class : Classes)
3827 BaseClassArrayData.push_back(
3828 ABI.getImageRelativeConstant(getBaseClassDescriptor(Class)));
3829 BaseClassArrayData.push_back(llvm::Constant::getNullValue(PtrType));
3830 BCA->setInitializer(llvm::ConstantArray::get(ArrType, BaseClassArrayData));
3834llvm::GlobalVariable *
3835MSRTTIBuilder::getBaseClassDescriptor(
const MSRTTIClass &Class) {
3838 uint32_t OffsetInVBTable = 0;
3839 int32_t VBPtrOffset = -1;
3840 if (
Class.VirtualRoot) {
3848 llvm::raw_svector_ostream Out(MangledName);
3849 ABI.getMangleContext().mangleCXXRTTIBaseClassDescriptor(
3850 Class.RD,
Class.OffsetInVBase, VBPtrOffset, OffsetInVBTable,
3855 if (
auto BCD =
Module.getNamedGlobal(MangledName))
3859 auto Type = ABI.getBaseClassDescriptorType();
3862 nullptr, MangledName);
3863 if (BCD->isWeakForLinker())
3864 BCD->setComdat(CGM.
getModule().getOrInsertComdat(BCD->getName()));
3867 llvm::Constant *Fields[] = {
3868 ABI.getImageRelativeConstant(
3870 llvm::ConstantInt::get(CGM.
IntTy,
Class.NumBases),
3871 llvm::ConstantInt::get(CGM.
IntTy,
Class.OffsetInVBase),
3872 llvm::ConstantInt::get(CGM.
IntTy, VBPtrOffset),
3873 llvm::ConstantInt::get(CGM.
IntTy, OffsetInVBTable),
3874 llvm::ConstantInt::get(CGM.
IntTy,
Class.Flags),
3875 ABI.getImageRelativeConstant(
3876 MSRTTIBuilder(ABI,
Class.RD).getClassHierarchyDescriptor()),
3878 BCD->setInitializer(llvm::ConstantStruct::get(
Type, Fields));
3882llvm::GlobalVariable *
3883MSRTTIBuilder::getCompleteObjectLocator(
const VPtrInfo &Info) {
3886 llvm::raw_svector_ostream Out(MangledName);
3887 ABI.getMangleContext().mangleCXXRTTICompleteObjectLocator(RD, Info.
MangledPath, Out);
3891 if (
auto COL =
Module.getNamedGlobal(MangledName))
3896 int VFPtrOffset = 0;
3902 ->second.hasVtorDisp())
3906 llvm::StructType *
Type = ABI.getCompleteObjectLocatorType();
3908 nullptr, MangledName);
3911 llvm::Constant *Fields[] = {
3912 llvm::ConstantInt::get(CGM.
IntTy, ABI.isImageRelative()),
3913 llvm::ConstantInt::get(CGM.
IntTy, OffsetToTop),
3914 llvm::ConstantInt::get(CGM.
IntTy, VFPtrOffset),
3915 ABI.getImageRelativeConstant(
3917 ABI.getImageRelativeConstant(getClassHierarchyDescriptor()),
3918 ABI.getImageRelativeConstant(COL),
3921 if (!ABI.isImageRelative())
3922 FieldsRef = FieldsRef.drop_back();
3923 COL->setInitializer(llvm::ConstantStruct::get(
Type, FieldsRef));
3924 if (COL->isWeakForLinker())
3925 COL->setComdat(CGM.
getModule().getOrInsertComdat(COL->getName()));
3930 bool &IsConst,
bool &IsVolatile,
3931 bool &IsUnaligned) {
3941 IsUnaligned =
false;
3943 if (!PointeeType.
isNull()) {
3964MicrosoftCXXABI::getAddrOfCXXCatchHandlerType(
QualType Type,
3969 bool IsConst, IsVolatile, IsUnaligned;
3993llvm::Constant *MicrosoftCXXABI::getAddrOfRTTIDescriptor(
QualType Type) {
3996 llvm::raw_svector_ostream Out(MangledName);
3997 getMangleContext().mangleCXXRTTI(
Type, Out);
4001 if (llvm::GlobalVariable *GV = CGM.
getModule().getNamedGlobal(MangledName))
4010 llvm::raw_svector_ostream Out(TypeInfoString);
4011 getMangleContext().mangleCXXRTTIName(
Type, Out);
4015 llvm::Constant *Fields[] = {
4017 llvm::ConstantPointerNull::get(CGM.
Int8PtrTy),
4018 llvm::ConstantDataArray::getString(CGM.
getLLVMContext(), TypeInfoString)};
4019 llvm::StructType *TypeDescriptorType =
4020 getTypeDescriptorType(TypeInfoString);
4021 auto *Var =
new llvm::GlobalVariable(
4022 CGM.
getModule(), TypeDescriptorType,
false,
4023 getLinkageForRTTI(
Type),
4024 llvm::ConstantStruct::get(TypeDescriptorType, Fields),
4026 if (Var->isWeakForLinker())
4027 Var->setComdat(CGM.
getModule().getOrInsertComdat(Var->getName()));
4032llvm::GlobalVariable *
4033MicrosoftCXXABI::getMSCompleteObjectLocator(
const CXXRecordDecl *RD,
4035 return MSRTTIBuilder(*
this, RD).getCompleteObjectLocator(Info);
4038void MicrosoftCXXABI::emitCXXStructor(
GlobalDecl GD) {
4039 if (
auto *ctor = dyn_cast<CXXConstructorDecl>(GD.
getDecl())) {
4041 llvm::Function *
Fn =
4047 auto *dtor = cast<CXXDestructorDecl>(GD.
getDecl());
4053 dtor->getParent()->getNumVBases() == 0)
4064 if (
Fn->isWeakForLinker())
4065 Fn->setComdat(CGM.
getModule().getOrInsertComdat(
Fn->getName()));
4075 llvm::raw_svector_ostream Out(ThunkName);
4076 getMangleContext().mangleName(
GlobalDecl(CD, CT), Out);
4079 if (llvm::GlobalValue *GV = CGM.
getModule().getNamedValue(ThunkName))
4080 return cast<llvm::Function>(GV);
4086 QualType RecordTy = getContext().getRecordType(RD);
4087 llvm::Function *ThunkFn = llvm::Function::Create(
4088 ThunkTy, getLinkageForRTTI(RecordTy), ThunkName.str(), &CGM.
getModule());
4089 ThunkFn->setCallingConv(
static_cast<llvm::CallingConv::ID
>(
4091 if (ThunkFn->isWeakForLinker())
4092 ThunkFn->setComdat(CGM.
getModule().getOrInsertComdat(ThunkFn->getName()));
4103 buildThisParam(CGF, FunctionArgs);
4109 &getContext().Idents.get(
"src"),
4110 getContext().getLValueReferenceType(RecordTy,
4112 ImplicitParamKind::Other);
4114 FunctionArgs.push_back(&SrcParam);
4121 &getContext().Idents.get(
"is_most_derived"),
4122 getContext().IntTy, ImplicitParamKind::Other);
4125 FunctionArgs.push_back(&IsMostDerived);
4133 setCXXABIThisValue(CGF, loadIncomingCXXThis(CGF));
4134 llvm::Value *
This = getThisValue(CGF);
4136 llvm::Value *SrcVal =
4153 assert(PD->hasDefaultArg() &&
"ctor closure lacks default args");
4154 ArgVec.push_back(PD->getDefaultArg());
4157 CodeGenFunction::RunCleanupsScope Cleanups(CGF);
4163 AddedStructorArgCounts ExtraArgs =
4168 llvm::Constant *CalleePtr =
4173 Args, CD,
Ctor_Complete, ExtraArgs.Prefix, ExtraArgs.Suffix);
4176 Cleanups.ForceCleanup();
4185llvm::Constant *MicrosoftCXXABI::getCatchableType(
QualType T,
4187 int32_t VBPtrOffset,
4199 uint32_t
Size = getContext().getTypeSizeInChars(
T).getQuantity();
4202 llvm::raw_svector_ostream Out(MangledName);
4203 getMangleContext().mangleCXXCatchableType(
T, CD, CT, Size, NVOffset,
4204 VBPtrOffset, VBIndex, Out);
4206 if (llvm::GlobalVariable *GV = CGM.
getModule().getNamedGlobal(MangledName))
4207 return getImageRelativeConstant(GV);
4211 llvm::Constant *TD = getImageRelativeConstant(getAddrOfRTTIDescriptor(
T));
4215 llvm::Constant *CopyCtor;
4222 CopyCtor = llvm::Constant::getNullValue(CGM.
Int8PtrTy);
4224 CopyCtor = getImageRelativeConstant(CopyCtor);
4226 bool IsScalar = !RD;
4227 bool HasVirtualBases =
false;
4228 bool IsStdBadAlloc =
false;
4243 if (HasVirtualBases)
4248 llvm::Constant *Fields[] = {
4249 llvm::ConstantInt::get(CGM.
IntTy, Flags),
4251 llvm::ConstantInt::get(CGM.
IntTy, NVOffset),
4252 llvm::ConstantInt::get(CGM.
IntTy, VBPtrOffset),
4253 llvm::ConstantInt::get(CGM.
IntTy, VBIndex),
4254 llvm::ConstantInt::get(CGM.
IntTy, Size),
4257 llvm::StructType *CTType = getCatchableTypeType();
4258 auto *GV =
new llvm::GlobalVariable(
4259 CGM.
getModule(), CTType,
true, getLinkageForRTTI(
T),
4260 llvm::ConstantStruct::get(CTType, Fields), MangledName);
4261 GV->setUnnamedAddr(llvm::GlobalValue::UnnamedAddr::Global);
4262 GV->setSection(
".xdata");
4263 if (GV->isWeakForLinker())
4264 GV->setComdat(CGM.
getModule().getOrInsertComdat(GV->getName()));
4265 return getImageRelativeConstant(GV);
4268llvm::GlobalVariable *MicrosoftCXXABI::getCatchableTypeArray(
QualType T) {
4272 llvm::GlobalVariable *&CTA = CatchableTypeArrays[
T];
4297 if (MostDerivedClass) {
4304 Classes.front().initialize(
nullptr,
nullptr);
4306 for (
const MSRTTIClass &Class : Classes) {
4309 (MSRTTIClass::IsPrivateOnPath | MSRTTIClass::IsAmbiguous))
4312 uint32_t OffsetInVBTable = 0;
4313 int32_t VBPtrOffset = -1;
4314 if (
Class.VirtualRoot) {
4325 CatchableTypes.insert(getCatchableType(RTTITy,
Class.OffsetInVBase,
4326 VBPtrOffset, OffsetInVBTable));
4334 CatchableTypes.insert(getCatchableType(
T));
4347 CatchableTypes.insert(getCatchableType(getContext().VoidPtrTy));
4358 CatchableTypes.insert(getCatchableType(getContext().VoidPtrTy));
4360 uint32_t NumEntries = CatchableTypes.size();
4361 llvm::Type *CTType =
4362 getImageRelativeType(getCatchableTypeType()->getPointerTo());
4363 llvm::ArrayType *AT = llvm::ArrayType::get(CTType, NumEntries);
4364 llvm::StructType *CTAType = getCatchableTypeArrayType(NumEntries);
4365 llvm::Constant *Fields[] = {
4366 llvm::ConstantInt::get(CGM.
IntTy, NumEntries),
4367 llvm::ConstantArray::get(
4369 CatchableTypes.end()))
4373 llvm::raw_svector_ostream Out(MangledName);
4374 getMangleContext().mangleCXXCatchableTypeArray(
T, NumEntries, Out);
4376 CTA =
new llvm::GlobalVariable(
4377 CGM.
getModule(), CTAType,
true, getLinkageForRTTI(
T),
4378 llvm::ConstantStruct::get(CTAType, Fields), MangledName);
4379 CTA->setUnnamedAddr(llvm::GlobalValue::UnnamedAddr::Global);
4380 CTA->setSection(
".xdata");
4381 if (CTA->isWeakForLinker())
4382 CTA->setComdat(CGM.
getModule().getOrInsertComdat(CTA->getName()));
4386llvm::GlobalVariable *MicrosoftCXXABI::getThrowInfo(
QualType T) {
4387 bool IsConst, IsVolatile, IsUnaligned;
4392 llvm::GlobalVariable *CTA = getCatchableTypeArray(
T);
4397 uint32_t NumEntries =
4398 cast<llvm::ConstantInt>(CTA->getInitializer()->getAggregateElement(0
U))
4399 ->getLimitedValue();
4403 llvm::raw_svector_ostream Out(MangledName);
4404 getMangleContext().mangleCXXThrowInfo(
T, IsConst, IsVolatile, IsUnaligned,
4410 if (llvm::GlobalVariable *GV = CGM.
getModule().getNamedGlobal(MangledName))
4426 llvm::Constant *CleanupFn = llvm::Constant::getNullValue(CGM.
Int8PtrTy);
4429 if (!DtorD->isTrivial())
4432 llvm::Constant *ForwardCompat =
4433 getImageRelativeConstant(llvm::Constant::getNullValue(CGM.
Int8PtrTy));
4434 llvm::Constant *PointerToCatchableTypes = getImageRelativeConstant(CTA);
4435 llvm::StructType *TIType = getThrowInfoType();
4436 llvm::Constant *Fields[] = {
4437 llvm::ConstantInt::get(CGM.
IntTy, Flags),
4438 getImageRelativeConstant(CleanupFn),
4440 PointerToCatchableTypes
4442 auto *GV =
new llvm::GlobalVariable(
4443 CGM.
getModule(), TIType,
true, getLinkageForRTTI(
T),
4444 llvm::ConstantStruct::get(TIType, Fields), MangledName.str());
4445 GV->setUnnamedAddr(llvm::GlobalValue::UnnamedAddr::Global);
4446 GV->setSection(
".xdata");
4447 if (GV->isWeakForLinker())
4448 GV->setComdat(CGM.
getModule().getOrInsertComdat(GV->getName()));
4453 const Expr *SubExpr =
E->getSubExpr();
4454 assert(SubExpr &&
"SubExpr cannot be null");
4464 llvm::GlobalVariable *TI = getThrowInfo(ThrowType);
4471std::pair<llvm::Value *, const CXXRecordDecl *>
4474 std::tie(This, std::ignore, RD) =
4479bool MicrosoftCXXABI::isPermittedToBeHomogeneousAggregate(
4511 if (
const CXXRecordDecl *FRD = B.getType()->getAsCXXRecordDecl()) {
4512 if (!isPermittedToBeHomogeneousAggregate(FRD))
static llvm::FunctionCallee getThrowFn(CodeGenModule &CGM)
static void emitTlsGuardCheck(CodeGenFunction &CGF, llvm::GlobalValue *TlsGuard, llvm::BasicBlock *DynInitBB, llvm::BasicBlock *ContinueBB)
static llvm::GlobalVariable * getTypeInfoVTable(CodeGenModule &CGM)
static bool hasDefaultCXXMethodCC(ASTContext &Context, const CXXMethodDecl *MD)
static bool isTrivialForMSVC(const CXXRecordDecl *RD, QualType Ty, CodeGenModule &CGM)
static llvm::FunctionCallee getInitThreadAbortFn(CodeGenModule &CGM)
static llvm::FunctionCallee getInitThreadHeaderFn(CodeGenModule &CGM)
static llvm::GlobalValue * getTlsGuardVar(CodeGenModule &CGM)
static QualType decomposeTypeForEH(ASTContext &Context, QualType T, bool &IsConst, bool &IsVolatile, bool &IsUnaligned)
static llvm::CallBase * emitRTtypeidCall(CodeGenFunction &CGF, llvm::Value *Argument)
static llvm::FunctionCallee getDynTlsOnDemandInitFn(CodeGenModule &CGM)
static void emitDynamicTlsInitializationCall(CodeGenFunction &CGF, llvm::GlobalValue *TlsGuard, llvm::BasicBlock *ContinueBB)
static void detectAmbiguousBases(SmallVectorImpl< MSRTTIClass > &Classes)
Find ambiguity among base classes.
static void emitDynamicTlsInitialization(CodeGenFunction &CGF)
static void serializeClassHierarchy(SmallVectorImpl< MSRTTIClass > &Classes, const CXXRecordDecl *RD)
Recursively serializes a class hierarchy in pre-order depth first order.
static llvm::FunctionCallee getInitThreadFooterFn(CodeGenModule &CGM)
static bool isDeletingDtor(GlobalDecl GD)
static void emitGlobalDtorWithTLRegDtor(CodeGenFunction &CGF, const VarDecl &VD, llvm::FunctionCallee Dtor, llvm::Constant *Addr)
static ConstantAddress getInitThreadEpochPtr(CodeGenModule &CGM)
static void mangleVFTableName(MicrosoftMangleContext &MangleContext, const CXXRecordDecl *RD, const VPtrInfo &VFPtr, SmallString< 256 > &Name)
Defines the C++ Decl subclasses, other than those for templates (found in DeclTemplate....
llvm::DenseSet< const void * > Visited
const NestedNameSpecifier * Specifier
APValue - This class implements a discriminated union of [uninitialized] [APSInt] [APFloat],...
bool isMemberPointerToDerivedMember() const
const ValueDecl * getMemberPointerDecl() const
ArrayRef< const CXXRecordDecl * > getMemberPointerPath() const
Holds long-lived AST nodes (such as types and decls) that can be referred to throughout the semantic ...
CharUnits getTypeAlignInChars(QualType T) const
Return the ABI-specified alignment of a (complete) type T, in characters.
uint64_t getFieldOffset(const ValueDecl *FD) const
Get the offset of a FieldDecl or IndirectFieldDecl, in bits.
QualType getMemberPointerType(QualType T, const Type *Cls) const
Return the uniqued reference to the type for a member pointer to the specified type in the specified ...
const ASTRecordLayout & getASTRecordLayout(const RecordDecl *D) const
Get or compute information about the layout of the specified record (struct/union/class) D,...
CallingConv getDefaultCallingConvention(bool IsVariadic, bool IsCXXMethod, bool IsBuiltin=false) const
Retrieves the default calling convention for the current target.
QualType getPointerType(QualType T) const
Return the uniqued reference to the type for a pointer to the specified type.
QualType getTypeDeclType(const TypeDecl *Decl, const TypeDecl *PrevDecl=nullptr) const
Return the unique reference to the type for the specified type declaration.
const LangOptions & getLangOpts() const
GVALinkage GetGVALinkageForFunction(const FunctionDecl *FD) const
CanQualType getSizeType() const
Return the unique type for "size_t" (C99 7.17), defined in <stddef.h>.
CharUnits getDeclAlign(const Decl *D, bool ForAlignof=false) const
Return a conservative estimate of the alignment of the specified decl D.
CharUnits getTypeSizeInChars(QualType T) const
Return the size of the specified (complete) type T, in characters.
QualType getExceptionObjectType(QualType T) const
const CXXConstructorDecl * getCopyConstructorForExceptionObject(CXXRecordDecl *RD)
const TargetInfo & getTargetInfo() const
CharUnits toCharUnitsFromBits(int64_t BitSize) const
Convert a size in bits to a size in characters.
ASTRecordLayout - This class contains layout information for one RecordDecl, which is a struct/union/...
llvm::DenseMap< const CXXRecordDecl *, VBaseInfo > VBaseOffsetsMapTy
CharUnits getVBPtrOffset() const
getVBPtrOffset - Get the offset for virtual base table pointer.
CharUnits getBaseClassOffset(const CXXRecordDecl *Base) const
getBaseClassOffset - Get the offset, in chars, for the given base class.
CharUnits getVBaseClassOffset(const CXXRecordDecl *VBase) const
getVBaseClassOffset - Get the offset, in chars, for the given base class.
const VBaseOffsetsMapTy & getVBaseOffsetsMap() const
bool hasExtendableVFPtr() const
hasVFPtr - Does this class have a virtual function table pointer that can be extended by a derived cl...
Represents a base class of a C++ class.
CXXCatchStmt - This represents a C++ catch block.
Represents a C++ constructor within a class.
Represents a delete expression for memory deallocation and destructor calls, e.g.
FunctionDecl * getOperatorDelete() const
bool isGlobalDelete() const
Represents a C++ destructor within a class.
Represents a call to a member function that may be written either with member call syntax (e....
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.
Represents a new-expression for memory allocation and constructor calls, e.g: "new CXXNewExpr(foo)".
Represents a C++ struct/union/class.
bool hasNonTrivialCopyAssignment() const
Determine whether this class has a non-trivial copy assignment operator (C++ [class....
bool hasPrivateFields() const
bool hasProtectedFields() const
bool hasNonTrivialDestructor() const
Determine whether this class has a non-trivial destructor (C++ [class.dtor]p3)
bool isPolymorphic() const
Whether this class is polymorphic (C++ [class.virtual]), which means that the class contains or inher...
unsigned getNumBases() const
Retrieves the number of base classes of this class.
CXXRecordDecl * getMostRecentNonInjectedDecl()
base_class_range vbases()
MSInheritanceModel getMSInheritanceModel() const
Returns the inheritance model used for this record.
bool nullFieldOffsetIsZero() const
In the Microsoft C++ ABI, use zero for the field offset of a null data member pointer if we can guara...
bool hasDefinition() const
bool isEmpty() const
Determine whether this is an empty class in the sense of (C++11 [meta.unary.prop]).
CXXDestructorDecl * getDestructor() const
Returns the destructor decl for this class.
bool hasNonTrivialDefaultConstructor() const
Determine whether this class has a non-trivial default constructor (C++11 [class.ctor]p5).
bool isVirtuallyDerivedFrom(const CXXRecordDecl *Base) const
Determine whether this class is virtually derived from the class Base.
bool needsImplicitCopyAssignment() const
Determine whether this class needs an implicit copy assignment operator to be lazily declared.
bool hasSimpleCopyAssignment() const
true if we know for sure that this class has a single, accessible, unambiguous copy assignment operat...
unsigned getNumVBases() const
Retrieves the number of virtual base classes of this class.
A C++ throw-expression (C++ [except.throw]).
CastExpr - Base class for type casts, including both implicit casts (ImplicitCastExpr) and explicit c...
const CXXBaseSpecifier *const * path_const_iterator
CharUnits - This is an opaque type for sizes expressed in character units.
bool isPositive() const
isPositive - Test whether the quantity is greater than zero.
bool isZero() const
isZero - Test whether the quantity equals zero.
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.
static CharUnits fromQuantity(QuantityType Quantity)
fromQuantity - Construct a CharUnits quantity from a raw integer type.
static CharUnits Zero()
Zero - Construct a CharUnits quantity of zero.
bool hasProfileIRInstr() const
Check if IR level profile instrumentation is on.
static ABIArgInfo getIndirect(CharUnits Alignment, bool ByVal=true, bool Realign=false, llvm::Type *Padding=nullptr)
void setSRetAfterThis(bool AfterThis)
bool isHomogeneousAggregate(QualType Ty, const Type *&Base, uint64_t &Members) const
isHomogeneousAggregate - Return true if a type is an ELFv2 homogeneous aggregate.
Like RawAddress, an abstract representation of an aligned address, but the pointer contained in this ...
llvm::Value * emitRawPointer(CodeGenFunction &CGF) const
Return the pointer contained in this class after authenticating it and adding offset to it if necessa...
Address withElementType(llvm::Type *ElemTy) const
Return address with different element type, but same pointer and alignment.
A scoped helper to set the current debug location to the specified location or preferred location of ...
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::StoreInst * CreateStore(llvm::Value *Val, Address Addr, bool IsVolatile=false)
Address CreateConstInBoundsByteGEP(Address Addr, CharUnits Offset, const llvm::Twine &Name="")
Given a pointer to i8, adjust it by a given constant offset.
Address CreateConstInBoundsGEP2_32(Address Addr, unsigned Idx0, unsigned Idx1, const llvm::Twine &Name="")
Address CreateGEP(CodeGenFunction &CGF, Address Addr, llvm::Value *Index, const llvm::Twine &Name="")
llvm::LoadInst * CreateLoad(Address Addr, const llvm::Twine &Name="")
Address CreateConstByteGEP(Address Addr, CharUnits Offset, const llvm::Twine &Name="")
llvm::LoadInst * CreateAlignedLoad(llvm::Type *Ty, llvm::Value *Addr, CharUnits Align, const llvm::Twine &Name="")
Address CreateInBoundsGEP(Address Addr, ArrayRef< llvm::Value * > IdxList, llvm::Type *ElementType, CharUnits Align, const Twine &Name="")
Implements C++ ABI-specific code generation functions.
virtual bool shouldEmitExactDynamicCast(QualType DestRecordTy)=0
llvm::Value *& getStructorImplicitParamValue(CodeGenFunction &CGF)
virtual void EmitCXXConstructors(const CXXConstructorDecl *D)=0
Emit constructor variants required by this ABI.
virtual llvm::Constant * getAddrOfRTTIDescriptor(QualType Ty)=0
virtual bool hasMostDerivedReturn(GlobalDecl GD) const
virtual bool HasThisReturn(GlobalDecl GD) const
Returns true if the given constructor or destructor is one of the kinds that the ABI says returns 'th...
virtual llvm::Value * getVTableAddressPointInStructor(CodeGenFunction &CGF, const CXXRecordDecl *RD, BaseSubobject Base, const CXXRecordDecl *NearestVBase)=0
Get the address point of the vtable for the given base subobject while building a constructor or a de...
virtual void emitBeginCatch(CodeGenFunction &CGF, const CXXCatchStmt *C)=0
virtual std::vector< CharUnits > getVBPtrOffsets(const CXXRecordDecl *RD)
Gets the offsets of all the virtual base pointers in a given class.
virtual void emitRethrow(CodeGenFunction &CGF, bool isNoReturn)=0
virtual void initializeHiddenVirtualInheritanceMembers(CodeGenFunction &CGF, const CXXRecordDecl *RD)
Emit the code to initialize hidden members required to handle virtual inheritance,...
virtual bool isMemberPointerConvertible(const MemberPointerType *MPT) const
Return whether or not a member pointers type is convertible to an IR type.
virtual size_t getSrcArgforCopyCtor(const CXXConstructorDecl *, FunctionArgList &Args) const =0
virtual bool isVirtualOffsetNeededForVTableField(CodeGenFunction &CGF, CodeGenFunction::VPtr Vptr)=0
Checks if ABI requires extra virtual offset for vtable field.
virtual void EmitGuardedInit(CodeGenFunction &CGF, const VarDecl &D, llvm::GlobalVariable *DeclPtr, bool PerformInit)=0
Emits the guarded initializer and destructor setup for the given variable, given that it couldn't be ...
virtual void EmitCXXDestructors(const CXXDestructorDecl *D)=0
Emit destructor variants required by this ABI.
virtual void EmitInstanceFunctionProlog(CodeGenFunction &CGF)=0
Emit the ABI-specific prolog for the function.
virtual bool useThunkForDtorVariant(const CXXDestructorDecl *Dtor, CXXDtorType DT) const =0
Returns true if the given destructor type should be emitted as a linkonce delegating thunk,...
virtual CharUnits getVirtualFunctionPrologueThisAdjustment(GlobalDecl GD)
Get the ABI-specific "this" parameter adjustment to apply in the prologue of a virtual function.
virtual void setCXXDestructorDLLStorage(llvm::GlobalValue *GV, const CXXDestructorDecl *Dtor, CXXDtorType DT) const
RecordArgABI
Specify how one should pass an argument of a record type.
virtual bool shouldTypeidBeNullChecked(QualType SrcRecordTy)=0
virtual llvm::Type * ConvertMemberPointerType(const MemberPointerType *MPT)
Find the LLVM type used to represent the given member pointer type.
virtual llvm::Value * performThisAdjustment(CodeGenFunction &CGF, Address This, const CXXRecordDecl *UnadjustedClass, const ThunkInfo &TI)=0
virtual llvm::Constant * EmitNullMemberPointer(const MemberPointerType *MPT)
Create a null member pointer of the given type.
virtual StringRef GetPureVirtualCallName()=0
Gets the pure virtual member call function.
virtual CharUnits getArrayCookieSizeImpl(QualType elementType)
Returns the extra size required in order to store the array cookie for the given type.
virtual bool isSRetParameterAfterThis() const
Returns true if the implicit 'sret' parameter comes after the implicit 'this' parameter of C++ instan...
virtual void registerGlobalDtor(CodeGenFunction &CGF, const VarDecl &D, llvm::FunctionCallee Dtor, llvm::Constant *Addr)=0
Emit code to force the execution of a destructor during global teardown.
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 StringRef GetDeletedVirtualCallName()=0
Gets the deleted virtual member call name.
virtual llvm::Value * EmitMemberPointerIsNotNull(CodeGenFunction &CGF, llvm::Value *MemPtr, const MemberPointerType *MPT)
Determine if a member pointer is non-null. Returns an i1.
virtual llvm::Value * performReturnAdjustment(CodeGenFunction &CGF, Address Ret, const CXXRecordDecl *UnadjustedClass, const ReturnAdjustment &RA)=0
virtual LValue EmitThreadLocalVarDeclLValue(CodeGenFunction &CGF, const VarDecl *VD, QualType LValType)=0
Emit a reference to a non-local thread_local variable (including triggering the initialization of all...
bool isEmittedWithConstantInitializer(const VarDecl *VD, bool InspectInitForWeakDef=false) const
Determine whether we will definitely emit this variable with a constant initializer,...
virtual llvm::Value * EmitMemberPointerComparison(CodeGenFunction &CGF, llvm::Value *L, llvm::Value *R, const MemberPointerType *MPT, bool Inequality)
Emit a comparison between two member pointers. Returns an i1.
virtual llvm::Constant * EmitMemberPointer(const APValue &MP, QualType MPT)
Create a member pointer for the given member pointer constant.
virtual llvm::Constant * getVTableAddressPoint(BaseSubobject Base, const CXXRecordDecl *VTableClass)=0
Get the address point of the vtable for the given base subobject.
virtual void addImplicitStructorParams(CodeGenFunction &CGF, QualType &ResTy, FunctionArgList &Params)=0
Insert any ABI-specific implicit parameters into the parameter list for a function.
virtual llvm::Value * readArrayCookieImpl(CodeGenFunction &IGF, Address ptr, CharUnits cookieSize)
Reads the array cookie for an allocation which is known to have one.
virtual llvm::Value * EmitMemberDataPointerAddress(CodeGenFunction &CGF, const Expr *E, Address Base, llvm::Value *MemPtr, const MemberPointerType *MPT)
Calculate an l-value from an object and a data member pointer.
virtual llvm::Value * getCXXDestructorImplicitParam(CodeGenFunction &CGF, const CXXDestructorDecl *DD, CXXDtorType Type, bool ForVirtualBase, bool Delegating)=0
Get the implicit (second) parameter that comes after the "this" pointer, or nullptr if there is isn't...
virtual bool requiresArrayCookie(const CXXDeleteExpr *E, QualType eltType)
virtual CatchTypeInfo getCatchAllTypeInfo()
virtual std::pair< llvm::Value *, const CXXRecordDecl * > LoadVTablePtr(CodeGenFunction &CGF, Address This, const CXXRecordDecl *RD)=0
Load a vtable from This, an object of polymorphic type RD, or from one of its virtual bases if it doe...
virtual void setThunkLinkage(llvm::Function *Thunk, bool ForVTable, GlobalDecl GD, bool ReturnAdjustment)=0
virtual Address adjustThisArgumentForVirtualFunctionCall(CodeGenFunction &CGF, GlobalDecl GD, Address This, bool VirtualCall)
Perform ABI-specific "this" argument adjustment required prior to a call of a virtual function.
virtual llvm::Value * EmitVirtualDestructorCall(CodeGenFunction &CGF, const CXXDestructorDecl *Dtor, CXXDtorType DtorType, Address This, DeleteOrMemberCallExpr E)=0
Emit the ABI-specific virtual destructor call.
bool mayNeedDestruction(const VarDecl *VD) const
virtual llvm::BasicBlock * EmitCtorCompleteObjectHandler(CodeGenFunction &CGF, const CXXRecordDecl *RD)
virtual bool doStructorsInitializeVPtrs(const CXXRecordDecl *VTableClass)=0
Checks if ABI requires to initialize vptrs for given dynamic class.
virtual void emitThrow(CodeGenFunction &CGF, const CXXThrowExpr *E)=0
virtual llvm::Value * GetVirtualBaseClassOffset(CodeGenFunction &CGF, Address This, const CXXRecordDecl *ClassDecl, const CXXRecordDecl *BaseClassDecl)=0
virtual bool isThisCompleteObject(GlobalDecl GD) const =0
Determine whether there's something special about the rules of the ABI tell us that 'this' is a compl...
virtual CGCallee getVirtualFunctionPointer(CodeGenFunction &CGF, GlobalDecl GD, Address This, llvm::Type *Ty, SourceLocation Loc)=0
Build a virtual function pointer in the ABI-specific way.
virtual bool classifyReturnType(CGFunctionInfo &FI) const =0
If the C++ ABI requires the given type be returned in a particular way, this method sets RetAI and re...
virtual void emitVirtualObjectDelete(CodeGenFunction &CGF, const CXXDeleteExpr *DE, Address Ptr, QualType ElementType, const CXXDestructorDecl *Dtor)=0
virtual CatchTypeInfo getAddrOfCXXCatchHandlerType(QualType Ty, QualType CatchHandlerType)=0
virtual void EmitThreadLocalInitFuncs(CodeGenModule &CGM, ArrayRef< const VarDecl * > CXXThreadLocals, ArrayRef< llvm::Function * > CXXThreadLocalInits, ArrayRef< const VarDecl * > CXXThreadLocalInitVars)=0
Emits ABI-required functions necessary to initialize thread_local variables in this translation unit.
virtual bool usesThreadWrapperFunction(const VarDecl *VD) const =0
virtual RecordArgABI getRecordArgABI(const CXXRecordDecl *RD) const =0
Returns how an argument of the given record type should be passed.
virtual llvm::Value * emitExactDynamicCast(CodeGenFunction &CGF, Address Value, QualType SrcRecordTy, QualType DestTy, QualType DestRecordTy, llvm::BasicBlock *CastSuccess, llvm::BasicBlock *CastFail)=0
Emit a dynamic_cast from SrcRecordTy to DestRecordTy.
virtual const CXXRecordDecl * getThisArgumentTypeForMethod(GlobalDecl GD)
Get the type of the implicit "this" parameter used by a method.
virtual void EmitDestructorCall(CodeGenFunction &CGF, const CXXDestructorDecl *DD, CXXDtorType Type, bool ForVirtualBase, bool Delegating, Address This, QualType ThisTy)=0
Emit the destructor call.
virtual llvm::GlobalVariable * getAddrOfVTable(const CXXRecordDecl *RD, CharUnits VPtrOffset)=0
Get the address of the vtable for the given record decl which should be used for the vptr at the give...
virtual bool EmitBadCastCall(CodeGenFunction &CGF)=0
virtual void adjustCallArgsForDestructorThunk(CodeGenFunction &CGF, GlobalDecl GD, CallArgList &CallArgs)
virtual llvm::Constant * EmitMemberDataPointer(const MemberPointerType *MPT, CharUnits offset)
Create a member pointer for the given field.
virtual llvm::Value * EmitTypeid(CodeGenFunction &CGF, QualType SrcRecordTy, Address ThisPtr, llvm::Type *StdTypeInfoPtrTy)=0
virtual void emitVirtualInheritanceTables(const CXXRecordDecl *RD)=0
Emit any tables needed to implement virtual inheritance.
virtual void emitVTableDefinitions(CodeGenVTables &CGVT, const CXXRecordDecl *RD)=0
Emits the VTable definitions required for the given record type.
virtual CGCallee EmitLoadOfMemberFunctionPointer(CodeGenFunction &CGF, const Expr *E, Address This, llvm::Value *&ThisPtrForCall, llvm::Value *MemPtr, const MemberPointerType *MPT)
Load a member function from an object and a member function pointer.
virtual bool isPermittedToBeHomogeneousAggregate(const CXXRecordDecl *RD) const
Returns true if the ABI permits the argument to be a homogeneous aggregate.
virtual void emitCXXStructor(GlobalDecl GD)=0
Emit a single constructor/destructor with the given type from a C++ constructor Decl.
virtual bool exportThunk()=0
virtual void EmitBadTypeidCall(CodeGenFunction &CGF)=0
virtual llvm::Value * emitDynamicCastToVoid(CodeGenFunction &CGF, Address Value, QualType SrcRecordTy)=0
virtual bool isZeroInitializable(const MemberPointerType *MPT)
Return true if the given member pointer can be zero-initialized (in the C++ sense) with an LLVM zeroi...
virtual llvm::Value * EmitMemberPointerConversion(CodeGenFunction &CGF, const CastExpr *E, llvm::Value *Src)
Perform a derived-to-base, base-to-derived, or bitcast member pointer conversion.
virtual llvm::Constant * EmitMemberFunctionPointer(const CXXMethodDecl *MD)
Create a member pointer for the given method.
virtual llvm::Value * emitDynamicCastCall(CodeGenFunction &CGF, Address Value, QualType SrcRecordTy, QualType DestTy, QualType DestRecordTy, llvm::BasicBlock *CastEnd)=0
virtual llvm::GlobalVariable * getThrowInfo(QualType T)
virtual llvm::GlobalValue::LinkageTypes getCXXDestructorLinkage(GVALinkage Linkage, const CXXDestructorDecl *Dtor, CXXDtorType DT) const
virtual Address InitializeArrayCookie(CodeGenFunction &CGF, Address NewPtr, llvm::Value *NumElements, const CXXNewExpr *expr, QualType ElementType)
Initialize the array cookie for the given allocation.
ASTContext & getContext() const
virtual bool shouldDynamicCastCallBeNullChecked(bool SrcIsPtr, QualType SrcRecordTy)=0
virtual AddedStructorArgCounts buildStructorSignature(GlobalDecl GD, SmallVectorImpl< CanQualType > &ArgTys)=0
Build the signature of the given constructor or destructor variant by adding any required parameters.
MangleContext & getMangleContext()
Gets the mangle context.
virtual AddedStructorArgs getImplicitConstructorArgs(CodeGenFunction &CGF, const CXXConstructorDecl *D, CXXCtorType Type, bool ForVirtualBase, bool Delegating)=0
All available information about a concrete callee.
static CGCallee forVirtual(const CallExpr *CE, GlobalDecl MD, Address Addr, llvm::FunctionType *FTy)
static CGCallee forDirect(llvm::Constant *functionPtr, const CGCalleeInfo &abstractInfo=CGCalleeInfo())
CGFunctionInfo - Class to encapsulate the information about a function definition.
bool isInstanceMethod() const
ABIArgInfo & getReturnInfo()
CanQualType getReturnType() const
unsigned getEffectiveCallingConvention() const
getEffectiveCallingConvention - Return the actual calling convention to use, which may depend on the ...
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 GenerateCXXGlobalInitFunc(llvm::Function *Fn, ArrayRef< llvm::Function * > CXXThreadLocals, ConstantAddress Guard=ConstantAddress::invalid())
GenerateCXXGlobalInitFunc - Generates code for initializing global variables.
void FinishFunction(SourceLocation EndLoc=SourceLocation())
FinishFunction - Complete IR generation of the current function.
GlobalDecl CurGD
CurGD - The GlobalDecl for the current function being compiled.
llvm::Value * GetVTablePtr(Address This, llvm::Type *VTableTy, const CXXRecordDecl *VTableClass, VTableAuthMode AuthMode=VTableAuthMode::Authenticate)
GetVTablePtr - Return the Value of the vtable pointer member pointed to by This.
llvm::Constant * createAtExitStub(const VarDecl &VD, llvm::FunctionCallee Dtor, llvm::Constant *Addr)
void EmitCallArgs(CallArgList &Args, PrototypeWrapper Prototype, llvm::iterator_range< CallExpr::const_arg_iterator > ArgRange, AbstractCallee AC=AbstractCallee(), unsigned ParamsToSkip=0, EvaluationOrder Order=EvaluationOrder::Default)
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)
llvm::Value * getAsNaturalPointerTo(Address Addr, QualType PointeeType)
void EmitDeleteCall(const FunctionDecl *DeleteFD, llvm::Value *Ptr, QualType DeleteTy, llvm::Value *NumElements=nullptr, CharUnits CookieSize=CharUnits())
llvm::BasicBlock * createBasicBlock(const Twine &name="", llvm::Function *parent=nullptr, llvm::BasicBlock *before=nullptr)
createBasicBlock - Create an LLVM basic block.
void EmitBlock(llvm::BasicBlock *BB, bool IsFinished=false)
EmitBlock - Emit the given block.
llvm::Value * EmitVTableTypeCheckedLoad(const CXXRecordDecl *RD, llvm::Value *VTable, llvm::Type *VTableTy, uint64_t VTableByteOffset)
Emit a type checked load from the given vtable.
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,...
void EmitAnyExprToMem(const Expr *E, Address Location, Qualifiers Quals, bool IsInitializer)
EmitAnyExprToMem - Emits the code necessary to evaluate an arbitrary expression into the given memory...
const Decl * CurCodeDecl
CurCodeDecl - This is the inner-most code context, which includes blocks.
RawAddress CreateMemTemp(QualType T, const Twine &Name="tmp", RawAddress *Alloca=nullptr)
CreateMemTemp - Create a temporary memory object of the given type, with appropriate alignmen and cas...
void EmitCXXGlobalVarDeclInit(const VarDecl &D, llvm::GlobalVariable *GV, bool PerformInit)
EmitCXXGlobalVarDeclInit - Create the initializer for a C++ variable with global storage.
bool ShouldEmitVTableTypeCheckedLoad(const CXXRecordDecl *RD)
Returns whether we should perform a type checked load when loading a virtual function for virtual cal...
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.
void registerGlobalDtorWithAtExit(const VarDecl &D, llvm::FunctionCallee fn, llvm::Constant *addr)
Call atexit() with a function that passes the given argument to the given function.
llvm::CallInst * EmitNounwindRuntimeCall(llvm::FunctionCallee callee, const Twine &name="")
ASTContext & getContext() const
const Decl * CurFuncDecl
CurFuncDecl - Holds the Decl for the current outermost non-closure context.
AutoVarEmission EmitAutoVarAlloca(const VarDecl &var)
void EmitAutoVarCleanups(const AutoVarEmission &emission)
void PopCleanupBlock(bool FallThroughIsBranchThrough=false, bool ForDeactivation=false)
PopCleanupBlock - Will pop the cleanup entry on the stack and process all branch fixups.
void EmitTypeMetadataCodeForVCall(const CXXRecordDecl *RD, llvm::Value *VTable, SourceLocation Loc)
If whole-program virtual table optimization is enabled, emit an assumption that VTable is a member of...
llvm::CallInst * EmitRuntimeCall(llvm::FunctionCallee callee, const Twine &name="")
void EmitNoreturnRuntimeCallOrInvoke(llvm::FunctionCallee callee, ArrayRef< llvm::Value * > args)
CodeGenTypes & getTypes() const
llvm::CallBase * EmitRuntimeCallOrInvoke(llvm::FunctionCallee callee, ArrayRef< llvm::Value * > args, const Twine &name="")
LValue MakeAddrLValue(Address Addr, QualType T, AlignmentSource Source=AlignmentSource::Type)
LValue EmitLoadOfReferenceLValue(LValue RefLVal)
Address GetAddrOfLocalVar(const VarDecl *VD)
GetAddrOfLocalVar - Return the address of a local variable.
Address ReturnValue
ReturnValue - The temporary alloca to hold the return value.