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);
334 return cir::GlobalLinkageKind::InternalLinkage;
338 const CXXMethodDecl *keyFunction = astContext.getCurrentKeyFunction(rd);
339 if (keyFunction && !rd->
hasAttr<DLLImportAttr>()) {
352 (def || codeGenOpts.OptimizationLevel > 0 ||
353 codeGenOpts.getDebugInfo() != llvm::codegenoptions::NoDebugInfo) &&
354 "Shouldn't query vtable linkage without key function, "
355 "optimizations, or debug info");
356 if (!def && codeGenOpts.OptimizationLevel > 0)
357 return cir::GlobalLinkageKind::AvailableExternallyLinkage;
360 return !astContext.getLangOpts().AppleKext
361 ? cir::GlobalLinkageKind::LinkOnceODRLinkage
362 : cir::GlobalLinkageKind::InternalLinkage;
363 return cir::GlobalLinkageKind::ExternalLinkage;
366 return cir::GlobalLinkageKind::LinkOnceODRLinkage;
369 return cir::GlobalLinkageKind::WeakODRLinkage;
372 llvm_unreachable(
"Should not have been asked to emit this");
377 if (astContext.getLangOpts().AppleKext)
378 return cir::GlobalLinkageKind::InternalLinkage;
380 auto discardableODRLinkage = cir::GlobalLinkageKind::LinkOnceODRLinkage;
381 auto nonDiscardableODRLinkage = cir::GlobalLinkageKind::WeakODRLinkage;
382 if (rd->
hasAttr<DLLExportAttr>()) {
384 discardableODRLinkage = nonDiscardableODRLinkage;
385 }
else if (rd->
hasAttr<DLLImportAttr>()) {
387 discardableODRLinkage = cir::GlobalLinkageKind::AvailableExternallyLinkage;
388 nonDiscardableODRLinkage =
389 cir::GlobalLinkageKind::AvailableExternallyLinkage;
396 return discardableODRLinkage;
400 "getVTableLinkage: explicit instantiation declaration");
401 return cir::GlobalLinkageKind::ExternalLinkage;
405 return nonDiscardableODRLinkage;
408 llvm_unreachable(
"Invalid TemplateSpecializationKind!");
412 assert(rd->
getNumVBases() &&
"Only classes with virtual bases need a VTT");
415 llvm::raw_svector_ostream
out(outName);
417 .mangleCXXVTT(rd,
out);
418 StringRef name = outName.str();
421 (void)cgm.getCXXABI().getAddrOfVTable(rd,
CharUnits());
423 VTTBuilder builder(cgm.getASTContext(), rd,
false);
425 auto arrayType = cir::ArrayType::get(cgm.getBuilder().getUInt8PtrTy(),
428 cgm.getDataLayout().getABITypeAlign(cgm.getBuilder().getUInt8PtrTy());
429 cir::GlobalOp vtt = cgm.createOrReplaceCXXRuntimeVariable(
432 cgm.setGVProperties(vtt, rd);
439 const VTTVTable &vtable, cir::GlobalLinkageKind linkage,
441 if (vtable.
getBase() == mostDerivedClass) {
443 "Most derived class vtable must have a zero offset!");
454 cir::GlobalLinkageKind linkage,
456 VTTBuilder builder(cgm.getASTContext(), rd,
true);
458 mlir::MLIRContext *mlirContext = &cgm.getMLIRContext();
460 auto arrayType = cir::ArrayType::get(cgm.getBuilder().getUInt8PtrTy(),
466 vtableAddressPoints.push_back(VTableAddressPointsMapTy());
468 vtableAddressPoints.back()));
474 cir::GlobalOp vtable = vtables[vttComponent.VTableIndex];
480 vttComponent.VTableBase);
482 addressPoint = vtableAddressPoints[vttComponent.VTableIndex].lookup(
483 vttComponent.VTableBase);
485 "Did not find ctor vtable address point!");
488 mlir::Attribute indices[2] = {
489 cgm.getBuilder().getI32IntegerAttr(addressPoint.
VTableIndex),
493 auto indicesAttr = mlir::ArrayAttr::get(mlirContext, indices);
494 cir::GlobalViewAttr init = cgm.getBuilder().getGlobalViewAttr(
495 cgm.getBuilder().getUInt8PtrTy(), vtable, indicesAttr);
497 vttComponents.push_back(init);
500 auto init = cir::ConstArrayAttr::get(
501 arrayType, mlir::ArrayAttr::get(mlirContext, vttComponents));
503 vttOp.setInitialValueAttr(init);
506 vttOp.setLinkage(linkage);
507 mlir::SymbolTable::setSymbolVisibility(
510 if (cgm.supportsCOMDAT() && vttOp.isWeakForLinker())
511 vttOp.setComdat(
true);
516 BaseSubobjectPairTy classSubobjectPair(rd, base);
518 SubVTTIndiciesMapTy::iterator it = subVTTIndicies.find(classSubobjectPair);
519 if (it != subVTTIndicies.end())
522 VTTBuilder builder(cgm.getASTContext(), rd,
false);
526 BaseSubobjectPairTy subclassSubobjectPair(rd, entry.first);
528 subVTTIndicies.insert(std::make_pair(subclassSubobjectPair, entry.second));
531 it = subVTTIndicies.find(classSubobjectPair);
532 assert(it != subVTTIndicies.end() &&
"Did not find index!");
539 auto it = secondaryVirtualPointerIndices.find(std::make_pair(rd, base));
541 if (it != secondaryVirtualPointerIndices.end())
544 VTTBuilder builder(cgm.getASTContext(), rd,
false);
548 std::pair<const CXXRecordDecl *, BaseSubobject> pair =
549 std::make_pair(rd, entry.first);
551 secondaryVirtualPointerIndices.insert(std::make_pair(pair, entry.second));
554 it = secondaryVirtualPointerIndices.find(std::make_pair(rd, base));
555 assert(it != secondaryVirtualPointerIndices.end() &&
"Did not find index!");
565 mlir::Value returnValue = rv.
getValue();
572 mlir::Location loc = returnValue.getLoc();
574 if (!nullCheckValue) {
576 cgf,
Address(returnValue, pointeeType, classAlign), classDecl,
583 cir::TernaryOp::create(
584 builder, loc, isNotNull,
585 [&](mlir::OpBuilder &, mlir::Location) {
587 cgf,
Address(returnValue, pointeeType, classAlign), classDecl,
591 [&](mlir::OpBuilder &, mlir::Location) {
592 mlir::Value nullVal =
593 builder.
getNullPtr(returnValue.getType(), loc).getResult();
603 bool isUnprototyped) {
604 assert(!
curGD.getDecl() &&
"curGD was already set!");
613 resultType =
cgm.getASTContext().VoidTy;
614 else if (
cgm.getCXXABI().hasThisReturn(gd))
615 resultType = thisType;
616 else if (
cgm.getCXXABI().hasMostDerivedReturn(gd))
617 resultType =
cgm.getASTContext().VoidPtrTy;
623 cgm.getCXXABI().buildThisParam(*
this, functionArgs);
626 if (!isUnprototyped) {
630 cgm.getCXXABI().addImplicitStructorParams(*
this, resultType,
637 cir::FuncType funcType =
cgm.getTypes().getFunctionType(fnInfo);
645 cgm.getCXXABI().emitInstanceFunctionProlog(md->
getLocation(), *
this);
662 bool isUnprototyped) {
664 "Please use a new CGF for this thunk");
673 mlir::Value adjustedThisPtr =
675 thisValueClass, *thunk)
686 cgm.errorUnsupported(
687 md,
"return-adjusting thunk with incomplete parameter type");
689 llvm_unreachable(
"shouldn't try to emit musttail return-adjusting "
690 "thunks for variadic functions");
692 cgm.errorUnsupported(
693 md,
"non-trivial argument copy for return-adjusting thunk");
702 callArgs.add(
RValue::get(adjustedThisPtr), thisType);
705 cgm.getCXXABI().adjustCallArgsForDestructorThunk(*
this,
curGD, callArgs);
708 unsigned prefixArgs = callArgs.size() - 1;
721 assert(callFnInfo.argTypeSize() ==
curFnInfo->argTypeSize());
726 :
cgm.getCXXABI().hasMostDerivedReturn(
curGD)
727 ?
cgm.getASTContext().VoidPtrTy
738 mlir::Location loc = builder.getUnknownLoc();
750 cgm.getCXXABI().emitReturnFromThunk(*
this, rv, resultType);
759 mlir::Value adjustedThisPtr,
760 cir::FuncOp callee) {
765 for (mlir::BlockArgument arg : entryBlock->getArguments())
769 assert(!args.empty() &&
"thunk must have at least 'this' argument");
770 if (adjustedThisPtr.getType() != args[0].getType())
771 adjustedThisPtr = builder.createBitcast(adjustedThisPtr, args[0].
getType());
772 args[0] = adjustedThisPtr;
774 mlir::Location loc =
curFn->getLoc();
775 cir::FuncType calleeTy = callee.getFunctionType();
776 mlir::Type retTy = calleeTy.getReturnType();
778 cir::CallOp call = builder.createCallOp(loc, callee, args);
779 call->setAttr(cir::CIRDialect::getMustTailAttrName(),
780 mlir::UnitAttr::get(builder.getContext()));
783 cir::ReturnOp::create(builder, loc);
785 cir::ReturnOp::create(builder, loc, call->getResult(0));
793 bool isUnprototyped) {
796 assert(fn.isDeclaration() &&
"Function already has body?");
797 mlir::Block *entryBb = fn.addEntryBlock();
798 builder.setInsertionPointToStart(entryBb);
818 cgm.errorNYI(
"unprototyped thunk placeholder type");
820 ty =
cgm.getTypes().getFunctionType(fnInfo);
822 cir::FuncOp calleeOp =
cgm.getAddrOfFunction(gd, ty,
true);
829 bool isUnprototyped,
bool forVTable) {
853 llvm::raw_svector_ostream
out(name);
861 if (cgm.getASTContext().useAbbreviatedThunkName(gd, name.str())) {
870 cir::FuncType thunkVTableTy = cgm.getTypes().getFunctionType(gd);
871 cir::FuncOp thunk = cgm.getAddrOfThunk(name, thunkVTableTy, gd);
874 bool isUnprototyped = !cgm.getTypes().isFuncTypeConvertible(
882 isUnprototyped ? (cgm.errorNYI(
"unprototyped must-tail thunk"),
883 cgm.getTypes().arrangeGlobalDeclaration(gd))
884 : cgm.getTypes().arrangeGlobalDeclaration(gd);
885 cir::FuncType thunkFnTy = cgm.getTypes().getFunctionType(fnInfo);
889 cir::FuncOp thunkFn = thunk;
890 if (thunk.getFunctionType() != thunkFnTy) {
891 cir::FuncOp oldThunkFn = thunkFn;
893 assert(oldThunkFn.isDeclaration() &&
"Shouldn't replace non-declaration");
896 oldThunkFn.setName(StringRef());
898 cir::FuncOp::create(cgm.getBuilder(), thunk->getLoc(), name.str(),
899 thunkFnTy, cir::GlobalLinkageKind::ExternalLinkage);
900 cgm.setCIRFunctionAttributes(md, fnInfo, thunkFn,
false);
902 if (!oldThunkFn->use_empty())
903 oldThunkFn->replaceAllUsesWith(thunkFn);
909 bool abiHasKeyFunctions = cgm.getTarget().getCXXABI().hasKeyFunctions();
910 bool useAvailableExternallyLinkage = forVTable && abiHasKeyFunctions;
914 if (!thunkFn.isDeclaration()) {
915 if (!abiHasKeyFunctions || useAvailableExternallyLinkage) {
934 bool shouldCloneVarArgs =
false;
935 if (!isUnprototyped && thunkFn.getFunctionType().isVarArg()) {
936 shouldCloneVarArgs =
true;
938 switch (cgm.getTriple().getArch()) {
939 case llvm::Triple::x86_64:
940 case llvm::Triple::x86:
941 case llvm::Triple::aarch64:
942 shouldCloneVarArgs =
false;
950 if (shouldCloneVarArgs) {
951 if (useAvailableExternallyLinkage)
953 cgm.errorNYI(
"varargs thunk cloning");
956 mlir::OpBuilder::InsertionGuard guard(cgm.getBuilder());
958 cgf.
generateThunk(thunkFn, fnInfo, gd, thunkAdjustments, isUnprototyped);
974 vtContext->getThunkInfo(gd);
976 if (!thunkInfoVector)
979 for (
const ThunkInfo &thunk : *thunkInfoVector)
1009 size_t savedSize = deferredVTables.size();
1013 vtables.generateClassData(rd);
1015 opportunisticVTables.push_back(rd);
1018 assert(savedSize == deferredVTables.size() &&
1019 "deferred extra vtables during vtable emission?");
1020 deferredVTables.clear();
1031 "Only emit opportunistic vtables with optimizations");
1035 "This queue should only contain external vtables");
1036 if (
getCXXABI().canSpeculativelyEmitVTable(rd))
1037 vtables.generateClassData(rd);
1039 opportunisticVTables.clear();
1043 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