From: John McCall Date: Mon, 28 Nov 2016 22:18:33 +0000 (+0000) Subject: Make CGVTables use ConstantInitBuilder. NFC. X-Git-Url: https://granicus.if.org/sourcecode?a=commitdiff_plain;h=01a821e2b6d7be3e171f07c232a9e9b40e634986;p=clang Make CGVTables use ConstantInitBuilder. NFC. git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@288081 91177308-0d34-0410-b5e6-96231b3b80d8 --- diff --git a/lib/CodeGen/CGVTables.cpp b/lib/CodeGen/CGVTables.cpp index 3999c158d9..be9c311512 100644 --- a/lib/CodeGen/CGVTables.cpp +++ b/lib/CodeGen/CGVTables.cpp @@ -14,6 +14,7 @@ #include "CGCXXABI.h" #include "CodeGenFunction.h" #include "CodeGenModule.h" +#include "ConstantBuilder.h" #include "clang/AST/CXXInheritance.h" #include "clang/AST/RecordLayout.h" #include "clang/CodeGen/CGFunctionInfo.h" @@ -524,29 +525,29 @@ void CodeGenVTables::EmitThunks(GlobalDecl GD) emitThunk(GD, Thunk, /*ForVTable=*/false); } -llvm::Constant *CodeGenVTables::CreateVTableComponent( - unsigned Idx, const VTableLayout &VTLayout, llvm::Constant *RTTI, - unsigned &NextVTableThunkIndex) { - VTableComponent Component = VTLayout.vtable_components()[Idx]; +void CodeGenVTables::addVTableComponent( + ConstantArrayBuilder &builder, const VTableLayout &layout, + unsigned idx, llvm::Constant *rtti, unsigned &nextVTableThunkIndex) { + auto &component = layout.vtable_components()[idx]; - auto OffsetConstant = [&](CharUnits Offset) { - return llvm::ConstantExpr::getIntToPtr( - llvm::ConstantInt::get(CGM.PtrDiffTy, Offset.getQuantity()), - CGM.Int8PtrTy); + auto addOffsetConstant = [&](CharUnits offset) { + builder.add(llvm::ConstantExpr::getIntToPtr( + llvm::ConstantInt::get(CGM.PtrDiffTy, offset.getQuantity()), + CGM.Int8PtrTy)); }; - switch (Component.getKind()) { + switch (component.getKind()) { case VTableComponent::CK_VCallOffset: - return OffsetConstant(Component.getVCallOffset()); + return addOffsetConstant(component.getVCallOffset()); case VTableComponent::CK_VBaseOffset: - return OffsetConstant(Component.getVBaseOffset()); + return addOffsetConstant(component.getVBaseOffset()); case VTableComponent::CK_OffsetToTop: - return OffsetConstant(Component.getOffsetToTop()); + return addOffsetConstant(component.getOffsetToTop()); case VTableComponent::CK_RTTI: - return RTTI; + return builder.add(llvm::ConstantExpr::getBitCast(rtti, CGM.Int8PtrTy)); case VTableComponent::CK_FunctionPointer: case VTableComponent::CK_CompleteDtorPointer: @@ -554,17 +555,17 @@ llvm::Constant *CodeGenVTables::CreateVTableComponent( GlobalDecl GD; // Get the right global decl. - switch (Component.getKind()) { + switch (component.getKind()) { default: llvm_unreachable("Unexpected vtable component kind"); case VTableComponent::CK_FunctionPointer: - GD = Component.getFunctionDecl(); + GD = component.getFunctionDecl(); break; case VTableComponent::CK_CompleteDtorPointer: - GD = GlobalDecl(Component.getDestructorDecl(), Dtor_Complete); + GD = GlobalDecl(component.getDestructorDecl(), Dtor_Complete); break; case VTableComponent::CK_DeletingDtorPointer: - GD = GlobalDecl(Component.getDestructorDecl(), Dtor_Deleting); + GD = GlobalDecl(component.getDestructorDecl(), Dtor_Deleting); break; } @@ -580,68 +581,69 @@ llvm::Constant *CodeGenVTables::CreateVTableComponent( ? MD->hasAttr() : (MD->hasAttr() || !MD->hasAttr()); if (!CanEmitMethod) - return llvm::ConstantExpr::getNullValue(CGM.Int8PtrTy); + return builder.addNullPointer(CGM.Int8PtrTy); // Method is acceptable, continue processing as usual. } - auto SpecialVirtualFn = [&](llvm::Constant *&Cache, StringRef Name) { - if (!Cache) { - llvm::FunctionType *Ty = - llvm::FunctionType::get(CGM.VoidTy, /*isVarArg=*/false); - Cache = CGM.CreateRuntimeFunction(Ty, Name); - if (auto *F = dyn_cast(Cache)) - F->setUnnamedAddr(llvm::GlobalValue::UnnamedAddr::Global); - Cache = llvm::ConstantExpr::getBitCast(Cache, CGM.Int8PtrTy); - } - return Cache; + auto getSpecialVirtualFn = [&](StringRef name) { + llvm::FunctionType *fnTy = + llvm::FunctionType::get(CGM.VoidTy, /*isVarArg=*/false); + llvm::Constant *fn = CGM.CreateRuntimeFunction(fnTy, name); + if (auto f = dyn_cast(fn)) + f->setUnnamedAddr(llvm::GlobalValue::UnnamedAddr::Global); + return llvm::ConstantExpr::getBitCast(fn, CGM.Int8PtrTy); }; - if (cast(GD.getDecl())->isPure()) - // We have a pure virtual member function. - return SpecialVirtualFn(PureVirtualFn, - CGM.getCXXABI().GetPureVirtualCallName()); - - if (cast(GD.getDecl())->isDeleted()) - return SpecialVirtualFn(DeletedVirtualFn, - CGM.getCXXABI().GetDeletedVirtualCallName()); - - // Check if we should use a thunk. - if (NextVTableThunkIndex < VTLayout.vtable_thunks().size() && - VTLayout.vtable_thunks()[NextVTableThunkIndex].first == Idx) { - const ThunkInfo &Thunk = - VTLayout.vtable_thunks()[NextVTableThunkIndex].second; - - maybeEmitThunkForVTable(GD, Thunk); - NextVTableThunkIndex++; - return CGM.GetAddrOfThunk(GD, Thunk); + llvm::Constant *fnPtr; + + // Pure virtual member functions. + if (cast(GD.getDecl())->isPure()) { + if (!PureVirtualFn) + PureVirtualFn = + getSpecialVirtualFn(CGM.getCXXABI().GetPureVirtualCallName()); + fnPtr = PureVirtualFn; + + // Deleted virtual member functions. + } else if (cast(GD.getDecl())->isDeleted()) { + if (!DeletedVirtualFn) + DeletedVirtualFn = + getSpecialVirtualFn(CGM.getCXXABI().GetDeletedVirtualCallName()); + fnPtr = DeletedVirtualFn; + + // Thunks. + } else if (nextVTableThunkIndex < layout.vtable_thunks().size() && + layout.vtable_thunks()[nextVTableThunkIndex].first == idx) { + auto &thunkInfo = layout.vtable_thunks()[nextVTableThunkIndex].second; + + maybeEmitThunkForVTable(GD, thunkInfo); + nextVTableThunkIndex++; + fnPtr = CGM.GetAddrOfThunk(GD, thunkInfo); + + // Otherwise we can use the method definition directly. + } else { + llvm::Type *fnTy = CGM.getTypes().GetFunctionTypeForVTable(GD); + fnPtr = CGM.GetAddrOfFunction(GD, fnTy, /*ForVTable=*/true); } - llvm::Type *Ty = CGM.getTypes().GetFunctionTypeForVTable(GD); - return CGM.GetAddrOfFunction(GD, Ty, /*ForVTable=*/true); + fnPtr = llvm::ConstantExpr::getBitCast(fnPtr, CGM.Int8PtrTy); + builder.add(fnPtr); + return; } case VTableComponent::CK_UnusedFunctionPointer: - return llvm::ConstantExpr::getNullValue(CGM.Int8PtrTy); + return builder.addNullPointer(CGM.Int8PtrTy); } llvm_unreachable("Unexpected vtable component kind"); } -llvm::Constant * -CodeGenVTables::CreateVTableInitializer(const VTableLayout &VTLayout, - llvm::Constant *RTTI) { - SmallVector Inits; - unsigned NextVTableThunkIndex = 0; - - for (unsigned I = 0, E = VTLayout.vtable_components().size(); I != E; ++I) { - llvm::Constant *Init = - CreateVTableComponent(I, VTLayout, RTTI, NextVTableThunkIndex); - Inits.push_back(llvm::ConstantExpr::getBitCast(Init, CGM.Int8PtrTy)); +void CodeGenVTables::createVTableInitializer(ConstantArrayBuilder &builder, + const VTableLayout &layout, + llvm::Constant *rtti) { + unsigned nextVTableThunkIndex = 0; + for (unsigned i = 0, e = layout.vtable_components().size(); i != e; ++i) { + addVTableComponent(builder, layout, i, rtti, nextVTableThunkIndex); } - - llvm::ArrayType *ArrayType = - llvm::ArrayType::get(CGM.Int8PtrTy, VTLayout.vtable_components().size()); - return llvm::ConstantArray::get(ArrayType, Inits); } llvm::GlobalVariable * @@ -691,8 +693,10 @@ CodeGenVTables::GenerateConstructionVTable(const CXXRecordDecl *RD, CGM.getContext().getTagDeclType(Base.getBase())); // Create and set the initializer. - llvm::Constant *Init = CreateVTableInitializer(*VTLayout, RTTI); - VTable->setInitializer(Init); + ConstantInitBuilder builder(CGM); + auto components = builder.beginArray(CGM.Int8PtrTy); + createVTableInitializer(components, *VTLayout, RTTI); + components.finishAndSetAsInitializer(VTable); CGM.EmitVTableTypeMetadata(VTable, *VTLayout.get()); diff --git a/lib/CodeGen/CGVTables.h b/lib/CodeGen/CGVTables.h index be58446dbe..1d06832c46 100644 --- a/lib/CodeGen/CGVTables.h +++ b/lib/CodeGen/CGVTables.h @@ -27,6 +27,7 @@ namespace clang { namespace CodeGen { class CodeGenModule; + class ConstantArrayBuilder; class CodeGenVTables { CodeGenModule &CGM; @@ -62,16 +63,17 @@ class CodeGenVTables { /// the ABI. void maybeEmitThunkForVTable(GlobalDecl GD, const ThunkInfo &Thunk); - llvm::Constant *CreateVTableComponent(unsigned Idx, - const VTableLayout &VTLayout, - llvm::Constant *RTTI, - unsigned &NextVTableThunkIndex); + void addVTableComponent(ConstantArrayBuilder &builder, + const VTableLayout &layout, unsigned idx, + llvm::Constant *rtti, + unsigned &nextVTableThunkIndex); public: - /// CreateVTableInitializer - Create a vtable initializer with the given - /// layout. - llvm::Constant *CreateVTableInitializer(const VTableLayout &VTLayout, - llvm::Constant *RTTI); + /// Add vtable components for the given vtable layout to the given + /// global initializer. + void createVTableInitializer(ConstantArrayBuilder &builder, + const VTableLayout &layout, + llvm::Constant *rtti); CodeGenVTables(CodeGenModule &CGM); diff --git a/lib/CodeGen/ItaniumCXXABI.cpp b/lib/CodeGen/ItaniumCXXABI.cpp index b54dd365b0..1fb7a36e21 100644 --- a/lib/CodeGen/ItaniumCXXABI.cpp +++ b/lib/CodeGen/ItaniumCXXABI.cpp @@ -24,6 +24,7 @@ #include "CGVTables.h" #include "CodeGenFunction.h" #include "CodeGenModule.h" +#include "ConstantBuilder.h" #include "TargetInfo.h" #include "clang/AST/Mangle.h" #include "clang/AST/Type.h" @@ -1479,8 +1480,10 @@ void ItaniumCXXABI::emitVTableDefinitions(CodeGenVTables &CGVT, CGM.GetAddrOfRTTIDescriptor(CGM.getContext().getTagDeclType(RD)); // Create and set the initializer. - llvm::Constant *Init = CGVT.CreateVTableInitializer(VTLayout, RTTI); - VTable->setInitializer(Init); + ConstantInitBuilder Builder(CGM); + auto Components = Builder.beginArray(CGM.Int8PtrTy); + CGVT.createVTableInitializer(Components, VTLayout, RTTI); + Components.finishAndSetAsInitializer(VTable); // Set the correct linkage. VTable->setLinkage(Linkage); diff --git a/lib/CodeGen/MicrosoftCXXABI.cpp b/lib/CodeGen/MicrosoftCXXABI.cpp index c87ac40972..5947fd7ebf 100644 --- a/lib/CodeGen/MicrosoftCXXABI.cpp +++ b/lib/CodeGen/MicrosoftCXXABI.cpp @@ -19,6 +19,7 @@ #include "CGVTables.h" #include "CodeGenModule.h" #include "CodeGenTypes.h" +#include "ConstantBuilder.h" #include "TargetInfo.h" #include "clang/AST/Decl.h" #include "clang/AST/DeclCXX.h" @@ -1584,9 +1585,10 @@ void MicrosoftCXXABI::emitVTableDefinitions(CodeGenVTables &CGVT, [](const VTableComponent &VTC) { return VTC.isRTTIKind(); })) RTTI = getMSCompleteObjectLocator(RD, *Info); - llvm::Constant *Init = CGVT.CreateVTableInitializer(VTLayout, RTTI); - - VTable->setInitializer(Init); + ConstantInitBuilder Builder(CGM); + auto Components = Builder.beginArray(CGM.Int8PtrTy); + CGVT.createVTableInitializer(Components, VTLayout, RTTI); + Components.finishAndSetAsInitializer(VTable); emitVTableTypeMetadata(*Info, RD, VTable); }