aboutsummaryrefslogtreecommitdiff
path: root/lib/CodeGen
diff options
context:
space:
mode:
authorAnders Carlsson <andersca@mac.com>2009-04-17 00:06:03 +0000
committerAnders Carlsson <andersca@mac.com>2009-04-17 00:06:03 +0000
commitb14095aa98c6fedd3625920c4ce834bcaf24d9f7 (patch)
treeff9e31b69547be398d8ccd80b87c8344ccc9386e /lib/CodeGen
parent025452fa0eda63e150cfaeebe64f0a19c96b3a06 (diff)
Implement basic code generation of constructor calls. We can now compile:
struct S { S(int, int); }; void f() { S s(10, 10); } git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@69330 91177308-0d34-0410-b5e6-96231b3b80d8
Diffstat (limited to 'lib/CodeGen')
-rw-r--r--lib/CodeGen/CGCXX.cpp45
-rw-r--r--lib/CodeGen/CGExpr.cpp2
-rw-r--r--lib/CodeGen/CGExprAgg.cpp14
-rw-r--r--lib/CodeGen/CodeGenFunction.h10
4 files changed, 71 insertions, 0 deletions
diff --git a/lib/CodeGen/CGCXX.cpp b/lib/CodeGen/CGCXX.cpp
index f76bd122a0..e61cef3498 100644
--- a/lib/CodeGen/CGCXX.cpp
+++ b/lib/CodeGen/CGCXX.cpp
@@ -145,6 +145,51 @@ void CodeGenModule::EmitCXXConstructor(const CXXConstructorDecl *D,
SetLLVMFunctionAttributesForDefinition(D, Fn);
}
+void
+CodeGenFunction::EmitCXXConstructorCall(const CXXConstructorDecl *D,
+ CXXCtorType Type,
+ llvm::Value *This,
+ CallExpr::const_arg_iterator ArgBeg,
+ CallExpr::const_arg_iterator ArgEnd) {
+ CallArgList Args;
+
+ // Push the 'this' pointer.
+ Args.push_back(std::make_pair(RValue::get(This),
+ D->getThisType(getContext())));
+
+ EmitCallArgs(Args, D->getType()->getAsFunctionProtoType(), ArgBeg, ArgEnd);
+
+ EmitCall(CGM.getTypes().getFunctionInfo(getContext().VoidTy, Args),
+ CGM.GetAddrOfCXXConstructor(D, Type), Args, D);
+}
+
+LValue
+CodeGenFunction::EmitCXXTemporaryObjectExprLValue(
+ const CXXTemporaryObjectExpr *E) {
+ // Allocate the destination.
+ llvm::Value *Dest = CreateTempAlloca(ConvertType(E->getType()), "tmp");
+
+ EmitCXXTemporaryObjectExpr(Dest, E);
+
+ return LValue::MakeAddr(Dest, E->getType().getCVRQualifiers(),
+ getContext().getObjCGCAttrKind(E->getType()));
+}
+
+void
+CodeGenFunction::EmitCXXTemporaryObjectExpr(llvm::Value *Dest,
+ const CXXTemporaryObjectExpr *E) {
+ assert(Dest && "Must have a destination!");
+
+ const CXXRecordDecl *RD =
+ cast<CXXRecordDecl>(E->getType()->getAsRecordType()->getDecl());
+ if (RD->hasTrivialConstructor())
+ return;
+
+ // Call the constructor.
+ EmitCXXConstructorCall(E->getConstructor(), Ctor_Complete, Dest,
+ E->arg_begin(), E->arg_end());
+}
+
static bool canGenerateCXXConstructor(const CXXConstructorDecl *D,
ASTContext &Context) {
const CXXRecordDecl *RD = D->getParent();
diff --git a/lib/CodeGen/CGExpr.cpp b/lib/CodeGen/CGExpr.cpp
index bf862e72c3..f7f3a5ea72 100644
--- a/lib/CodeGen/CGExpr.cpp
+++ b/lib/CodeGen/CGExpr.cpp
@@ -191,6 +191,8 @@ LValue CodeGenFunction::EmitLValue(const Expr *E) {
case Expr::CXXReinterpretCastExprClass:
case Expr::CXXConstCastExprClass:
return EmitCastLValue(cast<CastExpr>(E));
+ case Expr::CXXTemporaryObjectExprClass:
+ return EmitCXXTemporaryObjectExprLValue(cast<CXXTemporaryObjectExpr>(E));
}
}
diff --git a/lib/CodeGen/CGExprAgg.cpp b/lib/CodeGen/CGExprAgg.cpp
index acca396c0b..9d6df5c820 100644
--- a/lib/CodeGen/CGExprAgg.cpp
+++ b/lib/CodeGen/CGExprAgg.cpp
@@ -14,6 +14,7 @@
#include "CodeGenFunction.h"
#include "CodeGenModule.h"
#include "clang/AST/ASTContext.h"
+#include "clang/AST/DeclCXX.h"
#include "clang/AST/StmtVisitor.h"
#include "llvm/Constants.h"
#include "llvm/Function.h"
@@ -96,6 +97,7 @@ public:
void VisitCXXDefaultArgExpr(CXXDefaultArgExpr *DAE) {
Visit(DAE->getExpr());
}
+ void VisitCXXTemporaryObjectExpr(const CXXTemporaryObjectExpr *E);
void VisitVAArgExpr(VAArgExpr *E);
void EmitInitializationToLValue(Expr *E, LValue Address);
@@ -290,6 +292,18 @@ void AggExprEmitter::VisitVAArgExpr(VAArgExpr *VE) {
CGF.EmitAggregateCopy(DestPtr, ArgPtr, VE->getType());
}
+void
+AggExprEmitter::VisitCXXTemporaryObjectExpr(const CXXTemporaryObjectExpr *E) {
+ llvm::Value *This = 0;
+
+ if (DestPtr)
+ This = DestPtr;
+ else
+ This = CGF.CreateTempAlloca(CGF.ConvertType(E->getType()), "tmp");
+
+ CGF.EmitCXXTemporaryObjectExpr(This, E);
+}
+
void AggExprEmitter::EmitInitializationToLValue(Expr* E, LValue LV) {
// FIXME: Are initializers affected by volatile?
if (isa<ImplicitValueInitExpr>(E)) {
diff --git a/lib/CodeGen/CodeGenFunction.h b/lib/CodeGen/CodeGenFunction.h
index e96b815dd5..d511da3f7f 100644
--- a/lib/CodeGen/CodeGenFunction.h
+++ b/lib/CodeGen/CodeGenFunction.h
@@ -25,6 +25,7 @@
#include "CGBlocks.h"
#include "CGBuilder.h"
#include "CGCall.h"
+#include "CGCXX.h"
#include "CGValue.h"
namespace llvm {
@@ -458,6 +459,11 @@ public:
/// generating code for an C++ member function.
llvm::Value *LoadCXXThis();
+ void EmitCXXConstructorCall(const CXXConstructorDecl *D, CXXCtorType Type,
+ llvm::Value *This,
+ CallExpr::const_arg_iterator ArgBeg,
+ CallExpr::const_arg_iterator ArgEnd);
+
//===--------------------------------------------------------------------===//
// Declaration Emission
//===--------------------------------------------------------------------===//
@@ -632,6 +638,7 @@ public:
LValue EmitBlockDeclRefLValue(const BlockDeclRefExpr *E);
LValue EmitCXXConditionDeclLValue(const CXXConditionDeclExpr *E);
+ LValue EmitCXXTemporaryObjectExprLValue(const CXXTemporaryObjectExpr *E);
LValue EmitObjCMessageExprLValue(const ObjCMessageExpr *E);
LValue EmitObjCIvarRefLValue(const ObjCIvarRefExpr *E);
@@ -743,6 +750,9 @@ public:
void GenerateStaticCXXBlockVarDeclInit(const VarDecl &D,
llvm::GlobalVariable *GV);
+ void EmitCXXTemporaryObjectExpr(llvm::Value *Dest,
+ const CXXTemporaryObjectExpr *E);
+
//===--------------------------------------------------------------------===//
// Internal Helpers
//===--------------------------------------------------------------------===//