mirror of
https://github.com/capstone-engine/llvm-capstone.git
synced 2025-05-19 04:16:37 +00:00

new InitializationSequence. This fixes some bugs (e.g., PR5808), changed some diagnostics, and caused more churn than expected. What's new: - InitializationSequence now has a "C conversion sequence" category and step kind, which falls back to - Changed the diagnostics for returns to always have the result type of the function first and the type of the expression second. CheckSingleAssignmentConstraints to peform checking in C. - Improved ASTs for initialization of return values. The ASTs now capture all of the temporaries we need to create, but intentionally do not bind the tempoary that is actually returned, so that it won't get destroyed twice. - Make sure to perform an (elidable!) copy of the class object that is returned from a class. - Fix copy elision in CodeGen to properly see through the subexpressions that occur with elidable copies. - Give "new" its own entity kind; as with return values and thrown objects, we don't bind the expression so we don't call a destructor for it. Note that, with this patch, I've broken returning move-only types in C++0x. We'll fix it later, when we tackle NRVO. llvm-svn: 91669
1812 lines
73 KiB
C++
1812 lines
73 KiB
C++
//===--- CGDecl.cpp - Emit LLVM Code for declarations ---------------------===//
|
|
//
|
|
// The LLVM Compiler Infrastructure
|
|
//
|
|
// This file is distributed under the University of Illinois Open Source
|
|
// License. See LICENSE.TXT for details.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
//
|
|
// This contains code dealing with C++ code generation.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// We might split this into multiple files if it gets too unwieldy
|
|
|
|
#include "CodeGenFunction.h"
|
|
#include "CodeGenModule.h"
|
|
#include "Mangle.h"
|
|
#include "clang/AST/ASTContext.h"
|
|
#include "clang/AST/RecordLayout.h"
|
|
#include "clang/AST/Decl.h"
|
|
#include "clang/AST/DeclCXX.h"
|
|
#include "clang/AST/DeclObjC.h"
|
|
#include "clang/AST/StmtCXX.h"
|
|
#include "llvm/ADT/StringExtras.h"
|
|
using namespace clang;
|
|
using namespace CodeGen;
|
|
|
|
RValue CodeGenFunction::EmitCXXMemberCall(const CXXMethodDecl *MD,
|
|
llvm::Value *Callee,
|
|
llvm::Value *This,
|
|
CallExpr::const_arg_iterator ArgBeg,
|
|
CallExpr::const_arg_iterator ArgEnd) {
|
|
assert(MD->isInstance() &&
|
|
"Trying to emit a member call expr on a static method!");
|
|
|
|
const FunctionProtoType *FPT = MD->getType()->getAs<FunctionProtoType>();
|
|
|
|
CallArgList Args;
|
|
|
|
// Push the this ptr.
|
|
Args.push_back(std::make_pair(RValue::get(This),
|
|
MD->getThisType(getContext())));
|
|
|
|
// And the rest of the call args
|
|
EmitCallArgs(Args, FPT, ArgBeg, ArgEnd);
|
|
|
|
QualType ResultType = MD->getType()->getAs<FunctionType>()->getResultType();
|
|
return EmitCall(CGM.getTypes().getFunctionInfo(ResultType, Args),
|
|
Callee, Args, MD);
|
|
}
|
|
|
|
/// canDevirtualizeMemberFunctionCalls - Checks whether virtual calls on given
|
|
/// expr can be devirtualized.
|
|
static bool canDevirtualizeMemberFunctionCalls(const Expr *Base) {
|
|
if (const DeclRefExpr *DRE = dyn_cast<DeclRefExpr>(Base)) {
|
|
if (const VarDecl *VD = dyn_cast<VarDecl>(DRE->getDecl())) {
|
|
// This is a record decl. We know the type and can devirtualize it.
|
|
return VD->getType()->isRecordType();
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
// We can always devirtualize calls on temporary object expressions.
|
|
if (isa<CXXTemporaryObjectExpr>(Base))
|
|
return true;
|
|
|
|
// And calls on bound temporaries.
|
|
if (isa<CXXBindTemporaryExpr>(Base))
|
|
return true;
|
|
|
|
// Check if this is a call expr that returns a record type.
|
|
if (const CallExpr *CE = dyn_cast<CallExpr>(Base))
|
|
return CE->getCallReturnType()->isRecordType();
|
|
|
|
// We can't devirtualize the call.
|
|
return false;
|
|
}
|
|
|
|
RValue CodeGenFunction::EmitCXXMemberCallExpr(const CXXMemberCallExpr *CE) {
|
|
if (isa<BinaryOperator>(CE->getCallee()->IgnoreParens()))
|
|
return EmitCXXMemberPointerCallExpr(CE);
|
|
|
|
const MemberExpr *ME = cast<MemberExpr>(CE->getCallee()->IgnoreParens());
|
|
const CXXMethodDecl *MD = cast<CXXMethodDecl>(ME->getMemberDecl());
|
|
|
|
if (MD->isStatic()) {
|
|
// The method is static, emit it as we would a regular call.
|
|
llvm::Value *Callee = CGM.GetAddrOfFunction(MD);
|
|
return EmitCall(Callee, getContext().getPointerType(MD->getType()),
|
|
CE->arg_begin(), CE->arg_end(), 0);
|
|
|
|
}
|
|
|
|
const FunctionProtoType *FPT = MD->getType()->getAs<FunctionProtoType>();
|
|
|
|
const llvm::Type *Ty =
|
|
CGM.getTypes().GetFunctionType(CGM.getTypes().getFunctionInfo(MD),
|
|
FPT->isVariadic());
|
|
llvm::Value *This;
|
|
|
|
if (ME->isArrow())
|
|
This = EmitScalarExpr(ME->getBase());
|
|
else {
|
|
LValue BaseLV = EmitLValue(ME->getBase());
|
|
This = BaseLV.getAddress();
|
|
}
|
|
|
|
if (MD->isCopyAssignment() && MD->isTrivial()) {
|
|
// We don't like to generate the trivial copy assignment operator when
|
|
// it isn't necessary; just produce the proper effect here.
|
|
llvm::Value *RHS = EmitLValue(*CE->arg_begin()).getAddress();
|
|
EmitAggregateCopy(This, RHS, CE->getType());
|
|
return RValue::get(This);
|
|
}
|
|
|
|
// C++ [class.virtual]p12:
|
|
// Explicit qualification with the scope operator (5.1) suppresses the
|
|
// virtual call mechanism.
|
|
//
|
|
// We also don't emit a virtual call if the base expression has a record type
|
|
// because then we know what the type is.
|
|
llvm::Value *Callee;
|
|
if (const CXXDestructorDecl *Destructor
|
|
= dyn_cast<CXXDestructorDecl>(MD)) {
|
|
if (Destructor->isTrivial())
|
|
return RValue::get(0);
|
|
if (MD->isVirtual() && !ME->hasQualifier() &&
|
|
!canDevirtualizeMemberFunctionCalls(ME->getBase())) {
|
|
Callee = BuildVirtualCall(Destructor, Dtor_Complete, This, Ty);
|
|
} else {
|
|
Callee = CGM.GetAddrOfFunction(GlobalDecl(Destructor, Dtor_Complete), Ty);
|
|
}
|
|
} else if (MD->isVirtual() && !ME->hasQualifier() &&
|
|
!canDevirtualizeMemberFunctionCalls(ME->getBase())) {
|
|
Callee = BuildVirtualCall(MD, This, Ty);
|
|
} else {
|
|
Callee = CGM.GetAddrOfFunction(MD, Ty);
|
|
}
|
|
|
|
return EmitCXXMemberCall(MD, Callee, This,
|
|
CE->arg_begin(), CE->arg_end());
|
|
}
|
|
|
|
RValue
|
|
CodeGenFunction::EmitCXXMemberPointerCallExpr(const CXXMemberCallExpr *E) {
|
|
const BinaryOperator *BO =
|
|
cast<BinaryOperator>(E->getCallee()->IgnoreParens());
|
|
const Expr *BaseExpr = BO->getLHS();
|
|
const Expr *MemFnExpr = BO->getRHS();
|
|
|
|
const MemberPointerType *MPT =
|
|
MemFnExpr->getType()->getAs<MemberPointerType>();
|
|
const FunctionProtoType *FPT =
|
|
MPT->getPointeeType()->getAs<FunctionProtoType>();
|
|
const CXXRecordDecl *RD =
|
|
cast<CXXRecordDecl>(MPT->getClass()->getAs<RecordType>()->getDecl());
|
|
|
|
const llvm::FunctionType *FTy =
|
|
CGM.getTypes().GetFunctionType(CGM.getTypes().getFunctionInfo(RD, FPT),
|
|
FPT->isVariadic());
|
|
|
|
const llvm::Type *Int8PtrTy =
|
|
llvm::Type::getInt8Ty(VMContext)->getPointerTo();
|
|
|
|
// Get the member function pointer.
|
|
llvm::Value *MemFnPtr =
|
|
CreateTempAlloca(ConvertType(MemFnExpr->getType()), "mem.fn");
|
|
EmitAggExpr(MemFnExpr, MemFnPtr, /*VolatileDest=*/false);
|
|
|
|
// Emit the 'this' pointer.
|
|
llvm::Value *This;
|
|
|
|
if (BO->getOpcode() == BinaryOperator::PtrMemI)
|
|
This = EmitScalarExpr(BaseExpr);
|
|
else
|
|
This = EmitLValue(BaseExpr).getAddress();
|
|
|
|
// Adjust it.
|
|
llvm::Value *Adj = Builder.CreateStructGEP(MemFnPtr, 1);
|
|
Adj = Builder.CreateLoad(Adj, "mem.fn.adj");
|
|
|
|
llvm::Value *Ptr = Builder.CreateBitCast(This, Int8PtrTy, "ptr");
|
|
Ptr = Builder.CreateGEP(Ptr, Adj, "adj");
|
|
|
|
This = Builder.CreateBitCast(Ptr, This->getType(), "this");
|
|
|
|
llvm::Value *FnPtr = Builder.CreateStructGEP(MemFnPtr, 0, "mem.fn.ptr");
|
|
|
|
const llvm::Type *PtrDiffTy = ConvertType(getContext().getPointerDiffType());
|
|
|
|
llvm::Value *FnAsInt = Builder.CreateLoad(FnPtr, "fn");
|
|
|
|
// If the LSB in the function pointer is 1, the function pointer points to
|
|
// a virtual function.
|
|
llvm::Value *IsVirtual
|
|
= Builder.CreateAnd(FnAsInt, llvm::ConstantInt::get(PtrDiffTy, 1),
|
|
"and");
|
|
|
|
IsVirtual = Builder.CreateTrunc(IsVirtual,
|
|
llvm::Type::getInt1Ty(VMContext));
|
|
|
|
llvm::BasicBlock *FnVirtual = createBasicBlock("fn.virtual");
|
|
llvm::BasicBlock *FnNonVirtual = createBasicBlock("fn.nonvirtual");
|
|
llvm::BasicBlock *FnEnd = createBasicBlock("fn.end");
|
|
|
|
Builder.CreateCondBr(IsVirtual, FnVirtual, FnNonVirtual);
|
|
EmitBlock(FnVirtual);
|
|
|
|
const llvm::Type *VTableTy =
|
|
FTy->getPointerTo()->getPointerTo()->getPointerTo();
|
|
|
|
llvm::Value *VTable = Builder.CreateBitCast(This, VTableTy);
|
|
VTable = Builder.CreateLoad(VTable);
|
|
|
|
VTable = Builder.CreateGEP(VTable, FnAsInt, "fn");
|
|
|
|
// Since the function pointer is 1 plus the virtual table offset, we
|
|
// subtract 1 by using a GEP.
|
|
VTable = Builder.CreateConstGEP1_64(VTable, (uint64_t)-1);
|
|
|
|
llvm::Value *VirtualFn = Builder.CreateLoad(VTable, "virtualfn");
|
|
|
|
EmitBranch(FnEnd);
|
|
EmitBlock(FnNonVirtual);
|
|
|
|
// If the function is not virtual, just load the pointer.
|
|
llvm::Value *NonVirtualFn = Builder.CreateLoad(FnPtr, "fn");
|
|
NonVirtualFn = Builder.CreateIntToPtr(NonVirtualFn, FTy->getPointerTo());
|
|
|
|
EmitBlock(FnEnd);
|
|
|
|
llvm::PHINode *Callee = Builder.CreatePHI(FTy->getPointerTo());
|
|
Callee->reserveOperandSpace(2);
|
|
Callee->addIncoming(VirtualFn, FnVirtual);
|
|
Callee->addIncoming(NonVirtualFn, FnNonVirtual);
|
|
|
|
CallArgList Args;
|
|
|
|
QualType ThisType =
|
|
getContext().getPointerType(getContext().getTagDeclType(RD));
|
|
|
|
// Push the this ptr.
|
|
Args.push_back(std::make_pair(RValue::get(This), ThisType));
|
|
|
|
// And the rest of the call args
|
|
EmitCallArgs(Args, FPT, E->arg_begin(), E->arg_end());
|
|
QualType ResultType = BO->getType()->getAs<FunctionType>()->getResultType();
|
|
return EmitCall(CGM.getTypes().getFunctionInfo(ResultType, Args),
|
|
Callee, Args, 0);
|
|
}
|
|
|
|
RValue
|
|
CodeGenFunction::EmitCXXOperatorMemberCallExpr(const CXXOperatorCallExpr *E,
|
|
const CXXMethodDecl *MD) {
|
|
assert(MD->isInstance() &&
|
|
"Trying to emit a member call expr on a static method!");
|
|
|
|
if (MD->isCopyAssignment()) {
|
|
const CXXRecordDecl *ClassDecl = cast<CXXRecordDecl>(MD->getDeclContext());
|
|
if (ClassDecl->hasTrivialCopyAssignment()) {
|
|
assert(!ClassDecl->hasUserDeclaredCopyAssignment() &&
|
|
"EmitCXXOperatorMemberCallExpr - user declared copy assignment");
|
|
llvm::Value *This = EmitLValue(E->getArg(0)).getAddress();
|
|
llvm::Value *Src = EmitLValue(E->getArg(1)).getAddress();
|
|
QualType Ty = E->getType();
|
|
EmitAggregateCopy(This, Src, Ty);
|
|
return RValue::get(This);
|
|
}
|
|
}
|
|
|
|
const FunctionProtoType *FPT = MD->getType()->getAs<FunctionProtoType>();
|
|
const llvm::Type *Ty =
|
|
CGM.getTypes().GetFunctionType(CGM.getTypes().getFunctionInfo(MD),
|
|
FPT->isVariadic());
|
|
|
|
llvm::Value *This = EmitLValue(E->getArg(0)).getAddress();
|
|
|
|
llvm::Value *Callee;
|
|
if (MD->isVirtual() && !canDevirtualizeMemberFunctionCalls(E->getArg(0)))
|
|
Callee = BuildVirtualCall(MD, This, Ty);
|
|
else
|
|
Callee = CGM.GetAddrOfFunction(MD, Ty);
|
|
|
|
return EmitCXXMemberCall(MD, Callee, This,
|
|
E->arg_begin() + 1, E->arg_end());
|
|
}
|
|
|
|
llvm::Value *CodeGenFunction::LoadCXXThis() {
|
|
assert(isa<CXXMethodDecl>(CurFuncDecl) &&
|
|
"Must be in a C++ member function decl to load 'this'");
|
|
assert(cast<CXXMethodDecl>(CurFuncDecl)->isInstance() &&
|
|
"Must be in a C++ member function decl to load 'this'");
|
|
|
|
// FIXME: What if we're inside a block?
|
|
// ans: See how CodeGenFunction::LoadObjCSelf() uses
|
|
// CodeGenFunction::BlockForwardSelf() for how to do this.
|
|
return Builder.CreateLoad(LocalDeclMap[CXXThisDecl], "this");
|
|
}
|
|
|
|
/// EmitCXXAggrConstructorCall - This routine essentially creates a (nested)
|
|
/// for-loop to call the default constructor on individual members of the
|
|
/// array.
|
|
/// 'D' is the default constructor for elements of the array, 'ArrayTy' is the
|
|
/// array type and 'ArrayPtr' points to the beginning fo the array.
|
|
/// It is assumed that all relevant checks have been made by the caller.
|
|
void
|
|
CodeGenFunction::EmitCXXAggrConstructorCall(const CXXConstructorDecl *D,
|
|
const ConstantArrayType *ArrayTy,
|
|
llvm::Value *ArrayPtr,
|
|
CallExpr::const_arg_iterator ArgBeg,
|
|
CallExpr::const_arg_iterator ArgEnd) {
|
|
|
|
const llvm::Type *SizeTy = ConvertType(getContext().getSizeType());
|
|
llvm::Value * NumElements =
|
|
llvm::ConstantInt::get(SizeTy,
|
|
getContext().getConstantArrayElementCount(ArrayTy));
|
|
|
|
EmitCXXAggrConstructorCall(D, NumElements, ArrayPtr, ArgBeg, ArgEnd);
|
|
}
|
|
|
|
void
|
|
CodeGenFunction::EmitCXXAggrConstructorCall(const CXXConstructorDecl *D,
|
|
llvm::Value *NumElements,
|
|
llvm::Value *ArrayPtr,
|
|
CallExpr::const_arg_iterator ArgBeg,
|
|
CallExpr::const_arg_iterator ArgEnd) {
|
|
const llvm::Type *SizeTy = ConvertType(getContext().getSizeType());
|
|
|
|
// Create a temporary for the loop index and initialize it with 0.
|
|
llvm::Value *IndexPtr = CreateTempAlloca(SizeTy, "loop.index");
|
|
llvm::Value *Zero = llvm::Constant::getNullValue(SizeTy);
|
|
Builder.CreateStore(Zero, IndexPtr);
|
|
|
|
// Start the loop with a block that tests the condition.
|
|
llvm::BasicBlock *CondBlock = createBasicBlock("for.cond");
|
|
llvm::BasicBlock *AfterFor = createBasicBlock("for.end");
|
|
|
|
EmitBlock(CondBlock);
|
|
|
|
llvm::BasicBlock *ForBody = createBasicBlock("for.body");
|
|
|
|
// Generate: if (loop-index < number-of-elements fall to the loop body,
|
|
// otherwise, go to the block after the for-loop.
|
|
llvm::Value *Counter = Builder.CreateLoad(IndexPtr);
|
|
llvm::Value *IsLess = Builder.CreateICmpULT(Counter, NumElements, "isless");
|
|
// If the condition is true, execute the body.
|
|
Builder.CreateCondBr(IsLess, ForBody, AfterFor);
|
|
|
|
EmitBlock(ForBody);
|
|
|
|
llvm::BasicBlock *ContinueBlock = createBasicBlock("for.inc");
|
|
// Inside the loop body, emit the constructor call on the array element.
|
|
Counter = Builder.CreateLoad(IndexPtr);
|
|
llvm::Value *Address = Builder.CreateInBoundsGEP(ArrayPtr, Counter,
|
|
"arrayidx");
|
|
|
|
// C++ [class.temporary]p4:
|
|
// There are two contexts in which temporaries are destroyed at a different
|
|
// point than the end of the full-expression. The first context is when a
|
|
// default constructor is called to initialize an element of an array.
|
|
// If the constructor has one or more default arguments, the destruction of
|
|
// every temporary created in a default argument expression is sequenced
|
|
// before the construction of the next array element, if any.
|
|
|
|
// Keep track of the current number of live temporaries.
|
|
unsigned OldNumLiveTemporaries = LiveTemporaries.size();
|
|
|
|
EmitCXXConstructorCall(D, Ctor_Complete, Address, ArgBeg, ArgEnd);
|
|
|
|
// Pop temporaries.
|
|
while (LiveTemporaries.size() > OldNumLiveTemporaries)
|
|
PopCXXTemporary();
|
|
|
|
EmitBlock(ContinueBlock);
|
|
|
|
// Emit the increment of the loop counter.
|
|
llvm::Value *NextVal = llvm::ConstantInt::get(SizeTy, 1);
|
|
Counter = Builder.CreateLoad(IndexPtr);
|
|
NextVal = Builder.CreateAdd(Counter, NextVal, "inc");
|
|
Builder.CreateStore(NextVal, IndexPtr);
|
|
|
|
// Finally, branch back up to the condition for the next iteration.
|
|
EmitBranch(CondBlock);
|
|
|
|
// Emit the fall-through block.
|
|
EmitBlock(AfterFor, true);
|
|
}
|
|
|
|
/// EmitCXXAggrDestructorCall - calls the default destructor on array
|
|
/// elements in reverse order of construction.
|
|
void
|
|
CodeGenFunction::EmitCXXAggrDestructorCall(const CXXDestructorDecl *D,
|
|
const ArrayType *Array,
|
|
llvm::Value *This) {
|
|
const ConstantArrayType *CA = dyn_cast<ConstantArrayType>(Array);
|
|
assert(CA && "Do we support VLA for destruction ?");
|
|
uint64_t ElementCount = getContext().getConstantArrayElementCount(CA);
|
|
|
|
const llvm::Type *SizeLTy = ConvertType(getContext().getSizeType());
|
|
llvm::Value* ElementCountPtr = llvm::ConstantInt::get(SizeLTy, ElementCount);
|
|
EmitCXXAggrDestructorCall(D, ElementCountPtr, This);
|
|
}
|
|
|
|
/// EmitCXXAggrDestructorCall - calls the default destructor on array
|
|
/// elements in reverse order of construction.
|
|
void
|
|
CodeGenFunction::EmitCXXAggrDestructorCall(const CXXDestructorDecl *D,
|
|
llvm::Value *UpperCount,
|
|
llvm::Value *This) {
|
|
const llvm::Type *SizeLTy = ConvertType(getContext().getSizeType());
|
|
llvm::Value *One = llvm::ConstantInt::get(SizeLTy, 1);
|
|
|
|
// Create a temporary for the loop index and initialize it with count of
|
|
// array elements.
|
|
llvm::Value *IndexPtr = CreateTempAlloca(SizeLTy, "loop.index");
|
|
|
|
// Store the number of elements in the index pointer.
|
|
Builder.CreateStore(UpperCount, IndexPtr);
|
|
|
|
// Start the loop with a block that tests the condition.
|
|
llvm::BasicBlock *CondBlock = createBasicBlock("for.cond");
|
|
llvm::BasicBlock *AfterFor = createBasicBlock("for.end");
|
|
|
|
EmitBlock(CondBlock);
|
|
|
|
llvm::BasicBlock *ForBody = createBasicBlock("for.body");
|
|
|
|
// Generate: if (loop-index != 0 fall to the loop body,
|
|
// otherwise, go to the block after the for-loop.
|
|
llvm::Value* zeroConstant =
|
|
llvm::Constant::getNullValue(SizeLTy);
|
|
llvm::Value *Counter = Builder.CreateLoad(IndexPtr);
|
|
llvm::Value *IsNE = Builder.CreateICmpNE(Counter, zeroConstant,
|
|
"isne");
|
|
// If the condition is true, execute the body.
|
|
Builder.CreateCondBr(IsNE, ForBody, AfterFor);
|
|
|
|
EmitBlock(ForBody);
|
|
|
|
llvm::BasicBlock *ContinueBlock = createBasicBlock("for.inc");
|
|
// Inside the loop body, emit the constructor call on the array element.
|
|
Counter = Builder.CreateLoad(IndexPtr);
|
|
Counter = Builder.CreateSub(Counter, One);
|
|
llvm::Value *Address = Builder.CreateInBoundsGEP(This, Counter, "arrayidx");
|
|
EmitCXXDestructorCall(D, Dtor_Complete, Address);
|
|
|
|
EmitBlock(ContinueBlock);
|
|
|
|
// Emit the decrement of the loop counter.
|
|
Counter = Builder.CreateLoad(IndexPtr);
|
|
Counter = Builder.CreateSub(Counter, One, "dec");
|
|
Builder.CreateStore(Counter, IndexPtr);
|
|
|
|
// Finally, branch back up to the condition for the next iteration.
|
|
EmitBranch(CondBlock);
|
|
|
|
// Emit the fall-through block.
|
|
EmitBlock(AfterFor, true);
|
|
}
|
|
|
|
/// GenerateCXXAggrDestructorHelper - Generates a helper function which when
|
|
/// invoked, calls the default destructor on array elements in reverse order of
|
|
/// construction.
|
|
llvm::Constant *
|
|
CodeGenFunction::GenerateCXXAggrDestructorHelper(const CXXDestructorDecl *D,
|
|
const ArrayType *Array,
|
|
llvm::Value *This) {
|
|
FunctionArgList Args;
|
|
ImplicitParamDecl *Dst =
|
|
ImplicitParamDecl::Create(getContext(), 0,
|
|
SourceLocation(), 0,
|
|
getContext().getPointerType(getContext().VoidTy));
|
|
Args.push_back(std::make_pair(Dst, Dst->getType()));
|
|
|
|
llvm::SmallString<16> Name;
|
|
llvm::raw_svector_ostream(Name) << "__tcf_" << (++UniqueAggrDestructorCount);
|
|
QualType R = getContext().VoidTy;
|
|
const CGFunctionInfo &FI = CGM.getTypes().getFunctionInfo(R, Args);
|
|
const llvm::FunctionType *FTy = CGM.getTypes().GetFunctionType(FI, false);
|
|
llvm::Function *Fn =
|
|
llvm::Function::Create(FTy, llvm::GlobalValue::InternalLinkage,
|
|
Name.str(),
|
|
&CGM.getModule());
|
|
IdentifierInfo *II = &CGM.getContext().Idents.get(Name.str());
|
|
FunctionDecl *FD = FunctionDecl::Create(getContext(),
|
|
getContext().getTranslationUnitDecl(),
|
|
SourceLocation(), II, R, 0,
|
|
FunctionDecl::Static,
|
|
false, true);
|
|
StartFunction(FD, R, Fn, Args, SourceLocation());
|
|
QualType BaseElementTy = getContext().getBaseElementType(Array);
|
|
const llvm::Type *BasePtr = ConvertType(BaseElementTy);
|
|
BasePtr = llvm::PointerType::getUnqual(BasePtr);
|
|
llvm::Value *BaseAddrPtr = Builder.CreateBitCast(This, BasePtr);
|
|
EmitCXXAggrDestructorCall(D, Array, BaseAddrPtr);
|
|
FinishFunction();
|
|
llvm::Type *Ptr8Ty = llvm::PointerType::get(llvm::Type::getInt8Ty(VMContext),
|
|
0);
|
|
llvm::Constant *m = llvm::ConstantExpr::getBitCast(Fn, Ptr8Ty);
|
|
return m;
|
|
}
|
|
|
|
void
|
|
CodeGenFunction::EmitCXXConstructorCall(const CXXConstructorDecl *D,
|
|
CXXCtorType Type,
|
|
llvm::Value *This,
|
|
CallExpr::const_arg_iterator ArgBeg,
|
|
CallExpr::const_arg_iterator ArgEnd) {
|
|
if (D->isCopyConstructor(getContext())) {
|
|
const CXXRecordDecl *ClassDecl = cast<CXXRecordDecl>(D->getDeclContext());
|
|
if (ClassDecl->hasTrivialCopyConstructor()) {
|
|
assert(!ClassDecl->hasUserDeclaredCopyConstructor() &&
|
|
"EmitCXXConstructorCall - user declared copy constructor");
|
|
const Expr *E = (*ArgBeg);
|
|
QualType Ty = E->getType();
|
|
llvm::Value *Src = EmitLValue(E).getAddress();
|
|
EmitAggregateCopy(This, Src, Ty);
|
|
return;
|
|
}
|
|
} else if (D->isTrivial()) {
|
|
// FIXME: Track down why we're trying to generate calls to the trivial
|
|
// default constructor!
|
|
return;
|
|
}
|
|
|
|
llvm::Value *Callee = CGM.GetAddrOfCXXConstructor(D, Type);
|
|
|
|
EmitCXXMemberCall(D, Callee, This, ArgBeg, ArgEnd);
|
|
}
|
|
|
|
void CodeGenFunction::EmitCXXDestructorCall(const CXXDestructorDecl *DD,
|
|
CXXDtorType Type,
|
|
llvm::Value *This) {
|
|
llvm::Value *Callee = CGM.GetAddrOfCXXDestructor(DD, Type);
|
|
|
|
CallArgList Args;
|
|
|
|
// Push the this ptr.
|
|
Args.push_back(std::make_pair(RValue::get(This),
|
|
DD->getThisType(getContext())));
|
|
|
|
// Add a VTT parameter if necessary.
|
|
// FIXME: This should not be a dummy null parameter!
|
|
if (Type == Dtor_Base && DD->getParent()->getNumVBases() != 0) {
|
|
QualType T = getContext().getPointerType(getContext().VoidPtrTy);
|
|
|
|
Args.push_back(std::make_pair(RValue::get(CGM.EmitNullConstant(T)), T));
|
|
}
|
|
|
|
// FIXME: We should try to share this code with EmitCXXMemberCall.
|
|
|
|
QualType ResultType = DD->getType()->getAs<FunctionType>()->getResultType();
|
|
EmitCall(CGM.getTypes().getFunctionInfo(ResultType, Args), Callee, Args, DD);
|
|
}
|
|
|
|
void
|
|
CodeGenFunction::EmitCXXConstructExpr(llvm::Value *Dest,
|
|
const CXXConstructExpr *E) {
|
|
assert(Dest && "Must have a destination!");
|
|
const CXXConstructorDecl *CD = E->getConstructor();
|
|
const ConstantArrayType *Array =
|
|
getContext().getAsConstantArrayType(E->getType());
|
|
// For a copy constructor, even if it is trivial, must fall thru so
|
|
// its argument is code-gen'ed.
|
|
if (!CD->isCopyConstructor(getContext())) {
|
|
QualType InitType = E->getType();
|
|
if (Array)
|
|
InitType = getContext().getBaseElementType(Array);
|
|
const CXXRecordDecl *RD =
|
|
cast<CXXRecordDecl>(InitType->getAs<RecordType>()->getDecl());
|
|
if (RD->hasTrivialConstructor())
|
|
return;
|
|
}
|
|
// Code gen optimization to eliminate copy constructor and return
|
|
// its first argument instead.
|
|
if (getContext().getLangOptions().ElideConstructors && E->isElidable()) {
|
|
const Expr *Arg = E->getArg(0);
|
|
|
|
if (const ImplicitCastExpr *ICE = dyn_cast<ImplicitCastExpr>(Arg)) {
|
|
if (isa<CXXBindTemporaryExpr>(ICE->getSubExpr()))
|
|
Arg = cast<CXXBindTemporaryExpr>(ICE->getSubExpr())->getSubExpr();
|
|
} else if (const CXXBindTemporaryExpr *BindExpr =
|
|
dyn_cast<CXXBindTemporaryExpr>(Arg))
|
|
Arg = BindExpr->getSubExpr();
|
|
|
|
EmitAggExpr(Arg, Dest, false);
|
|
return;
|
|
}
|
|
if (Array) {
|
|
QualType BaseElementTy = getContext().getBaseElementType(Array);
|
|
const llvm::Type *BasePtr = ConvertType(BaseElementTy);
|
|
BasePtr = llvm::PointerType::getUnqual(BasePtr);
|
|
llvm::Value *BaseAddrPtr =
|
|
Builder.CreateBitCast(Dest, BasePtr);
|
|
|
|
EmitCXXAggrConstructorCall(CD, Array, BaseAddrPtr,
|
|
E->arg_begin(), E->arg_end());
|
|
}
|
|
else
|
|
// Call the constructor.
|
|
EmitCXXConstructorCall(CD, Ctor_Complete, Dest,
|
|
E->arg_begin(), E->arg_end());
|
|
}
|
|
|
|
void CodeGenModule::EmitCXXConstructors(const CXXConstructorDecl *D) {
|
|
EmitGlobal(GlobalDecl(D, Ctor_Complete));
|
|
EmitGlobal(GlobalDecl(D, Ctor_Base));
|
|
}
|
|
|
|
void CodeGenModule::EmitCXXConstructor(const CXXConstructorDecl *D,
|
|
CXXCtorType Type) {
|
|
|
|
llvm::Function *Fn = GetAddrOfCXXConstructor(D, Type);
|
|
|
|
CodeGenFunction(*this).GenerateCode(GlobalDecl(D, Type), Fn);
|
|
|
|
SetFunctionDefinitionAttributes(D, Fn);
|
|
SetLLVMFunctionAttributesForDefinition(D, Fn);
|
|
}
|
|
|
|
llvm::Function *
|
|
CodeGenModule::GetAddrOfCXXConstructor(const CXXConstructorDecl *D,
|
|
CXXCtorType Type) {
|
|
const FunctionProtoType *FPT = D->getType()->getAs<FunctionProtoType>();
|
|
const llvm::FunctionType *FTy =
|
|
getTypes().GetFunctionType(getTypes().getFunctionInfo(D, Type),
|
|
FPT->isVariadic());
|
|
|
|
const char *Name = getMangledCXXCtorName(D, Type);
|
|
return cast<llvm::Function>(
|
|
GetOrCreateLLVMFunction(Name, FTy, GlobalDecl(D, Type)));
|
|
}
|
|
|
|
const char *CodeGenModule::getMangledCXXCtorName(const CXXConstructorDecl *D,
|
|
CXXCtorType Type) {
|
|
llvm::SmallString<256> Name;
|
|
getMangleContext().mangleCXXCtor(D, Type, Name);
|
|
|
|
Name += '\0';
|
|
return UniqueMangledName(Name.begin(), Name.end());
|
|
}
|
|
|
|
void CodeGenModule::EmitCXXDestructors(const CXXDestructorDecl *D) {
|
|
if (D->isVirtual())
|
|
EmitGlobal(GlobalDecl(D, Dtor_Deleting));
|
|
EmitGlobal(GlobalDecl(D, Dtor_Complete));
|
|
EmitGlobal(GlobalDecl(D, Dtor_Base));
|
|
}
|
|
|
|
void CodeGenModule::EmitCXXDestructor(const CXXDestructorDecl *D,
|
|
CXXDtorType Type) {
|
|
llvm::Function *Fn = GetAddrOfCXXDestructor(D, Type);
|
|
|
|
CodeGenFunction(*this).GenerateCode(GlobalDecl(D, Type), Fn);
|
|
|
|
SetFunctionDefinitionAttributes(D, Fn);
|
|
SetLLVMFunctionAttributesForDefinition(D, Fn);
|
|
}
|
|
|
|
llvm::Function *
|
|
CodeGenModule::GetAddrOfCXXDestructor(const CXXDestructorDecl *D,
|
|
CXXDtorType Type) {
|
|
const llvm::FunctionType *FTy =
|
|
getTypes().GetFunctionType(getTypes().getFunctionInfo(D, Type), false);
|
|
|
|
const char *Name = getMangledCXXDtorName(D, Type);
|
|
return cast<llvm::Function>(
|
|
GetOrCreateLLVMFunction(Name, FTy, GlobalDecl(D, Type)));
|
|
}
|
|
|
|
const char *CodeGenModule::getMangledCXXDtorName(const CXXDestructorDecl *D,
|
|
CXXDtorType Type) {
|
|
llvm::SmallString<256> Name;
|
|
getMangleContext().mangleCXXDtor(D, Type, Name);
|
|
|
|
Name += '\0';
|
|
return UniqueMangledName(Name.begin(), Name.end());
|
|
}
|
|
|
|
llvm::Constant *
|
|
CodeGenFunction::GenerateThunk(llvm::Function *Fn, GlobalDecl GD,
|
|
bool Extern,
|
|
const ThunkAdjustment &ThisAdjustment) {
|
|
return GenerateCovariantThunk(Fn, GD, Extern,
|
|
CovariantThunkAdjustment(ThisAdjustment,
|
|
ThunkAdjustment()));
|
|
}
|
|
|
|
llvm::Value *
|
|
CodeGenFunction::DynamicTypeAdjust(llvm::Value *V,
|
|
const ThunkAdjustment &Adjustment) {
|
|
const llvm::Type *Int8PtrTy = llvm::Type::getInt8PtrTy(VMContext);
|
|
|
|
const llvm::Type *OrigTy = V->getType();
|
|
if (Adjustment.NonVirtual) {
|
|
// Do the non-virtual adjustment
|
|
V = Builder.CreateBitCast(V, Int8PtrTy);
|
|
V = Builder.CreateConstInBoundsGEP1_64(V, Adjustment.NonVirtual);
|
|
V = Builder.CreateBitCast(V, OrigTy);
|
|
}
|
|
|
|
if (!Adjustment.Virtual)
|
|
return V;
|
|
|
|
assert(Adjustment.Virtual % (LLVMPointerWidth / 8) == 0 &&
|
|
"vtable entry unaligned");
|
|
|
|
// Do the virtual this adjustment
|
|
const llvm::Type *PtrDiffTy = ConvertType(getContext().getPointerDiffType());
|
|
const llvm::Type *PtrDiffPtrTy = PtrDiffTy->getPointerTo();
|
|
|
|
llvm::Value *ThisVal = Builder.CreateBitCast(V, Int8PtrTy);
|
|
V = Builder.CreateBitCast(V, PtrDiffPtrTy->getPointerTo());
|
|
V = Builder.CreateLoad(V, "vtable");
|
|
|
|
llvm::Value *VTablePtr = V;
|
|
uint64_t VirtualAdjustment = Adjustment.Virtual / (LLVMPointerWidth / 8);
|
|
V = Builder.CreateConstInBoundsGEP1_64(VTablePtr, VirtualAdjustment);
|
|
V = Builder.CreateLoad(V);
|
|
V = Builder.CreateGEP(ThisVal, V);
|
|
|
|
return Builder.CreateBitCast(V, OrigTy);
|
|
}
|
|
|
|
llvm::Constant *
|
|
CodeGenFunction::GenerateCovariantThunk(llvm::Function *Fn,
|
|
GlobalDecl GD, bool Extern,
|
|
const CovariantThunkAdjustment &Adjustment) {
|
|
const CXXMethodDecl *MD = cast<CXXMethodDecl>(GD.getDecl());
|
|
QualType ResultType = MD->getType()->getAs<FunctionType>()->getResultType();
|
|
|
|
FunctionArgList Args;
|
|
ImplicitParamDecl *ThisDecl =
|
|
ImplicitParamDecl::Create(getContext(), 0, SourceLocation(), 0,
|
|
MD->getThisType(getContext()));
|
|
Args.push_back(std::make_pair(ThisDecl, ThisDecl->getType()));
|
|
for (FunctionDecl::param_const_iterator i = MD->param_begin(),
|
|
e = MD->param_end();
|
|
i != e; ++i) {
|
|
ParmVarDecl *D = *i;
|
|
Args.push_back(std::make_pair(D, D->getType()));
|
|
}
|
|
IdentifierInfo *II
|
|
= &CGM.getContext().Idents.get("__thunk_named_foo_");
|
|
FunctionDecl *FD = FunctionDecl::Create(getContext(),
|
|
getContext().getTranslationUnitDecl(),
|
|
SourceLocation(), II, ResultType, 0,
|
|
Extern
|
|
? FunctionDecl::Extern
|
|
: FunctionDecl::Static,
|
|
false, true);
|
|
StartFunction(FD, ResultType, Fn, Args, SourceLocation());
|
|
|
|
// generate body
|
|
const FunctionProtoType *FPT = MD->getType()->getAs<FunctionProtoType>();
|
|
const llvm::Type *Ty =
|
|
CGM.getTypes().GetFunctionType(CGM.getTypes().getFunctionInfo(MD),
|
|
FPT->isVariadic());
|
|
llvm::Value *Callee = CGM.GetAddrOfFunction(GD, Ty);
|
|
|
|
CallArgList CallArgs;
|
|
|
|
bool ShouldAdjustReturnPointer = true;
|
|
QualType ArgType = MD->getThisType(getContext());
|
|
llvm::Value *Arg = Builder.CreateLoad(LocalDeclMap[ThisDecl], "this");
|
|
if (!Adjustment.ThisAdjustment.isEmpty()) {
|
|
// Do the this adjustment.
|
|
const llvm::Type *OrigTy = Callee->getType();
|
|
Arg = DynamicTypeAdjust(Arg, Adjustment.ThisAdjustment);
|
|
|
|
if (!Adjustment.ReturnAdjustment.isEmpty()) {
|
|
const CovariantThunkAdjustment &ReturnAdjustment =
|
|
CovariantThunkAdjustment(ThunkAdjustment(),
|
|
Adjustment.ReturnAdjustment);
|
|
|
|
Callee = CGM.BuildCovariantThunk(GD, Extern, ReturnAdjustment);
|
|
|
|
Callee = Builder.CreateBitCast(Callee, OrigTy);
|
|
ShouldAdjustReturnPointer = false;
|
|
}
|
|
}
|
|
|
|
CallArgs.push_back(std::make_pair(RValue::get(Arg), ArgType));
|
|
|
|
for (FunctionDecl::param_const_iterator i = MD->param_begin(),
|
|
e = MD->param_end();
|
|
i != e; ++i) {
|
|
ParmVarDecl *D = *i;
|
|
QualType ArgType = D->getType();
|
|
|
|
// llvm::Value *Arg = CGF.GetAddrOfLocalVar(Dst);
|
|
Expr *Arg = new (getContext()) DeclRefExpr(D, ArgType.getNonReferenceType(),
|
|
SourceLocation());
|
|
CallArgs.push_back(std::make_pair(EmitCallArg(Arg, ArgType), ArgType));
|
|
}
|
|
|
|
RValue RV = EmitCall(CGM.getTypes().getFunctionInfo(ResultType, CallArgs),
|
|
Callee, CallArgs, MD);
|
|
if (ShouldAdjustReturnPointer && !Adjustment.ReturnAdjustment.isEmpty()) {
|
|
bool CanBeZero = !(ResultType->isReferenceType()
|
|
// FIXME: attr nonnull can't be zero either
|
|
/* || ResultType->hasAttr<NonNullAttr>() */ );
|
|
// Do the return result adjustment.
|
|
if (CanBeZero) {
|
|
llvm::BasicBlock *NonZeroBlock = createBasicBlock();
|
|
llvm::BasicBlock *ZeroBlock = createBasicBlock();
|
|
llvm::BasicBlock *ContBlock = createBasicBlock();
|
|
|
|
const llvm::Type *Ty = RV.getScalarVal()->getType();
|
|
llvm::Value *Zero = llvm::Constant::getNullValue(Ty);
|
|
Builder.CreateCondBr(Builder.CreateICmpNE(RV.getScalarVal(), Zero),
|
|
NonZeroBlock, ZeroBlock);
|
|
EmitBlock(NonZeroBlock);
|
|
llvm::Value *NZ =
|
|
DynamicTypeAdjust(RV.getScalarVal(), Adjustment.ReturnAdjustment);
|
|
EmitBranch(ContBlock);
|
|
EmitBlock(ZeroBlock);
|
|
llvm::Value *Z = RV.getScalarVal();
|
|
EmitBlock(ContBlock);
|
|
llvm::PHINode *RVOrZero = Builder.CreatePHI(Ty);
|
|
RVOrZero->reserveOperandSpace(2);
|
|
RVOrZero->addIncoming(NZ, NonZeroBlock);
|
|
RVOrZero->addIncoming(Z, ZeroBlock);
|
|
RV = RValue::get(RVOrZero);
|
|
} else
|
|
RV = RValue::get(DynamicTypeAdjust(RV.getScalarVal(),
|
|
Adjustment.ReturnAdjustment));
|
|
}
|
|
|
|
if (!ResultType->isVoidType())
|
|
EmitReturnOfRValue(RV, ResultType);
|
|
|
|
FinishFunction();
|
|
return Fn;
|
|
}
|
|
|
|
llvm::Constant *
|
|
CodeGenModule::GetAddrOfThunk(GlobalDecl GD,
|
|
const ThunkAdjustment &ThisAdjustment) {
|
|
const CXXMethodDecl *MD = cast<CXXMethodDecl>(GD.getDecl());
|
|
|
|
// Compute mangled name
|
|
llvm::SmallString<256> OutName;
|
|
if (const CXXDestructorDecl* DD = dyn_cast<CXXDestructorDecl>(MD))
|
|
getMangleContext().mangleCXXDtorThunk(DD, GD.getDtorType(), ThisAdjustment,
|
|
OutName);
|
|
else
|
|
getMangleContext().mangleThunk(MD, ThisAdjustment, OutName);
|
|
OutName += '\0';
|
|
const char* Name = UniqueMangledName(OutName.begin(), OutName.end());
|
|
|
|
// Get function for mangled name
|
|
const llvm::Type *Ty = getTypes().GetFunctionTypeForVtable(MD);
|
|
return GetOrCreateLLVMFunction(Name, Ty, GlobalDecl());
|
|
}
|
|
|
|
llvm::Constant *
|
|
CodeGenModule::GetAddrOfCovariantThunk(GlobalDecl GD,
|
|
const CovariantThunkAdjustment &Adjustment) {
|
|
const CXXMethodDecl *MD = cast<CXXMethodDecl>(GD.getDecl());
|
|
|
|
// Compute mangled name
|
|
llvm::SmallString<256> OutName;
|
|
getMangleContext().mangleCovariantThunk(MD, Adjustment, OutName);
|
|
OutName += '\0';
|
|
const char* Name = UniqueMangledName(OutName.begin(), OutName.end());
|
|
|
|
// Get function for mangled name
|
|
const llvm::Type *Ty = getTypes().GetFunctionTypeForVtable(MD);
|
|
return GetOrCreateLLVMFunction(Name, Ty, GlobalDecl());
|
|
}
|
|
|
|
void CodeGenModule::BuildThunksForVirtual(GlobalDecl GD) {
|
|
CGVtableInfo::AdjustmentVectorTy *AdjPtr = getVtableInfo().getAdjustments(GD);
|
|
if (!AdjPtr)
|
|
return;
|
|
CGVtableInfo::AdjustmentVectorTy &Adj = *AdjPtr;
|
|
const CXXMethodDecl *MD = cast<CXXMethodDecl>(GD.getDecl());
|
|
for (unsigned i = 0; i < Adj.size(); i++) {
|
|
GlobalDecl OGD = Adj[i].first;
|
|
const CXXMethodDecl *OMD = cast<CXXMethodDecl>(OGD.getDecl());
|
|
QualType nc_oret = OMD->getType()->getAs<FunctionType>()->getResultType();
|
|
CanQualType oret = getContext().getCanonicalType(nc_oret);
|
|
QualType nc_ret = MD->getType()->getAs<FunctionType>()->getResultType();
|
|
CanQualType ret = getContext().getCanonicalType(nc_ret);
|
|
ThunkAdjustment ReturnAdjustment;
|
|
if (oret != ret) {
|
|
QualType qD = nc_ret->getPointeeType();
|
|
QualType qB = nc_oret->getPointeeType();
|
|
CXXRecordDecl *D = cast<CXXRecordDecl>(qD->getAs<RecordType>()->getDecl());
|
|
CXXRecordDecl *B = cast<CXXRecordDecl>(qB->getAs<RecordType>()->getDecl());
|
|
ReturnAdjustment = ComputeThunkAdjustment(D, B);
|
|
}
|
|
ThunkAdjustment ThisAdjustment = Adj[i].second;
|
|
bool Extern = !cast<CXXRecordDecl>(OMD->getDeclContext())->isInAnonymousNamespace();
|
|
if (!ReturnAdjustment.isEmpty() || !ThisAdjustment.isEmpty()) {
|
|
CovariantThunkAdjustment CoAdj(ThisAdjustment, ReturnAdjustment);
|
|
llvm::Constant *FnConst;
|
|
if (!ReturnAdjustment.isEmpty())
|
|
FnConst = GetAddrOfCovariantThunk(GD, CoAdj);
|
|
else
|
|
FnConst = GetAddrOfThunk(GD, ThisAdjustment);
|
|
if (!isa<llvm::Function>(FnConst)) {
|
|
llvm::Constant *SubExpr =
|
|
cast<llvm::ConstantExpr>(FnConst)->getOperand(0);
|
|
llvm::Function *OldFn = cast<llvm::Function>(SubExpr);
|
|
std::string Name = OldFn->getNameStr();
|
|
GlobalDeclMap.erase(UniqueMangledName(Name.data(),
|
|
Name.data() + Name.size() + 1));
|
|
llvm::Constant *NewFnConst;
|
|
if (!ReturnAdjustment.isEmpty())
|
|
NewFnConst = GetAddrOfCovariantThunk(GD, CoAdj);
|
|
else
|
|
NewFnConst = GetAddrOfThunk(GD, ThisAdjustment);
|
|
llvm::Function *NewFn = cast<llvm::Function>(NewFnConst);
|
|
NewFn->takeName(OldFn);
|
|
llvm::Constant *NewPtrForOldDecl =
|
|
llvm::ConstantExpr::getBitCast(NewFn, OldFn->getType());
|
|
OldFn->replaceAllUsesWith(NewPtrForOldDecl);
|
|
OldFn->eraseFromParent();
|
|
FnConst = NewFn;
|
|
}
|
|
llvm::Function *Fn = cast<llvm::Function>(FnConst);
|
|
if (Fn->isDeclaration()) {
|
|
llvm::GlobalVariable::LinkageTypes linktype;
|
|
linktype = llvm::GlobalValue::WeakAnyLinkage;
|
|
if (!Extern)
|
|
linktype = llvm::GlobalValue::InternalLinkage;
|
|
Fn->setLinkage(linktype);
|
|
if (!Features.Exceptions && !Features.ObjCNonFragileABI)
|
|
Fn->addFnAttr(llvm::Attribute::NoUnwind);
|
|
Fn->setAlignment(2);
|
|
CodeGenFunction(*this).GenerateCovariantThunk(Fn, GD, Extern, CoAdj);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
llvm::Constant *
|
|
CodeGenModule::BuildThunk(GlobalDecl GD, bool Extern,
|
|
const ThunkAdjustment &ThisAdjustment) {
|
|
const CXXMethodDecl *MD = cast<CXXMethodDecl>(GD.getDecl());
|
|
llvm::SmallString<256> OutName;
|
|
if (const CXXDestructorDecl *D = dyn_cast<CXXDestructorDecl>(MD)) {
|
|
getMangleContext().mangleCXXDtorThunk(D, GD.getDtorType(), ThisAdjustment,
|
|
OutName);
|
|
} else
|
|
getMangleContext().mangleThunk(MD, ThisAdjustment, OutName);
|
|
|
|
llvm::GlobalVariable::LinkageTypes linktype;
|
|
linktype = llvm::GlobalValue::WeakAnyLinkage;
|
|
if (!Extern)
|
|
linktype = llvm::GlobalValue::InternalLinkage;
|
|
llvm::Type *Ptr8Ty=llvm::PointerType::get(llvm::Type::getInt8Ty(VMContext),0);
|
|
const FunctionProtoType *FPT = MD->getType()->getAs<FunctionProtoType>();
|
|
const llvm::FunctionType *FTy =
|
|
getTypes().GetFunctionType(getTypes().getFunctionInfo(MD),
|
|
FPT->isVariadic());
|
|
|
|
llvm::Function *Fn = llvm::Function::Create(FTy, linktype, OutName.str(),
|
|
&getModule());
|
|
CodeGenFunction(*this).GenerateThunk(Fn, GD, Extern, ThisAdjustment);
|
|
llvm::Constant *m = llvm::ConstantExpr::getBitCast(Fn, Ptr8Ty);
|
|
return m;
|
|
}
|
|
|
|
llvm::Constant *
|
|
CodeGenModule::BuildCovariantThunk(const GlobalDecl &GD, bool Extern,
|
|
const CovariantThunkAdjustment &Adjustment) {
|
|
const CXXMethodDecl *MD = cast<CXXMethodDecl>(GD.getDecl());
|
|
llvm::SmallString<256> OutName;
|
|
getMangleContext().mangleCovariantThunk(MD, Adjustment, OutName);
|
|
llvm::GlobalVariable::LinkageTypes linktype;
|
|
linktype = llvm::GlobalValue::WeakAnyLinkage;
|
|
if (!Extern)
|
|
linktype = llvm::GlobalValue::InternalLinkage;
|
|
llvm::Type *Ptr8Ty=llvm::PointerType::get(llvm::Type::getInt8Ty(VMContext),0);
|
|
const FunctionProtoType *FPT = MD->getType()->getAs<FunctionProtoType>();
|
|
const llvm::FunctionType *FTy =
|
|
getTypes().GetFunctionType(getTypes().getFunctionInfo(MD),
|
|
FPT->isVariadic());
|
|
|
|
llvm::Function *Fn = llvm::Function::Create(FTy, linktype, OutName.str(),
|
|
&getModule());
|
|
CodeGenFunction(*this).GenerateCovariantThunk(Fn, MD, Extern, Adjustment);
|
|
llvm::Constant *m = llvm::ConstantExpr::getBitCast(Fn, Ptr8Ty);
|
|
return m;
|
|
}
|
|
|
|
llvm::Value *
|
|
CodeGenFunction::GetVirtualCXXBaseClassOffset(llvm::Value *This,
|
|
const CXXRecordDecl *ClassDecl,
|
|
const CXXRecordDecl *BaseClassDecl) {
|
|
const llvm::Type *Int8PtrTy =
|
|
llvm::Type::getInt8Ty(VMContext)->getPointerTo();
|
|
|
|
llvm::Value *VTablePtr = Builder.CreateBitCast(This,
|
|
Int8PtrTy->getPointerTo());
|
|
VTablePtr = Builder.CreateLoad(VTablePtr, "vtable");
|
|
|
|
int64_t VBaseOffsetIndex =
|
|
CGM.getVtableInfo().getVirtualBaseOffsetIndex(ClassDecl, BaseClassDecl);
|
|
|
|
llvm::Value *VBaseOffsetPtr =
|
|
Builder.CreateConstGEP1_64(VTablePtr, VBaseOffsetIndex, "vbase.offset.ptr");
|
|
const llvm::Type *PtrDiffTy =
|
|
ConvertType(getContext().getPointerDiffType());
|
|
|
|
VBaseOffsetPtr = Builder.CreateBitCast(VBaseOffsetPtr,
|
|
PtrDiffTy->getPointerTo());
|
|
|
|
llvm::Value *VBaseOffset = Builder.CreateLoad(VBaseOffsetPtr, "vbase.offset");
|
|
|
|
return VBaseOffset;
|
|
}
|
|
|
|
static llvm::Value *BuildVirtualCall(CodeGenFunction &CGF, uint64_t VtableIndex,
|
|
llvm::Value *This, const llvm::Type *Ty) {
|
|
Ty = Ty->getPointerTo()->getPointerTo()->getPointerTo();
|
|
|
|
llvm::Value *Vtable = CGF.Builder.CreateBitCast(This, Ty);
|
|
Vtable = CGF.Builder.CreateLoad(Vtable);
|
|
|
|
llvm::Value *VFuncPtr =
|
|
CGF.Builder.CreateConstInBoundsGEP1_64(Vtable, VtableIndex, "vfn");
|
|
return CGF.Builder.CreateLoad(VFuncPtr);
|
|
}
|
|
|
|
llvm::Value *
|
|
CodeGenFunction::BuildVirtualCall(const CXXMethodDecl *MD, llvm::Value *This,
|
|
const llvm::Type *Ty) {
|
|
MD = MD->getCanonicalDecl();
|
|
uint64_t VtableIndex = CGM.getVtableInfo().getMethodVtableIndex(MD);
|
|
|
|
return ::BuildVirtualCall(*this, VtableIndex, This, Ty);
|
|
}
|
|
|
|
llvm::Value *
|
|
CodeGenFunction::BuildVirtualCall(const CXXDestructorDecl *DD, CXXDtorType Type,
|
|
llvm::Value *&This, const llvm::Type *Ty) {
|
|
DD = cast<CXXDestructorDecl>(DD->getCanonicalDecl());
|
|
uint64_t VtableIndex =
|
|
CGM.getVtableInfo().getMethodVtableIndex(GlobalDecl(DD, Type));
|
|
|
|
return ::BuildVirtualCall(*this, VtableIndex, This, Ty);
|
|
}
|
|
|
|
/// EmitClassAggrMemberwiseCopy - This routine generates code to copy a class
|
|
/// array of objects from SrcValue to DestValue. Copying can be either a bitwise
|
|
/// copy or via a copy constructor call.
|
|
// FIXME. Consolidate this with EmitCXXAggrConstructorCall.
|
|
void CodeGenFunction::EmitClassAggrMemberwiseCopy(llvm::Value *Dest,
|
|
llvm::Value *Src,
|
|
const ArrayType *Array,
|
|
const CXXRecordDecl *BaseClassDecl,
|
|
QualType Ty) {
|
|
const ConstantArrayType *CA = dyn_cast<ConstantArrayType>(Array);
|
|
assert(CA && "VLA cannot be copied over");
|
|
bool BitwiseCopy = BaseClassDecl->hasTrivialCopyConstructor();
|
|
|
|
// Create a temporary for the loop index and initialize it with 0.
|
|
llvm::Value *IndexPtr = CreateTempAlloca(llvm::Type::getInt64Ty(VMContext),
|
|
"loop.index");
|
|
llvm::Value* zeroConstant =
|
|
llvm::Constant::getNullValue(llvm::Type::getInt64Ty(VMContext));
|
|
Builder.CreateStore(zeroConstant, IndexPtr);
|
|
// Start the loop with a block that tests the condition.
|
|
llvm::BasicBlock *CondBlock = createBasicBlock("for.cond");
|
|
llvm::BasicBlock *AfterFor = createBasicBlock("for.end");
|
|
|
|
EmitBlock(CondBlock);
|
|
|
|
llvm::BasicBlock *ForBody = createBasicBlock("for.body");
|
|
// Generate: if (loop-index < number-of-elements fall to the loop body,
|
|
// otherwise, go to the block after the for-loop.
|
|
uint64_t NumElements = getContext().getConstantArrayElementCount(CA);
|
|
llvm::Value * NumElementsPtr =
|
|
llvm::ConstantInt::get(llvm::Type::getInt64Ty(VMContext), NumElements);
|
|
llvm::Value *Counter = Builder.CreateLoad(IndexPtr);
|
|
llvm::Value *IsLess = Builder.CreateICmpULT(Counter, NumElementsPtr,
|
|
"isless");
|
|
// If the condition is true, execute the body.
|
|
Builder.CreateCondBr(IsLess, ForBody, AfterFor);
|
|
|
|
EmitBlock(ForBody);
|
|
llvm::BasicBlock *ContinueBlock = createBasicBlock("for.inc");
|
|
// Inside the loop body, emit the constructor call on the array element.
|
|
Counter = Builder.CreateLoad(IndexPtr);
|
|
Src = Builder.CreateInBoundsGEP(Src, Counter, "srcaddress");
|
|
Dest = Builder.CreateInBoundsGEP(Dest, Counter, "destaddress");
|
|
if (BitwiseCopy)
|
|
EmitAggregateCopy(Dest, Src, Ty);
|
|
else if (CXXConstructorDecl *BaseCopyCtor =
|
|
BaseClassDecl->getCopyConstructor(getContext(), 0)) {
|
|
llvm::Value *Callee = CGM.GetAddrOfCXXConstructor(BaseCopyCtor,
|
|
Ctor_Complete);
|
|
CallArgList CallArgs;
|
|
// Push the this (Dest) ptr.
|
|
CallArgs.push_back(std::make_pair(RValue::get(Dest),
|
|
BaseCopyCtor->getThisType(getContext())));
|
|
|
|
// Push the Src ptr.
|
|
CallArgs.push_back(std::make_pair(RValue::get(Src),
|
|
BaseCopyCtor->getParamDecl(0)->getType()));
|
|
QualType ResultType =
|
|
BaseCopyCtor->getType()->getAs<FunctionType>()->getResultType();
|
|
EmitCall(CGM.getTypes().getFunctionInfo(ResultType, CallArgs),
|
|
Callee, CallArgs, BaseCopyCtor);
|
|
}
|
|
EmitBlock(ContinueBlock);
|
|
|
|
// Emit the increment of the loop counter.
|
|
llvm::Value *NextVal = llvm::ConstantInt::get(Counter->getType(), 1);
|
|
Counter = Builder.CreateLoad(IndexPtr);
|
|
NextVal = Builder.CreateAdd(Counter, NextVal, "inc");
|
|
Builder.CreateStore(NextVal, IndexPtr);
|
|
|
|
// Finally, branch back up to the condition for the next iteration.
|
|
EmitBranch(CondBlock);
|
|
|
|
// Emit the fall-through block.
|
|
EmitBlock(AfterFor, true);
|
|
}
|
|
|
|
/// EmitClassAggrCopyAssignment - This routine generates code to assign a class
|
|
/// array of objects from SrcValue to DestValue. Assignment can be either a
|
|
/// bitwise assignment or via a copy assignment operator function call.
|
|
/// FIXME. This can be consolidated with EmitClassAggrMemberwiseCopy
|
|
void CodeGenFunction::EmitClassAggrCopyAssignment(llvm::Value *Dest,
|
|
llvm::Value *Src,
|
|
const ArrayType *Array,
|
|
const CXXRecordDecl *BaseClassDecl,
|
|
QualType Ty) {
|
|
const ConstantArrayType *CA = dyn_cast<ConstantArrayType>(Array);
|
|
assert(CA && "VLA cannot be asssigned");
|
|
bool BitwiseAssign = BaseClassDecl->hasTrivialCopyAssignment();
|
|
|
|
// Create a temporary for the loop index and initialize it with 0.
|
|
llvm::Value *IndexPtr = CreateTempAlloca(llvm::Type::getInt64Ty(VMContext),
|
|
"loop.index");
|
|
llvm::Value* zeroConstant =
|
|
llvm::Constant::getNullValue(llvm::Type::getInt64Ty(VMContext));
|
|
Builder.CreateStore(zeroConstant, IndexPtr);
|
|
// Start the loop with a block that tests the condition.
|
|
llvm::BasicBlock *CondBlock = createBasicBlock("for.cond");
|
|
llvm::BasicBlock *AfterFor = createBasicBlock("for.end");
|
|
|
|
EmitBlock(CondBlock);
|
|
|
|
llvm::BasicBlock *ForBody = createBasicBlock("for.body");
|
|
// Generate: if (loop-index < number-of-elements fall to the loop body,
|
|
// otherwise, go to the block after the for-loop.
|
|
uint64_t NumElements = getContext().getConstantArrayElementCount(CA);
|
|
llvm::Value * NumElementsPtr =
|
|
llvm::ConstantInt::get(llvm::Type::getInt64Ty(VMContext), NumElements);
|
|
llvm::Value *Counter = Builder.CreateLoad(IndexPtr);
|
|
llvm::Value *IsLess = Builder.CreateICmpULT(Counter, NumElementsPtr,
|
|
"isless");
|
|
// If the condition is true, execute the body.
|
|
Builder.CreateCondBr(IsLess, ForBody, AfterFor);
|
|
|
|
EmitBlock(ForBody);
|
|
llvm::BasicBlock *ContinueBlock = createBasicBlock("for.inc");
|
|
// Inside the loop body, emit the assignment operator call on array element.
|
|
Counter = Builder.CreateLoad(IndexPtr);
|
|
Src = Builder.CreateInBoundsGEP(Src, Counter, "srcaddress");
|
|
Dest = Builder.CreateInBoundsGEP(Dest, Counter, "destaddress");
|
|
const CXXMethodDecl *MD = 0;
|
|
if (BitwiseAssign)
|
|
EmitAggregateCopy(Dest, Src, Ty);
|
|
else {
|
|
bool hasCopyAssign = BaseClassDecl->hasConstCopyAssignment(getContext(),
|
|
MD);
|
|
assert(hasCopyAssign && "EmitClassAggrCopyAssignment - No user assign");
|
|
(void)hasCopyAssign;
|
|
const FunctionProtoType *FPT = MD->getType()->getAs<FunctionProtoType>();
|
|
const llvm::Type *LTy =
|
|
CGM.getTypes().GetFunctionType(CGM.getTypes().getFunctionInfo(MD),
|
|
FPT->isVariadic());
|
|
llvm::Constant *Callee = CGM.GetAddrOfFunction(MD, LTy);
|
|
|
|
CallArgList CallArgs;
|
|
// Push the this (Dest) ptr.
|
|
CallArgs.push_back(std::make_pair(RValue::get(Dest),
|
|
MD->getThisType(getContext())));
|
|
|
|
// Push the Src ptr.
|
|
CallArgs.push_back(std::make_pair(RValue::get(Src),
|
|
MD->getParamDecl(0)->getType()));
|
|
QualType ResultType = MD->getType()->getAs<FunctionType>()->getResultType();
|
|
EmitCall(CGM.getTypes().getFunctionInfo(ResultType, CallArgs),
|
|
Callee, CallArgs, MD);
|
|
}
|
|
EmitBlock(ContinueBlock);
|
|
|
|
// Emit the increment of the loop counter.
|
|
llvm::Value *NextVal = llvm::ConstantInt::get(Counter->getType(), 1);
|
|
Counter = Builder.CreateLoad(IndexPtr);
|
|
NextVal = Builder.CreateAdd(Counter, NextVal, "inc");
|
|
Builder.CreateStore(NextVal, IndexPtr);
|
|
|
|
// Finally, branch back up to the condition for the next iteration.
|
|
EmitBranch(CondBlock);
|
|
|
|
// Emit the fall-through block.
|
|
EmitBlock(AfterFor, true);
|
|
}
|
|
|
|
/// EmitClassMemberwiseCopy - This routine generates code to copy a class
|
|
/// object from SrcValue to DestValue. Copying can be either a bitwise copy
|
|
/// or via a copy constructor call.
|
|
void CodeGenFunction::EmitClassMemberwiseCopy(
|
|
llvm::Value *Dest, llvm::Value *Src,
|
|
const CXXRecordDecl *ClassDecl,
|
|
const CXXRecordDecl *BaseClassDecl, QualType Ty) {
|
|
if (ClassDecl) {
|
|
Dest = GetAddressOfBaseClass(Dest, ClassDecl, BaseClassDecl,
|
|
/*NullCheckValue=*/false);
|
|
Src = GetAddressOfBaseClass(Src, ClassDecl, BaseClassDecl,
|
|
/*NullCheckValue=*/false);
|
|
}
|
|
if (BaseClassDecl->hasTrivialCopyConstructor()) {
|
|
EmitAggregateCopy(Dest, Src, Ty);
|
|
return;
|
|
}
|
|
|
|
if (CXXConstructorDecl *BaseCopyCtor =
|
|
BaseClassDecl->getCopyConstructor(getContext(), 0)) {
|
|
llvm::Value *Callee = CGM.GetAddrOfCXXConstructor(BaseCopyCtor,
|
|
Ctor_Complete);
|
|
CallArgList CallArgs;
|
|
// Push the this (Dest) ptr.
|
|
CallArgs.push_back(std::make_pair(RValue::get(Dest),
|
|
BaseCopyCtor->getThisType(getContext())));
|
|
|
|
// Push the Src ptr.
|
|
CallArgs.push_back(std::make_pair(RValue::get(Src),
|
|
BaseCopyCtor->getParamDecl(0)->getType()));
|
|
QualType ResultType =
|
|
BaseCopyCtor->getType()->getAs<FunctionType>()->getResultType();
|
|
EmitCall(CGM.getTypes().getFunctionInfo(ResultType, CallArgs),
|
|
Callee, CallArgs, BaseCopyCtor);
|
|
}
|
|
}
|
|
|
|
/// EmitClassCopyAssignment - This routine generates code to copy assign a class
|
|
/// object from SrcValue to DestValue. Assignment can be either a bitwise
|
|
/// assignment of via an assignment operator call.
|
|
// FIXME. Consolidate this with EmitClassMemberwiseCopy as they share a lot.
|
|
void CodeGenFunction::EmitClassCopyAssignment(
|
|
llvm::Value *Dest, llvm::Value *Src,
|
|
const CXXRecordDecl *ClassDecl,
|
|
const CXXRecordDecl *BaseClassDecl,
|
|
QualType Ty) {
|
|
if (ClassDecl) {
|
|
Dest = GetAddressOfBaseClass(Dest, ClassDecl, BaseClassDecl,
|
|
/*NullCheckValue=*/false);
|
|
Src = GetAddressOfBaseClass(Src, ClassDecl, BaseClassDecl,
|
|
/*NullCheckValue=*/false);
|
|
}
|
|
if (BaseClassDecl->hasTrivialCopyAssignment()) {
|
|
EmitAggregateCopy(Dest, Src, Ty);
|
|
return;
|
|
}
|
|
|
|
const CXXMethodDecl *MD = 0;
|
|
bool ConstCopyAssignOp = BaseClassDecl->hasConstCopyAssignment(getContext(),
|
|
MD);
|
|
assert(ConstCopyAssignOp && "EmitClassCopyAssignment - missing copy assign");
|
|
(void)ConstCopyAssignOp;
|
|
|
|
const FunctionProtoType *FPT = MD->getType()->getAs<FunctionProtoType>();
|
|
const llvm::Type *LTy =
|
|
CGM.getTypes().GetFunctionType(CGM.getTypes().getFunctionInfo(MD),
|
|
FPT->isVariadic());
|
|
llvm::Constant *Callee = CGM.GetAddrOfFunction(MD, LTy);
|
|
|
|
CallArgList CallArgs;
|
|
// Push the this (Dest) ptr.
|
|
CallArgs.push_back(std::make_pair(RValue::get(Dest),
|
|
MD->getThisType(getContext())));
|
|
|
|
// Push the Src ptr.
|
|
CallArgs.push_back(std::make_pair(RValue::get(Src),
|
|
MD->getParamDecl(0)->getType()));
|
|
QualType ResultType =
|
|
MD->getType()->getAs<FunctionType>()->getResultType();
|
|
EmitCall(CGM.getTypes().getFunctionInfo(ResultType, CallArgs),
|
|
Callee, CallArgs, MD);
|
|
}
|
|
|
|
/// SynthesizeDefaultConstructor - synthesize a default constructor
|
|
void
|
|
CodeGenFunction::SynthesizeDefaultConstructor(const CXXConstructorDecl *Ctor,
|
|
CXXCtorType Type,
|
|
llvm::Function *Fn,
|
|
const FunctionArgList &Args) {
|
|
assert(!Ctor->isTrivial() && "shouldn't need to generate trivial ctor");
|
|
StartFunction(GlobalDecl(Ctor, Type), Ctor->getResultType(), Fn, Args,
|
|
SourceLocation());
|
|
EmitCtorPrologue(Ctor, Type);
|
|
FinishFunction();
|
|
}
|
|
|
|
/// SynthesizeCXXCopyConstructor - This routine implicitly defines body of a
|
|
/// copy constructor, in accordance with section 12.8 (p7 and p8) of C++03
|
|
/// The implicitly-defined copy constructor for class X performs a memberwise
|
|
/// copy of its subobjects. The order of copying is the same as the order of
|
|
/// initialization of bases and members in a user-defined constructor
|
|
/// Each subobject is copied in the manner appropriate to its type:
|
|
/// if the subobject is of class type, the copy constructor for the class is
|
|
/// used;
|
|
/// if the subobject is an array, each element is copied, in the manner
|
|
/// appropriate to the element type;
|
|
/// if the subobject is of scalar type, the built-in assignment operator is
|
|
/// used.
|
|
/// Virtual base class subobjects shall be copied only once by the
|
|
/// implicitly-defined copy constructor
|
|
|
|
void
|
|
CodeGenFunction::SynthesizeCXXCopyConstructor(const CXXConstructorDecl *Ctor,
|
|
CXXCtorType Type,
|
|
llvm::Function *Fn,
|
|
const FunctionArgList &Args) {
|
|
const CXXRecordDecl *ClassDecl = Ctor->getParent();
|
|
assert(!ClassDecl->hasUserDeclaredCopyConstructor() &&
|
|
"SynthesizeCXXCopyConstructor - copy constructor has definition already");
|
|
assert(!Ctor->isTrivial() && "shouldn't need to generate trivial ctor");
|
|
StartFunction(GlobalDecl(Ctor, Type), Ctor->getResultType(), Fn, Args,
|
|
SourceLocation());
|
|
|
|
FunctionArgList::const_iterator i = Args.begin();
|
|
const VarDecl *ThisArg = i->first;
|
|
llvm::Value *ThisObj = GetAddrOfLocalVar(ThisArg);
|
|
llvm::Value *LoadOfThis = Builder.CreateLoad(ThisObj, "this");
|
|
const VarDecl *SrcArg = (i+1)->first;
|
|
llvm::Value *SrcObj = GetAddrOfLocalVar(SrcArg);
|
|
llvm::Value *LoadOfSrc = Builder.CreateLoad(SrcObj);
|
|
|
|
for (CXXRecordDecl::base_class_const_iterator Base = ClassDecl->bases_begin();
|
|
Base != ClassDecl->bases_end(); ++Base) {
|
|
// FIXME. copy constrution of virtual base NYI
|
|
if (Base->isVirtual())
|
|
continue;
|
|
|
|
CXXRecordDecl *BaseClassDecl
|
|
= cast<CXXRecordDecl>(Base->getType()->getAs<RecordType>()->getDecl());
|
|
EmitClassMemberwiseCopy(LoadOfThis, LoadOfSrc, ClassDecl, BaseClassDecl,
|
|
Base->getType());
|
|
}
|
|
|
|
for (CXXRecordDecl::field_iterator I = ClassDecl->field_begin(),
|
|
E = ClassDecl->field_end(); I != E; ++I) {
|
|
const FieldDecl *Field = *I;
|
|
|
|
QualType FieldType = getContext().getCanonicalType(Field->getType());
|
|
const ConstantArrayType *Array =
|
|
getContext().getAsConstantArrayType(FieldType);
|
|
if (Array)
|
|
FieldType = getContext().getBaseElementType(FieldType);
|
|
|
|
if (const RecordType *FieldClassType = FieldType->getAs<RecordType>()) {
|
|
CXXRecordDecl *FieldClassDecl
|
|
= cast<CXXRecordDecl>(FieldClassType->getDecl());
|
|
LValue LHS = EmitLValueForField(LoadOfThis, Field, false, 0);
|
|
LValue RHS = EmitLValueForField(LoadOfSrc, Field, false, 0);
|
|
if (Array) {
|
|
const llvm::Type *BasePtr = ConvertType(FieldType);
|
|
BasePtr = llvm::PointerType::getUnqual(BasePtr);
|
|
llvm::Value *DestBaseAddrPtr =
|
|
Builder.CreateBitCast(LHS.getAddress(), BasePtr);
|
|
llvm::Value *SrcBaseAddrPtr =
|
|
Builder.CreateBitCast(RHS.getAddress(), BasePtr);
|
|
EmitClassAggrMemberwiseCopy(DestBaseAddrPtr, SrcBaseAddrPtr, Array,
|
|
FieldClassDecl, FieldType);
|
|
}
|
|
else
|
|
EmitClassMemberwiseCopy(LHS.getAddress(), RHS.getAddress(),
|
|
0 /*ClassDecl*/, FieldClassDecl, FieldType);
|
|
continue;
|
|
}
|
|
|
|
if (Field->getType()->isReferenceType()) {
|
|
unsigned FieldIndex = CGM.getTypes().getLLVMFieldNo(Field);
|
|
|
|
llvm::Value *LHS = Builder.CreateStructGEP(LoadOfThis, FieldIndex,
|
|
"lhs.ref");
|
|
|
|
llvm::Value *RHS = Builder.CreateStructGEP(LoadOfThis, FieldIndex,
|
|
"rhs.ref");
|
|
|
|
// Load the value in RHS.
|
|
RHS = Builder.CreateLoad(RHS);
|
|
|
|
// And store it in the LHS
|
|
Builder.CreateStore(RHS, LHS);
|
|
|
|
continue;
|
|
}
|
|
// Do a built-in assignment of scalar data members.
|
|
LValue LHS = EmitLValueForField(LoadOfThis, Field, false, 0);
|
|
LValue RHS = EmitLValueForField(LoadOfSrc, Field, false, 0);
|
|
|
|
if (!hasAggregateLLVMType(Field->getType())) {
|
|
RValue RVRHS = EmitLoadOfLValue(RHS, Field->getType());
|
|
EmitStoreThroughLValue(RVRHS, LHS, Field->getType());
|
|
} else if (Field->getType()->isAnyComplexType()) {
|
|
ComplexPairTy Pair = LoadComplexFromAddr(RHS.getAddress(),
|
|
RHS.isVolatileQualified());
|
|
StoreComplexToAddr(Pair, LHS.getAddress(), LHS.isVolatileQualified());
|
|
} else {
|
|
EmitAggregateCopy(LHS.getAddress(), RHS.getAddress(), Field->getType());
|
|
}
|
|
}
|
|
|
|
InitializeVtablePtrs(ClassDecl);
|
|
FinishFunction();
|
|
}
|
|
|
|
/// SynthesizeCXXCopyAssignment - Implicitly define copy assignment operator.
|
|
/// Before the implicitly-declared copy assignment operator for a class is
|
|
/// implicitly defined, all implicitly- declared copy assignment operators for
|
|
/// its direct base classes and its nonstatic data members shall have been
|
|
/// implicitly defined. [12.8-p12]
|
|
/// The implicitly-defined copy assignment operator for class X performs
|
|
/// memberwise assignment of its subob- jects. The direct base classes of X are
|
|
/// assigned first, in the order of their declaration in
|
|
/// the base-specifier-list, and then the immediate nonstatic data members of X
|
|
/// are assigned, in the order in which they were declared in the class
|
|
/// definition.Each subobject is assigned in the manner appropriate to its type:
|
|
/// if the subobject is of class type, the copy assignment operator for the
|
|
/// class is used (as if by explicit qualification; that is, ignoring any
|
|
/// possible virtual overriding functions in more derived classes);
|
|
///
|
|
/// if the subobject is an array, each element is assigned, in the manner
|
|
/// appropriate to the element type;
|
|
///
|
|
/// if the subobject is of scalar type, the built-in assignment operator is
|
|
/// used.
|
|
void CodeGenFunction::SynthesizeCXXCopyAssignment(const CXXMethodDecl *CD,
|
|
llvm::Function *Fn,
|
|
const FunctionArgList &Args) {
|
|
|
|
const CXXRecordDecl *ClassDecl = cast<CXXRecordDecl>(CD->getDeclContext());
|
|
assert(!ClassDecl->hasUserDeclaredCopyAssignment() &&
|
|
"SynthesizeCXXCopyAssignment - copy assignment has user declaration");
|
|
StartFunction(CD, CD->getResultType(), Fn, Args, SourceLocation());
|
|
|
|
FunctionArgList::const_iterator i = Args.begin();
|
|
const VarDecl *ThisArg = i->first;
|
|
llvm::Value *ThisObj = GetAddrOfLocalVar(ThisArg);
|
|
llvm::Value *LoadOfThis = Builder.CreateLoad(ThisObj, "this");
|
|
const VarDecl *SrcArg = (i+1)->first;
|
|
llvm::Value *SrcObj = GetAddrOfLocalVar(SrcArg);
|
|
llvm::Value *LoadOfSrc = Builder.CreateLoad(SrcObj);
|
|
|
|
for (CXXRecordDecl::base_class_const_iterator Base = ClassDecl->bases_begin();
|
|
Base != ClassDecl->bases_end(); ++Base) {
|
|
// FIXME. copy assignment of virtual base NYI
|
|
if (Base->isVirtual())
|
|
continue;
|
|
|
|
CXXRecordDecl *BaseClassDecl
|
|
= cast<CXXRecordDecl>(Base->getType()->getAs<RecordType>()->getDecl());
|
|
EmitClassCopyAssignment(LoadOfThis, LoadOfSrc, ClassDecl, BaseClassDecl,
|
|
Base->getType());
|
|
}
|
|
|
|
for (CXXRecordDecl::field_iterator Field = ClassDecl->field_begin(),
|
|
FieldEnd = ClassDecl->field_end();
|
|
Field != FieldEnd; ++Field) {
|
|
QualType FieldType = getContext().getCanonicalType((*Field)->getType());
|
|
const ConstantArrayType *Array =
|
|
getContext().getAsConstantArrayType(FieldType);
|
|
if (Array)
|
|
FieldType = getContext().getBaseElementType(FieldType);
|
|
|
|
if (const RecordType *FieldClassType = FieldType->getAs<RecordType>()) {
|
|
CXXRecordDecl *FieldClassDecl
|
|
= cast<CXXRecordDecl>(FieldClassType->getDecl());
|
|
LValue LHS = EmitLValueForField(LoadOfThis, *Field, false, 0);
|
|
LValue RHS = EmitLValueForField(LoadOfSrc, *Field, false, 0);
|
|
if (Array) {
|
|
const llvm::Type *BasePtr = ConvertType(FieldType);
|
|
BasePtr = llvm::PointerType::getUnqual(BasePtr);
|
|
llvm::Value *DestBaseAddrPtr =
|
|
Builder.CreateBitCast(LHS.getAddress(), BasePtr);
|
|
llvm::Value *SrcBaseAddrPtr =
|
|
Builder.CreateBitCast(RHS.getAddress(), BasePtr);
|
|
EmitClassAggrCopyAssignment(DestBaseAddrPtr, SrcBaseAddrPtr, Array,
|
|
FieldClassDecl, FieldType);
|
|
}
|
|
else
|
|
EmitClassCopyAssignment(LHS.getAddress(), RHS.getAddress(),
|
|
0 /*ClassDecl*/, FieldClassDecl, FieldType);
|
|
continue;
|
|
}
|
|
// Do a built-in assignment of scalar data members.
|
|
LValue LHS = EmitLValueForField(LoadOfThis, *Field, false, 0);
|
|
LValue RHS = EmitLValueForField(LoadOfSrc, *Field, false, 0);
|
|
if (!hasAggregateLLVMType(Field->getType())) {
|
|
RValue RVRHS = EmitLoadOfLValue(RHS, Field->getType());
|
|
EmitStoreThroughLValue(RVRHS, LHS, Field->getType());
|
|
} else if (Field->getType()->isAnyComplexType()) {
|
|
ComplexPairTy Pair = LoadComplexFromAddr(RHS.getAddress(),
|
|
RHS.isVolatileQualified());
|
|
StoreComplexToAddr(Pair, LHS.getAddress(), LHS.isVolatileQualified());
|
|
} else {
|
|
EmitAggregateCopy(LHS.getAddress(), RHS.getAddress(), Field->getType());
|
|
}
|
|
}
|
|
|
|
// return *this;
|
|
Builder.CreateStore(LoadOfThis, ReturnValue);
|
|
|
|
FinishFunction();
|
|
}
|
|
|
|
static void EmitBaseInitializer(CodeGenFunction &CGF,
|
|
const CXXRecordDecl *ClassDecl,
|
|
CXXBaseOrMemberInitializer *BaseInit,
|
|
CXXCtorType CtorType) {
|
|
assert(BaseInit->isBaseInitializer() &&
|
|
"Must have base initializer!");
|
|
|
|
llvm::Value *ThisPtr = CGF.LoadCXXThis();
|
|
|
|
const Type *BaseType = BaseInit->getBaseClass();
|
|
CXXRecordDecl *BaseClassDecl =
|
|
cast<CXXRecordDecl>(BaseType->getAs<RecordType>()->getDecl());
|
|
llvm::Value *V = CGF.GetAddressOfBaseClass(ThisPtr, ClassDecl,
|
|
BaseClassDecl,
|
|
/*NullCheckValue=*/false);
|
|
CGF.EmitCXXConstructorCall(BaseInit->getConstructor(),
|
|
CtorType, V,
|
|
BaseInit->const_arg_begin(),
|
|
BaseInit->const_arg_end());
|
|
}
|
|
|
|
static void EmitMemberInitializer(CodeGenFunction &CGF,
|
|
const CXXRecordDecl *ClassDecl,
|
|
CXXBaseOrMemberInitializer *MemberInit) {
|
|
assert(MemberInit->isMemberInitializer() &&
|
|
"Must have member initializer!");
|
|
|
|
// non-static data member initializers.
|
|
FieldDecl *Field = MemberInit->getMember();
|
|
QualType FieldType = CGF.getContext().getCanonicalType(Field->getType());
|
|
|
|
llvm::Value *ThisPtr = CGF.LoadCXXThis();
|
|
LValue LHS;
|
|
if (FieldType->isReferenceType()) {
|
|
// FIXME: This is really ugly; should be refactored somehow
|
|
unsigned idx = CGF.CGM.getTypes().getLLVMFieldNo(Field);
|
|
llvm::Value *V = CGF.Builder.CreateStructGEP(ThisPtr, idx, "tmp");
|
|
assert(!FieldType.getObjCGCAttr() && "fields cannot have GC attrs");
|
|
LHS = LValue::MakeAddr(V, CGF.MakeQualifiers(FieldType));
|
|
} else {
|
|
LHS = CGF.EmitLValueForField(ThisPtr, Field, ClassDecl->isUnion(), 0);
|
|
}
|
|
|
|
// If we are initializing an anonymous union field, drill down to the field.
|
|
if (MemberInit->getAnonUnionMember()) {
|
|
Field = MemberInit->getAnonUnionMember();
|
|
LHS = CGF.EmitLValueForField(LHS.getAddress(), Field,
|
|
/*IsUnion=*/true, 0);
|
|
FieldType = Field->getType();
|
|
}
|
|
|
|
// If the field is an array, branch based on the element type.
|
|
const ConstantArrayType *Array =
|
|
CGF.getContext().getAsConstantArrayType(FieldType);
|
|
if (Array)
|
|
FieldType = CGF.getContext().getBaseElementType(FieldType);
|
|
|
|
// We lose the constructor for anonymous union members, so handle them
|
|
// explicitly.
|
|
// FIXME: This is somwhat ugly.
|
|
if (MemberInit->getAnonUnionMember() && FieldType->getAs<RecordType>()) {
|
|
if (MemberInit->getNumArgs())
|
|
CGF.EmitAggExpr(*MemberInit->arg_begin(), LHS.getAddress(),
|
|
LHS.isVolatileQualified());
|
|
else
|
|
CGF.EmitAggregateClear(LHS.getAddress(), Field->getType());
|
|
return;
|
|
}
|
|
|
|
if (FieldType->getAs<RecordType>()) {
|
|
assert(MemberInit->getConstructor() &&
|
|
"EmitCtorPrologue - no constructor to initialize member");
|
|
if (Array) {
|
|
const llvm::Type *BasePtr = CGF.ConvertType(FieldType);
|
|
BasePtr = llvm::PointerType::getUnqual(BasePtr);
|
|
llvm::Value *BaseAddrPtr =
|
|
CGF.Builder.CreateBitCast(LHS.getAddress(), BasePtr);
|
|
CGF.EmitCXXAggrConstructorCall(MemberInit->getConstructor(),
|
|
Array, BaseAddrPtr,
|
|
MemberInit->const_arg_begin(),
|
|
MemberInit->const_arg_end());
|
|
}
|
|
else
|
|
CGF.EmitCXXConstructorCall(MemberInit->getConstructor(),
|
|
Ctor_Complete, LHS.getAddress(),
|
|
MemberInit->const_arg_begin(),
|
|
MemberInit->const_arg_end());
|
|
return;
|
|
}
|
|
|
|
assert(MemberInit->getNumArgs() == 1 && "Initializer count must be 1 only");
|
|
Expr *RhsExpr = *MemberInit->arg_begin();
|
|
RValue RHS;
|
|
if (FieldType->isReferenceType()) {
|
|
RHS = CGF.EmitReferenceBindingToExpr(RhsExpr, FieldType,
|
|
/*IsInitializer=*/true);
|
|
CGF.EmitStoreThroughLValue(RHS, LHS, FieldType);
|
|
} else if (Array) {
|
|
CGF.EmitMemSetToZero(LHS.getAddress(), Field->getType());
|
|
} else if (!CGF.hasAggregateLLVMType(RhsExpr->getType())) {
|
|
RHS = RValue::get(CGF.EmitScalarExpr(RhsExpr, true));
|
|
CGF.EmitStoreThroughLValue(RHS, LHS, FieldType);
|
|
} else if (RhsExpr->getType()->isAnyComplexType()) {
|
|
CGF.EmitComplexExprIntoAddr(RhsExpr, LHS.getAddress(),
|
|
LHS.isVolatileQualified());
|
|
} else {
|
|
// Handle member function pointers; other aggregates shouldn't get this far.
|
|
CGF.EmitAggExpr(RhsExpr, LHS.getAddress(), LHS.isVolatileQualified());
|
|
}
|
|
}
|
|
|
|
/// EmitCtorPrologue - This routine generates necessary code to initialize
|
|
/// base classes and non-static data members belonging to this constructor.
|
|
/// FIXME: This needs to take a CXXCtorType.
|
|
void CodeGenFunction::EmitCtorPrologue(const CXXConstructorDecl *CD,
|
|
CXXCtorType CtorType) {
|
|
const CXXRecordDecl *ClassDecl = CD->getParent();
|
|
|
|
// FIXME: Add vbase initialization
|
|
|
|
for (CXXConstructorDecl::init_const_iterator B = CD->init_begin(),
|
|
E = CD->init_end();
|
|
B != E; ++B) {
|
|
CXXBaseOrMemberInitializer *Member = (*B);
|
|
|
|
assert(LiveTemporaries.empty() &&
|
|
"Should not have any live temporaries at initializer start!");
|
|
|
|
if (Member->isBaseInitializer())
|
|
EmitBaseInitializer(*this, ClassDecl, Member, CtorType);
|
|
else
|
|
EmitMemberInitializer(*this, ClassDecl, Member);
|
|
|
|
// Pop any live temporaries that the initializers might have pushed.
|
|
while (!LiveTemporaries.empty())
|
|
PopCXXTemporary();
|
|
}
|
|
|
|
InitializeVtablePtrs(ClassDecl);
|
|
}
|
|
|
|
void CodeGenFunction::InitializeVtablePtrs(const CXXRecordDecl *ClassDecl) {
|
|
if (!ClassDecl->isDynamicClass())
|
|
return;
|
|
|
|
// Initialize the vtable pointer.
|
|
// FIXME: This needs to initialize secondary vtable pointers too.
|
|
llvm::Value *ThisPtr = LoadCXXThis();
|
|
|
|
llvm::Constant *Vtable = CGM.getVtableInfo().getVtable(ClassDecl);
|
|
uint64_t AddressPoint = CGM.getVtableInfo().getVtableAddressPoint(ClassDecl);
|
|
|
|
llvm::Value *VtableAddressPoint =
|
|
Builder.CreateConstInBoundsGEP2_64(Vtable, 0, AddressPoint);
|
|
|
|
llvm::Value *VtableField =
|
|
Builder.CreateBitCast(ThisPtr,
|
|
VtableAddressPoint->getType()->getPointerTo());
|
|
|
|
Builder.CreateStore(VtableAddressPoint, VtableField);
|
|
}
|
|
|
|
/// EmitDtorEpilogue - Emit all code that comes at the end of class's
|
|
/// destructor. This is to call destructors on members and base classes
|
|
/// in reverse order of their construction.
|
|
/// FIXME: This needs to take a CXXDtorType.
|
|
void CodeGenFunction::EmitDtorEpilogue(const CXXDestructorDecl *DD,
|
|
CXXDtorType DtorType) {
|
|
assert(!DD->isTrivial() &&
|
|
"Should not emit dtor epilogue for trivial dtor!");
|
|
|
|
const CXXRecordDecl *ClassDecl = DD->getParent();
|
|
|
|
// Collect the fields.
|
|
llvm::SmallVector<const FieldDecl *, 16> FieldDecls;
|
|
for (CXXRecordDecl::field_iterator I = ClassDecl->field_begin(),
|
|
E = ClassDecl->field_end(); I != E; ++I) {
|
|
const FieldDecl *Field = *I;
|
|
|
|
QualType FieldType = getContext().getCanonicalType(Field->getType());
|
|
FieldType = getContext().getBaseElementType(FieldType);
|
|
|
|
const RecordType *RT = FieldType->getAs<RecordType>();
|
|
if (!RT)
|
|
continue;
|
|
|
|
CXXRecordDecl *FieldClassDecl = cast<CXXRecordDecl>(RT->getDecl());
|
|
if (FieldClassDecl->hasTrivialDestructor())
|
|
continue;
|
|
|
|
FieldDecls.push_back(Field);
|
|
}
|
|
|
|
// Now destroy the fields.
|
|
for (size_t i = FieldDecls.size(); i > 0; --i) {
|
|
const FieldDecl *Field = FieldDecls[i - 1];
|
|
|
|
QualType FieldType = Field->getType();
|
|
const ConstantArrayType *Array =
|
|
getContext().getAsConstantArrayType(FieldType);
|
|
if (Array)
|
|
FieldType = getContext().getBaseElementType(FieldType);
|
|
|
|
const RecordType *RT = FieldType->getAs<RecordType>();
|
|
CXXRecordDecl *FieldClassDecl = cast<CXXRecordDecl>(RT->getDecl());
|
|
|
|
llvm::Value *ThisPtr = LoadCXXThis();
|
|
|
|
LValue LHS = EmitLValueForField(ThisPtr, Field,
|
|
/*isUnion=*/false,
|
|
// FIXME: Qualifiers?
|
|
/*CVRQualifiers=*/0);
|
|
if (Array) {
|
|
const llvm::Type *BasePtr = ConvertType(FieldType);
|
|
BasePtr = llvm::PointerType::getUnqual(BasePtr);
|
|
llvm::Value *BaseAddrPtr =
|
|
Builder.CreateBitCast(LHS.getAddress(), BasePtr);
|
|
EmitCXXAggrDestructorCall(FieldClassDecl->getDestructor(getContext()),
|
|
Array, BaseAddrPtr);
|
|
} else
|
|
EmitCXXDestructorCall(FieldClassDecl->getDestructor(getContext()),
|
|
Dtor_Complete, LHS.getAddress());
|
|
}
|
|
|
|
// Destroy non-virtual bases.
|
|
for (CXXRecordDecl::reverse_base_class_const_iterator I =
|
|
ClassDecl->bases_rbegin(), E = ClassDecl->bases_rend(); I != E; ++I) {
|
|
const CXXBaseSpecifier &Base = *I;
|
|
|
|
// Ignore virtual bases.
|
|
if (Base.isVirtual())
|
|
continue;
|
|
|
|
CXXRecordDecl *BaseClassDecl
|
|
= cast<CXXRecordDecl>(Base.getType()->getAs<RecordType>()->getDecl());
|
|
|
|
// Ignore trivial destructors.
|
|
if (BaseClassDecl->hasTrivialDestructor())
|
|
continue;
|
|
const CXXDestructorDecl *D = BaseClassDecl->getDestructor(getContext());
|
|
|
|
llvm::Value *V = GetAddressOfBaseClass(LoadCXXThis(),
|
|
ClassDecl, BaseClassDecl,
|
|
/*NullCheckValue=*/false);
|
|
EmitCXXDestructorCall(D, Dtor_Base, V);
|
|
}
|
|
|
|
// If we're emitting a base destructor, we don't want to emit calls to the
|
|
// virtual bases.
|
|
if (DtorType == Dtor_Base)
|
|
return;
|
|
|
|
// Handle virtual bases.
|
|
for (CXXRecordDecl::reverse_base_class_const_iterator I =
|
|
ClassDecl->vbases_rbegin(), E = ClassDecl->vbases_rend(); I != E; ++I) {
|
|
const CXXBaseSpecifier &Base = *I;
|
|
CXXRecordDecl *BaseClassDecl
|
|
= cast<CXXRecordDecl>(Base.getType()->getAs<RecordType>()->getDecl());
|
|
|
|
// Ignore trivial destructors.
|
|
if (BaseClassDecl->hasTrivialDestructor())
|
|
continue;
|
|
const CXXDestructorDecl *D = BaseClassDecl->getDestructor(getContext());
|
|
llvm::Value *V = GetAddressOfBaseClass(LoadCXXThis(),
|
|
ClassDecl, BaseClassDecl,
|
|
/*NullCheckValue=*/false);
|
|
EmitCXXDestructorCall(D, Dtor_Base, V);
|
|
}
|
|
|
|
// If we have a deleting destructor, emit a call to the delete operator.
|
|
if (DtorType == Dtor_Deleting) {
|
|
assert(DD->getOperatorDelete() &&
|
|
"operator delete missing - EmitDtorEpilogue");
|
|
EmitDeleteCall(DD->getOperatorDelete(), LoadCXXThis(),
|
|
getContext().getTagDeclType(ClassDecl));
|
|
}
|
|
}
|
|
|
|
void CodeGenFunction::SynthesizeDefaultDestructor(const CXXDestructorDecl *Dtor,
|
|
CXXDtorType DtorType,
|
|
llvm::Function *Fn,
|
|
const FunctionArgList &Args) {
|
|
assert(!Dtor->getParent()->hasUserDeclaredDestructor() &&
|
|
"SynthesizeDefaultDestructor - destructor has user declaration");
|
|
|
|
StartFunction(GlobalDecl(Dtor, DtorType), Dtor->getResultType(), Fn, Args,
|
|
SourceLocation());
|
|
|
|
EmitDtorEpilogue(Dtor, DtorType);
|
|
FinishFunction();
|
|
}
|