<div dir="ltr"><div class="gmail_extra"><div class="gmail_quote">On Wed, Apr 8, 2015 at 8:09 PM, Richard Smith <span dir="ltr"><<a href="mailto:richard@metafoo.co.uk" target="_blank">richard@metafoo.co.uk</a>></span> wrote:<br><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex"><div dir="ltr">I suspect this caused PR23165, can you investigate? Looks like the global constant we emit for the temporary in that case has the wrong vptr!</div><div class="HOEnZb"><div class="h5"><div class="gmail_extra"><br><div class="gmail_quote">On Sat, Mar 7, 2015 at 5:37 AM, Benjamin Kramer <span dir="ltr"><<a href="mailto:benny.kra@googlemail.com" target="_blank">benny.kra@googlemail.com</a>></span> wrote:<br><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex">Author: d0k<br>
Date: Sat Mar 7 07:37:13 2015<br>
New Revision: 231564<br>
<br>
URL: <a href="http://llvm.org/viewvc/llvm-project?rev=231564&view=rev" target="_blank">http://llvm.org/viewvc/llvm-project?rev=231564&view=rev</a><br>
Log:<br>
Reapply r231508 "CodeGen: Emit constant temporaries into read-only globals."<br>
<br>
I disabled putting the new global into the same COMDAT as the function for now.<br>
There's a fundamental problem when we inline references to the global but still<br>
have the global in a COMDAT linked to the inlined function. Since this is only<br>
an optimization there may be other versions of the COMDAT around that are<br>
missing the new global and hell breaks loose at link time.<br>
<br>
I hope the chromium build doesn't break this time :)<br>
<br>
Modified:<br>
cfe/trunk/lib/CodeGen/CGExpr.cpp<br>
cfe/trunk/test/CodeGenCXX/compound-literals.cpp<br>
cfe/trunk/test/CodeGenCXX/cxx0x-initializer-array.cpp<br>
cfe/trunk/test/CodeGenCXX/cxx0x-initializer-references.cpp<br>
cfe/trunk/test/CodeGenCXX/cxx0x-initializer-stdinitializerlist.cpp<br>
<br>
Modified: cfe/trunk/lib/CodeGen/CGExpr.cpp<br>
URL: <a href="http://llvm.org/viewvc/llvm-project/cfe/trunk/lib/CodeGen/CGExpr.cpp?rev=231564&r1=231563&r2=231564&view=diff" target="_blank">http://llvm.org/viewvc/llvm-project/cfe/trunk/lib/CodeGen/CGExpr.cpp?rev=231564&r1=231563&r2=231564&view=diff</a><br>
==============================================================================<br>
--- cfe/trunk/lib/CodeGen/CGExpr.cpp (original)<br>
+++ cfe/trunk/lib/CodeGen/CGExpr.cpp Sat Mar 7 07:37:13 2015<br>
@@ -301,6 +301,23 @@ createReferenceTemporary(CodeGenFunction<br>
switch (M->getStorageDuration()) {<br>
case SD_FullExpression:<br>
case SD_Automatic:<br>
+ // If we have a constant temporary array or record try to promote it into a<br>
+ // constant global under the same rules a normal constant would've been<br>
+ // promoted. This is easier on the optimizer and generally emits fewer<br>
+ // instructions.<br>
+ if (CGF.CGM.getCodeGenOpts().MergeAllConstants &&<br>
+ (M->getType()->isArrayType() || M->getType()->isRecordType()) &&<br>
+ CGF.CGM.isTypeConstant(M->getType(), true))<br>
+ if (llvm::Constant *Init =<br>
+ CGF.CGM.EmitConstantExpr(Inner, M->getType(), &CGF)) {<br></blockquote></div></div></div></div></blockquote><div><br></div><div>This is using the wrong type; you're assuming that M's type and Inner's type are the same, and they're not in general (M can refer to a subobject of the object initialized by Inner). You shouldn't be looking at M at all in this code; it's passed in so that the static / thread storage cases can work out the linkage and mangled name for their global.</div><div> </div><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex"><div class="HOEnZb"><div class="h5"><div class="gmail_extra"><div class="gmail_quote"><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex">
+ auto *GV = new llvm::GlobalVariable(<br>
+ CGF.CGM.getModule(), Init->getType(), /*isConstant=*/true,<br>
+ llvm::GlobalValue::PrivateLinkage, Init, ".ref.tmp");<br>
+ GV->setAlignment(<br>
+ CGF.getContext().getTypeAlignInChars(M->getType()).getQuantity());<br>
+ // FIXME: Should we put the new global into a COMDAT?<br>
+ return GV;<br>
+ }<br>
return CGF.CreateMemTemp(Inner->getType(), "ref.tmp");<br>
<br>
case SD_Thread:<br>
@@ -370,8 +387,9 @@ EmitMaterializeTemporaryExpr(const Mater<br>
// Create and initialize the reference temporary.<br>
llvm::Value *Object = createReferenceTemporary(*this, M, E);<br>
if (auto *Var = dyn_cast<llvm::GlobalVariable>(Object)) {<br>
- // If the temporary is a global and has a constant initializer, we may<br>
- // have already initialized it.<br>
+ // If the temporary is a global and has a constant initializer or is a<br>
+ // constant temporary that we promoted to a global, we may have already<br>
+ // initialized it.<br>
if (!Var->hasInitializer()) {<br>
Var->setInitializer(CGM.EmitNullConstant(E->getType()));<br>
EmitAnyExprToMem(E, Object, Qualifiers(), /*IsInit*/true);<br>
<br>
Modified: cfe/trunk/test/CodeGenCXX/compound-literals.cpp<br>
URL: <a href="http://llvm.org/viewvc/llvm-project/cfe/trunk/test/CodeGenCXX/compound-literals.cpp?rev=231564&r1=231563&r2=231564&view=diff" target="_blank">http://llvm.org/viewvc/llvm-project/cfe/trunk/test/CodeGenCXX/compound-literals.cpp?rev=231564&r1=231563&r2=231564&view=diff</a><br>
==============================================================================<br>
--- cfe/trunk/test/CodeGenCXX/compound-literals.cpp (original)<br>
+++ cfe/trunk/test/CodeGenCXX/compound-literals.cpp Sat Mar 7 07:37:13 2015<br>
@@ -28,7 +28,7 @@ int f() {<br>
<br>
// CHECK-LABEL: define i32 @_Z1gv()<br>
int g() {<br>
- // CHECK: store [2 x i32]* %{{[a-z0-9.]+}}, [2 x i32]** [[V:%[a-z0-9.]+]]<br>
+ // CHECK: store [2 x i32]* @{{.*}}, [2 x i32]** [[V:%[a-z0-9.]+]]<br>
const int (&v)[2] = (int [2]) {1,2};<br>
<br>
// CHECK: [[A:%[a-z0-9.]+]] = load [2 x i32]*, [2 x i32]** [[V]]<br>
<br>
Modified: cfe/trunk/test/CodeGenCXX/cxx0x-initializer-array.cpp<br>
URL: <a href="http://llvm.org/viewvc/llvm-project/cfe/trunk/test/CodeGenCXX/cxx0x-initializer-array.cpp?rev=231564&r1=231563&r2=231564&view=diff" target="_blank">http://llvm.org/viewvc/llvm-project/cfe/trunk/test/CodeGenCXX/cxx0x-initializer-array.cpp?rev=231564&r1=231563&r2=231564&view=diff</a><br>
==============================================================================<br>
--- cfe/trunk/test/CodeGenCXX/cxx0x-initializer-array.cpp (original)<br>
+++ cfe/trunk/test/CodeGenCXX/cxx0x-initializer-array.cpp Sat Mar 7 07:37:13 2015<br>
@@ -42,10 +42,11 @@ namespace ValueInitArrayOfMemPtr {<br>
// CHECK: call void @llvm.memcpy.p0i8.p0i8.i32(i8* %{{.*}}, i8* bitcast ([3 x i32]* @[[THREE_NULL_MEMPTRS]] to i8*), i32 12, i32 4, i1 false)<br>
}<br>
<br>
- // CHECK-LABEL: define void @_ZN22ValueInitArrayOfMemPtr1gEv<br>
- void g() {<br>
+ // Test dynamic initialization.<br>
+ // CHECK-LABEL: define void @_ZN22ValueInitArrayOfMemPtr1gEMNS_1SEi<br>
+ void g(p ptr) {<br>
// CHECK: store i32 -1,<br>
- f(a{});<br>
+ f(a{ptr});<br>
}<br>
}<br>
<br>
<br>
Modified: cfe/trunk/test/CodeGenCXX/cxx0x-initializer-references.cpp<br>
URL: <a href="http://llvm.org/viewvc/llvm-project/cfe/trunk/test/CodeGenCXX/cxx0x-initializer-references.cpp?rev=231564&r1=231563&r2=231564&view=diff" target="_blank">http://llvm.org/viewvc/llvm-project/cfe/trunk/test/CodeGenCXX/cxx0x-initializer-references.cpp?rev=231564&r1=231563&r2=231564&view=diff</a><br>
==============================================================================<br>
--- cfe/trunk/test/CodeGenCXX/cxx0x-initializer-references.cpp (original)<br>
+++ cfe/trunk/test/CodeGenCXX/cxx0x-initializer-references.cpp Sat Mar 7 07:37:13 2015<br>
@@ -35,22 +35,30 @@ namespace reference {<br>
// CHECK-NEXT: ret<br>
}<br>
<br>
- void reference_to_aggregate() {<br>
+ void reference_to_aggregate(int i) {<br>
// CHECK: getelementptr {{.*}}, i32 0, i32 0<br>
// CHECK-NEXT: store i32 1<br>
// CHECK-NEXT: getelementptr {{.*}}, i32 0, i32 1<br>
- // CHECK-NEXT: store i32 2<br>
+ // CHECK-NEXT: %[[I1:.*]] = load i32, i32*<br>
+ // CHECK-NEXT: store i32 %[[I1]]<br>
// CHECK-NEXT: store %{{.*}}* %{{.*}}, %{{.*}}** %{{.*}}, align<br>
- const A &ra1{1, 2};<br>
+ const A &ra1{1, i};<br>
<br>
// CHECK-NEXT: getelementptr inbounds [3 x i32], [3 x i32]* %{{.*}}, i{{32|64}} 0, i{{32|64}} 0<br>
// CHECK-NEXT: store i32 1<br>
// CHECK-NEXT: getelementptr inbounds i32, i32* %{{.*}}, i{{32|64}} 1<br>
// CHECK-NEXT: store i32 2<br>
// CHECK-NEXT: getelementptr inbounds i32, i32* %{{.*}}, i{{32|64}} 1<br>
- // CHECK-NEXT: store i32 3<br>
+ // CHECK-NEXT: %[[I2:.*]] = load i32, i32*<br>
+ // CHECK-NEXT: store i32 %[[I2]]<br>
// CHECK-NEXT: store [3 x i32]* %{{.*}}, [3 x i32]** %{{.*}}, align<br>
- const int (&arrayRef)[] = {1, 2, 3};<br>
+ const int (&arrayRef)[] = {1, 2, i};<br>
+<br>
+ // CHECK: store %{{.*}}* @{{.*}}, %{{.*}}** %{{.*}}, align<br>
+ const A &constra1{1, 2};<br>
+<br>
+ // CHECK-NEXT: store [3 x i32]* @{{.*}}, [3 x i32]** %{{.*}}, align<br>
+ const int (&constarrayRef)[] = {1, 2, 3};<br>
<br>
// CHECK-NEXT: ret<br>
}<br>
<br>
Modified: cfe/trunk/test/CodeGenCXX/cxx0x-initializer-stdinitializerlist.cpp<br>
URL: <a href="http://llvm.org/viewvc/llvm-project/cfe/trunk/test/CodeGenCXX/cxx0x-initializer-stdinitializerlist.cpp?rev=231564&r1=231563&r2=231564&view=diff" target="_blank">http://llvm.org/viewvc/llvm-project/cfe/trunk/test/CodeGenCXX/cxx0x-initializer-stdinitializerlist.cpp?rev=231564&r1=231563&r2=231564&view=diff</a><br>
==============================================================================<br>
--- cfe/trunk/test/CodeGenCXX/cxx0x-initializer-stdinitializerlist.cpp (original)<br>
+++ cfe/trunk/test/CodeGenCXX/cxx0x-initializer-stdinitializerlist.cpp Sat Mar 7 07:37:13 2015<br>
@@ -72,6 +72,9 @@ namespace thread_local_global_array {<br>
// CHECK: @[[PARTLY_CONSTANT_SECOND:_ZGRN15partly_constant2ilE.*]] = internal global [2 x i32] zeroinitializer, align 4<br>
// CHECK: @[[PARTLY_CONSTANT_THIRD:_ZGRN15partly_constant2ilE.*]] = internal constant [4 x i32] [i32 5, i32 6, i32 7, i32 8], align 4<br>
<br>
+// CHECK: @[[REFTMP1:.*]] = private constant [2 x i32] [i32 42, i32 43], align 4<br>
+// CHECK: @[[REFTMP2:.*]] = private constant [3 x %{{.*}}] [%{{.*}} { i32 1 }, %{{.*}} { i32 2 }, %{{.*}} { i32 3 }], align 4<br>
+<br>
// CHECK: appending global<br>
<br>
<br>
@@ -215,17 +218,16 @@ void fn9() {<br>
<br>
struct haslist1 {<br>
std::initializer_list<int> il;<br>
- haslist1();<br>
+ haslist1(int i);<br>
};<br>
<br>
-// CHECK-LABEL: define void @_ZN8haslist1C2Ev<br>
-haslist1::haslist1()<br>
+// CHECK-LABEL: define void @_ZN8haslist1C2Ei<br>
+haslist1::haslist1(int i)<br>
// CHECK: alloca [3 x i32]<br>
-// CHECK: store i32 1<br>
+// CHECK: store i32 %<br>
// CHECK: store i32 2<br>
// CHECK: store i32 3<br>
-// CHECK: store i{{32|64}} 3<br>
- : il{1, 2, 3}<br>
+ : il{i, 2, 3}<br>
{<br>
destroyme2 dm2;<br>
}<br>
@@ -244,16 +246,15 @@ haslist2::haslist2()<br>
// CHECK: call void @_ZN10destroyme1D1Ev<br>
}<br>
<br>
-void fn10() {<br>
- // CHECK-LABEL: define void @_Z4fn10v<br>
+void fn10(int i) {<br>
+ // CHECK-LABEL: define void @_Z4fn10i<br>
// CHECK: alloca [3 x i32]<br>
// CHECK: call noalias i8* @_Znw{{[jm]}}<br>
- // CHECK: store i32 1<br>
+ // CHECK: store i32 %<br>
// CHECK: store i32 2<br>
// CHECK: store i32 3<br>
// CHECK: store i32*<br>
- // CHECK: store i{{32|64}} 3<br>
- (void) new std::initializer_list<int> {1, 2, 3};<br>
+ (void) new std::initializer_list<int> {i, 2, 3};<br>
}<br>
<br>
void fn11() {<br>
@@ -462,6 +463,22 @@ namespace PR20445 {<br>
template<int x> void f() { new MyClass({42, 43}); }<br>
template void f<0>();<br>
// CHECK-LABEL: define {{.*}} @_ZN7PR204451fILi0EEEvv(<br>
+ // CHECK: store i32* getelementptr inbounds ([2 x i32]* @[[REFTMP1]], i64 0, i64 0)<br>
// CHECK: call void @_ZN7PR204456vectorC1ESt16initializer_listIiE(<br>
// CHECK: call void @_ZN7PR204457MyClassC1ERKNS_6vectorE(<br>
}<br>
+<br>
+namespace ConstExpr {<br>
+ class C {<br>
+ int x;<br>
+ public:<br>
+ constexpr C(int x) : x(x) {}<br>
+ };<br>
+ void f(std::initializer_list<C>);<br>
+ void g() {<br>
+// CHECK-LABEL: _ZN9ConstExpr1gEv<br>
+// CHECK: store %"class.ConstExpr::C"* getelementptr inbounds ([3 x %"class.ConstExpr::C"]* @[[REFTMP2]], i64 0, i64 0)<br>
+// CHECK: call void @_ZN9ConstExpr1fESt16initializer_listINS_1CEE<br>
+ f({C(1), C(2), C(3)});<br>
+ }<br>
+}<br>
<br>
<br>
_______________________________________________<br>
cfe-commits mailing list<br>
<a href="mailto:cfe-commits@cs.uiuc.edu" target="_blank">cfe-commits@cs.uiuc.edu</a><br>
<a href="http://lists.cs.uiuc.edu/mailman/listinfo/cfe-commits" target="_blank">http://lists.cs.uiuc.edu/mailman/listinfo/cfe-commits</a><br>
</blockquote></div><br></div>
</div></div></blockquote></div><br></div></div>