17#include "mlir/IR/Types.h"
20#include "llvm/ADT/SmallVector.h"
27 : cgm(cgm), vtContext(cgm.getASTContext().getVTableContext()) {}
36 cir::FuncOp thunkFn,
bool forVTable,
47 cgm.
setDSOLocal(
static_cast<mlir::Operation *
>(thunkFn));
51 thunkFn.setComdat(
true);
55 mlir::Type ptrTy = builder.getUInt8PtrTy();
60mlir::Type CIRGenVTables::getVTableComponentType() {
66 mlir::Type componentType = getVTableComponentType();
68 tys.push_back(cir::ArrayType::get(componentType, layout.
getVTableSize(i)));
72 return cgm.getBuilder().getAnonRecordTy(tys,
false);
87 assert(rd->
isDynamicClass() &&
"Non-dynamic classes have no VTable.");
91 if (cgm.getTarget().getCXXABI().isMicrosoft())
115 return !keyFunction->
hasBody();
125 vtables.generateClassData(rd);
132 cgm.getCXXABI().emitVirtualInheritanceTables(rd);
134 cgm.getCXXABI().emitVTableDefinitions(*
this, rd);
137mlir::Attribute CIRGenVTables::getVTableComponent(
138 const VTableLayout &layout,
unsigned componentIndex, mlir::Attribute rtti,
139 unsigned &nextVTableThunkIndex,
unsigned vtableAddressPoint,
140 bool vtableHasLocalLinkage) {
164 assert((mlir::isa<cir::GlobalViewAttr>(rtti) ||
165 mlir::isa<cir::ConstPtrAttr>(rtti)) &&
166 "expected GlobalViewAttr or ConstPtrAttr");
178 auto getSpecialVirtFn = [&](StringRef name) -> cir::FuncOp {
184 "getVTableComponent for OMP Device NVPTX");
199 fnPtr = pureVirtualFn;
201 if (!deletedVirtualFn)
203 getSpecialVirtFn(cgm.getCXXABI().getDeletedVirtualCallName());
204 fnPtr = deletedVirtualFn;
205 }
else if (nextVTableThunkIndex < layout.
vtable_thunks().size() &&
208 const ThunkInfo &thunkInfo =
210 nextVTableThunkIndex++;
215 cir::FuncType fnTy = cgm.getTypes().getFunctionType(gd);
216 fnPtr = cgm.getAddrOfFunction(gd, fnTy,
true);
219 return cir::GlobalViewAttr::get(
221 mlir::FlatSymbolRefAttr::get(fnPtr.getSymNameAttr()));
225 llvm_unreachable(
"Unexpected vtable component kind");
230 mlir::Attribute rtti,
231 bool vtableHasLocalLinkage) {
232 mlir::Type componentType = getVTableComponentType();
236 unsigned nextVTableThunkIndex = 0;
238 mlir::MLIRContext *mlirContext = &cgm.getMLIRContext();
241 for (
auto [vtableIndex, addressPoint] : llvm::enumerate(addressPoints)) {
244 size_t vtableEnd = vtableStart + layout.
getVTableSize(vtableIndex);
246 components.reserve(vtableEnd - vtableStart);
247 for (
size_t componentIndex : llvm::seq(vtableStart, vtableEnd))
248 components.push_back(
249 getVTableComponent(layout, componentIndex, rtti, nextVTableThunkIndex,
250 addressPoint, vtableHasLocalLinkage));
252 auto arr = cir::ConstArrayAttr::get(
253 cir::ArrayType::get(componentType, components.size()),
254 mlir::ArrayAttr::get(mlirContext, components));
255 vtables.push_back(arr);
259 const auto members = mlir::ArrayAttr::get(mlirContext, vtables);
260 cir::ConstRecordAttr record = cgm.getBuilder().getAnonConstRecord(members);
263 auto vtableAttr = cir::VTableAttr::get(record.getType(), record.getMembers());
266 cgm.setInitializer(vtableOp, vtableAttr);
271 cir::GlobalLinkageKind linkage, VTableAddressPointsMapTy &addressPoints) {
274 std::unique_ptr<VTableLayout> vtLayout(
279 addressPoints = vtLayout->getAddressPoints();
283 llvm::raw_svector_ostream
out(outName);
298 if (linkage == cir::GlobalLinkageKind::AvailableExternallyLinkage)
299 linkage = cir::GlobalLinkageKind::InternalLinkage;
301 llvm::Align align = cgm.getDataLayout().getABITypeAlign(vtType);
305 cir::GlobalOp vtable = cgm.createOrReplaceCXXRuntimeVariable(
311 mlir::Attribute rtti = cgm.getAddrOfRTTIDescriptor(
312 loc, cgm.getASTContext().getCanonicalTagType(base.
getBase()));
320 assert(!vtable.isDeclaration() &&
"Shouldn't set properties on declaration");
321 cgm.setGVProperties(vtable, rd);
337 return cir::GlobalLinkageKind::InternalLinkage;
341 const CXXMethodDecl *keyFunction = astContext.getCurrentKeyFunction(rd);
342 if (keyFunction && !rd->
hasAttr<DLLImportAttr>()) {
355 (def || codeGenOpts.OptimizationLevel > 0 ||
356 codeGenOpts.getDebugInfo() != llvm::codegenoptions::NoDebugInfo) &&
357 "Shouldn't query vtable linkage without key function, "
358 "optimizations, or debug info");
359 if (!def && codeGenOpts.OptimizationLevel > 0)
360 return cir::GlobalLinkageKind::AvailableExternallyLinkage;
363 return !astContext.getLangOpts().AppleKext
364 ? cir::GlobalLinkageKind::LinkOnceODRLinkage
365 : cir::GlobalLinkageKind::InternalLinkage;
366 return cir::GlobalLinkageKind::ExternalLinkage;
369 return cir::GlobalLinkageKind::LinkOnceODRLinkage;
372 return cir::GlobalLinkageKind::WeakODRLinkage;
375 return !def ? cir::GlobalLinkageKind::AvailableExternallyLinkage
376 : cir::GlobalLinkageKind::ExternalLinkage;
381 if (astContext.getLangOpts().AppleKext)
382 return cir::GlobalLinkageKind::InternalLinkage;
384 auto discardableODRLinkage = cir::GlobalLinkageKind::LinkOnceODRLinkage;
385 auto nonDiscardableODRLinkage = cir::GlobalLinkageKind::WeakODRLinkage;
386 if (rd->
hasAttr<DLLExportAttr>()) {
388 discardableODRLinkage = nonDiscardableODRLinkage;
389 }
else if (rd->
hasAttr<DLLImportAttr>()) {
391 discardableODRLinkage = cir::GlobalLinkageKind::AvailableExternallyLinkage;
392 nonDiscardableODRLinkage =
393 cir::GlobalLinkageKind::AvailableExternallyLinkage;
400 return discardableODRLinkage;
406 return discardableODRLinkage;
408 ? cir::GlobalLinkageKind::AvailableExternallyLinkage
409 : cir::GlobalLinkageKind::ExternalLinkage;
412 return nonDiscardableODRLinkage;
415 llvm_unreachable(
"Invalid TemplateSpecializationKind!");
419 assert(rd->
getNumVBases() &&
"Only classes with virtual bases need a VTT");
422 llvm::raw_svector_ostream
out(outName);
424 .mangleCXXVTT(rd,
out);
425 StringRef name = outName.str();
428 (void)cgm.getCXXABI().getAddrOfVTable(rd,
CharUnits());
430 VTTBuilder builder(cgm.getASTContext(), rd,
false);
432 auto arrayType = cir::ArrayType::get(cgm.getBuilder().getUInt8PtrTy(),
435 cgm.getDataLayout().getABITypeAlign(cgm.getBuilder().getUInt8PtrTy());
436 cir::GlobalOp vtt = cgm.createOrReplaceCXXRuntimeVariable(
439 cgm.setGVProperties(vtt, rd);
446 const VTTVTable &vtable, cir::GlobalLinkageKind linkage,
448 if (vtable.
getBase() == mostDerivedClass) {
450 "Most derived class vtable must have a zero offset!");
461 cir::GlobalLinkageKind linkage,
463 VTTBuilder builder(cgm.getASTContext(), rd,
true);
465 mlir::MLIRContext *mlirContext = &cgm.getMLIRContext();
467 auto arrayType = cir::ArrayType::get(cgm.getBuilder().getUInt8PtrTy(),
473 vtableAddressPoints.push_back(VTableAddressPointsMapTy());
475 vtableAddressPoints.back()));
481 cir::GlobalOp vtable = vtables[vttComponent.VTableIndex];
487 vttComponent.VTableBase);
489 addressPoint = vtableAddressPoints[vttComponent.VTableIndex].lookup(
490 vttComponent.VTableBase);
492 "Did not find ctor vtable address point!");
495 mlir::Attribute indices[2] = {
496 cgm.getBuilder().getI32IntegerAttr(addressPoint.
VTableIndex),
500 auto indicesAttr = mlir::ArrayAttr::get(mlirContext, indices);
501 cir::GlobalViewAttr init = cgm.getBuilder().getGlobalViewAttr(
502 cgm.getBuilder().getUInt8PtrTy(), vtable, indicesAttr);
504 vttComponents.push_back(init);
507 auto init = cir::ConstArrayAttr::get(
508 arrayType, mlir::ArrayAttr::get(mlirContext, vttComponents));
510 vttOp.setInitialValueAttr(init);
513 vttOp.setLinkage(linkage);
514 mlir::SymbolTable::setSymbolVisibility(
517 if (cgm.supportsCOMDAT() && vttOp.isWeakForLinker())
518 vttOp.setComdat(
true);
523 BaseSubobjectPairTy classSubobjectPair(rd, base);
525 SubVTTIndiciesMapTy::iterator it = subVTTIndicies.find(classSubobjectPair);
526 if (it != subVTTIndicies.end())
529 VTTBuilder builder(cgm.getASTContext(), rd,
false);
533 BaseSubobjectPairTy subclassSubobjectPair(rd, entry.first);
535 subVTTIndicies.insert(std::make_pair(subclassSubobjectPair, entry.second));
538 it = subVTTIndicies.find(classSubobjectPair);
539 assert(it != subVTTIndicies.end() &&
"Did not find index!");
546 auto it = secondaryVirtualPointerIndices.find(std::make_pair(rd, base));
548 if (it != secondaryVirtualPointerIndices.end())
551 VTTBuilder builder(cgm.getASTContext(), rd,
false);
555 std::pair<const CXXRecordDecl *, BaseSubobject> pair =
556 std::make_pair(rd, entry.first);
558 secondaryVirtualPointerIndices.insert(std::make_pair(pair, entry.second));
561 it = secondaryVirtualPointerIndices.find(std::make_pair(rd, base));
562 assert(it != secondaryVirtualPointerIndices.end() &&
"Did not find index!");
572 mlir::Value returnValue = rv.
getValue();
579 mlir::Location loc = returnValue.getLoc();
581 if (!nullCheckValue) {
583 cgf,
Address(returnValue, pointeeType, classAlign), classDecl,
590 cir::TernaryOp::create(
591 builder, loc, isNotNull,
592 [&](mlir::OpBuilder &, mlir::Location) {
594 cgf,
Address(returnValue, pointeeType, classAlign), classDecl,
598 [&](mlir::OpBuilder &, mlir::Location) {
599 mlir::Value nullVal =
600 builder.
getNullPtr(returnValue.getType(), loc).getResult();
610 bool isUnprototyped) {
611 assert(!
curGD.getDecl() &&
"curGD was already set!");
620 resultType =
cgm.getASTContext().VoidTy;
621 else if (
cgm.getCXXABI().hasThisReturn(gd))
622 resultType = thisType;
623 else if (
cgm.getCXXABI().hasMostDerivedReturn(gd))
624 resultType =
cgm.getASTContext().VoidPtrTy;
630 cgm.getCXXABI().buildThisParam(*
this, functionArgs);
633 if (!isUnprototyped) {
637 cgm.getCXXABI().addImplicitStructorParams(*
this, resultType,
644 cir::FuncType funcType =
cgm.getTypes().getFunctionType(fnInfo);
652 cgm.getCXXABI().emitInstanceFunctionProlog(md->
getLocation(), *
this);
669 bool isUnprototyped) {
671 "Please use a new CGF for this thunk");
680 mlir::Value adjustedThisPtr =
682 thisValueClass, *thunk)
693 cgm.errorUnsupported(
694 md,
"return-adjusting thunk with incomplete parameter type");
696 llvm_unreachable(
"shouldn't try to emit musttail return-adjusting "
697 "thunks for variadic functions");
699 cgm.errorUnsupported(
700 md,
"non-trivial argument copy for return-adjusting thunk");
709 callArgs.add(
RValue::get(adjustedThisPtr), thisType);
712 cgm.getCXXABI().adjustCallArgsForDestructorThunk(*
this,
curGD, callArgs);
715 unsigned prefixArgs = callArgs.size() - 1;
728 assert(callFnInfo.argTypeSize() ==
curFnInfo->argTypeSize());
733 :
cgm.getCXXABI().hasMostDerivedReturn(
curGD)
734 ?
cgm.getASTContext().VoidPtrTy
745 mlir::Location loc = builder.getUnknownLoc();
757 cgm.getCXXABI().emitReturnFromThunk(*
this, rv, resultType);
766 mlir::Value adjustedThisPtr,
767 cir::FuncOp callee) {
772 for (mlir::BlockArgument arg : entryBlock->getArguments())
776 assert(!args.empty() &&
"thunk must have at least 'this' argument");
777 if (adjustedThisPtr.getType() != args[0].getType())
778 adjustedThisPtr = builder.createBitcast(adjustedThisPtr, args[0].
getType());
779 args[0] = adjustedThisPtr;
781 mlir::Location loc =
curFn->getLoc();
782 cir::FuncType calleeTy = callee.getFunctionType();
783 mlir::Type retTy = calleeTy.getReturnType();
785 cir::CallOp call = builder.createCallOp(loc, callee, args);
786 call->setAttr(cir::CIRDialect::getMustTailAttrName(),
787 mlir::UnitAttr::get(builder.getContext()));
790 cir::ReturnOp::create(builder, loc);
792 cir::ReturnOp::create(builder, loc, call->getResult(0));
800 bool isUnprototyped) {
803 assert(fn.isDeclaration() &&
"Function already has body?");
804 mlir::Block *entryBb = fn.addEntryBlock();
805 builder.setInsertionPointToStart(entryBb);
825 cgm.errorNYI(
"unprototyped thunk placeholder type");
827 ty =
cgm.getTypes().getFunctionType(fnInfo);
829 cir::FuncOp calleeOp =
cgm.getAddrOfFunction(gd, ty,
true);
836 bool isUnprototyped,
bool forVTable) {
860 llvm::raw_svector_ostream
out(name);
868 if (cgm.getASTContext().useAbbreviatedThunkName(gd, name.str())) {
877 cir::FuncType thunkVTableTy = cgm.getTypes().getFunctionType(gd);
878 cir::FuncOp thunk = cgm.getAddrOfThunk(name, thunkVTableTy, gd);
881 bool isUnprototyped = !cgm.getTypes().isFuncTypeConvertible(
889 isUnprototyped ? (cgm.errorNYI(
"unprototyped must-tail thunk"),
890 cgm.getTypes().arrangeGlobalDeclaration(gd))
891 : cgm.getTypes().arrangeGlobalDeclaration(gd);
892 cir::FuncType thunkFnTy = cgm.getTypes().getFunctionType(fnInfo);
896 cir::FuncOp thunkFn = thunk;
897 if (thunk.getFunctionType() != thunkFnTy) {
898 cir::FuncOp oldThunkFn = thunkFn;
900 assert(oldThunkFn.isDeclaration() &&
"Shouldn't replace non-declaration");
903 cgm.eraseGlobalSymbol(oldThunkFn);
904 oldThunkFn.setName(StringRef());
906 cir::FuncOp::create(cgm.getBuilder(), thunk->getLoc(), name.str(),
907 thunkFnTy, cir::GlobalLinkageKind::ExternalLinkage);
908 cgm.insertGlobalSymbol(thunkFn);
909 cgm.setCIRFunctionAttributes(md, fnInfo, thunkFn,
false);
911 if (!oldThunkFn->use_empty())
912 oldThunkFn->replaceAllUsesWith(thunkFn);
918 bool abiHasKeyFunctions = cgm.getTarget().getCXXABI().hasKeyFunctions();
919 bool useAvailableExternallyLinkage = forVTable && abiHasKeyFunctions;
923 if (!thunkFn.isDeclaration()) {
924 if (!abiHasKeyFunctions || useAvailableExternallyLinkage) {
943 bool shouldCloneVarArgs =
false;
944 if (!isUnprototyped && thunkFn.getFunctionType().isVarArg()) {
945 shouldCloneVarArgs =
true;
947 switch (cgm.getTriple().getArch()) {
948 case llvm::Triple::x86_64:
949 case llvm::Triple::x86:
950 case llvm::Triple::aarch64:
951 shouldCloneVarArgs =
false;
959 if (shouldCloneVarArgs) {
960 if (useAvailableExternallyLinkage)
962 cgm.errorNYI(
"varargs thunk cloning");
965 mlir::OpBuilder::InsertionGuard guard(cgm.getBuilder());
967 cgf.
generateThunk(thunkFn, fnInfo, gd, thunkAdjustments, isUnprototyped);
983 vtContext->getThunkInfo(gd);
985 if (!thunkInfoVector)
988 for (
const ThunkInfo &thunk : *thunkInfoVector)
1018 size_t savedSize = deferredVTables.size();
1022 vtables.generateClassData(rd);
1024 opportunisticVTables.push_back(rd);
1027 assert(savedSize == deferredVTables.size() &&
1028 "deferred extra vtables during vtable emission?");
1029 deferredVTables.clear();
1040 "Only emit opportunistic vtables with optimizations");
1044 "This queue should only contain external vtables");
1045 if (
getCXXABI().canSpeculativelyEmitVTable(rd))
1046 vtables.generateClassData(rd);
1048 opportunisticVTables.clear();
1052 return codeGenOpts.OptimizationLevel > 0;
static RValue performReturnAdjustment(CIRGenFunction &cgf, QualType resultType, RValue rv, const ThunkInfo &thunk)
static cir::GlobalOp getAddrOfVTTVTable(CIRGenVTables &cgvt, CIRGenModule &cgm, const CXXRecordDecl *mostDerivedClass, const VTTVTable &vtable, cir::GlobalLinkageKind linkage, VTableLayout::AddressPointsMapTy &addressPoints)
static bool shouldEmitAvailableExternallyVTable(const CIRGenModule &cgm, const CXXRecordDecl *rd)
static bool shouldEmitVTableAtEndOfTranslationUnit(CIRGenModule &cgm, const CXXRecordDecl *rd)
Given that we're currently at the end of the translation unit, and we've emitted a reference to the v...
static void setThunkProperties(CIRGenModule &cgm, const ThunkInfo &thunk, cir::FuncOp thunkFn, bool forVTable, GlobalDecl gd)
static bool shouldEmitVTableThunk(CIRGenModule &cgm, const CXXMethodDecl *md, bool isUnprototyped, bool forVTable)
mlir::TypedAttr getConstNullPtrAttr(mlir::Type t)
cir::ConstantOp getNullPtr(mlir::Type ty, mlir::Location loc)
mlir::Value createPtrIsNotNull(mlir::Value ptr)
mlir::TypedAttr getConstPtrAttr(mlir::Type type, int64_t value)
cir::YieldOp createYield(mlir::Location loc, mlir::ValueRange value={})
Create a yield operation.
const CXXMethodDecl * getCurrentKeyFunction(const CXXRecordDecl *RD)
Get our current best idea for the key function of the given record decl, or nullptr if there isn't on...
const LangOptions & getLangOpts() const
const TargetInfo & getTargetInfo() const
const CXXRecordDecl * getBase() const
getBase - Returns the base class declaration.
CharUnits getBaseOffset() const
getBaseOffset - Returns the base class offset.
cir::PointerType getUInt8PtrTy()
cir::FuncType getFuncType(llvm::ArrayRef< mlir::Type > params, mlir::Type retTy, bool isVarArg=false)
virtual bool exportThunk()=0
Returns true if the thunk should be exported.
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 cir::GlobalOp 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 void setThunkLinkage(cir::FuncOp thunk, bool forVTable, GlobalDecl gd, bool returnAdjustment)=0
Set the linkage and visibility of a thunk function.
virtual mlir::Value performReturnAdjustment(CIRGenFunction &cgf, Address ret, const CXXRecordDecl *unadjustedClass, const ReturnAdjustment &ra)=0
Perform adjustment on a return pointer for a thunk (covariant returns).
virtual llvm::StringRef getPureVirtualCallName()=0
static CIRGenCallee forDirect(mlir::Operation *funcPtr, const CIRGenCalleeInfo &abstractInfo=CIRGenCalleeInfo())
void generateThunk(cir::FuncOp fn, const CIRGenFunctionInfo &fnInfo, GlobalDecl gd, const ThunkInfo &thunk, bool isUnprototyped)
Generate code for a thunk function.
clang::GlobalDecl curGD
The GlobalDecl for the current function being compiled or the global variable currently being initial...
bool curFuncIsThunk
In C++, whether we are code generating a thunk.
mlir::Block * getCurFunctionEntryBlock()
mlir::Value loadCXXThis()
Load the value for 'this'.
const clang::Decl * curFuncDecl
Address loadCXXThisAddress()
void emitMustTailThunk(GlobalDecl gd, mlir::Value adjustedThisPtr, cir::FuncOp callee)
Emit a musttail call for a thunk with a potentially different ABI.
llvm::ScopedHashTableScope< const clang::Decl *, mlir::Value > SymTableScopeTy
mlir::Operation * curFn
The current function or global initializer that is generated code for.
void startThunk(cir::FuncOp fn, GlobalDecl gd, const CIRGenFunctionInfo &fnInfo, bool isUnprototyped)
Start generating a thunk function.
mlir::Type convertTypeForMem(QualType t)
Address returnValue
The temporary alloca to hold the return value.
void emitCallAndReturnForThunk(cir::FuncOp callee, const ThunkInfo *thunk, bool isUnprototyped)
Emit the call and return for a thunk function.
static bool hasAggregateEvaluationKind(clang::QualType type)
void finishFunction(SourceLocation endLoc)
RValue emitCall(const CIRGenFunctionInfo &funcInfo, const CIRGenCallee &callee, ReturnValueSlot returnValue, const CallArgList &args, cir::CIRCallOpInterface *callOp, mlir::Location loc)
const clang::Decl * curCodeDecl
This is the inner-most code context, which includes blocks.
CIRGenBuilderTy & getBuilder()
void startFunction(clang::GlobalDecl gd, clang::QualType returnType, cir::FuncOp fn, cir::FuncType funcType, FunctionArgList args, clang::SourceLocation loc, clang::SourceLocation startLoc)
Emit code for the start of a function.
void emitDelegateCallArg(CallArgList &args, const clang::VarDecl *param, clang::SourceLocation loc)
We are performing a delegate call; that is, the current function is delegating to another one.
const CIRGenFunctionInfo * curFnInfo
mlir::Value cxxabiThisValue
void finishThunk()
Finish generating a thunk function.
This class organizes the cross-function state that is used while generating CIR code.
DiagnosticBuilder errorNYI(SourceLocation, llvm::StringRef)
Helpers to emit "not yet implemented" error diagnostics.
clang::ASTContext & getASTContext() const
CIRGenBuilderTy & getBuilder()
void setDSOLocal(mlir::Operation *op) const
void setGVProperties(mlir::Operation *op, const NamedDecl *d) const
Set visibility, dllimport/dllexport and dso_local.
clang::CharUnits getClassPointerAlignment(const clang::CXXRecordDecl *rd)
Return the best known alignment for an unknown pointer to a particular class.
const clang::TargetInfo & getTarget() const
const llvm::Triple & getTriple() const
bool shouldOpportunisticallyEmitVTables()
static mlir::SymbolTable::Visibility getMLIRVisibility(Visibility v)
mlir::Type getVTableComponentType()
cir::FuncOp createRuntimeFunction(cir::FuncType ty, llvm::StringRef name, mlir::NamedAttrList extraAttrs={}, bool isLocal=false, bool assumeConvergent=false)
cir::FuncOp getAddrOfThunk(StringRef name, mlir::Type fnTy, GlobalDecl gd)
Get or create a thunk function with the given name and type.
const clang::CodeGenOptions & getCodeGenOpts() const
void emitDeferredVTables()
Emit any vtables which we deferred and still have a use for.
const clang::LangOptions & getLangOpts() const
cir::FuncOp getOrCreateCIRFunction(llvm::StringRef mangledName, mlir::Type funcType, clang::GlobalDecl gd, bool forVTable, bool dontDefer=false, bool isThunk=false, ForDefinition_t isForDefinition=NotForDefinition, mlir::NamedAttrList extraAttrs={})
void emitVTablesOpportunistically()
Try to emit external vtables as available_externally if they have emitted all inlined virtual functio...
bool supportsCOMDAT() const
CIRGenCXXABI & getCXXABI() const
CIRGenVTables & getVTables()
void setFunctionLinkage(GlobalDecl gd, cir::FuncOp f)
void emitVTable(const CXXRecordDecl *rd)
This is a callback from Sema to tell us that a particular vtable is required to be emitted in this tr...
cir::GlobalLinkageKind getVTableLinkage(const CXXRecordDecl *rd)
Return the appropriate linkage for the vtable, VTT, and type information of the given class.
cir::RecordType getVTableType(const clang::VTableLayout &layout)
Returns the type of a vtable with the given layout.
void createVTableInitializer(cir::GlobalOp &vtable, const clang::VTableLayout &layout, mlir::Attribute rtti, bool vtableHasLocalLinkage)
Add vtable components for the given vtable layout to the given global initializer.
cir::GlobalOp generateConstructionVTable(const CXXRecordDecl *rd, const BaseSubobject &base, bool baseIsVirtual, cir::GlobalLinkageKind linkage, VTableAddressPointsMapTy &addressPoints)
Generate a construction vtable for the given base subobject.
uint64_t getSubVTTIndex(const CXXRecordDecl *rd, BaseSubobject base)
Return the index of the sub-VTT for the base class of the given record decl.
void emitThunks(GlobalDecl gd)
Emit the associated thunks for the given global decl.
void emitVTTDefinition(cir::GlobalOp vttOp, cir::GlobalLinkageKind linkage, const CXXRecordDecl *rd)
Emit the definition of the given vtable.
CIRGenVTables(CIRGenModule &cgm)
cir::FuncOp maybeEmitThunk(GlobalDecl gd, const ThunkInfo &thunkAdjustments, bool forVTable)
Emit a thunk for the given global decl if needed, or return an existing thunk.
void generateClassData(const CXXRecordDecl *rd)
Generate all the class data required to be generated upon definition of a KeyFunction.
cir::GlobalOp getAddrOfVTT(const CXXRecordDecl *rd)
Get the address of the VTT for the given record decl.
clang::ItaniumVTableContext & getItaniumVTableContext()
bool isVTableExternal(const clang::CXXRecordDecl *rd)
At this point in the translation unit, does it appear that can we rely on the vtable being defined el...
uint64_t getSecondaryVirtualPointerIndex(const CXXRecordDecl *rd, BaseSubobject base)
Return the index in the VTT where the virtual pointer for the given subobject is located.
Type for representing both the decl and type of parameters to a function.
This trivial value class is used to represent the result of an expression that is evaluated.
static RValue get(mlir::Value v)
mlir::Value getValue() const
Return the value of this scalar value.
static RequiredArgs getFromProtoWithExtraSlots(const clang::FunctionProtoType *prototype, unsigned additional)
Compute the arguments required by the given formal prototype, given that there may be some additional...
Contains the address where the return value of a function can be stored, and whether the address is v...
Represents a C++ destructor within a class.
Represents a static or instance method of a struct/union/class.
QualType getThisType() const
Return the type of the this pointer.
Represents a C++ struct/union/class.
TemplateSpecializationKind getTemplateSpecializationKind() const
Determine whether this particular class is a specialization or instantiation of a class template or m...
bool isDynamicClass() const
unsigned getNumVBases() const
Retrieves the number of virtual base classes of this class.
CharUnits - This is an opaque type for sizes expressed in character units.
bool isZero() const
isZero - Test whether the quantity equals zero.
QuantityType getQuantity() const
getQuantity - Get the raw integer representation of this quantity.
static CharUnits fromQuantity(QuantityType Quantity)
fromQuantity - Construct a CharUnits quantity from a raw integer type.
ASTContext & getASTContext() const LLVM_READONLY
SourceLocation getLocation() const
virtual SourceRange getSourceRange() const LLVM_READONLY
Source range that this declaration covers.
Represents a function declaration or definition.
param_iterator param_end()
bool isInlined() const
Determine whether this function should be inlined, because it is either marked "inline" or "constexpr...
ArrayRef< ParmVarDecl * > parameters() const
param_iterator param_begin()
TemplateSpecializationKind getTemplateSpecializationKind() const
Determine what kind of template instantiation this function represents.
bool hasBody(const FunctionDecl *&Definition) const
Returns true if the function has a body.
Represents a prototype with parameter type info, e.g.
FunctionType - C99 6.7.5.3 - Function Declarators.
QualType getReturnType() const
GlobalDecl - represents a global declaration.
CXXDtorType getDtorType() const
const Decl * getDecl() const
const VTableLayout & getVTableLayout(const CXXRecordDecl *RD)
MangleContext - Context for tracking state which persists across multiple calls to the C++ name mangl...
virtual void mangleCXXDtorThunk(const CXXDestructorDecl *DD, CXXDtorType Type, const ThunkInfo &Thunk, bool ElideOverrideInfo, raw_ostream &)=0
virtual void mangleThunk(const CXXMethodDecl *MD, const ThunkInfo &Thunk, bool ElideOverrideInfo, raw_ostream &)=0
bool isExternallyVisible() const
Represents a parameter to a function.
A (possibly-)qualified type.
Encodes a location in the source.
SourceRange getSourceRange() const override LLVM_READONLY
Source range that this declaration covers.
bool isMicrosoft() const
Is this ABI an MSVC-compatible ABI?
virtual bool emitVectorDeletingDtors(const LangOptions &) const
Controls whether to emit MSVC vector deleting destructors.
TargetCXXABI getCXXABI() const
Get the C++ ABI currently in use.
CXXRecordDecl * getAsCXXRecordDecl() const
Retrieves the CXXRecordDecl that this type refers to, either because the type is a RecordType or beca...
const T * castAs() const
Member-template castAs<specific type>.
bool isReferenceType() const
const CXXRecordDecl * getPointeeCXXRecordDecl() const
If this is a pointer or reference to a RecordType, return the CXXRecordDecl that the type refers to.
QualType getPointeeType() const
If this is a pointer, ObjC object pointer, or block pointer, this returns the respective pointee.
Class for building VTT layout information.
const llvm::DenseMap< BaseSubobject, uint64_t > & getSecondaryVirtualPointerIndices() const
Returns a reference to the secondary virtual pointer indices.
const llvm::DenseMap< BaseSubobject, uint64_t > & getSubVTTIndices() const
Returns a reference to the sub-VTT indices.
const VTTComponentsVectorTy & getVTTComponents() const
const VTTVTablesVectorTy & getVTTVTables() const
CharUnits getBaseOffset() const
const CXXRecordDecl * getBase() const
BaseSubobject getBaseSubobject() const
Represents a single component in a vtable.
CharUnits getVBaseOffset() const
Kind getKind() const
Get the kind of this vtable component.
@ CK_DeletingDtorPointer
A pointer to the deleting destructor.
@ CK_UnusedFunctionPointer
An entry that is never used.
@ CK_CompleteDtorPointer
A pointer to the complete destructor.
CharUnits getOffsetToTop() const
GlobalDecl getGlobalDecl(bool HasVectorDeletingDtors) const
CharUnits getVCallOffset() const
SmallVector< ThunkInfo, 1 > ThunkInfoVectorTy
const AddressPointsIndexMapTy & getAddressPointIndices() const
size_t getVTableOffset(size_t i) const
llvm::DenseMap< BaseSubobject, AddressPointLocation > AddressPointsMapTy
AddressPointLocation getAddressPoint(BaseSubobject Base) const
ArrayRef< VTableComponent > vtable_components() const
size_t getNumVTables() const
ArrayRef< VTableThunkTy > vtable_thunks() const
size_t getVTableSize(size_t i) const
static bool isLocalLinkage(GlobalLinkageKind linkage)
const AstTypeMatcher< ArrayType > arrayType
The JSON file list parser is used to communicate input to InstallAPI.
bool isa(CodeGen::Address addr)
@ Dtor_Base
Base object dtor.
TemplateSpecializationKind
Describes the kind of template specialization that a particular template specialization declaration r...
@ TSK_ExplicitInstantiationDefinition
This template specialization was instantiated from a template due to an explicit instantiation defini...
@ TSK_ExplicitInstantiationDeclaration
This template specialization was instantiated from a template due to an explicit instantiation declar...
@ TSK_ExplicitSpecialization
This template specialization was declared or defined by an explicit specialization (C++ [temp....
@ TSK_ImplicitInstantiation
This template specialization was implicitly instantiated from a template.
@ TSK_Undeclared
This template specialization was formed from a template-id but has not yet been declared,...
U cast(CodeGen::Address addr)
Diagnostic wrappers for TextAPI types for error reporting.
static bool objCLifetime()
static bool opGlobalUnnamedAddr()
static bool vtableEmitMetadata()
static bool setDLLStorageClass()
static bool opCallInAlloca()
static bool pointerAuthentication()
static bool opCallMustTail()
static bool returnValueSlotFeatures()
static bool cudaSupport()
static bool generateDebugInfo()
static bool vtableRelativeLayout()
Represents a scope, including function bodies, compound statements, and the substatements of if/while...
The this pointer adjustment as well as an optional return adjustment for a thunk.
ReturnAdjustment Return
The return adjustment.
unsigned AddressPointIndex