clang 23.0.0git
CIRGenVTables.cpp
Go to the documentation of this file.
1//===----------------------------------------------------------------------===//
2//
3// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4// See https://llvm.org/LICENSE.txt for license information.
5// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6//
7//===----------------------------------------------------------------------===//
8//
9// This contains code dealing with C++ code generation of virtual tables.
10//
11//===----------------------------------------------------------------------===//
12
13#include "CIRGenVTables.h"
14
15#include "CIRGenCXXABI.h"
16#include "CIRGenModule.h"
17#include "mlir/IR/Types.h"
20#include "llvm/ADT/SmallVector.h"
21
22using namespace llvm;
23using namespace clang;
24using namespace clang::CIRGen;
25
27 : cgm(cgm), vtContext(cgm.getASTContext().getVTableContext()) {}
28
29cir::FuncOp CIRGenModule::getAddrOfThunk(StringRef name, mlir::Type fnTy,
30 GlobalDecl gd) {
31 return getOrCreateCIRFunction(name, fnTy, gd, /*forVTable=*/true,
32 /*dontDefer=*/true, /*isThunk=*/true);
33}
34
35static void setThunkProperties(CIRGenModule &cgm, const ThunkInfo &thunk,
36 cir::FuncOp thunkFn, bool forVTable,
37 GlobalDecl gd) {
38 cgm.setFunctionLinkage(gd, thunkFn);
39 cgm.getCXXABI().setThunkLinkage(thunkFn, forVTable, gd,
40 !thunk.Return.isEmpty());
41
42 // Set the right visibility.
43 cgm.setGVProperties(thunkFn, cast<NamedDecl>(gd.getDecl()));
44
45 if (!cgm.getCXXABI().exportThunk()) {
47 cgm.setDSOLocal(static_cast<mlir::Operation *>(thunkFn));
48 }
49
50 if (cgm.supportsCOMDAT() && thunkFn.isWeakForLinker())
51 thunkFn.setComdat(true);
52}
53
55 mlir::Type ptrTy = builder.getUInt8PtrTy();
57 return ptrTy;
58}
59
60mlir::Type CIRGenVTables::getVTableComponentType() {
61 return cgm.getVTableComponentType();
62}
63
64cir::RecordType CIRGenVTables::getVTableType(const VTableLayout &layout) {
66 mlir::Type componentType = getVTableComponentType();
67 for (unsigned i = 0, e = layout.getNumVTables(); i != e; ++i)
68 tys.push_back(cir::ArrayType::get(componentType, layout.getVTableSize(i)));
69
70 // FIXME(cir): should VTableLayout be encoded like we do for some
71 // AST nodes?
72 return cgm.getBuilder().getAnonRecordTy(tys, /*incomplete=*/false);
73}
74
75/// At this point in the translation unit, does it appear that can we
76/// rely on the vtable being defined elsewhere in the program?
77///
78/// The response is really only definitive when called at the end of
79/// the translation unit.
80///
81/// The only semantic restriction here is that the object file should
82/// not contain a vtable definition when that vtable is defined
83/// strongly elsewhere. Otherwise, we'd just like to avoid emitting
84/// vtables when unnecessary.
85/// TODO(cir): this should be merged into common AST helper for codegen.
87 assert(rd->isDynamicClass() && "Non-dynamic classes have no VTable.");
88
89 // We always synthesize vtables if they are needed in the MS ABI. MSVC doesn't
90 // emit them even if there is an explicit template instantiation.
91 if (cgm.getTarget().getCXXABI().isMicrosoft())
92 return false;
93
94 // If we have an explicit instantiation declaration (and not a
95 // definition), the vtable is defined elsewhere.
98 return true;
99
100 // Otherwise, if the class is an instantiated template, the
101 // vtable must be defined here.
102 if (tsk == TSK_ImplicitInstantiation ||
104 return false;
105
106 // Otherwise, if the class doesn't have a key function (possibly
107 // anymore), the vtable must be defined here.
108 const CXXMethodDecl *keyFunction =
110 if (!keyFunction)
111 return false;
112
113 // Otherwise, if we don't have a definition of the key function, the
114 // vtable must be defined somewhere else.
115 return !keyFunction->hasBody();
116}
117
118/// This is a callback from Sema to tell us that a particular vtable is
119/// required to be emitted in this translation unit.
120///
121/// This is only called for vtables that _must_ be emitted (mainly due to key
122/// functions). For weak vtables, CodeGen tracks when they are needed and
123/// emits them as-needed.
125 vtables.generateClassData(rd);
126}
127
130
131 if (rd->getNumVBases())
132 cgm.getCXXABI().emitVirtualInheritanceTables(rd);
133
134 cgm.getCXXABI().emitVTableDefinitions(*this, rd);
135}
136
137mlir::Attribute CIRGenVTables::getVTableComponent(
138 const VTableLayout &layout, unsigned componentIndex, mlir::Attribute rtti,
139 unsigned &nextVTableThunkIndex, unsigned vtableAddressPoint,
140 bool vtableHasLocalLinkage) {
141 const VTableComponent &component = layout.vtable_components()[componentIndex];
142
143 CIRGenBuilderTy builder = cgm.getBuilder();
144
146
147 switch (component.getKind()) {
149 return builder.getConstNullPtrAttr(builder.getUInt8PtrTy());
150
152 return builder.getConstPtrAttr(builder.getUInt8PtrTy(),
153 component.getVCallOffset().getQuantity());
154
156 return builder.getConstPtrAttr(builder.getUInt8PtrTy(),
157 component.getVBaseOffset().getQuantity());
158
160 return builder.getConstPtrAttr(builder.getUInt8PtrTy(),
161 component.getOffsetToTop().getQuantity());
162
164 assert((mlir::isa<cir::GlobalViewAttr>(rtti) ||
165 mlir::isa<cir::ConstPtrAttr>(rtti)) &&
166 "expected GlobalViewAttr or ConstPtrAttr");
167 return rtti;
168
172 GlobalDecl gd = component.getGlobalDecl(
174 cgm.getASTContext().getLangOpts()));
175
177
178 auto getSpecialVirtFn = [&](StringRef name) -> cir::FuncOp {
180
181 if (cgm.getLangOpts().OpenMP && cgm.getLangOpts().OpenMPIsTargetDevice &&
182 cgm.getTriple().isNVPTX())
183 cgm.errorNYI(gd.getDecl()->getSourceRange(),
184 "getVTableComponent for OMP Device NVPTX");
185
186 cir::FuncType fnTy =
187 cgm.getBuilder().getFuncType({}, cgm.getBuilder().getVoidTy());
188 cir::FuncOp fnPtr = cgm.createRuntimeFunction(fnTy, name);
189
191 return fnPtr;
192 };
193
194 cir::FuncOp fnPtr;
195 if (cast<CXXMethodDecl>(gd.getDecl())->isPureVirtual()) {
196 if (!pureVirtualFn)
197 pureVirtualFn =
198 getSpecialVirtFn(cgm.getCXXABI().getPureVirtualCallName());
199 fnPtr = pureVirtualFn;
200 } else if (cast<CXXMethodDecl>(gd.getDecl())->isDeleted()) {
201 if (!deletedVirtualFn)
202 deletedVirtualFn =
203 getSpecialVirtFn(cgm.getCXXABI().getDeletedVirtualCallName());
204 fnPtr = deletedVirtualFn;
205 } else if (nextVTableThunkIndex < layout.vtable_thunks().size() &&
206 layout.vtable_thunks()[nextVTableThunkIndex].first ==
207 componentIndex) {
208 const ThunkInfo &thunkInfo =
209 layout.vtable_thunks()[nextVTableThunkIndex].second;
210 nextVTableThunkIndex++;
211 fnPtr = maybeEmitThunk(gd, thunkInfo, /*forVTable=*/true);
213 } else {
214 // Otherwise we can use the method definition directly.
215 cir::FuncType fnTy = cgm.getTypes().getFunctionType(gd);
216 fnPtr = cgm.getAddrOfFunction(gd, fnTy, /*ForVTable=*/true);
217 }
218
219 return cir::GlobalViewAttr::get(
220 builder.getUInt8PtrTy(),
221 mlir::FlatSymbolRefAttr::get(fnPtr.getSymNameAttr()));
222 }
223 }
224
225 llvm_unreachable("Unexpected vtable component kind");
226}
227
228void CIRGenVTables::createVTableInitializer(cir::GlobalOp &vtableOp,
229 const clang::VTableLayout &layout,
230 mlir::Attribute rtti,
231 bool vtableHasLocalLinkage) {
232 mlir::Type componentType = getVTableComponentType();
233
234 const llvm::SmallVectorImpl<unsigned> &addressPoints =
235 layout.getAddressPointIndices();
236 unsigned nextVTableThunkIndex = 0;
237
238 mlir::MLIRContext *mlirContext = &cgm.getMLIRContext();
239
241 for (auto [vtableIndex, addressPoint] : llvm::enumerate(addressPoints)) {
242 // Build a ConstArrayAttr of the vtable components.
243 size_t vtableStart = layout.getVTableOffset(vtableIndex);
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));
251 // Create a ConstArrayAttr to hold the components.
252 auto arr = cir::ConstArrayAttr::get(
253 cir::ArrayType::get(componentType, components.size()),
254 mlir::ArrayAttr::get(mlirContext, components));
255 vtables.push_back(arr);
256 }
257
258 // Create a ConstRecordAttr to hold the component array.
259 const auto members = mlir::ArrayAttr::get(mlirContext, vtables);
260 cir::ConstRecordAttr record = cgm.getBuilder().getAnonConstRecord(members);
261
262 // Create a VTableAttr
263 auto vtableAttr = cir::VTableAttr::get(record.getType(), record.getMembers());
264
265 // Add the vtable initializer to the vtable global op.
266 cgm.setInitializer(vtableOp, vtableAttr);
267}
268
270 const CXXRecordDecl *rd, const BaseSubobject &base, bool baseIsVirtual,
271 cir::GlobalLinkageKind linkage, VTableAddressPointsMapTy &addressPoints) {
273
274 std::unique_ptr<VTableLayout> vtLayout(
275 getItaniumVTableContext().createConstructionVTableLayout(
276 base.getBase(), base.getBaseOffset(), baseIsVirtual, rd));
277
278 // Add the address points.
279 addressPoints = vtLayout->getAddressPoints();
280
281 // Get the mangled construction vtable name.
282 SmallString<256> outName;
283 llvm::raw_svector_ostream out(outName);
284 cast<ItaniumMangleContext>(cgm.getCXXABI().getMangleContext())
285 .mangleCXXCtorVTable(rd, base.getBaseOffset().getQuantity(),
286 base.getBase(), out);
287 SmallString<256> name(outName);
288
290
291 cir::RecordType vtType = getVTableType(*vtLayout);
292
293 // Construction vtable symbols are not part of the Itanium ABI, so we cannot
294 // guarantee that they actually will be available externally. Instead, when
295 // emitting an available_externally VTT, we provide references to an internal
296 // linkage construction vtable. The ABI only requires complete-object vtables
297 // to be the same for all instances of a type, not construction vtables.
298 if (linkage == cir::GlobalLinkageKind::AvailableExternallyLinkage)
299 linkage = cir::GlobalLinkageKind::InternalLinkage;
300
301 llvm::Align align = cgm.getDataLayout().getABITypeAlign(vtType);
302 mlir::Location loc = cgm.getLoc(rd->getSourceRange());
303
304 // Create the variable that will hold the construction vtable.
305 cir::GlobalOp vtable = cgm.createOrReplaceCXXRuntimeVariable(
306 loc, name, vtType, linkage, CharUnits::fromQuantity(align));
307
308 // V-tables are always unnamed_addr.
310
311 mlir::Attribute rtti = cgm.getAddrOfRTTIDescriptor(
312 loc, cgm.getASTContext().getCanonicalTagType(base.getBase()));
313
314 // Create and set the initializer.
315 createVTableInitializer(vtable, *vtLayout, rtti,
316 cir::isLocalLinkage(vtable.getLinkage()));
317
318 // Set properties only after the initializer has been set to ensure that the
319 // GV is treated as definition and not declaration.
320 assert(!vtable.isDeclaration() && "Shouldn't set properties on declaration");
321 cgm.setGVProperties(vtable, rd);
322
325
326 return vtable;
327}
328
329/// Compute the required linkage of the vtable for the given class.
330///
331/// Note that we only call this at the end of the translation unit.
332cir::GlobalLinkageKind CIRGenModule::getVTableLinkage(const CXXRecordDecl *rd) {
333 if (!rd->isExternallyVisible())
334 return cir::GlobalLinkageKind::InternalLinkage;
335
336 // We're at the end of the translation unit, so the current key
337 // function is fully correct.
338 const CXXMethodDecl *keyFunction = astContext.getCurrentKeyFunction(rd);
339 if (keyFunction && !rd->hasAttr<DLLImportAttr>()) {
340 // If this class has a key function, use that to determine the
341 // linkage of the vtable.
342 const FunctionDecl *def = nullptr;
343 if (keyFunction->hasBody(def))
344 keyFunction = cast<CXXMethodDecl>(def);
345
346 // All of the cases below do something different with AppleKext enabled.
348 switch (keyFunction->getTemplateSpecializationKind()) {
349 case TSK_Undeclared:
351 assert(
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;
358
359 if (keyFunction->isInlined())
360 return !astContext.getLangOpts().AppleKext
361 ? cir::GlobalLinkageKind::LinkOnceODRLinkage
362 : cir::GlobalLinkageKind::InternalLinkage;
363 return cir::GlobalLinkageKind::ExternalLinkage;
364
366 return cir::GlobalLinkageKind::LinkOnceODRLinkage;
367
369 return cir::GlobalLinkageKind::WeakODRLinkage;
370
372 llvm_unreachable("Should not have been asked to emit this");
373 }
374 }
375 // -fapple-kext mode does not support weak linkage, so we must use
376 // internal linkage.
377 if (astContext.getLangOpts().AppleKext)
378 return cir::GlobalLinkageKind::InternalLinkage;
379
380 auto discardableODRLinkage = cir::GlobalLinkageKind::LinkOnceODRLinkage;
381 auto nonDiscardableODRLinkage = cir::GlobalLinkageKind::WeakODRLinkage;
382 if (rd->hasAttr<DLLExportAttr>()) {
383 // Cannot discard exported vtables.
384 discardableODRLinkage = nonDiscardableODRLinkage;
385 } else if (rd->hasAttr<DLLImportAttr>()) {
386 // Imported vtables are available externally.
387 discardableODRLinkage = cir::GlobalLinkageKind::AvailableExternallyLinkage;
388 nonDiscardableODRLinkage =
389 cir::GlobalLinkageKind::AvailableExternallyLinkage;
390 }
391
392 switch (rd->getTemplateSpecializationKind()) {
393 case TSK_Undeclared:
396 return discardableODRLinkage;
397
400 "getVTableLinkage: explicit instantiation declaration");
401 return cir::GlobalLinkageKind::ExternalLinkage;
402 }
403
405 return nonDiscardableODRLinkage;
406 }
407
408 llvm_unreachable("Invalid TemplateSpecializationKind!");
409}
410
412 assert(rd->getNumVBases() && "Only classes with virtual bases need a VTT");
413
414 SmallString<256> outName;
415 llvm::raw_svector_ostream out(outName);
416 cast<ItaniumMangleContext>(cgm.getCXXABI().getMangleContext())
417 .mangleCXXVTT(rd, out);
418 StringRef name = outName.str();
419
420 // This will also defer the definition of the VTT.
421 (void)cgm.getCXXABI().getAddrOfVTable(rd, CharUnits());
422
423 VTTBuilder builder(cgm.getASTContext(), rd, /*GenerateDefinition=*/false);
424
425 auto arrayType = cir::ArrayType::get(cgm.getBuilder().getUInt8PtrTy(),
426 builder.getVTTComponents().size());
427 llvm::Align align =
428 cgm.getDataLayout().getABITypeAlign(cgm.getBuilder().getUInt8PtrTy());
429 cir::GlobalOp vtt = cgm.createOrReplaceCXXRuntimeVariable(
430 cgm.getLoc(rd->getSourceRange()), name, arrayType,
431 cir::GlobalLinkageKind::ExternalLinkage, CharUnits::fromQuantity(align));
432 cgm.setGVProperties(vtt, rd);
433 return vtt;
434}
435
436static cir::GlobalOp
438 const CXXRecordDecl *mostDerivedClass,
439 const VTTVTable &vtable, cir::GlobalLinkageKind linkage,
440 VTableLayout::AddressPointsMapTy &addressPoints) {
441 if (vtable.getBase() == mostDerivedClass) {
442 assert(vtable.getBaseOffset().isZero() &&
443 "Most derived class vtable must have a zero offset!");
444 // This is a regular vtable.
445 return cgm.getCXXABI().getAddrOfVTable(mostDerivedClass, CharUnits());
446 }
447 return cgvt.generateConstructionVTable(
448 mostDerivedClass, vtable.getBaseSubobject(), vtable.isVirtual(), linkage,
449 addressPoints);
450}
451
452/// Emit the definition of the given vtable.
453void CIRGenVTables::emitVTTDefinition(cir::GlobalOp vttOp,
454 cir::GlobalLinkageKind linkage,
455 const CXXRecordDecl *rd) {
456 VTTBuilder builder(cgm.getASTContext(), rd, /*GenerateDefinition=*/true);
457
458 mlir::MLIRContext *mlirContext = &cgm.getMLIRContext();
459
460 auto arrayType = cir::ArrayType::get(cgm.getBuilder().getUInt8PtrTy(),
461 builder.getVTTComponents().size());
462
464 SmallVector<VTableAddressPointsMapTy> vtableAddressPoints;
465 for (const VTTVTable &vtt : builder.getVTTVTables()) {
466 vtableAddressPoints.push_back(VTableAddressPointsMapTy());
467 vtables.push_back(getAddrOfVTTVTable(*this, cgm, rd, vtt, linkage,
468 vtableAddressPoints.back()));
469 }
470
471 SmallVector<mlir::Attribute> vttComponents;
472 for (const VTTComponent &vttComponent : builder.getVTTComponents()) {
473 const VTTVTable &vttVT = builder.getVTTVTables()[vttComponent.VTableIndex];
474 cir::GlobalOp vtable = vtables[vttComponent.VTableIndex];
476 if (vttVT.getBase() == rd) {
477 // Just get the address point for the regular vtable.
478 addressPoint =
480 vttComponent.VTableBase);
481 } else {
482 addressPoint = vtableAddressPoints[vttComponent.VTableIndex].lookup(
483 vttComponent.VTableBase);
484 assert(addressPoint.AddressPointIndex != 0 &&
485 "Did not find ctor vtable address point!");
486 }
487
488 mlir::Attribute indices[2] = {
489 cgm.getBuilder().getI32IntegerAttr(addressPoint.VTableIndex),
490 cgm.getBuilder().getI32IntegerAttr(addressPoint.AddressPointIndex),
491 };
492
493 auto indicesAttr = mlir::ArrayAttr::get(mlirContext, indices);
494 cir::GlobalViewAttr init = cgm.getBuilder().getGlobalViewAttr(
495 cgm.getBuilder().getUInt8PtrTy(), vtable, indicesAttr);
496
497 vttComponents.push_back(init);
498 }
499
500 auto init = cir::ConstArrayAttr::get(
501 arrayType, mlir::ArrayAttr::get(mlirContext, vttComponents));
502
503 vttOp.setInitialValueAttr(init);
504
505 // Set the correct linkage.
506 vttOp.setLinkage(linkage);
507 mlir::SymbolTable::setSymbolVisibility(
508 vttOp, CIRGenModule::getMLIRVisibility(vttOp));
509
510 if (cgm.supportsCOMDAT() && vttOp.isWeakForLinker())
511 vttOp.setComdat(true);
512}
513
515 BaseSubobject base) {
516 BaseSubobjectPairTy classSubobjectPair(rd, base);
517
518 SubVTTIndiciesMapTy::iterator it = subVTTIndicies.find(classSubobjectPair);
519 if (it != subVTTIndicies.end())
520 return it->second;
521
522 VTTBuilder builder(cgm.getASTContext(), rd, /*GenerateDefinition=*/false);
523
524 for (const auto &entry : builder.getSubVTTIndices()) {
525 // Insert all indices.
526 BaseSubobjectPairTy subclassSubobjectPair(rd, entry.first);
527
528 subVTTIndicies.insert(std::make_pair(subclassSubobjectPair, entry.second));
529 }
530
531 it = subVTTIndicies.find(classSubobjectPair);
532 assert(it != subVTTIndicies.end() && "Did not find index!");
533
534 return it->second;
535}
536
538 BaseSubobject base) {
539 auto it = secondaryVirtualPointerIndices.find(std::make_pair(rd, base));
540
541 if (it != secondaryVirtualPointerIndices.end())
542 return it->second;
543
544 VTTBuilder builder(cgm.getASTContext(), rd, /*GenerateDefinition=*/false);
545
546 // Insert all secondary vpointer indices.
547 for (const auto &entry : builder.getSecondaryVirtualPointerIndices()) {
548 std::pair<const CXXRecordDecl *, BaseSubobject> pair =
549 std::make_pair(rd, entry.first);
550
551 secondaryVirtualPointerIndices.insert(std::make_pair(pair, entry.second));
552 }
553
554 it = secondaryVirtualPointerIndices.find(std::make_pair(rd, base));
555 assert(it != secondaryVirtualPointerIndices.end() && "Did not find index!");
556
557 return it->second;
558}
559
561 RValue rv, const ThunkInfo &thunk) {
562 // Emit the return adjustment. For non-reference pointer returns, match
563 // classic codegen: skip the adjustment when the returned pointer is null.
564 bool nullCheckValue = !resultType->isReferenceType();
565 mlir::Value returnValue = rv.getValue();
566
567 const CXXRecordDecl *classDecl =
568 resultType->getPointeeType()->getAsCXXRecordDecl();
569 CharUnits classAlign = cgf.cgm.getClassPointerAlignment(classDecl);
570 mlir::Type pointeeType = cgf.convertTypeForMem(resultType->getPointeeType());
571 CIRGenBuilderTy &builder = cgf.getBuilder();
572 mlir::Location loc = returnValue.getLoc();
573
574 if (!nullCheckValue) {
575 returnValue = cgf.cgm.getCXXABI().performReturnAdjustment(
576 cgf, Address(returnValue, pointeeType, classAlign), classDecl,
577 thunk.Return);
578 return RValue::get(returnValue);
579 }
580
581 mlir::Value isNotNull = builder.createPtrIsNotNull(returnValue);
582 returnValue =
583 cir::TernaryOp::create(
584 builder, loc, isNotNull,
585 [&](mlir::OpBuilder &, mlir::Location) {
586 mlir::Value adjusted = cgf.cgm.getCXXABI().performReturnAdjustment(
587 cgf, Address(returnValue, pointeeType, classAlign), classDecl,
588 thunk.Return);
589 builder.createYield(loc, adjusted);
590 },
591 [&](mlir::OpBuilder &, mlir::Location) {
592 mlir::Value nullVal =
593 builder.getNullPtr(returnValue.getType(), loc).getResult();
594 builder.createYield(loc, nullVal);
595 })
596 .getResult();
597
598 return RValue::get(returnValue);
599}
600
602 const CIRGenFunctionInfo &fnInfo,
603 bool isUnprototyped) {
604 assert(!curGD.getDecl() && "curGD was already set!");
605 curGD = gd;
606 curFuncIsThunk = true;
607
608 // Build FunctionArgs.
609 const CXXMethodDecl *md = cast<CXXMethodDecl>(gd.getDecl());
610 QualType thisType = md->getThisType();
611 QualType resultType;
612 if (isUnprototyped)
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;
618 else
619 resultType = md->getType()->castAs<FunctionProtoType>()->getReturnType();
620 FunctionArgList functionArgs;
621
622 // Create the implicit 'this' parameter declaration.
623 cgm.getCXXABI().buildThisParam(*this, functionArgs);
624
625 // Add the rest of the parameters, if we have a prototype to work with.
626 if (!isUnprototyped) {
627 functionArgs.append(md->param_begin(), md->param_end());
628
630 cgm.getCXXABI().addImplicitStructorParams(*this, resultType,
631 functionArgs);
632 }
633
635
636 // Start defining the function.
637 cir::FuncType funcType = cgm.getTypes().getFunctionType(fnInfo);
638 startFunction(GlobalDecl(), resultType, fn, funcType, functionArgs,
639 md->getLocation(), md->getLocation());
640 // TODO(cir): Move this into startFunction.
641 curFnInfo = &fnInfo;
643
644 // Since we didn't pass a GlobalDecl to startFunction, do this ourselves.
645 cgm.getCXXABI().emitInstanceFunctionProlog(md->getLocation(), *this);
647 curCodeDecl = md;
648 curFuncDecl = md;
649}
650
652 // Clear these to restore the invariants expected by
653 // startFunction/finishFunction.
654 curCodeDecl = nullptr;
655 curFuncDecl = nullptr;
656
658}
659
661 const ThunkInfo *thunk,
662 bool isUnprototyped) {
663 assert(isa<CXXMethodDecl>(curGD.getDecl()) &&
664 "Please use a new CGF for this thunk");
665 const CXXMethodDecl *md = cast<CXXMethodDecl>(curGD.getDecl());
666
667 // Determine the this pointer class (may differ from md's class for thunks).
668 const CXXRecordDecl *thisValueClass =
670 if (thunk)
671 thisValueClass = thunk->ThisType->getPointeeCXXRecordDecl();
672
673 mlir::Value adjustedThisPtr =
674 thunk ? cgm.getCXXABI().performThisAdjustment(*this, loadCXXThisAddress(),
675 thisValueClass, *thunk)
676 : loadCXXThis();
677
678 // If perfect forwarding is required a variadic method, a method using
679 // inalloca, or an unprototyped thunk, use musttail. Emit an error if this
680 // thunk requires a return adjustment, since that is impossible with musttail.
682 if ((curFnInfo && curFnInfo->isVariadic()) || isUnprototyped) {
683 // Error if return adjustment is needed (can't do with musttail).
684 if (thunk && !thunk->Return.isEmpty()) {
685 if (isUnprototyped)
686 cgm.errorUnsupported(
687 md, "return-adjusting thunk with incomplete parameter type");
688 else if (curFnInfo && curFnInfo->isVariadic())
689 llvm_unreachable("shouldn't try to emit musttail return-adjusting "
690 "thunks for variadic functions");
691 else
692 cgm.errorUnsupported(
693 md, "non-trivial argument copy for return-adjusting thunk");
694 }
695 emitMustTailThunk(curGD, adjustedThisPtr, callee);
696 return;
697 }
698
699 // Build the call argument list.
700 CallArgList callArgs;
701 QualType thisType = md->getThisType();
702 callArgs.add(RValue::get(adjustedThisPtr), thisType);
703
705 cgm.getCXXABI().adjustCallArgsForDestructorThunk(*this, curGD, callArgs);
706
707#ifndef NDEBUG
708 unsigned prefixArgs = callArgs.size() - 1;
709#endif
710
711 // Add the rest of the method parameters.
712 for (const ParmVarDecl *pd : md->parameters())
713 emitDelegateCallArg(callArgs, pd, SourceLocation());
714
715 const FunctionProtoType *fpt = md->getType()->castAs<FunctionProtoType>();
716
717#ifndef NDEBUG
718 const CIRGenFunctionInfo &callFnInfo = cgm.getTypes().arrangeCXXMethodCall(
719 callArgs, fpt, RequiredArgs::getFromProtoWithExtraSlots(fpt, 1),
720 prefixArgs);
721 assert(callFnInfo.argTypeSize() == curFnInfo->argTypeSize());
722#endif
723
724 // Determine whether we have a return value slot to use.
725 QualType resultType = cgm.getCXXABI().hasThisReturn(curGD) ? thisType
726 : cgm.getCXXABI().hasMostDerivedReturn(curGD)
727 ? cgm.getASTContext().VoidPtrTy
728 : fpt->getReturnType();
729
730 ReturnValueSlot slot;
731 // This should also be tracking volatile, unused, and externally destructed.
733 if (!resultType->isVoidType() && hasAggregateEvaluationKind(resultType))
735
736 // Now emit our call.
737 CIRGenCallee cirCallee = CIRGenCallee::forDirect(callee, curGD);
738 mlir::Location loc = builder.getUnknownLoc();
739 RValue rv = emitCall(*curFnInfo, cirCallee, slot, callArgs,
740 /*callOrTryCall=*/nullptr, loc);
741
742 // Consider return adjustment if we have ThunkInfo.
743 if (thunk && !thunk->Return.isEmpty())
744 rv = performReturnAdjustment(*this, resultType, rv, *thunk);
745 else
747
748 // Emit return.
749 if (!resultType->isVoidType() && slot.isNull())
750 cgm.getCXXABI().emitReturnFromThunk(*this, rv, resultType);
751
752 // Disable final ARC autorelease.
754
755 finishThunk();
756}
757
759 mlir::Value adjustedThisPtr,
760 cir::FuncOp callee) {
761 // Forward all function arguments, replacing 'this' with the adjusted pointer.
762 // The call is marked musttail so varargs are forwarded correctly.
763 mlir::Block *entryBlock = getCurFunctionEntryBlock();
765 for (mlir::BlockArgument arg : entryBlock->getArguments())
766 args.push_back(arg);
767
768 // Replace the 'this' argument (first arg) with the adjusted pointer.
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;
773
774 mlir::Location loc = curFn->getLoc();
775 cir::FuncType calleeTy = callee.getFunctionType();
776 mlir::Type retTy = calleeTy.getReturnType();
777
778 cir::CallOp call = builder.createCallOp(loc, callee, args);
779 call->setAttr(cir::CIRDialect::getMustTailAttrName(),
780 mlir::UnitAttr::get(builder.getContext()));
781
782 if (isa<cir::VoidType>(retTy))
783 cir::ReturnOp::create(builder, loc);
784 else
785 cir::ReturnOp::create(builder, loc, call->getResult(0));
786
787 finishThunk();
788}
789
791 const CIRGenFunctionInfo &fnInfo,
792 GlobalDecl gd, const ThunkInfo &thunk,
793 bool isUnprototyped) {
794 // Create entry block and set up the builder's insertion point.
795 // This must be done before calling startThunk() which calls startFunction().
796 assert(fn.isDeclaration() && "Function already has body?");
797 mlir::Block *entryBb = fn.addEntryBlock();
798 builder.setInsertionPointToStart(entryBb);
799
800 // Create a scope in the symbol table to hold variable declarations.
801 // This is required before startFunction processes parameters, as it will
802 // insert them into the symbolTable (ScopedHashTable) which requires an
803 // active scope.
805
806 // Create lexical scope - must stay alive for entire thunk generation.
807 // startFunction() requires currLexScope to be set.
808 SourceLocRAIIObject locRAII(*this, fn.getLoc());
809 LexicalScope lexScope{*this, fn.getLoc(), entryBb};
810
811 startThunk(fn, gd, fnInfo, isUnprototyped);
813
814 // Get our callee. Use a placeholder type if this method is unprototyped so
815 // that CIRGenModule doesn't try to set attributes.
816 mlir::Type ty;
817 if (isUnprototyped)
818 cgm.errorNYI("unprototyped thunk placeholder type");
819 else
820 ty = cgm.getTypes().getFunctionType(fnInfo);
821
822 cir::FuncOp calleeOp = cgm.getAddrOfFunction(gd, ty, /*forVTable=*/true);
823
824 // Make the call and return the result.
825 emitCallAndReturnForThunk(calleeOp, &thunk, isUnprototyped);
826}
827
829 bool isUnprototyped, bool forVTable) {
830 // Always emit thunks in the MS C++ ABI. We cannot rely on other TUs to
831 // provide thunks for us.
832 if (cgm.getTarget().getCXXABI().isMicrosoft())
833 return true;
834
835 // In the Itanium C++ ABI, vtable thunks are provided by TUs that provide
836 // definitions of the main method. Therefore, emitting thunks with the vtable
837 // is purely an optimization. Emit the thunk if optimizations are enabled and
838 // all of the parameter types are complete.
839 if (forVTable)
840 return cgm.getCodeGenOpts().OptimizationLevel && !isUnprototyped;
841
842 // Always emit thunks along with the method definition.
843 return true;
844}
845
847 const ThunkInfo &thunkAdjustments,
848 bool forVTable) {
849 const CXXMethodDecl *md = cast<CXXMethodDecl>(gd.getDecl());
850 SmallString<256> name;
851 MangleContext &mCtx = cgm.getCXXABI().getMangleContext();
852
853 llvm::raw_svector_ostream out(name);
854 if (const CXXDestructorDecl *dd = dyn_cast<CXXDestructorDecl>(md)) {
855 mCtx.mangleCXXDtorThunk(dd, gd.getDtorType(), thunkAdjustments,
856 /*elideOverrideInfo=*/false, out);
857 } else {
858 mCtx.mangleThunk(md, thunkAdjustments, /*elideOverrideInfo=*/false, out);
859 }
860
861 if (cgm.getASTContext().useAbbreviatedThunkName(gd, name.str())) {
862 name = "";
863 if (const CXXDestructorDecl *dd = dyn_cast<CXXDestructorDecl>(md))
864 mCtx.mangleCXXDtorThunk(dd, gd.getDtorType(), thunkAdjustments,
865 /*elideOverrideInfo=*/true, out);
866 else
867 mCtx.mangleThunk(md, thunkAdjustments, /*elideOverrideInfo=*/true, out);
868 }
869
870 cir::FuncType thunkVTableTy = cgm.getTypes().getFunctionType(gd);
871 cir::FuncOp thunk = cgm.getAddrOfThunk(name, thunkVTableTy, gd);
872
873 // If we don't need to emit a definition, return this declaration as is.
874 bool isUnprototyped = !cgm.getTypes().isFuncTypeConvertible(
875 md->getType()->castAs<FunctionType>());
876 if (!shouldEmitVTableThunk(cgm, md, isUnprototyped, forVTable))
877 return thunk;
878
879 // Arrange a function prototype appropriate for a function definition. In some
880 // cases in the MS ABI, we may need to build an unprototyped musttail thunk.
881 const CIRGenFunctionInfo &fnInfo =
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);
886
887 // This is to replace OG's casting to a function, keeping it here to
888 // streamline the 1-to-1 mapping from OG starting below.
889 cir::FuncOp thunkFn = thunk;
890 if (thunk.getFunctionType() != thunkFnTy) {
891 cir::FuncOp oldThunkFn = thunkFn;
892
893 assert(oldThunkFn.isDeclaration() && "Shouldn't replace non-declaration");
894
895 // Remove the name from the old thunk function and get a new thunk.
896 oldThunkFn.setName(StringRef());
897 thunkFn =
898 cir::FuncOp::create(cgm.getBuilder(), thunk->getLoc(), name.str(),
899 thunkFnTy, cir::GlobalLinkageKind::ExternalLinkage);
900 cgm.setCIRFunctionAttributes(md, fnInfo, thunkFn, /*isThunk=*/false);
901
902 if (!oldThunkFn->use_empty())
903 oldThunkFn->replaceAllUsesWith(thunkFn);
904
905 // Remove the old thunk.
906 oldThunkFn->erase();
907 }
908
909 bool abiHasKeyFunctions = cgm.getTarget().getCXXABI().hasKeyFunctions();
910 bool useAvailableExternallyLinkage = forVTable && abiHasKeyFunctions;
911
912 // If the type of the underlying GlobalValue is wrong, we'll have to replace
913 // it. It should be a declaration.
914 if (!thunkFn.isDeclaration()) {
915 if (!abiHasKeyFunctions || useAvailableExternallyLinkage) {
916 // There is already a thunk emitted for this function, do nothing.
917 return thunkFn;
918 }
919
920 setThunkProperties(cgm, thunkAdjustments, thunkFn, forVTable, gd);
921 return thunkFn;
922 }
923
924 // TODO(cir): Add "thunk" attribute if unprototyped.
925
926 cgm.setCIRFunctionAttributesForDefinition(cast<FunctionDecl>(gd.getDecl()),
927 thunkFn);
928
929 // Thunks for variadic methods are special because in general variadic
930 // arguments cannot be perfectly forwarded. In the general case, clang
931 // implements such thunks by cloning the original function body. However, for
932 // thunks with no return adjustment on targets that support musttail, we can
933 // use musttail to perfectly forward the variadic arguments.
934 bool shouldCloneVarArgs = false;
935 if (!isUnprototyped && thunkFn.getFunctionType().isVarArg()) {
936 shouldCloneVarArgs = true;
937 if (thunkAdjustments.Return.isEmpty()) {
938 switch (cgm.getTriple().getArch()) {
939 case llvm::Triple::x86_64:
940 case llvm::Triple::x86:
941 case llvm::Triple::aarch64:
942 shouldCloneVarArgs = false;
943 break;
944 default:
945 break;
946 }
947 }
948 }
949
950 if (shouldCloneVarArgs) {
951 if (useAvailableExternallyLinkage)
952 return thunkFn;
953 cgm.errorNYI("varargs thunk cloning");
954 } else {
955 // Normal thunk body generation.
956 mlir::OpBuilder::InsertionGuard guard(cgm.getBuilder());
957 CIRGenFunction cgf(cgm, cgm.getBuilder());
958 cgf.generateThunk(thunkFn, fnInfo, gd, thunkAdjustments, isUnprototyped);
959 }
960
961 setThunkProperties(cgm, thunkAdjustments, thunkFn, forVTable, gd);
962 return thunkFn;
963}
964
966 const CXXMethodDecl *md =
967 cast<CXXMethodDecl>(gd.getDecl())->getCanonicalDecl();
968
969 // We don't need to generate thunks for the base destructor.
971 return;
972
973 const VTableContextBase::ThunkInfoVectorTy *thunkInfoVector =
974 vtContext->getThunkInfo(gd);
975
976 if (!thunkInfoVector)
977 return;
978
979 for (const ThunkInfo &thunk : *thunkInfoVector)
980 maybeEmitThunk(gd, thunk, /*forVTable=*/false);
981}
982
984 const CXXRecordDecl *rd) {
985 return cgm.getCodeGenOpts().OptimizationLevel > 0 &&
987}
988
989/// Given that we're currently at the end of the translation unit, and
990/// we've emitted a reference to the vtable for this class, should
991/// we define that vtable?
993 const CXXRecordDecl *rd) {
994 // If vtable is internal then it has to be done.
995 if (!cgm.getVTables().isVTableExternal(rd))
996 return true;
997
998 // If it's external then maybe we will need it as available_externally.
1000}
1001
1002/// Given that at some point we emitted a reference to one or more
1003/// vtables, and that we are now at the end of the translation unit,
1004/// decide whether we should emit them.
1006#ifndef NDEBUG
1007 // Remember the size of DeferredVTables, because we're going to assume
1008 // that this entire operation doesn't modify it.
1009 size_t savedSize = deferredVTables.size();
1010#endif
1011 for (const CXXRecordDecl *rd : deferredVTables) {
1013 vtables.generateClassData(rd);
1015 opportunisticVTables.push_back(rd);
1016 }
1017
1018 assert(savedSize == deferredVTables.size() &&
1019 "deferred extra vtables during vtable emission?");
1020 deferredVTables.clear();
1021}
1022
1024 // Try to emit external vtables as available_externally if they have emitted
1025 // all inlined virtual functions. It runs after EmitDeferred() and therefore
1026 // is not allowed to create new references to things that need to be emitted
1027 // lazily. Note that it also uses fact that we eagerly emitting RTTI.
1028
1029 assert(
1030 (opportunisticVTables.empty() || shouldOpportunisticallyEmitVTables()) &&
1031 "Only emit opportunistic vtables with optimizations");
1032
1033 for (const CXXRecordDecl *rd : opportunisticVTables) {
1034 assert(getVTables().isVTableExternal(rd) &&
1035 "This queue should only contain external vtables");
1036 if (getCXXABI().canSpeculativelyEmitVTable(rd))
1037 vtables.generateClassData(rd);
1038 }
1039 opportunisticVTables.clear();
1040}
1041
1043 return codeGenOpts.OptimizationLevel > 0;
1044}
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)
TokenType getType() const
Returns the token's type, e.g.
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
Definition ASTContext.h:952
const TargetInfo & getTargetInfo() const
Definition ASTContext.h:917
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())
Definition CIRGenCall.h:92
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
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
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
static mlir::SymbolTable::Visibility getMLIRVisibility(Visibility v)
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...
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.
Definition CIRGenCall.h:193
This trivial value class is used to represent the result of an expression that is evaluated.
Definition CIRGenValue.h:33
static RValue get(mlir::Value v)
Definition CIRGenValue.h:83
mlir::Value getValue() const
Return the value of this scalar value.
Definition CIRGenValue.h:57
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...
Definition CIRGenCall.h:260
Represents a C++ destructor within a class.
Definition DeclCXX.h:2889
Represents a static or instance method of a struct/union/class.
Definition DeclCXX.h:2136
QualType getThisType() const
Return the type of the this pointer.
Definition DeclCXX.cpp:2861
Represents a C++ struct/union/class.
Definition DeclCXX.h:258
TemplateSpecializationKind getTemplateSpecializationKind() const
Determine whether this particular class is a specialization or instantiation of a class template or m...
Definition DeclCXX.cpp:2060
bool isDynamicClass() const
Definition DeclCXX.h:574
unsigned getNumVBases() const
Retrieves the number of virtual base classes of this class.
Definition DeclCXX.h:623
CharUnits - This is an opaque type for sizes expressed in character units.
Definition CharUnits.h:38
bool isZero() const
isZero - Test whether the quantity equals zero.
Definition CharUnits.h:122
QuantityType getQuantity() const
getQuantity - Get the raw integer representation of this quantity.
Definition CharUnits.h:185
static CharUnits fromQuantity(QuantityType Quantity)
fromQuantity - Construct a CharUnits quantity from a raw integer type.
Definition CharUnits.h:63
ASTContext & getASTContext() const LLVM_READONLY
Definition DeclBase.cpp:547
SourceLocation getLocation() const
Definition DeclBase.h:447
bool hasAttr() const
Definition DeclBase.h:585
virtual SourceRange getSourceRange() const LLVM_READONLY
Source range that this declaration covers.
Definition DeclBase.h:435
Represents a function declaration or definition.
Definition Decl.h:2015
param_iterator param_end()
Definition Decl.h:2802
bool isInlined() const
Determine whether this function should be inlined, because it is either marked "inline" or "constexpr...
Definition Decl.h:2936
ArrayRef< ParmVarDecl * > parameters() const
Definition Decl.h:2789
param_iterator param_begin()
Definition Decl.h:2801
TemplateSpecializationKind getTemplateSpecializationKind() const
Determine what kind of template instantiation this function represents.
Definition Decl.cpp:4419
bool hasBody(const FunctionDecl *&Definition) const
Returns true if the function has a body.
Definition Decl.cpp:3201
Represents a prototype with parameter type info, e.g.
Definition TypeBase.h:5357
FunctionType - C99 6.7.5.3 - Function Declarators.
Definition TypeBase.h:4553
QualType getReturnType() const
Definition TypeBase.h:4893
GlobalDecl - represents a global declaration.
Definition GlobalDecl.h:57
CXXDtorType getDtorType() const
Definition GlobalDecl.h:113
const Decl * getDecl() const
Definition GlobalDecl.h:106
const VTableLayout & getVTableLayout(const CXXRecordDecl *RD)
MangleContext - Context for tracking state which persists across multiple calls to the C++ name mangl...
Definition Mangle.h:56
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
Definition Decl.h:433
Represents a parameter to a function.
Definition Decl.h:1805
A (possibly-)qualified type.
Definition TypeBase.h:937
Encodes a location in the source.
SourceRange getSourceRange() const override LLVM_READONLY
Source range that this declaration covers.
Definition Decl.cpp:4895
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.
bool isVoidType() const
Definition TypeBase.h:9034
CXXRecordDecl * getAsCXXRecordDecl() const
Retrieves the CXXRecordDecl that this type refers to, either because the type is a RecordType or beca...
Definition Type.h:26
const T * castAs() const
Member-template castAs<specific type>.
Definition TypeBase.h:9328
bool isReferenceType() const
Definition TypeBase.h:8692
const CXXRecordDecl * getPointeeCXXRecordDecl() const
If this is a pointer or reference to a RecordType, return the CXXRecordDecl that the type refers to.
Definition Type.cpp:1923
QualType getPointeeType() const
If this is a pointer, ObjC object pointer, or block pointer, this returns the respective pointee.
Definition Type.cpp:754
Class for building VTT layout information.
Definition VTTBuilder.h:71
const llvm::DenseMap< BaseSubobject, uint64_t > & getSecondaryVirtualPointerIndices() const
Returns a reference to the secondary virtual pointer indices.
Definition VTTBuilder.h:157
const llvm::DenseMap< BaseSubobject, uint64_t > & getSubVTTIndices() const
Returns a reference to the sub-VTT indices.
Definition VTTBuilder.h:151
const VTTComponentsVectorTy & getVTTComponents() const
Definition VTTBuilder.h:141
const VTTVTablesVectorTy & getVTTVTables() const
Definition VTTBuilder.h:146
CharUnits getBaseOffset() const
Definition VTTBuilder.h:48
bool isVirtual() const
Definition VTTBuilder.h:52
const CXXRecordDecl * getBase() const
Definition VTTBuilder.h:44
BaseSubobject getBaseSubobject() const
Definition VTTBuilder.h:56
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
QualType getType() const
Definition Decl.h:723
static bool isLocalLinkage(GlobalLinkageKind linkage)
Definition CIROpsEnums.h:51
const AstTypeMatcher< ArrayType > arrayType
The JSON file list parser is used to communicate input to InstallAPI.
bool isa(CodeGen::Address addr)
Definition Address.h:330
@ Dtor_Base
Base object dtor.
Definition ABI.h:37
TemplateSpecializationKind
Describes the kind of template specialization that a particular template specialization declaration r...
Definition Specifiers.h:189
@ TSK_ExplicitInstantiationDefinition
This template specialization was instantiated from a template due to an explicit instantiation defini...
Definition Specifiers.h:207
@ TSK_ExplicitInstantiationDeclaration
This template specialization was instantiated from a template due to an explicit instantiation declar...
Definition Specifiers.h:203
@ TSK_ExplicitSpecialization
This template specialization was declared or defined by an explicit specialization (C++ [temp....
Definition Specifiers.h:199
@ TSK_ImplicitInstantiation
This template specialization was implicitly instantiated from a template.
Definition Specifiers.h:195
@ TSK_Undeclared
This template specialization was formed from a template-id but has not yet been declared,...
Definition Specifiers.h:192
U cast(CodeGen::Address addr)
Definition Address.h:327
Diagnostic wrappers for TextAPI types for error reporting.
Definition Dominators.h:30
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...
bool isEmpty() const
Definition Thunk.h:70
The this pointer adjustment as well as an optional return adjustment for a thunk.
Definition Thunk.h:157
ReturnAdjustment Return
The return adjustment.
Definition Thunk.h:162
const Type * ThisType
Definition Thunk.h:173