1 // RUN: %clang_cc1 -fsyntax-only -std=c++11 -Wloop-analysis -verify %s
2 // RUN: %clang_cc1 -fsyntax-only -std=c++11 -Wrange-loop-analysis -verify %s
3
4 template <typename return_type>
5 struct Iterator {
6 return_type operator*();
7 Iterator operator++();
8 bool operator!=(const Iterator);
9 };
10
11 template <typename T>
12 struct Container {
13 typedef Iterator<T> I;
14
15 I begin();
16 I end();
17 };
18
19 struct Foo {};
20 struct Bar {
21 Bar(Foo);
22 Bar(int);
23 operator int();
24 };
25
26 // Testing notes:
27 // test0 checks that the full text of the warnings and notes is correct. The
28 // rest of the tests checks a smaller portion of the text.
29 // test1-6 are set in pairs, the odd numbers are the non-reference returning
30 // versions of the even numbers.
31 // test7-9 use an array instead of a range object
32 // tests use all four versions of the loop varaible, const &T, const T, T&, and
33 // T. Versions producing errors and are commented out.
34 //
35 // Conversion chart:
36 // double <=> int
37 // int <=> Bar
38 // double => Bar
39 // Foo => Bar
40 //
41 // Conversions during tests:
42 // test1-2
43 // int => int
44 // int => double
45 // int => Bar
46 // test3-4
47 // Bar => Bar
48 // Bar => int
49 // test5-6
50 // Foo => Bar
51 // test7
52 // double => double
53 // double => int
54 // double => Bar
55 // test8
56 // Foo => Foo
57 // Foo => Bar
58 // test9
59 // Bar => Bar
60 // Bar => int
61
test0()62 void test0() {
63 Container<int> int_non_ref_container;
64 Container<int&> int_container;
65 Container<Bar&> bar_container;
66
67 for (const int &x : int_non_ref_container) {}
68 // expected-warning@-1 {{loop variable 'x' is always a copy because the range of type 'Container<int>' does not return a reference}}
69 // expected-note@-2 {{use non-reference type 'int'}}
70
71 for (const double &x : int_container) {}
72 // expected-warning@-1 {{loop variable 'x' has type 'const double &' but is initialized with type 'int' resulting in a copy}}
73 // expected-note@-2 {{use non-reference type 'double' to keep the copy or type 'const int &' to prevent copying}}
74
75 for (const Bar x : bar_container) {}
76 // expected-warning@-1 {{loop variable 'x' of type 'const Bar' creates a copy from type 'const Bar'}}
77 // expected-note@-2 {{use reference type 'const Bar &' to prevent copying}}
78 }
79
test1()80 void test1() {
81 Container<int> A;
82
83 for (const int &x : A) {}
84 // expected-warning@-1 {{always a copy}}
85 // expected-note@-2 {{'int'}}
86 for (const int x : A) {}
87 // No warning, non-reference type indicates copy is made
88 //for (int &x : A) {}
89 // Binding error
90 for (int x : A) {}
91 // No warning, non-reference type indicates copy is made
92
93 for (const double &x : A) {}
94 // expected-warning@-1 {{always a copy}}
95 // expected-note@-2 {{'double'}}
96 for (const double x : A) {}
97 // No warning, non-reference type indicates copy is made
98 //for (double &x : A) {}
99 // Binding error
100 for (double x : A) {}
101 // No warning, non-reference type indicates copy is made
102
103 for (const Bar &x : A) {}
104 // expected-warning@-1 {{always a copy}}
105 // expected-note@-2 {{'Bar'}}
106 for (const Bar x : A) {}
107 // No warning, non-reference type indicates copy is made
108 //for (Bar &x : A) {}
109 // Binding error
110 for (Bar x : A) {}
111 // No warning, non-reference type indicates copy is made
112 }
113
test2()114 void test2() {
115 Container<int&> B;
116
117 for (const int &x : B) {}
118 // No warning, this reference is not a temporary
119 for (const int x : B) {}
120 // No warning on POD copy
121 for (int &x : B) {}
122 // No warning
123 for (int x : B) {}
124 // No warning
125
126 for (const double &x : B) {}
127 // expected-warning@-1 {{resulting in a copy}}
128 // expected-note-re@-2 {{'double'{{.*}}'const int &'}}
129 for (const double x : B) {}
130 //for (double &x : B) {}
131 // Binding error
132 for (double x : B) {}
133 // No warning
134
135 for (const Bar &x : B) {}
136 // expected-warning@-1 {{resulting in a copy}}
137 // expected-note@-2 {{'Bar'}}
138 for (const Bar x : B) {}
139 //for (Bar &x : B) {}
140 // Binding error
141 for (Bar x : B) {}
142 // No warning
143 }
144
test3()145 void test3() {
146 Container<Bar> C;
147
148 for (const Bar &x : C) {}
149 // expected-warning@-1 {{always a copy}}
150 // expected-note@-2 {{'Bar'}}
151 for (const Bar x : C) {}
152 // No warning, non-reference type indicates copy is made
153 //for (Bar &x : C) {}
154 // Binding error
155 for (Bar x : C) {}
156 // No warning, non-reference type indicates copy is made
157
158 for (const int &x : C) {}
159 // expected-warning@-1 {{always a copy}}
160 // expected-note@-2 {{'int'}}
161 for (const int x : C) {}
162 // No warning, copy made
163 //for (int &x : C) {}
164 // Binding error
165 for (int x : C) {}
166 // No warning, copy made
167 }
168
test4()169 void test4() {
170 Container<Bar&> D;
171
172 for (const Bar &x : D) {}
173 // No warning, this reference is not a temporary
174 for (const Bar x : D) {}
175 // expected-warning@-1 {{creates a copy}}
176 // expected-note@-2 {{'const Bar &'}}
177 for (Bar &x : D) {}
178 // No warning
179 for (Bar x : D) {}
180 // No warning
181
182 for (const int &x : D) {}
183 // expected-warning@-1 {{resulting in a copy}}
184 // expected-note-re@-2 {{'int'{{.*}}'const Bar &'}}
185 for (const int x : D) {}
186 // No warning
187 //for (int &x : D) {}
188 // Binding error
189 for (int x : D) {}
190 // No warning
191 }
192
test5()193 void test5() {
194 Container<Foo> E;
195
196 for (const Bar &x : E) {}
197 // expected-warning@-1 {{always a copy}}
198 // expected-note@-2 {{'Bar'}}
199 for (const Bar x : E) {}
200 // No warning, non-reference type indicates copy is made
201 //for (Bar &x : E) {}
202 // Binding error
203 for (Bar x : E) {}
204 // No warning, non-reference type indicates copy is made
205 }
206
test6()207 void test6() {
208 Container<Foo&> F;
209
210 for (const Bar &x : F) {}
211 // expected-warning@-1 {{resulting in a copy}}
212 // expected-note-re@-2 {{'Bar'{{.*}}'const Foo &'}}
213 for (const Bar x : F) {}
214 // No warning.
215 //for (Bar &x : F) {}
216 // Binding error
217 for (Bar x : F) {}
218 // No warning
219 }
220
test7()221 void test7() {
222 double G[2];
223
224 for (const double &x : G) {}
225 // No warning
226 for (const double x : G) {}
227 // No warning on POD copy
228 for (double &x : G) {}
229 // No warning
230 for (double x : G) {}
231 // No warning
232
233 for (const int &x : G) {}
234 // expected-warning@-1 {{resulting in a copy}}
235 // expected-note-re@-2 {{'int'{{.*}}'const double &'}}
236 for (const int x : G) {}
237 // No warning
238 //for (int &x : G) {}
239 // Binding error
240 for (int x : G) {}
241 // No warning
242
243 for (const Bar &x : G) {}
244 // expected-warning@-1 {{resulting in a copy}}
245 // expected-note-re@-2 {{'Bar'{{.*}}'const double &'}}
246 for (const Bar x : G) {}
247 // No warning
248 //for (int &Bar : G) {}
249 // Binding error
250 for (int Bar : G) {}
251 // No warning
252 }
253
test8()254 void test8() {
255 Foo H[2];
256
257 for (const Foo &x : H) {}
258 // No warning
259 for (const Foo x : H) {}
260 // No warning on POD copy
261 for (Foo &x : H) {}
262 // No warning
263 for (Foo x : H) {}
264 // No warning
265
266 for (const Bar &x : H) {}
267 // expected-warning@-1 {{resulting in a copy}}
268 // expected-note-re@-2 {{'Bar'{{.*}}'const Foo &'}}
269 for (const Bar x : H) {}
270 // No warning
271 //for (Bar &x: H) {}
272 // Binding error
273 for (Bar x: H) {}
274 // No warning
275 }
276
test9()277 void test9() {
278 Bar I[2] = {1,2};
279
280 for (const Bar &x : I) {}
281 // No warning
282 for (const Bar x : I) {}
283 // expected-warning@-1 {{creates a copy}}
284 // expected-note@-2 {{'const Bar &'}}
285 for (Bar &x : I) {}
286 // No warning
287 for (Bar x : I) {}
288 // No warning
289
290 for (const int &x : I) {}
291 // expected-warning@-1 {{resulting in a copy}}
292 // expected-note-re@-2 {{'int'{{.*}}'const Bar &'}}
293 for (const int x : I) {}
294 // No warning
295 //for (int &x : I) {}
296 // Binding error
297 for (int x : I) {}
298 // No warning
299 }
300