1; RUN: opt -basicaa -loop-idiom < %s -S | FileCheck %s
2target datalayout = "e-p:64:64:64-i1:8:8-i8:8:8-i16:16:16-i32:32:32-i64:64:64-f32:32:32-f64:64:64-v64:64:64-v128:128:128-a0:0:64-s0:64:64-f80:128:128-n8:16:32:64"
3
4target triple = "x86_64-apple-darwin10.0.0"
5
6%struct.foo = type { i32, i32 }
7%struct.foo1 = type { i32, i32, i32 }
8%struct.foo2 = type { i32, i16, i16 }
9
10;void bar1(foo_t *f, unsigned n) {
11;  for (unsigned i = 0; i < n; ++i) {
12;    f[i].a = 0;
13;    f[i].b = 0;
14;  }
15;}
16define void @bar1(%struct.foo* %f, i32 %n) nounwind ssp {
17entry:
18  %cmp1 = icmp eq i32 %n, 0
19  br i1 %cmp1, label %for.end, label %for.body.preheader
20
21for.body.preheader:                               ; preds = %entry
22  br label %for.body
23
24for.body:                                         ; preds = %for.body.preheader, %for.body
25  %indvars.iv = phi i64 [ 0, %for.body.preheader ], [ %indvars.iv.next, %for.body ]
26  %a = getelementptr inbounds %struct.foo, %struct.foo* %f, i64 %indvars.iv, i32 0
27  store i32 0, i32* %a, align 4
28  %b = getelementptr inbounds %struct.foo, %struct.foo* %f, i64 %indvars.iv, i32 1
29  store i32 0, i32* %b, align 4
30  %indvars.iv.next = add nuw nsw i64 %indvars.iv, 1
31  %lftr.wideiv = trunc i64 %indvars.iv.next to i32
32  %exitcond = icmp ne i32 %lftr.wideiv, %n
33  br i1 %exitcond, label %for.body, label %for.end.loopexit
34
35for.end.loopexit:                                 ; preds = %for.body
36  br label %for.end
37
38for.end:                                          ; preds = %for.end.loopexit, %entry
39  ret void
40; CHECK-LABEL: @bar1(
41; CHECK: call void @llvm.memset
42; CHECK-NOT: store
43}
44
45;void bar2(foo_t *f, unsigned n) {
46;  for (unsigned i = 0; i < n; ++i) {
47;    f[i].b = 0;
48;    f[i].a = 0;
49;  }
50;}
51define void @bar2(%struct.foo* %f, i32 %n) nounwind ssp {
52entry:
53  %cmp1 = icmp eq i32 %n, 0
54  br i1 %cmp1, label %for.end, label %for.body.preheader
55
56for.body.preheader:                               ; preds = %entry
57  br label %for.body
58
59for.body:                                         ; preds = %for.body.preheader, %for.body
60  %indvars.iv = phi i64 [ 0, %for.body.preheader ], [ %indvars.iv.next, %for.body ]
61  %b = getelementptr inbounds %struct.foo, %struct.foo* %f, i64 %indvars.iv, i32 1
62  store i32 0, i32* %b, align 4
63  %a = getelementptr inbounds %struct.foo, %struct.foo* %f, i64 %indvars.iv, i32 0
64  store i32 0, i32* %a, align 4
65  %indvars.iv.next = add nuw nsw i64 %indvars.iv, 1
66  %lftr.wideiv = trunc i64 %indvars.iv.next to i32
67  %exitcond = icmp ne i32 %lftr.wideiv, %n
68  br i1 %exitcond, label %for.body, label %for.end.loopexit
69
70for.end.loopexit:                                 ; preds = %for.body
71  br label %for.end
72
73for.end:                                          ; preds = %for.end.loopexit, %entry
74  ret void
75; CHECK-LABEL: @bar2(
76; CHECK: call void @llvm.memset
77; CHECK-NOT: store
78}
79
80;void bar3(foo_t *f, unsigned n) {
81;  for (unsigned i = n; i > 0; --i) {
82;    f[i].a = 0;
83;    f[i].b = 0;
84;  }
85;}
86define void @bar3(%struct.foo* nocapture %f, i32 %n) nounwind ssp {
87entry:
88  %cmp1 = icmp eq i32 %n, 0
89  br i1 %cmp1, label %for.end, label %for.body.preheader
90
91for.body.preheader:                               ; preds = %entry
92  %0 = zext i32 %n to i64
93  br label %for.body
94
95for.body:                                         ; preds = %for.body.preheader, %for.body
96  %indvars.iv = phi i64 [ %0, %for.body.preheader ], [ %indvars.iv.next, %for.body ]
97  %a = getelementptr inbounds %struct.foo, %struct.foo* %f, i64 %indvars.iv, i32 0
98  store i32 0, i32* %a, align 4
99  %b = getelementptr inbounds %struct.foo, %struct.foo* %f, i64 %indvars.iv, i32 1
100  store i32 0, i32* %b, align 4
101  %1 = trunc i64 %indvars.iv to i32
102  %dec = add i32 %1, -1
103  %cmp = icmp eq i32 %dec, 0
104  %indvars.iv.next = add nsw i64 %indvars.iv, -1
105  br i1 %cmp, label %for.end.loopexit, label %for.body
106
107for.end.loopexit:                                 ; preds = %for.body
108  br label %for.end
109
110for.end:                                          ; preds = %for.end.loopexit, %entry
111  ret void
112; CHECK-LABEL: @bar3(
113; CHECK: call void @llvm.memset
114; CHECK-NOT: store
115}
116
117;void bar4(foo_t *f, unsigned n) {
118;  for (unsigned i = 0; i < n; ++i) {
119;    f[i].a = 0;
120;    f[i].b = 1;
121;  }
122;}
123define void @bar4(%struct.foo* nocapture %f, i32 %n) nounwind ssp {
124entry:
125  %cmp1 = icmp eq i32 %n, 0
126  br i1 %cmp1, label %for.end, label %for.body.preheader
127
128for.body.preheader:                               ; preds = %entry
129  br label %for.body
130
131for.body:                                         ; preds = %for.body.preheader, %for.body
132  %indvars.iv = phi i64 [ 0, %for.body.preheader ], [ %indvars.iv.next, %for.body ]
133  %a = getelementptr inbounds %struct.foo, %struct.foo* %f, i64 %indvars.iv, i32 0
134  store i32 0, i32* %a, align 4
135  %b = getelementptr inbounds %struct.foo, %struct.foo* %f, i64 %indvars.iv, i32 1
136  store i32 1, i32* %b, align 4
137  %indvars.iv.next = add nuw nsw i64 %indvars.iv, 1
138  %lftr.wideiv = trunc i64 %indvars.iv.next to i32
139  %exitcond = icmp ne i32 %lftr.wideiv, %n
140  br i1 %exitcond, label %for.body, label %for.end.loopexit
141
142for.end.loopexit:                                 ; preds = %for.body
143  br label %for.end
144
145for.end:                                          ; preds = %for.end.loopexit, %entry
146  ret void
147; CHECK-LABEL: @bar4(
148; CHECK-NOT: call void @llvm.memset
149}
150
151;void bar5(foo1_t *f, unsigned n) {
152;  for (unsigned i = 0; i < n; ++i) {
153;    f[i].a = 0;
154;    f[i].b = 0;
155;  }
156;}
157define void @bar5(%struct.foo1* nocapture %f, i32 %n) nounwind ssp {
158entry:
159  %cmp1 = icmp eq i32 %n, 0
160  br i1 %cmp1, label %for.end, label %for.body.preheader
161
162for.body.preheader:                               ; preds = %entry
163  br label %for.body
164
165for.body:                                         ; preds = %for.body.preheader, %for.body
166  %indvars.iv = phi i64 [ 0, %for.body.preheader ], [ %indvars.iv.next, %for.body ]
167  %a = getelementptr inbounds %struct.foo1, %struct.foo1* %f, i64 %indvars.iv, i32 0
168  store i32 0, i32* %a, align 4
169  %b = getelementptr inbounds %struct.foo1, %struct.foo1* %f, i64 %indvars.iv, i32 1
170  store i32 0, i32* %b, align 4
171  %indvars.iv.next = add nuw nsw i64 %indvars.iv, 1
172  %lftr.wideiv = trunc i64 %indvars.iv.next to i32
173  %exitcond = icmp ne i32 %lftr.wideiv, %n
174  br i1 %exitcond, label %for.body, label %for.end.loopexit
175
176for.end.loopexit:                                 ; preds = %for.body
177  br label %for.end
178
179for.end:                                          ; preds = %for.end.loopexit, %entry
180  ret void
181; CHECK-LABEL: @bar5(
182; CHECK-NOT: call void @llvm.memset
183}
184
185;void bar6(foo2_t *f, unsigned n) {
186;  for (unsigned i = 0; i < n; ++i) {
187;    f[i].a = 0;
188;    f[i].b = 0;
189;    f[i].c = 0;
190;  }
191;}
192define void @bar6(%struct.foo2* nocapture %f, i32 %n) nounwind ssp {
193entry:
194  %cmp1 = icmp eq i32 %n, 0
195  br i1 %cmp1, label %for.end, label %for.body.preheader
196
197for.body.preheader:                               ; preds = %entry
198  br label %for.body
199
200for.body:                                         ; preds = %for.body.preheader, %for.body
201  %indvars.iv = phi i64 [ 0, %for.body.preheader ], [ %indvars.iv.next, %for.body ]
202  %a = getelementptr inbounds %struct.foo2, %struct.foo2* %f, i64 %indvars.iv, i32 0
203  store i32 0, i32* %a, align 4
204  %b = getelementptr inbounds %struct.foo2, %struct.foo2* %f, i64 %indvars.iv, i32 1
205  store i16 0, i16* %b, align 4
206  %c = getelementptr inbounds %struct.foo2, %struct.foo2* %f, i64 %indvars.iv, i32 2
207  store i16 0, i16* %c, align 2
208  %indvars.iv.next = add nuw nsw i64 %indvars.iv, 1
209  %lftr.wideiv = trunc i64 %indvars.iv.next to i32
210  %exitcond = icmp ne i32 %lftr.wideiv, %n
211  br i1 %exitcond, label %for.body, label %for.end.loopexit
212
213for.end.loopexit:                                 ; preds = %for.body
214  br label %for.end
215
216for.end:                                          ; preds = %for.end.loopexit, %entry
217  ret void
218; CHECK-LABEL: @bar6(
219; CHECK: call void @llvm.memset
220; CHECK-NOT: store
221}
222