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* align 4 %[[REST]], i8 0, i64 %[[SIZE]], i1 false) |
158 | // |
159 | // CHECK: } |
160 | |
161 | |