
This fixes all the places that hit the new assertion added in https://github.com/llvm/llvm-project/pull/106524 in tests. That is, cases where the value passed to the APInt constructor is not an N-bit signed/unsigned integer, where N is the bit width and signedness is determined by the isSigned flag. The fixes either set the correct value for isSigned, set the implicitTrunc flag, or perform more calculations inside APInt. Note that the assertion is currently still disabled by default, so this patch is mostly NFC.
195 lines
7.7 KiB
C++
195 lines
7.7 KiB
C++
//===--- CGVTT.cpp - Emit LLVM Code for C++ VTTs --------------------------===//
|
|
//
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
//
|
|
// This contains code dealing with C++ code generation of VTTs (vtable tables).
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "CodeGenModule.h"
|
|
#include "CGCXXABI.h"
|
|
#include "clang/AST/RecordLayout.h"
|
|
#include "clang/AST/VTTBuilder.h"
|
|
using namespace clang;
|
|
using namespace CodeGen;
|
|
|
|
static llvm::GlobalVariable *
|
|
GetAddrOfVTTVTable(CodeGenVTables &CGVT, CodeGenModule &CGM,
|
|
const CXXRecordDecl *MostDerivedClass,
|
|
const VTTVTable &VTable,
|
|
llvm::GlobalVariable::LinkageTypes Linkage,
|
|
VTableLayout::AddressPointsMapTy &AddressPoints) {
|
|
if (VTable.getBase() == MostDerivedClass) {
|
|
assert(VTable.getBaseOffset().isZero() &&
|
|
"Most derived class vtable must have a zero offset!");
|
|
// This is a regular vtable.
|
|
return CGM.getCXXABI().getAddrOfVTable(MostDerivedClass, CharUnits());
|
|
}
|
|
|
|
return CGVT.GenerateConstructionVTable(MostDerivedClass,
|
|
VTable.getBaseSubobject(),
|
|
VTable.isVirtual(),
|
|
Linkage,
|
|
AddressPoints);
|
|
}
|
|
|
|
void
|
|
CodeGenVTables::EmitVTTDefinition(llvm::GlobalVariable *VTT,
|
|
llvm::GlobalVariable::LinkageTypes Linkage,
|
|
const CXXRecordDecl *RD) {
|
|
VTTBuilder Builder(CGM.getContext(), RD, /*GenerateDefinition=*/true);
|
|
llvm::ArrayType *ArrayType = llvm::ArrayType::get(
|
|
CGM.GlobalsInt8PtrTy, Builder.getVTTComponents().size());
|
|
|
|
SmallVector<llvm::GlobalVariable *, 8> VTables;
|
|
SmallVector<VTableAddressPointsMapTy, 8> VTableAddressPoints;
|
|
for (const VTTVTable *i = Builder.getVTTVTables().begin(),
|
|
*e = Builder.getVTTVTables().end(); i != e; ++i) {
|
|
VTableAddressPoints.push_back(VTableAddressPointsMapTy());
|
|
VTables.push_back(GetAddrOfVTTVTable(*this, CGM, RD, *i, Linkage,
|
|
VTableAddressPoints.back()));
|
|
}
|
|
|
|
SmallVector<llvm::Constant *, 8> VTTComponents;
|
|
for (const VTTComponent *i = Builder.getVTTComponents().begin(),
|
|
*e = Builder.getVTTComponents().end(); i != e; ++i) {
|
|
const VTTVTable &VTTVT = Builder.getVTTVTables()[i->VTableIndex];
|
|
llvm::GlobalVariable *VTable = VTables[i->VTableIndex];
|
|
VTableLayout::AddressPointLocation AddressPoint;
|
|
if (VTTVT.getBase() == RD) {
|
|
// Just get the address point for the regular vtable.
|
|
AddressPoint =
|
|
getItaniumVTableContext().getVTableLayout(RD).getAddressPoint(
|
|
i->VTableBase);
|
|
} else {
|
|
AddressPoint = VTableAddressPoints[i->VTableIndex].lookup(i->VTableBase);
|
|
assert(AddressPoint.AddressPointIndex != 0 &&
|
|
"Did not find ctor vtable address point!");
|
|
}
|
|
|
|
llvm::Value *Idxs[] = {
|
|
llvm::ConstantInt::get(CGM.Int32Ty, 0),
|
|
llvm::ConstantInt::get(CGM.Int32Ty, AddressPoint.VTableIndex),
|
|
llvm::ConstantInt::get(CGM.Int32Ty, AddressPoint.AddressPointIndex),
|
|
};
|
|
|
|
// Add inrange attribute to indicate that only the VTableIndex can be
|
|
// accessed.
|
|
unsigned ComponentSize =
|
|
CGM.getDataLayout().getTypeAllocSize(getVTableComponentType());
|
|
unsigned VTableSize = CGM.getDataLayout().getTypeAllocSize(
|
|
cast<llvm::StructType>(VTable->getValueType())
|
|
->getElementType(AddressPoint.VTableIndex));
|
|
unsigned Offset = ComponentSize * AddressPoint.AddressPointIndex;
|
|
llvm::ConstantRange InRange(
|
|
llvm::APInt(32, (int)-Offset, true),
|
|
llvm::APInt(32, (int)(VTableSize - Offset), true));
|
|
llvm::Constant *Init = llvm::ConstantExpr::getGetElementPtr(
|
|
VTable->getValueType(), VTable, Idxs, /*InBounds=*/true, InRange);
|
|
|
|
if (const auto &Schema =
|
|
CGM.getCodeGenOpts().PointerAuth.CXXVTTVTablePointers)
|
|
Init = CGM.getConstantSignedPointer(Init, Schema, nullptr, GlobalDecl(),
|
|
QualType());
|
|
|
|
VTTComponents.push_back(Init);
|
|
}
|
|
|
|
llvm::Constant *Init = llvm::ConstantArray::get(ArrayType, VTTComponents);
|
|
|
|
VTT->setInitializer(Init);
|
|
|
|
// Set the correct linkage.
|
|
VTT->setLinkage(Linkage);
|
|
|
|
if (CGM.supportsCOMDAT() && VTT->isWeakForLinker())
|
|
VTT->setComdat(CGM.getModule().getOrInsertComdat(VTT->getName()));
|
|
|
|
// Set the visibility. This will already have been set on the VTT declaration.
|
|
// Set it again, now that we have a definition, as the implicit visibility can
|
|
// apply differently to definitions.
|
|
CGM.setGVProperties(VTT, RD);
|
|
}
|
|
|
|
llvm::GlobalVariable *CodeGenVTables::GetAddrOfVTT(const CXXRecordDecl *RD) {
|
|
assert(RD->getNumVBases() && "Only classes with virtual bases need a VTT");
|
|
|
|
SmallString<256> OutName;
|
|
llvm::raw_svector_ostream Out(OutName);
|
|
cast<ItaniumMangleContext>(CGM.getCXXABI().getMangleContext())
|
|
.mangleCXXVTT(RD, Out);
|
|
StringRef Name = OutName.str();
|
|
|
|
// This will also defer the definition of the VTT.
|
|
(void) CGM.getCXXABI().getAddrOfVTable(RD, CharUnits());
|
|
|
|
VTTBuilder Builder(CGM.getContext(), RD, /*GenerateDefinition=*/false);
|
|
|
|
llvm::ArrayType *ArrayType = llvm::ArrayType::get(
|
|
CGM.GlobalsInt8PtrTy, Builder.getVTTComponents().size());
|
|
llvm::Align Align = CGM.getDataLayout().getABITypeAlign(CGM.GlobalsInt8PtrTy);
|
|
|
|
llvm::GlobalVariable *GV = CGM.CreateOrReplaceCXXRuntimeVariable(
|
|
Name, ArrayType, llvm::GlobalValue::ExternalLinkage, Align);
|
|
GV->setUnnamedAddr(llvm::GlobalValue::UnnamedAddr::Global);
|
|
CGM.setGVProperties(GV, RD);
|
|
return GV;
|
|
}
|
|
|
|
uint64_t CodeGenVTables::getSubVTTIndex(const CXXRecordDecl *RD,
|
|
BaseSubobject Base) {
|
|
BaseSubobjectPairTy ClassSubobjectPair(RD, Base);
|
|
|
|
SubVTTIndicesMapTy::iterator I = SubVTTIndices.find(ClassSubobjectPair);
|
|
if (I != SubVTTIndices.end())
|
|
return I->second;
|
|
|
|
VTTBuilder Builder(CGM.getContext(), RD, /*GenerateDefinition=*/false);
|
|
|
|
for (llvm::DenseMap<BaseSubobject, uint64_t>::const_iterator
|
|
I = Builder.getSubVTTIndices().begin(),
|
|
E = Builder.getSubVTTIndices().end();
|
|
I != E; ++I) {
|
|
// Insert all indices.
|
|
BaseSubobjectPairTy ClassSubobjectPair(RD, I->first);
|
|
|
|
SubVTTIndices.insert(std::make_pair(ClassSubobjectPair, I->second));
|
|
}
|
|
|
|
I = SubVTTIndices.find(ClassSubobjectPair);
|
|
assert(I != SubVTTIndices.end() && "Did not find index!");
|
|
|
|
return I->second;
|
|
}
|
|
|
|
uint64_t
|
|
CodeGenVTables::getSecondaryVirtualPointerIndex(const CXXRecordDecl *RD,
|
|
BaseSubobject Base) {
|
|
SecondaryVirtualPointerIndicesMapTy::iterator I =
|
|
SecondaryVirtualPointerIndices.find(std::make_pair(RD, Base));
|
|
|
|
if (I != SecondaryVirtualPointerIndices.end())
|
|
return I->second;
|
|
|
|
VTTBuilder Builder(CGM.getContext(), RD, /*GenerateDefinition=*/false);
|
|
|
|
// Insert all secondary vpointer indices.
|
|
for (llvm::DenseMap<BaseSubobject, uint64_t>::const_iterator I =
|
|
Builder.getSecondaryVirtualPointerIndices().begin(),
|
|
E = Builder.getSecondaryVirtualPointerIndices().end(); I != E; ++I) {
|
|
std::pair<const CXXRecordDecl *, BaseSubobject> Pair =
|
|
std::make_pair(RD, I->first);
|
|
|
|
SecondaryVirtualPointerIndices.insert(std::make_pair(Pair, I->second));
|
|
}
|
|
|
|
I = SecondaryVirtualPointerIndices.find(std::make_pair(RD, Base));
|
|
assert(I != SecondaryVirtualPointerIndices.end() && "Did not find index!");
|
|
|
|
return I->second;
|
|
}
|