diff options
author | Anders Carlsson <andersca@mac.com> | 2010-06-27 17:23:46 +0000 |
---|---|---|
committer | Anders Carlsson <andersca@mac.com> | 2010-06-27 17:23:46 +0000 |
commit | 656746cd8c3a45901bcdec89ca1eb4495ee79b36 (patch) | |
tree | 9c8938cb1e6df7c06124d3302ebdb48bfcd685e1 /lib/CodeGen | |
parent | dca7ab2ea50a57545976e9d4c0472279fa397917 (diff) |
Add a CreateReferenceTemporary that will do the right thing for variables with global storage.
git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@106982 91177308-0d34-0410-b5e6-96231b3b80d8
Diffstat (limited to 'lib/CodeGen')
-rw-r--r-- | lib/CodeGen/CGExpr.cpp | 40 |
1 files changed, 34 insertions, 6 deletions
diff --git a/lib/CodeGen/CGExpr.cpp b/lib/CodeGen/CGExpr.cpp index a9c0951197..0d5369d0d3 100644 --- a/lib/CodeGen/CGExpr.cpp +++ b/lib/CodeGen/CGExpr.cpp @@ -169,10 +169,34 @@ struct SubobjectAdjustment { }; static llvm::Value * +CreateReferenceTemporary(CodeGenFunction& CGF, QualType Type, + const NamedDecl *InitializedDecl) { + if (const VarDecl *VD = dyn_cast_or_null<VarDecl>(InitializedDecl)) { + if (VD->hasGlobalStorage()) { + llvm::SmallString<256> Name; + CGF.CGM.getMangleContext().mangleReferenceTemporary(VD, Name); + + const llvm::Type *RefTempTy = CGF.ConvertTypeForMem(Type); + + // Create the reference temporary. + llvm::GlobalValue *RefTemp = + new llvm::GlobalVariable(CGF.CGM.getModule(), + RefTempTy, /*isConstant=*/false, + llvm::GlobalValue::InternalLinkage, + llvm::Constant::getNullValue(RefTempTy), + Name.str()); + return RefTemp; + } + } + + return CGF.CreateMemTemp(Type, "ref.tmp"); +} + +static llvm::Value * EmitExprForReferenceBinding(CodeGenFunction& CGF, const Expr* E, llvm::Value *&ReferenceTemporary, const CXXDestructorDecl *&ReferenceTemporaryDtor, - bool IsInitializer) { + const NamedDecl *InitializedDecl) { if (const CXXDefaultArgExpr *DAE = dyn_cast<CXXDefaultArgExpr>(E)) E = DAE->getExpr(); @@ -182,7 +206,7 @@ EmitExprForReferenceBinding(CodeGenFunction& CGF, const Expr* E, return EmitExprForReferenceBinding(CGF, TE->getSubExpr(), ReferenceTemporary, ReferenceTemporaryDtor, - IsInitializer); + InitializedDecl); } RValue RV; @@ -240,11 +264,13 @@ EmitExprForReferenceBinding(CodeGenFunction& CGF, const Expr* E, // Create a reference temporary if necessary. if (CGF.hasAggregateLLVMType(E->getType()) && !E->getType()->isAnyComplexType()) - ReferenceTemporary = CGF.CreateMemTemp(E->getType(), "ref.tmp"); + ReferenceTemporary = CreateReferenceTemporary(CGF, E->getType(), + InitializedDecl); + RV = CGF.EmitAnyExpr(E, ReferenceTemporary, /*IsAggLocVolatile=*/false, - /*IgnoreResult=*/false, IsInitializer); + /*IgnoreResult=*/false, InitializedDecl); - if (IsInitializer) { + if (InitializedDecl) { // Get the destructor for the reference temporary. if (const RecordType *RT = E->getType()->getAs<RecordType>()) { CXXRecordDecl *ClassDecl = cast<CXXRecordDecl>(RT->getDecl()); @@ -301,7 +327,9 @@ EmitExprForReferenceBinding(CodeGenFunction& CGF, const Expr* E, return RV.getAggregateAddr(); // Create a temporary variable that we can bind the reference to. - ReferenceTemporary = CGF.CreateMemTemp(E->getType(), "ref.tmp"); + ReferenceTemporary = CreateReferenceTemporary(CGF, E->getType(), + InitializedDecl); + if (RV.isScalar()) CGF.EmitStoreOfScalar(RV.getScalarVal(), ReferenceTemporary, /*Volatile=*/false, E->getType()); |