From: Argyrios Kyrtzidis Date: Thu, 28 Apr 2011 18:53:58 +0000 (+0000) Subject: When value-initializing the elements of an array not not included in the initializer... X-Git-Url: https://granicus.if.org/sourcecode?a=commitdiff_plain;h=3b4d490b09347e0b68ec0511ddfae79dfaba77a6;p=clang When value-initializing the elements of an array not not included in the initializer make sure that a non-trivial C++ constructor gets called. Fixes rdar://9347552 & http://llvm.org/PR9801 git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@130421 91177308-0d34-0410-b5e6-96231b3b80d8 --- diff --git a/lib/CodeGen/CGExprAgg.cpp b/lib/CodeGen/CGExprAgg.cpp index 53bc58b69c..457e44d086 100644 --- a/lib/CodeGen/CGExprAgg.cpp +++ b/lib/CodeGen/CGExprAgg.cpp @@ -622,6 +622,13 @@ void AggExprEmitter::VisitInitListExpr(InitListExpr *E) { QualType ElementType = CGF.getContext().getCanonicalType(E->getType()); ElementType = CGF.getContext().getAsArrayType(ElementType)->getElementType(); + bool hasNonTrivialCXXConstructor = false; + if (CGF.getContext().getLangOptions().CPlusPlus) + if (const RecordType *RT = ElementType->getAs()) { + const CXXRecordDecl *RD = cast(RT->getDecl()); + hasNonTrivialCXXConstructor = !RD->hasTrivialConstructor(); + } + // FIXME: were we intentionally ignoring address spaces and GC attributes? for (uint64_t i = 0; i != NumArrayElements; ++i) { @@ -629,7 +636,8 @@ void AggExprEmitter::VisitInitListExpr(InitListExpr *E) { // then we're done. if (i == NumInitElements && Dest.isZeroed() && - CGF.getTypes().isZeroInitializable(ElementType)) + CGF.getTypes().isZeroInitializable(ElementType) && + !hasNonTrivialCXXConstructor) break; llvm::Value *NextVal = Builder.CreateStructGEP(DestPtr, i, ".array"); diff --git a/test/CodeGenCXX/value-init.cpp b/test/CodeGenCXX/value-init.cpp index 188c8f5a14..890b144de3 100644 --- a/test/CodeGenCXX/value-init.cpp +++ b/test/CodeGenCXX/value-init.cpp @@ -83,6 +83,47 @@ namespace ptrmem { } } +namespace PR9801 { + +struct Test { + Test() : i(10) {} + Test(int i) : i(i) {} + int i; +private: + int j; +}; + +struct Test2 { + Test t; +}; + +struct Test3 : public Test { }; + +// CHECK: define void @_ZN6PR98011fEv +void f() { + // CHECK: call void @_ZN6PR98014TestC1Ei + // CHECK: call void @_ZN6PR98014TestC1Ev + // CHECK: call void @_ZN6PR98014TestC1Ev + Test partial[3] = { 1 }; + + // CHECK: call void @_ZN6PR98014TestC1Ev + // CHECK: call void @_ZN6PR98014TestC1Ev + // CHECK: call void @_ZN6PR98014TestC1Ev + Test empty[3] = {}; + + // CHECK: call void @_ZN6PR98015Test2C1Ev + // CHECK: call void @_ZN6PR98015Test2C1Ev + // CHECK: call void @_ZN6PR98015Test2C1Ev + Test2 empty2[3] = {}; + + // CHECK: call void @_ZN6PR98015Test3C1Ev + // CHECK: call void @_ZN6PR98015Test3C1Ev + // CHECK: call void @_ZN6PR98015Test3C1Ev + Test3 empty3[3] = {}; +} + +} + namespace zeroinit { struct S { int i; };