Home | History | Annotate | Download | only in CodeGenCXX
      1 // RUN: %clang_cc1 -triple x86_64-linux-gnu -std=c++11 %s -emit-llvm -o - | FileCheck %s
      2 
      3 // PR10878
      4 
      5 struct S { S(); S(int); ~S(); int n; };
      6 
      7 void *p = new S[2][3]{ { 1, 2, 3 }, { 4, 5, 6 } };
      8 
      9 // CHECK-LABEL: define
     10 // CHECK: %[[ALLOC:.*]] = call i8* @_Znam(i64 32)
     11 // CHECK: %[[COOKIE:.*]] = bitcast i8* %[[ALLOC]] to i64*
     12 // CHECK: store i64 6, i64* %[[COOKIE]]
     13 // CHECK: %[[START_AS_i8:.*]] = getelementptr inbounds i8, i8* %[[ALLOC]], i64 8
     14 // CHECK: %[[START_AS_S:.*]] = bitcast i8* %[[START_AS_i8]] to %[[S:.*]]*
     15 //
     16 // Explicit initializers:
     17 //
     18 // { 1, 2, 3 }
     19 //
     20 // CHECK: %[[S_0:.*]] = bitcast %[[S]]* %[[START_AS_S]] to [3 x %[[S]]]*
     21 //
     22 // CHECK: %[[S_0_0:.*]] = getelementptr inbounds [3 x %[[S]]], [3 x %[[S]]]* %[[S_0]], i64 0, i64 0
     23 // CHECK: call void @_ZN1SC1Ei(%[[S]]* %[[S_0_0]], i32 1)
     24 // CHECK: %[[S_0_1:.*]] = getelementptr inbounds %[[S]], %[[S]]* %[[S_0_0]], i64 1
     25 // CHECK: call void @_ZN1SC1Ei(%[[S]]* %[[S_0_1]], i32 2)
     26 // CHECK: %[[S_0_2:.*]] = getelementptr inbounds %[[S]], %[[S]]* %[[S_0_1]], i64 1
     27 // CHECK: call void @_ZN1SC1Ei(%[[S]]* %[[S_0_2]], i32 3)
     28 //
     29 // { 4, 5, 6 }
     30 //
     31 // CHECK: %[[S_1:.*]] = getelementptr inbounds [3 x %[[S]]], [3 x %[[S]]]* %[[S_0]], i64 1
     32 //
     33 // CHECK: %[[S_1_0:.*]] = getelementptr inbounds [3 x %[[S]]], [3 x %[[S]]]* %[[S_1]], i64 0, i64 0
     34 // CHECK: call void @_ZN1SC1Ei(%[[S]]* %[[S_1_0]], i32 4)
     35 // CHECK: %[[S_1_1:.*]] = getelementptr inbounds %[[S]], %[[S]]* %[[S_1_0]], i64 1
     36 // CHECK: call void @_ZN1SC1Ei(%[[S]]* %[[S_1_1]], i32 5)
     37 // CHECK: %[[S_1_2:.*]] = getelementptr inbounds %[[S]], %[[S]]* %[[S_1_1]], i64 1
     38 // CHECK: call void @_ZN1SC1Ei(%[[S]]* %[[S_1_2]], i32 6)
     39 //
     40 // CHECK-NOT: br i1
     41 // CHECK-NOT: call
     42 // CHECK: }
     43 
     44 int n;
     45 void *q = new S[n][3]{ { 1, 2, 3 }, { 4, 5, 6 } };
     46 
     47 // CHECK-LABEL: define
     48 //
     49 // CHECK: load i32, i32* @n
     50 // CHECK: call {{.*}} @llvm.umul.with.overflow.i64(i64 %[[N:.*]], i64 12)
     51 // CHECK: %[[ELTS:.*]] = mul i64 %[[N]], 3
     52 // CHECK: call {{.*}} @llvm.uadd.with.overflow.i64(i64 %{{.*}}, i64 8)
     53 // CHECK: %[[ALLOC:.*]] = call i8* @_Znam(i64 %{{.*}})
     54 //
     55 // CHECK: %[[COOKIE:.*]] = bitcast i8* %[[ALLOC]] to i64*
     56 // CHECK: store i64 %[[ELTS]], i64* %[[COOKIE]]
     57 // CHECK: %[[START_AS_i8:.*]] = getelementptr inbounds i8, i8* %[[ALLOC]], i64 8
     58 // CHECK: %[[START_AS_S:.*]] = bitcast i8* %[[START_AS_i8]] to %[[S]]*
     59 //
     60 // Explicit initializers:
     61 //
     62 // { 1, 2, 3 }
     63 //
     64 // CHECK: %[[S_0:.*]] = bitcast %[[S]]* %[[START_AS_S]] to [3 x %[[S]]]*
     65 //
     66 // CHECK: %[[S_0_0:.*]] = getelementptr inbounds [3 x %[[S]]], [3 x %[[S]]]* %[[S_0]], i64 0, i64 0
     67 // CHECK: call void @_ZN1SC1Ei(%[[S]]* %[[S_0_0]], i32 1)
     68 // CHECK: %[[S_0_1:.*]] = getelementptr inbounds %[[S]], %[[S]]* %[[S_0_0]], i64 1
     69 // CHECK: call void @_ZN1SC1Ei(%[[S]]* %[[S_0_1]], i32 2)
     70 // CHECK: %[[S_0_2:.*]] = getelementptr inbounds %[[S]], %[[S]]* %[[S_0_1]], i64 1
     71 // CHECK: call void @_ZN1SC1Ei(%[[S]]* %[[S_0_2]], i32 3)
     72 //
     73 // { 4, 5, 6 }
     74 //
     75 // CHECK: %[[S_1:.*]] = getelementptr inbounds [3 x %[[S]]], [3 x %[[S]]]* %[[S_0]], i64 1
     76 //
     77 // CHECK: %[[S_1_0:.*]] = getelementptr inbounds [3 x %[[S]]], [3 x %[[S]]]* %[[S_1]], i64 0, i64 0
     78 // CHECK: call void @_ZN1SC1Ei(%[[S]]* %[[S_1_0]], i32 4)
     79 // CHECK: %[[S_1_1:.*]] = getelementptr inbounds %[[S]], %[[S]]* %[[S_1_0]], i64 1
     80 // CHECK: call void @_ZN1SC1Ei(%[[S]]* %[[S_1_1]], i32 5)
     81 // CHECK: %[[S_1_2:.*]] = getelementptr inbounds %[[S]], %[[S]]* %[[S_1_1]], i64 1
     82 // CHECK: call void @_ZN1SC1Ei(%[[S]]* %[[S_1_2]], i32 6)
     83 //
     84 // And the rest.
     85 //
     86 // CHECK: %[[S_2:.*]] = getelementptr inbounds [3 x %[[S]]], [3 x %[[S]]]* %[[S_1]], i64 1
     87 // CHECK: %[[S_2_AS_S:.*]] = bitcast [3 x %[[S]]]* %[[S_2]] to %[[S]]*
     88 //
     89 // CHECK: %[[REST:.*]] = sub i64 %[[ELTS]], 6
     90 // CHECK: icmp eq i64 %[[REST]], 0
     91 // CHECK: br i1
     92 //
     93 // CHECK: %[[END:.*]] = getelementptr inbounds %[[S]], %[[S]]* %[[S_2_AS_S]], i64 %[[REST]]
     94 // CHECK: br label
     95 //
     96 // CHECK: %[[CUR:.*]] = phi %[[S]]* [ %[[S_2_AS_S]], {{.*}} ], [ %[[NEXT:.*]], {{.*}} ]
     97 // CHECK: call void @_ZN1SC1Ev(%[[S]]* %[[CUR]])
     98 // CHECK: %[[NEXT]] = getelementptr inbounds %[[S]], %[[S]]* %[[CUR]], i64 1
     99 // CHECK: icmp eq %[[S]]* %[[NEXT]], %[[END]]
    100 // CHECK: br i1
    101 //
    102 // CHECK: }
    103 
    104 struct T { int a; };
    105 void *r = new T[n][3]{ { 1, 2, 3 }, { 4, 5, 6 } };
    106 
    107 // CHECK-LABEL: define
    108 //
    109 // CHECK: load i32, i32* @n
    110 // CHECK: call {{.*}} @llvm.umul.with.overflow.i64(i64 %[[N:.*]], i64 12)
    111 // CHECK: %[[ELTS:.*]] = mul i64 %[[N]], 3
    112 //
    113 // No cookie.
    114 // CHECK-NOT: @llvm.uadd.with.overflow
    115 //
    116 // CHECK: %[[ALLOC:.*]] = call i8* @_Znam(i64 %{{.*}})
    117 //
    118 // CHECK: %[[START_AS_T:.*]] = bitcast i8* %[[ALLOC]] to %[[T:.*]]*
    119 //
    120 // Explicit initializers:
    121 //
    122 // { 1, 2, 3 }
    123 //
    124 // CHECK: %[[T_0:.*]] = bitcast %[[T]]* %[[START_AS_T]] to [3 x %[[T]]]*
    125 //
    126 // CHECK: %[[T_0_0:.*]] = getelementptr inbounds [3 x %[[T]]], [3 x %[[T]]]* %[[T_0]], i64 0, i64 0
    127 // CHECK: %[[T_0_0_0:.*]] = getelementptr inbounds %[[T]], %[[T]]* %[[T_0_0]], i32 0, i32 0
    128 // CHECK: store i32 1, i32* %[[T_0_0_0]]
    129 // CHECK: %[[T_0_1:.*]] = getelementptr inbounds %[[T]], %[[T]]* %[[T_0_0]], i64 1
    130 // CHECK: %[[T_0_1_0:.*]] = getelementptr inbounds %[[T]], %[[T]]* %[[T_0_1]], i32 0, i32 0
    131 // CHECK: store i32 2, i32* %[[T_0_1_0]]
    132 // CHECK: %[[T_0_2:.*]] = getelementptr inbounds %[[T]], %[[T]]* %[[T_0_1]], i64 1
    133 // CHECK: %[[T_0_2_0:.*]] = getelementptr inbounds %[[T]], %[[T]]* %[[T_0_2]], i32 0, i32 0
    134 // CHECK: store i32 3, i32* %[[T_0_2_0]]
    135 //
    136 // { 4, 5, 6 }
    137 //
    138 // CHECK: %[[T_1:.*]] = getelementptr inbounds [3 x %[[T]]], [3 x %[[T]]]* %[[T_0]], i64 1
    139 //
    140 // CHECK: %[[T_1_0:.*]] = getelementptr inbounds [3 x %[[T]]], [3 x %[[T]]]* %[[T_1]], i64 0, i64 0
    141 // CHECK: %[[T_1_0_0:.*]] = getelementptr inbounds %[[T]], %[[T]]* %[[T_1_0]], i32 0, i32 0
    142 // CHECK: store i32 4, i32* %[[T_1_0_0]]
    143 // CHECK: %[[T_1_1:.*]] = getelementptr inbounds %[[T]], %[[T]]* %[[T_1_0]], i64 1
    144 // CHECK: %[[T_1_1_0:.*]] = getelementptr inbounds %[[T]], %[[T]]* %[[T_1_1]], i32 0, i32 0
    145 // CHECK: store i32 5, i32* %[[T_1_1_0]]
    146 // CHECK: %[[T_1_2:.*]] = getelementptr inbounds %[[T]], %[[T]]* %[[T_1_1]], i64 1
    147 // CHECK: %[[T_1_2_0:.*]] = getelementptr inbounds %[[T]], %[[T]]* %[[T_1_2]], i32 0, i32 0
    148 // CHECK: store i32 6, i32* %[[T_1_2_0]]
    149 //
    150 // And the rest gets memset to 0.
    151 //
    152 // CHECK: %[[T_2:.*]] = getelementptr inbounds [3 x %[[T]]], [3 x %[[T]]]* %[[T_1]], i64 1
    153 // CHECK: %[[T_2_AS_T:.*]] = bitcast [3 x %[[T]]]* %[[T_2]] to %[[T]]*
    154 //
    155 // CHECK: %[[SIZE:.*]] = sub i64 %{{.*}}, 24
    156 // CHECK: %[[REST:.*]] = bitcast %[[T]]* %[[T_2_AS_T]] to i8*
    157 // CHECK: call void @llvm.memset.p0i8.i64(i8* %[[REST]], i8 0, i64 %[[SIZE]], i32 4, i1 false)
    158 //
    159 // CHECK: }
    160 
    161