8 #include <clang/AST/VTableBuilder.h>
26 using namespace asthelpers;
30 static llvm::DenseMap<std::pair<const CXXRecordDecl*, const CXXRecordDecl*>,
int>
mThisPointerOffset{};
33 static MemberExpr*
AccessMember(std::string_view name,
const ValueDecl* vd, QualType type)
46 vtableRecorDecl =
Struct(
"__mptr"sv);
47 auto AddField = [&](FieldDecl* field) { vtableRecorDecl->addDecl(field); };
52 f =
mkFieldDecl(vtableRecorDecl,
"f"sv, vptpTypedef);
55 vtableRecorDecl->completeDefinition();
57 vtableRecordType = QualType{vtableRecorDecl->getTypeForDecl(), 0u};
81 CodeGeneratorVariant::CodeGenerators::CodeGenerators(
OutputFormatHelper& _outputFormatHelper,
83 CodeGenerator::ProcessingPrimaryTemplate processingPrimaryTemplate)
87 _outputFormatHelper, lambdaStack, CodeGenerator::LambdaInInitCapture::No, processingPrimaryTemplate};
90 _outputFormatHelper, lambdaStack, CodeGenerator::LambdaInInitCapture::No, processingPrimaryTemplate};
95 CodeGeneratorVariant::CodeGenerators::CodeGenerators(OutputFormatHelper& _outputFormatHelper,
96 CodeGenerator::LambdaInInitCapture lambdaInitCapture)
99 new(&cfcg) CfrontCodeGenerator{_outputFormatHelper, lambdaInitCapture};
101 new(&cg)
CodeGenerator{_outputFormatHelper, lambdaInitCapture};
106 CodeGeneratorVariant::CodeGenerators::~CodeGenerators()
109 cfcg.~CfrontCodeGenerator();
116 void CodeGeneratorVariant::Set()
128 return ctor and (ctor->isCopyConstructor() or ctor->isMoveConstructor());
134 return stmt and (stmt->isCopyAssignmentOperator() or stmt->isMoveAssignmentOperator());
140 if(
const auto* md = dyn_cast_or_null<CXXMethodDecl>(vd)) {
142 if(
const auto* ar = dyn_cast_or_null<ArrayType>(type)) {
143 type = ar->getElementType();
148 if(
const auto* ctor = dyn_cast_or_null<CXXConstructorDecl>(md)) {
149 if(ctor->isCopyConstructor()) {
150 return StrCat(
"CopyConstructor_"sv, rdName);
151 }
else if(ctor->isMoveConstructor()) {
152 return StrCat(
"MoveConstructor_"sv, rdName);
155 return StrCat(
"Constructor_"sv, rdName);
157 }
else if(isa<CXXDestructorDecl>(md)) {
158 return StrCat(
"Destructor_"sv, rdName);
168 if(
const auto* md = dyn_cast_or_null<CXXMethodDecl>(vd)) {
184 if(
auto* cxxRecordDecl = t->getAsCXXRecordDecl(); cxxRecordDecl and not cxxRecordDecl->isTrivial()) {
194 if(
auto* cxxRecordDecl = t->getAsCXXRecordDecl(); cxxRecordDecl and not cxxRecordDecl->hasTrivialDestructor()) {
202 static auto*
CallVecDeleteOrDtor(Expr* objectParam, QualType allocatedType, std::string_view name, uint64_t size)
206 SmallVector<Expr*, 4> args{objectParam,
211 return Call(name, args);
223 static auto*
CallVecDtor(Expr* objectParam,
const ConstantArrayType* ar)
233 auto destroyedType = stmt->getDestroyedType();
234 auto* arg =
const_cast<Expr*
>(stmt->getArgument());
238 if(
const auto hasDtor{
HasDtor(destroyedType)}; stmt->isArrayForm() and hasDtor) {
246 bodyStmts.Add(
Call(
"free"sv, {arg}));
255 QualType allocatedType,
257 std::string_view funName)
262 auto dtorName = [&]() {
267 return std::string{
kwNull};
270 SmallVector<Expr*, 6> args{objectParam,
277 return Call(funName, args);
281 static auto*
CallVecNew(std::string_view ctorName, Expr* objectParam, QualType allocatedType, Expr* arraySizeExpr)
285 return CallVecNewOrCtor(ctorName, objectParam, allocatedType, arraySizeExpr,
"__cxa_vec_new"sv);
290 CallVecCtor(std::string_view ctorName,
const VarDecl* objectParam, QualType allocatedType, Expr* arraySizeExpr)
300 auto* stmt =
const_cast<CXXNewExpr*
>(cstmt);
302 auto allocatedType = stmt->getAllocatedType();
303 auto ctorName =
StrCat(
"Constructor_"sv,
GetName(allocatedType));
305 if(stmt->isArray()) {
308 auto* arraySizeExpr = stmt->getArraySize().value();
309 auto* callMalloc =
Call(
"malloc"sv, {
Mul(
Sizeof(allocatedType), arraySizeExpr)});
316 assign =
Cast(callMalloc,
Ptr(allocatedType));
319 if(allocatedType->getAsCXXRecordDecl()->ctors().empty()) {
325 assign =
Cast(
CallVecNew(ctorName, callMalloc, allocatedType, arraySizeExpr),
Ptr(allocatedType));
330 if(stmt->hasInitializer() and allocatedType->isBuiltinType()) {
346 if(
auto* inits = dyn_cast_or_null<InitListExpr>(stmt->getInitializer())) {
347 auto* expr = [&]() ->
const Expr* {
348 if(
auto* vd = dyn_cast_or_null<VarDecl>(
mLastDecl)) {
370 auto* mallocCall = [&]() -> Expr* {
371 if(stmt->getNumPlacementArgs()) {
372 return stmt->getPlacementArg(0);
377 return Call(
"malloc"sv, {
Sizeof(allocatedType)});
380 mallocCall =
Cast(mallocCall,
Ptr(allocatedType));
382 if(allocatedType->isBuiltinType()) {
385 if(stmt->hasInitializer()) {
413 SmallVector<Expr*, 16> args{mallocCall};
414 if(
auto* ncCtorExpr =
const_cast<CXXConstructExpr*
>(stmt->getConstructExpr())) {
415 args.append(ncCtorExpr->arg_begin(), ncCtorExpr->arg_end());
424 const auto* callee = stmt->getDirectCallee();
426 if(
const auto* cxxCallee = dyn_cast_or_null<CXXMethodDecl>(callee);
IsCopyOrMoveAssign(cxxCallee)) {
427 SmallVector<Expr*, 16> args{};
429 for(
const auto& arg : stmt->arguments()) {
430 args.push_back(
Ref(arg));
436 InsertArg(dyn_cast_or_null<CallExpr>(stmt));
443 if(cur->isPolymorphic() and (0 == cur->getNumBases())) {
454 }
else if(cur->isPolymorphic() and (0 < cur->getNumBases()) and (cur != stmt->getParent())) {
455 for(
const auto& base : cur->bases()) {
456 InsertVtblPtr(stmt, base.getType()->getAsCXXRecordDecl(), bodyStmts);
466 (
IsCopyOrMoveAssign(stmt) or (not stmt->hasBody() and not isa<CXXConstructorDecl>(stmt))));
472 if(stmt->isConst()) {
473 recordDeclType.addConst();
476 auto parentType =
Ptr(recordDeclType);
477 auto* body = stmt->getBody();
480 auto retType = stmt->getReturnType();
482 auto processBaseClassesAndFields = [&](
const CXXRecordDecl* parent) {
485 auto insertFields = [&](
const RecordDecl* rd) {
486 for(
auto* fieldDecl : rd->fields()) {
487 if(
const auto* cxxRecordDecl = fieldDecl->getType()->getAsCXXRecordDecl()) {
491 auto lvalueRefType =
GetGlobalAST().getLValueReferenceType(parentType);
493 auto* rhsMemberExpr =
AccessMember(
"__rhs"sv, fieldDecl, lvalueRefType);
497 {
Ref(lhsMemberExpr),
Ref(rhsMemberExpr)}));
500 auto* rhsMemberExpr =
AccessMember(
"__rhs"sv, fieldDecl, parentType);
502 bodyStmts.AddBodyStmts(
Assign(thisOfParent, fieldDecl, rhsMemberExpr));
507 for(
const auto& base : parent->bases()) {
508 const auto rd = base.getType()->getAsRecordDecl();
516 bodyStmts.AddBodyStmts(callAssignBase);
520 insertFields(parent);
524 if(
const auto* ctorDecl = dyn_cast_or_null<CXXConstructorDecl>(stmt)) {
525 const auto* parent = stmt->getParent();
527 if(not stmt->doesThisDeclarationHaveABody()) {
529 if(
const bool showSpecialMemberFunc{stmt->isUserProvided() or stmt->isExplicitlyDefaulted()};
530 not showSpecialMemberFunc) {
534 processBaseClassesAndFields(parent);
536 }
else if(ctorDecl->isDefaultConstructor()) {
537 auto insertFields = [&](
const RecordDecl* rd) {
538 for(
auto* fieldDecl : rd->fields()) {
539 if(
auto* initializer = fieldDecl->getInClassInitializer();
540 initializer and fieldDecl->hasInClassInitializer()) {
541 const bool isConstructorExpr{isa<CXXConstructExpr>(initializer) or
542 isa<ExprWithCleanups>(initializer)};
544 if(not isConstructorExpr) {
547 bodyStmts.AddBodyStmts(
Assign(lhsMemberExpr, fieldDecl, initializer));
558 }
else if(
const auto* cxxRecordDecl = fieldDecl->getType()->getAsCXXRecordDecl()) {
562 if(
auto lhsType = fieldDecl->getType();
HasCtor(lhsType)) {
579 for(
const auto& base : parent->bases()) {
580 auto baseType = base.getType();
582 if(
const auto* baseRd = baseType->getAsCXXRecordDecl();
583 baseRd and baseRd->hasNonTrivialDefaultConstructor()) {
584 bodyStmts.AddBodyStmts(
CallConstructor(baseType,
Ptr(baseType),
nullptr, {}, DoCast::Yes));
592 for(
const auto& base : parent->bases()) {
593 InsertVtblPtr(stmt, base.getType()->getAsCXXRecordDecl(), bodyStmts);
597 insertFields(parent);
601 for(
const auto* init : ctorDecl->inits()) {
602 auto* member = init->getMember();
604 if(not isa<CXXConstructExpr>(init->getInit())) {
605 if(not init->getAnyMember()) {
611 bodyStmts.AddBodyStmts(
Assign(lhsMemberExpr, member, init->getInit()));
614 }
else if(init->isBaseInitializer()) {
615 bodyStmts.AddBodyStmts(init->getInit());
619 auto ctorType = member->getType();
624 callParams.insert(callParams.begin(),
Ref(lhsMemberExpr));
626 bodyStmts.AddBodyStmts(
632 bodyStmts.AddBodyStmts(body);
638 retType = parentType;
642 if(not stmt->doesThisDeclarationHaveABody() or stmt->isDefaulted()) {
647 processBaseClassesAndFields(stmt->getParent());
649 bodyStmts.AddBodyStmts(body);
655 retType = parentType;
657 }
else if(
const auto* dtor = dyn_cast_or_null<CXXDestructorDecl>(stmt)) {
661 bodyStmts.AddBodyStmts(body);
668 for(
const auto& base : llvm::reverse(dtor->getParent()->bases())) {
669 if(not dtor->isVirtual()) {
674 auto* cast =
Cast(lhsDeclRef, lhsDeclRef->getType());
684 params.reserve(stmt->getNumParams() + 1);
690 for(
const auto& param : stmt->parameters()) {
691 std::string name{
GetName(*param)};
692 auto type = param->getType();
695 if((0 == name.length()) and
698 type =
Ptr(type.getNonReferenceType());
701 params.emplace_back(name, type);
705 callSpecialMemberFn->setInlineSpecified(stmt->isInlined());
706 callSpecialMemberFn->setStorageClass((stmt->isStatic() or
IsStaticStorageClass(stmt)) ? SC_Static : SC_None);
707 callSpecialMemberFn->setBody(body);
716 const QualType& castDestType,
718 const CastKind& kind)
722 if(not castDestType->isRValueReferenceType() and not(CastKind::CK_UncheckedDerivedToBase == kind)) {
728 if(
is{kind}.
any_of(CastKind::CK_DerivedToBase, CastKind::CK_BaseToDerived)) {
731 sign] = [&]() -> std::pair<std::pair<const CXXRecordDecl*, const CXXRecordDecl*>, std::string_view> {
732 auto plainType = [](QualType t) {
733 if(
const auto* pt = dyn_cast_or_null<PointerType>(t.getTypePtrOrNull())) {
734 return pt->getPointeeType()->getAsCXXRecordDecl();
737 return t->getAsCXXRecordDecl();
740 auto base = plainType(castDestType);
741 auto derived = plainType(subExpr->getType());
743 if((CastKind::CK_BaseToDerived == kind)) {
744 return {{base, derived},
"-"sv};
747 return {{derived, base},
"+"sv};
781 if(stmt->getMessage()) {
798 static void ProcessFields(CXXRecordDecl* recordDecl,
const CXXRecordDecl* rd)
802 auto AddField = [&](
const FieldDecl* field) {
807 for(
const auto& base : rd->bases()) {
809 if(
const auto* rdBase = dyn_cast_or_null<CXXRecordDecl>(base.getType().getCanonicalType()->getAsRecordDecl())) {
815 if(rd->isPolymorphic() and (rd->getNumBases() == 0)) {
820 for(
const auto* d : rd->fields()) {
824 if(recordDecl->field_empty()) {
832 std::string ret{
GetName(*decl)};
834 if(
const auto* rdecl = dyn_cast_or_null<CXXRecordDecl>(decl); rdecl->getNumBases() > 1) {
835 for(
const auto& base : rdecl->bases()) {
836 if(
const auto* rd = base.getType()->getAsRecordDecl(); rd == to) {
851 if(stmt->hasDefinition() and stmt->isPolymorphic()) {
853 static_cast<ItaniumVTableContext*
>(
const_cast<ASTContext&
>(
GetGlobalAST()).getVTableContext())) {
856 auto* mc =
const_cast<ASTContext&
>(
GetGlobalAST()).createMangleContext(
nullptr);
857 SmallString<256> rttiName{};
858 llvm::raw_svector_ostream out(rttiName);
859 mc->mangleCXXRTTI(QualType(stmt->getTypeForDecl(), 0), out);
860 DPrint(
"name: %s\n", rttiName.c_str());
863 SmallVector<Expr*, 16> mInitExprs{};
864 SmallVector<QualType, 5> baseList{};
866 if(stmt->getNumBases() == 0) {
870 for(
const auto& base : stmt->bases()) {
871 baseList.push_back(base.getType());
874 llvm::DenseMap<uint64_t, ThunkInfo> thunkMap{};
875 const VTableLayout& layout{itctx->getVTableLayout(stmt)};
877 for(
const auto& [idx, thunk] : layout.vtable_thunks()) {
878 thunkMap[idx] = thunk;
885 auto pushVtable = [&] {
890 auto* thisRd = baseList[clsIdx - 1]->getAsCXXRecordDecl();
893 vtabl->setInit(
InitList(mInitExprs, vtblData.vtableRecordType));
903 for(
unsigned i = 0;
const auto& vc : layout.vtable_components()) {
904 switch(vc.getKind()) {
905 case VTableComponent::CK_OffsetToTop: {
906 auto off = layout.getVTableOffset(clsIdx);
907 if(
auto rem = (off % 4)) {
919 case VTableComponent::CK_RTTI:
927 case VTableComponent::CK_CompleteDtorPointer:
929 case VTableComponent::CK_DeletingDtorPointer:
930 case VTableComponent::CK_FunctionPointer: {
931 auto* thunkOffset = [&] {
932 if(ThunkInfo thunk = thunkMap.lookup(i); not thunk.isEmpty() and not thunk.This.isEmpty()) {
933 return Int32(thunk.This.NonVirtual);
939 const auto* md = dyn_cast_or_null<FunctionDecl>(vc.getFunctionDecl());
942 if(md->isPureVirtual()) {
955 mInitExprs.push_back(
InitList({thunkOffset,
Int32(0), reicast}, vtblData.vtableRecordType));
970 if(stmt->hasDefinition()) {
972 recordDecl->completeDefinition();
983 auto et = ctx.getElaboratedType(ElaboratedTypeKeyword::ETK_Struct,
nullptr,
GetRecordDeclType(recordDecl),
nullptr);
989 for(
OnceTrue firstRecordDecl{};
const auto* d : stmt->decls()) {
990 if((isa<CXXRecordDecl>(d) and firstRecordDecl)
991 or (stmt->isLambda() and isa<CXXDestructorDecl>(d))
993 or isa<AccessSpecDecl>(d)
999 if((stmt->isTrivial() and isa<CXXConstructorDecl>(d)) or
1000 (stmt->hasTrivialDestructor() and isa<CXXDestructorDecl>(d))) {
1012 if(
const auto* rdecl = dyn_cast_or_null<CXXRecordDecl>(decl); rdecl->getNumBases() >= 1) {
1013 for(
const auto& base : rdecl->bases()) {
1014 const auto* rd = base.getType()->getAsRecordDecl();
1016 if(
const auto* cxxRd = dyn_cast_or_null<CXXRecordDecl>(rd); not cxxRd or not cxxRd->isPolymorphic()) {
1026 return dyn_cast_or_null<CXXRecordDecl>(decl);
1032 if(
const auto* me = dyn_cast_or_null<MemberExpr>(stmt->getCallee())) {
1033 auto* obj = me->getBase();
1034 const bool isPointer{obj->getType()->isPointerType()};
1036 if(
const bool isReference =
IsReferenceType(dyn_cast_or_null<VarDecl>(obj->getReferencedDeclOfCallee()));
1037 not isPointer and not isReference) {
1041 if(
const auto* matExpr = dyn_cast_or_null<MaterializeTemporaryExpr>(me->getBase())) {
1042 if(
const auto* tmpExpr = dyn_cast_or_null<CXXBindTemporaryExpr>(matExpr->getSubExpr())) {
1043 if(
const auto* tmpObjExpr = dyn_cast_or_null<CXXTemporaryObjectExpr>(tmpExpr->getSubExpr())) {
1044 obj =
const_cast<CXXTemporaryObjectExpr*
>(tmpObjExpr);
1049 auto* memDecl = me->getMemberDecl();
1051 if(
const auto* ar = dyn_cast_or_null<ConstantArrayType>(obj->getType())) {
1052 if(
const auto* dtor = dyn_cast_or_null<CXXDestructorDecl>(memDecl)) {
1059 SmallVector<Expr*, 16> params{obj};
1060 auto* ncStmt =
const_cast<CXXMemberCallExpr*
>(stmt);
1061 params.append(ncStmt->arg_begin(), ncStmt->arg_end());
1063 if(
auto* md = dyn_cast_or_null<CXXMethodDecl>(memDecl); md and md->isVirtual()) {
1065 auto* cls = md->getParent();
1071 auto destType = not isPointer ?
Ptr(obj->getType()) : obj->getType();
1072 auto atype = isPointer ? obj->getType()->getPointeeType() : obj->getType();
1073 auto idx = mVirtualFunctions[{md, {atype->getAsCXXRecordDecl(), vRecordDecl}}];
1077 auto* vtblArrayPos =
ArraySubscript(accessVptr, idx, vtblField->getType());
1079 auto* p =
Paren(vtblArrayPos);
1080 auto* accessMemberF =
AccessMember(p, vtblData.f,
false);
1085 auto* reicast =
ReinterpretCast(funcPtrFuncDecl->getType(), accessMemberF,
true);
1087 auto* p4 =
Paren(reicast);
1089 auto* p6 =
Paren(p5);
1098 auto* p10 =
Paren(p9);
1099 auto* p11 =
Plus(p10, p7);
1100 auto* p12 =
Paren(p11);
1118 if(not stmt->isMain()) {
1124 SmallVector<Expr*, 16> args{};
1126 for(
auto* param : stmt->parameters()) {
1127 params.emplace_back(
GetName(*param), param->getType());
1131 auto mainName{
"main"sv};
1135 intMain->setBody(stmt->getBody());
1136 intMain->setHasImplicitReturnZero(
true);
1141 auto* mainRetVar =
Variable(
"ret"sv, stmt->getReturnType());
1142 mainRetVar->setInit(
Call(trampolinMainName, args));
1144 auto* mainRetVarDeclStmt =
mkDeclStmt(mainRetVar);
1150 modMain->setBody(body);
1158 if(
P0315Visitor dt{*
this}; not dt.TraverseType(stmt->getType())) {
1166 auto ctor = stmt->getConstructor();
1168 auto* vd = dyn_cast_or_null<VarDecl>(
mLastDecl);
1173 ar = dyn_cast_or_null<ConstantArrayType>(vd->getType());
1184 auto InsertCallCtor = [&](Expr* varNameRef) {
1185 SmallVector<Expr*, 16> args{
Cast(varNameRef,
Ptr(stmt->getType()))};
1187 for(
int i = 0;
auto* arg : stmt->arguments()) {
1188 if(
IsCopyOrMoveCtor(ctor) or ctor->getParamDecl(i)->getType()->isReferenceType()) {
1189 args.push_back(
Ref(arg));
1192 args.push_back(
const_cast<Expr*
>(arg));
1202 if(
const auto* ar = dyn_cast_or_null<ConstantArrayType>(stmt->getType())) {
1203 if(not
HasCtor(ar->getElementType())) {
1204 mInsertSemi =
false;
1210 }
else if(
const auto* tmpObjectExpr = dyn_cast_or_null<CXXTemporaryObjectExpr>(stmt); vd and not tmpObjectExpr) {
1211 if(not
HasCtor(vd->getType())) {
1212 mInsertSemi =
false;
1216 InsertCallCtor(varNameRef);
1219 }
else if(tmpObjectExpr) {
1222 if(not
HasCtor(stmt->getType())) {
1227 InsertCallCtor(varNameRef);
const ASTContext & GetGlobalAST()
Get access to the ASTContext.
const InsightsOptions & GetInsightsOptions()
Get the global C++ Insights options.
constexpr std::string_view kwNull
constexpr std::string_view cxaStart
constexpr std::string_view kwInternalThis
constexpr std::string_view kwTypedefSpace
constexpr std::string_view cxaAtExit
#define RETURN_IF(cond)
! A helper inspired by https://github.com/Microsoft/wil/wiki/Error-handling-helpers
constexpr bool empty() const noexcept
A special generator for coroutines. It is only activated, if -show-coroutines-transformation is given...
virtual void InsertArg(const Decl *stmt)
void InsertCXXMethodDecl(const CXXMethodDecl *, CodeGenerator::SkipBody) override
constexpr CodeGenerator(OutputFormatHelper &_outputFormatHelper, LambdaStackType &lambdaStack, LambdaInInitCapture lambdaInitCapture, ProcessingPrimaryTemplate processingPrimaryTemplate)
! We do not want to transform a primary template which contains a Coroutine.
static CfrontVtableData & VtableData()
void FormatCast(const std::string_view, const QualType &, const Expr *, const CastKind &) override
void WrapInParens(void_func_ref lambda, const AddSpaceAtTheEnd addSpaceAtTheEnd=AddSpaceAtTheEnd::No)
virtual void InsertArg(const Decl *stmt)
LambdaStackType & mLambdaStack
OutputFormatHelper & mOutputFormatHelper
! Find a LambdaExpr inside a Decltype
BinaryOperator * Plus(Expr *var, Expr *assignExpr)
CXXReinterpretCastExpr * ReinterpretCast(QualType toType, const Expr *toExpr, bool makePointer)
CXXRecordDecl * Struct(std::string_view name)
FieldDecl * mkFieldDecl(DeclContext *dc, std::string_view name, QualType type)
params_vector to_params_view(params_store ¶ms)
DeclStmt * mkDeclStmt(Dcls... dcls)
DeclRefExpr * mkDeclRefExpr(const ValueDecl *vd)
QualType Typedef(std::string_view name, QualType underlayingType)
std::vector< std::pair< std::string_view, QualType > > params_vector
CallExpr * Call(const FunctionDecl *fd, ArrayRef< Expr * > params)
UnaryOperator * Ref(const Expr *e)
VarDecl * Variable(std::string_view name, QualType type, DeclContext *dc)
MemberExpr * AccessMember(const Expr *expr, const ValueDecl *vd, bool isArrow)
DeclRefExpr * mkVarDeclRefExpr(std::string_view name, QualType type)
SmallVector< Expr *, 5 > ArgsToExprVector(const Expr *expr)
ArraySubscriptExpr * ArraySubscript(const Expr *lhs, uint64_t index, QualType type)
ReturnStmt * Return(Expr *stmt)
CXXStaticCastExpr * Cast(const Expr *toExpr, QualType toType)
CXXStaticCastExpr * CastToVoidFunPtr(std::string_view name)
Stmt * Comment(std::string_view comment)
ParenExpr * Paren(Expr *expr)
IfStmt * If(const Expr *condition, ArrayRef< Stmt * > bodyStmts)
BinaryOperator * Mul(Expr *lhs, Expr *rhs)
InitListExpr * InitList(ArrayRef< Expr * > initExprs, QualType t)
UnaryExprOrTypeTraitExpr * Sizeof(QualType toType)
QualType GetRecordDeclType(const CXXMethodDecl *md)
static CallExpr * CallConstructor(QualType ctorType, DeclRefExpr *lhsDeclRef, Expr *lhsMemberExpr, ArrayRef< Expr * > callParams, DoCast doCast, AsReference asReference)
std::vector< std::pair< std::string, QualType > > params_store
FunctionDecl * Function(std::string_view name, QualType returnType, const params_vector ¶meters)
CompoundStmt * mkCompoundStmt(ArrayRef< Stmt * > bodyStmts, SourceLocation beginLoc, SourceLocation endLoc)
IntegerLiteral * Int32(uint64_t value)
QualType ContantArrayTy(QualType t, int size)
UnaryOperator * Dref(const Expr *stmt)
QualType Ptr(QualType srcType)
CXXStaticCastExpr * StaticCast(QualType toType, const Expr *toExpr, bool makePointer)
BinaryOperator * Assign(const VarDecl *var, Expr *assignExpr)
uint64_t GetSize(const ConstantArrayType *arrayType)
@ HeaderStdlib
Track whether we need to insert <stdlib.h> in Cfront mode.
@ HeaderAssert
Track whether we need to insert <assert.h> in Cfront mode.
@ HeaderStddef
Track whether we need to insert <stddef.h> in Cfront mode.
void DPrint([[maybe_unused]] const char *fmt, [[maybe_unused]] const auto &... args)
Debug print which is disabled in release-mode.
void PushVtableEntry(const CXXRecordDecl *record, const CXXRecordDecl *recordB, VarDecl *decl)
int GetGlobalVtablePos(const CXXRecordDecl *record, const CXXRecordDecl *recordB)
static llvm::DenseMap< std::pair< const CXXRecordDecl *, const CXXRecordDecl * >, int > mThisPointerOffset
Store the this pointer offset from derived to base class.
static auto * CallVecCtor(std::string_view ctorName, const VarDecl *objectParam, QualType allocatedType, Expr *arraySizeExpr)
std::string GetName(const NamedDecl &nd, const QualifiedName qualifiedName)
static void ProcessFields(CXXRecordDecl *recordDecl, const CXXRecordDecl *rd)
std::string BuildInternalVarName(const std::string_view &varName)
static bool IsCopyOrMoveAssign(const CXXMethodDecl *stmt)
static FieldDecl * AddField(CoroutineASTData &astData, std::string_view name, QualType type)
static bool HasDtor(QualType t)
static MemberExpr * AccessMember(std::string_view name, const ValueDecl *vd, QualType type)
static bool HasCtor(QualType t)
static void InsertVtblPtr(const CXXMethodDecl *stmt, const CXXRecordDecl *cur, StmtsContainer &bodyStmts)
static const CXXRecordDecl * GetFirstPolymorphicBase(const RecordDecl *decl)
! Find the first polymorphic base class.
bool IsReferenceType(const ValueDecl *decl)
static auto * CallVecDelete(Expr *objectParam, QualType allocatedType)
static std::string GetSpecialMemberName(const ValueDecl *vd, QualType type)
static bool IsCopyOrMoveCtor(const CXXConstructorDecl *ctor)
bool IsStaticStorageClass(const CXXMethodDecl *md)
static std::string GetFirstPolymorphicBaseName(const RecordDecl *decl, const RecordDecl *to)
void EnableGlobalInsert(GlobalInserts idx)
static auto * CallVecDtor(Expr *objectParam, const ConstantArrayType *ar)
static auto * CallVecNewOrCtor(std::string_view ctorName, Expr *objectParam, QualType allocatedType, Expr *arraySizeExpr, std::string_view funName)
static auto * CallVecDeleteOrDtor(Expr *objectParam, QualType allocatedType, std::string_view name, uint64_t size)
std::string StrCat(const auto &... args)
static auto * CallVecNew(std::string_view ctorName, Expr *objectParam, QualType allocatedType, Expr *arraySizeExpr)
QualType vtableRecordType
VarDecl * VtblArrayVar(int size)
FieldDecl * VtblPtrField(const CXXRecordDecl *parent)
! A helper type to have a container for ArrayRef
void Add(const Stmt *stmt)
void AddBodyStmts(Stmt *body)
Handy helper to avoid longish comparisons.
constexpr bool any_of(const auto &... ts) const