blob: c960e0651a0c21f55a70f31d1db3e4c6cbaf5f8f [file] [log] [blame]
Alexander Musmana8e9d2e2014-06-03 10:16:47 +00001// RUN: %clang_cc1 -fsyntax-only -fopenmp=libiomp5 -x c++ -std=c++11 -fexceptions -fcxx-exceptions -verify %s
2
3static int sii;
4#pragma omp threadprivate(sii)
5
6int test_iteration_spaces() {
7 const int N = 100;
8 float a[N], b[N], c[N];
9 int ii, jj, kk;
10 float fii;
11 double dii;
12 #pragma omp simd
13 for (int i = 0; i < 10; i+=1) {
14 c[i] = a[i] + b[i];
15 }
16 #pragma omp simd
17 for (char i = 0; i < 10; i++) {
18 c[i] = a[i] + b[i];
19 }
20 #pragma omp simd
21 for (char i = 0; i < 10; i+='\1') {
22 c[i] = a[i] + b[i];
23 }
24 #pragma omp simd
25 for (long long i = 0; i < 10; i++) {
26 c[i] = a[i] + b[i];
27 }
28 // expected-error@+2 {{expression must have integral or unscoped enumeration type, not 'double'}}
29 #pragma omp simd
30 for (long long i = 0; i < 10; i+=1.5) {
31 c[i] = a[i] + b[i];
32 }
33 #pragma omp simd
34 for (long long i = 0; i < 'z'; i+=1u) {
35 c[i] = a[i] + b[i];
36 }
37 // expected-error@+2 {{variable must be of integer or random access iterator type}}
38 #pragma omp simd
39 for (float fi = 0; fi < 10.0; fi++) {
40 c[(int)fi] = a[(int)fi] + b[(int)fi];
41 }
42 // expected-error@+2 {{variable must be of integer or random access iterator type}}
43 #pragma omp simd
44 for (double fi = 0; fi < 10.0; fi++) {
45 c[(int)fi] = a[(int)fi] + b[(int)fi];
46 }
47 // expected-error@+2 {{variable must be of integer or random access iterator type}}
48 #pragma omp simd
49 for (int &ref = ii; ref < 10; ref++) {
50 }
51 // expected-error@+2 {{initialization clause of OpenMP for loop must be of the form 'var = init' or 'T var = init'}}
52 #pragma omp simd
53 for (int i; i < 10; i++)
54 c[i] = a[i];
55
56 // expected-error@+2 {{initialization clause of OpenMP for loop must be of the form 'var = init' or 'T var = init'}}
57 #pragma omp simd
58 for (int i = 0, j = 0; i < 10; ++i)
59 c[i] = a[i];
60
61 // expected-error@+2 {{initialization clause of OpenMP for loop must be of the form 'var = init' or 'T var = init'}}
62 #pragma omp simd
63 for (;ii < 10; ++ii)
64 c[ii] = a[ii];
65
66 // expected-warning@+3 {{expression result unused}}
67 // expected-error@+2 {{initialization clause of OpenMP for loop must be of the form 'var = init' or 'T var = init'}}
68 #pragma omp simd
69 for (ii + 1;ii < 10; ++ii)
70 c[ii] = a[ii];
71
72 // expected-error@+2 {{initialization clause of OpenMP for loop must be of the form 'var = init' or 'T var = init'}}
73 #pragma omp simd
74 for (c[ii] = 0;ii < 10; ++ii)
75 c[ii] = a[ii];
76
77 // Ok to skip parenthesises.
78 #pragma omp simd
79 for (((ii)) = 0;ii < 10; ++ii)
80 c[ii] = a[ii];
81
82 // expected-error@+2 {{condition of OpenMP for loop must be a relational comparison ('<', '<=', '>', or '>=') of loop variable 'i'}}
83 #pragma omp simd
84 for (int i = 0; i; i++)
85 c[i] = a[i];
86
87 // expected-error@+3 {{condition of OpenMP for loop must be a relational comparison ('<', '<=', '>', or '>=') of loop variable 'i'}}
88 // expected-error@+2 {{increment clause of OpenMP for loop must perform simple addition or subtraction on loop variable 'i'}}
89 #pragma omp simd
90 for (int i = 0; jj < kk; ii++)
91 c[i] = a[i];
92
93 // expected-error@+2 {{condition of OpenMP for loop must be a relational comparison ('<', '<=', '>', or '>=') of loop variable 'i'}}
94 #pragma omp simd
95 for (int i = 0; !!i; i++)
96 c[i] = a[i];
97
98 // expected-error@+2 {{condition of OpenMP for loop must be a relational comparison ('<', '<=', '>', or '>=') of loop variable 'i'}}
99 #pragma omp simd
100 for (int i = 0; i != 1; i++)
101 c[i] = a[i];
102
103 // expected-error@+2 {{condition of OpenMP for loop must be a relational comparison ('<', '<=', '>', or '>=') of loop variable 'i'}}
104 #pragma omp simd
105 for (int i = 0; ; i++)
106 c[i] = a[i];
107
108 // Ok.
109 #pragma omp simd
110 for (int i = 11; i > 10; i--)
111 c[i] = a[i];
112
113 // Ok.
114 #pragma omp simd
115 for (int i = 0; i < 10; ++i)
116 c[i] = a[i];
117
118 // Ok.
119 #pragma omp simd
120 for (ii = 0; ii < 10; ++ii)
121 c[ii] = a[ii];
122
123 // expected-error@+2 {{increment clause of OpenMP for loop must perform simple addition or subtraction on loop variable 'ii'}}
124 #pragma omp simd
125 for (ii = 0; ii < 10; ++jj)
126 c[ii] = a[jj];
127
128 // expected-error@+2 {{increment clause of OpenMP for loop must perform simple addition or subtraction on loop variable 'ii'}}
129 #pragma omp simd
130 for (ii = 0; ii < 10; ++ ++ ii)
131 c[ii] = a[ii];
132
133 // Ok but undefined behavior (in general, cannot check that incr
134 // is really loop-invariant).
135 #pragma omp simd
136 for (ii = 0; ii < 10; ii = ii + ii)
137 c[ii] = a[ii];
138
139 // expected-error@+2 {{expression must have integral or unscoped enumeration type, not 'float'}}
140 #pragma omp simd
141 for (ii = 0; ii < 10; ii = ii + 1.0f)
142 c[ii] = a[ii];
143
144 // Ok - step was converted to integer type.
145 #pragma omp simd
146 for (ii = 0; ii < 10; ii = ii + (int)1.1f)
147 c[ii] = a[ii];
148
149 // expected-error@+2 {{increment clause of OpenMP for loop must perform simple addition or subtraction on loop variable 'ii'}}
150 #pragma omp simd
151 for (ii = 0; ii < 10; jj = ii + 2)
152 c[ii] = a[ii];
153
154 // expected-warning@+3 {{relational comparison result unused}}
155 // expected-error@+2 {{increment clause of OpenMP for loop must perform simple addition or subtraction on loop variable 'ii'}}
156 #pragma omp simd
157 for (ii = 0; ii < 10; jj > kk + 2)
158 c[ii] = a[ii];
159
160 // expected-error@+2 {{increment clause of OpenMP for loop must perform simple addition or subtraction on loop variable 'ii'}}
161 #pragma omp simd
162 for (ii = 0; ii < 10;)
163 c[ii] = a[ii];
164
165 // expected-warning@+3 {{expression result unused}}
166 // expected-error@+2 {{increment clause of OpenMP for loop must perform simple addition or subtraction on loop variable 'ii'}}
167 #pragma omp simd
168 for (ii = 0; ii < 10; !ii)
169 c[ii] = a[ii];
170
171 // expected-error@+2 {{increment clause of OpenMP for loop must perform simple addition or subtraction on loop variable 'ii'}}
172 #pragma omp simd
173 for (ii = 0; ii < 10; ii ? ++ii : ++jj)
174 c[ii] = a[ii];
175
176 // expected-error@+2 {{increment clause of OpenMP for loop must perform simple addition or subtraction on loop variable 'ii'}}
177 #pragma omp simd
178 for (ii = 0; ii < 10; ii = ii < 10)
179 c[ii] = a[ii];
180
181 // expected-note@+3 {{loop step is expected to be positive due to this condition}}
182 // expected-error@+2 {{increment expression must cause 'ii' to increase on each iteration of OpenMP for loop}}
183 #pragma omp simd
184 for (ii = 0; ii < 10; ii = ii + 0)
185 c[ii] = a[ii];
186
187 // expected-note@+3 {{loop step is expected to be positive due to this condition}}
188 // expected-error@+2 {{increment expression must cause 'ii' to increase on each iteration of OpenMP for loop}}
189 #pragma omp simd
190 for (ii = 0; ii < 10; ii = ii + (int)(0.8 - 0.45))
191 c[ii] = a[ii];
192
193 // expected-note@+3 {{loop step is expected to be positive due to this condition}}
194 // expected-error@+2 {{increment expression must cause 'ii' to increase on each iteration of OpenMP for loop}}
195 #pragma omp simd
196 for (ii = 0; (ii) < 10; ii-=25)
197 c[ii] = a[ii];
198
199 // expected-note@+3 {{loop step is expected to be positive due to this condition}}
200 // expected-error@+2 {{increment expression must cause 'ii' to increase on each iteration of OpenMP for loop}}
201 #pragma omp simd
202 for (ii = 0; (ii < 10); ii-=0)
203 c[ii] = a[ii];
204
205 // expected-note@+3 {{loop step is expected to be negative due to this condition}}
206 // expected-error@+2 {{increment expression must cause 'ii' to decrease on each iteration of OpenMP for loop}}
207 #pragma omp simd
208 for (ii = 0; ii > 10; (ii+=0))
209 c[ii] = a[ii];
210
211 // expected-note@+3 {{loop step is expected to be positive due to this condition}}
212 // expected-error@+2 {{increment expression must cause 'ii' to increase on each iteration of OpenMP for loop}}
213 #pragma omp simd
214 for (ii = 0; ii < 10; (ii) = (1-1)+(ii))
215 c[ii] = a[ii];
216
217 // expected-note@+3 {{loop step is expected to be negative due to this condition}}
218 // expected-error@+2 {{increment expression must cause 'ii' to decrease on each iteration of OpenMP for loop}}
219 #pragma omp simd
220 for ((ii = 0); ii > 10; (ii-=0))
221 c[ii] = a[ii];
222
223 // expected-note@+3 {{loop step is expected to be positive due to this condition}}
224 // expected-error@+2 {{increment expression must cause 'ii' to increase on each iteration of OpenMP for loop}}
225 #pragma omp simd
226 for (ii = 0; (ii < 10); (ii-=0))
227 c[ii] = a[ii];
228
229 // expected-note@+2 {{defined as private}}
230 // expected-error@+2 {{loop iteration variable may not be private}}
231 #pragma omp simd private(ii)
232 for (ii = 0; ii < 10; ii++)
233 c[ii] = a[ii];
234
235 // expected-error@+3 {{unexpected OpenMP clause 'shared' in directive '#pragma omp simd'}}
236 // expected-note@+2 {{defined as shared}}
237 // expected-error@+2 {{loop iteration variable may not be shared}}
238 #pragma omp simd shared(ii)
239 for (ii = 0; ii < 10; ii++)
240 c[ii] = a[ii];
241
242 #pragma omp simd linear(ii)
243 for (ii = 0; ii < 10; ii++)
244 c[ii] = a[ii];
245
246 // TODO: Add test for lastprivate.
247
248 #pragma omp parallel
249 {
250 #pragma omp simd
251 for (sii = 0; sii < 10; sii+=1)
252 c[sii] = a[sii];
253 }
254
255 // expected-error@+2 {{statement after '#pragma omp simd' must be a for loop}}
256 #pragma omp simd
257 for (auto &item : a) {
258 item = item + 1;
259 }
260
261 // expected-note@+3 {{loop step is expected to be positive due to this condition}}
262 // expected-error@+2 {{increment expression must cause 'i' to increase on each iteration of OpenMP for loop}}
263 #pragma omp simd
264 for (unsigned i = 9; i < 10; i--) {
265 c[i] = a[i] + b[i];
266 }
267
268 int (*lb)[4] = nullptr;
269 #pragma omp simd
270 for (int (*p)[4] = lb; p < lb + 8; ++p) {
271 }
272
273 // expected-warning@+2 {{initialization clause of OpenMP for loop is not in canonical form ('var = init' or 'T var = init')}}
274 #pragma omp simd
275 for (int a{0}; a<10; ++a) {
276 }
277
278 return 0;
279}
280
281// Iterators allowed in openmp for-loops.
282namespace std {
283struct random_access_iterator_tag { };
284template <class Iter> struct iterator_traits {
285 typedef typename Iter::difference_type difference_type;
286 typedef typename Iter::iterator_category iterator_category;
287};
288template <class Iter>
289typename iterator_traits<Iter>::difference_type
290distance(Iter first, Iter last) { return first - last; }
291}
292class Iter0 {
293 public:
294 Iter0() { }
295 Iter0(const Iter0 &) { }
296 Iter0 operator ++() { return *this; }
297 Iter0 operator --() { return *this; }
298 bool operator <(Iter0 a) { return true; }
299};
300int operator -(Iter0 a, Iter0 b) { return 0; }
301class Iter1 {
302 public:
303 Iter1(float f=0.0f, double d=0.0) { }
304 Iter1(const Iter1 &) { }
305 Iter1 operator ++() { return *this; }
306 Iter1 operator --() { return *this; }
307 bool operator <(Iter1 a) { return true; }
308 bool operator >=(Iter1 a) { return false; }
309};
310class GoodIter {
311 public:
312 GoodIter() { }
313 GoodIter(const GoodIter &) { }
314 GoodIter(int fst, int snd) { }
315 GoodIter &operator =(const GoodIter &that) { return *this; }
316 GoodIter &operator =(const Iter0 &that) { return *this; }
317 GoodIter &operator +=(int x) { return *this; }
318 explicit GoodIter(void *) { }
319 GoodIter operator ++() { return *this; }
320 GoodIter operator --() { return *this; }
321 bool operator !() { return true; }
322 bool operator <(GoodIter a) { return true; }
323 bool operator <=(GoodIter a) { return true; }
324 bool operator >=(GoodIter a) { return false; }
325 typedef int difference_type;
326 typedef std::random_access_iterator_tag iterator_category;
327};
328int operator -(GoodIter a, GoodIter b) { return 0; }
329GoodIter operator -(GoodIter a) { return a; }
330GoodIter operator -(GoodIter a, int v) { return GoodIter(); }
331GoodIter operator +(GoodIter a, int v) { return GoodIter(); }
332GoodIter operator -(int v, GoodIter a) { return GoodIter(); }
333GoodIter operator +(int v, GoodIter a) { return GoodIter(); }
334
335int test_with_random_access_iterator() {
336 GoodIter begin, end;
337 Iter0 begin0, end0;
338 #pragma omp simd
339 for (GoodIter I = begin; I < end; ++I)
340 ++I;
341 // expected-error@+2 {{variable must be of integer or random access iterator type}}
342 #pragma omp simd
343 for (GoodIter &I = begin; I < end; ++I)
344 ++I;
345 #pragma omp simd
346 for (GoodIter I = begin; I >= end; --I)
347 ++I;
348 // expected-warning@+2 {{initialization clause of OpenMP for loop is not in canonical form ('var = init' or 'T var = init')}}
349 #pragma omp simd
350 for (GoodIter I(begin); I < end; ++I)
351 ++I;
352 // expected-warning@+2 {{initialization clause of OpenMP for loop is not in canonical form ('var = init' or 'T var = init')}}
353 #pragma omp simd
354 for (GoodIter I(nullptr); I < end; ++I)
355 ++I;
356 // expected-warning@+2 {{initialization clause of OpenMP for loop is not in canonical form ('var = init' or 'T var = init')}}
357 #pragma omp simd
358 for (GoodIter I(0); I < end; ++I)
359 ++I;
360 // expected-warning@+2 {{initialization clause of OpenMP for loop is not in canonical form ('var = init' or 'T var = init')}}
361 #pragma omp simd
362 for (GoodIter I(1,2); I < end; ++I)
363 ++I;
364 #pragma omp simd
365 for (begin = GoodIter(0); begin < end; ++begin)
366 ++begin;
367 #pragma omp simd
368 for (begin = begin0; begin < end; ++begin)
369 ++begin;
370 // expected-error@+2 {{initialization clause of OpenMP for loop must be of the form 'var = init' or 'T var = init'}}
371 #pragma omp simd
372 for (++begin; begin < end; ++begin)
373 ++begin;
374 #pragma omp simd
375 for (begin = end; begin < end; ++begin)
376 ++begin;
377 // expected-error@+2 {{condition of OpenMP for loop must be a relational comparison ('<', '<=', '>', or '>=') of loop variable 'I'}}
378 #pragma omp simd
379 for (GoodIter I = begin; I - I; ++I)
380 ++I;
381 // expected-error@+2 {{condition of OpenMP for loop must be a relational comparison ('<', '<=', '>', or '>=') of loop variable 'I'}}
382 #pragma omp simd
383 for (GoodIter I = begin; begin < end; ++I)
384 ++I;
385 // expected-error@+2 {{condition of OpenMP for loop must be a relational comparison ('<', '<=', '>', or '>=') of loop variable 'I'}}
386 #pragma omp simd
387 for (GoodIter I = begin; !I; ++I)
388 ++I;
389 // expected-note@+3 {{loop step is expected to be negative due to this condition}}
390 // expected-error@+2 {{increment expression must cause 'I' to decrease on each iteration of OpenMP for loop}}
391 #pragma omp simd
392 for (GoodIter I = begin; I >= end; I = I + 1)
393 ++I;
394 #pragma omp simd
395 for (GoodIter I = begin; I >= end; I = I - 1)
396 ++I;
397 // expected-error@+2 {{increment clause of OpenMP for loop must perform simple addition or subtraction on loop variable 'I'}}
398 #pragma omp simd
399 for (GoodIter I = begin; I >= end; I = -I)
400 ++I;
401 // expected-note@+3 {{loop step is expected to be negative due to this condition}}
402 // expected-error@+2 {{increment expression must cause 'I' to decrease on each iteration of OpenMP for loop}}
403 #pragma omp simd
404 for (GoodIter I = begin; I >= end; I = 2 + I)
405 ++I;
406 // expected-error@+2 {{increment clause of OpenMP for loop must perform simple addition or subtraction on loop variable 'I'}}
407 #pragma omp simd
408 for (GoodIter I = begin; I >= end; I = 2 - I)
409 ++I;
410 #pragma omp simd
411 for (Iter0 I = begin0; I < end0; ++I)
412 ++I;
413 // Initializer is constructor without params.
414 // expected-warning@+2 {{initialization clause of OpenMP for loop is not in canonical form ('var = init' or 'T var = init')}}
415 #pragma omp simd
416 for (Iter0 I; I < end0; ++I)
417 ++I;
418 Iter1 begin1, end1;
419 #pragma omp simd
420 for (Iter1 I = begin1; I < end1; ++I)
421 ++I;
422 // expected-note@+3 {{loop step is expected to be negative due to this condition}}
423 // expected-error@+2 {{increment expression must cause 'I' to decrease on each iteration of OpenMP for loop}}
424 #pragma omp simd
425 for (Iter1 I = begin1; I >= end1; ++I)
426 ++I;
427 // Initializer is constructor with all default params.
428 // expected-warning@+2 {{initialization clause of OpenMP for loop is not in canonical form ('var = init' or 'T var = init')}}
429 #pragma omp simd
430 for (Iter1 I; I < end1; ++I) {
431 }
432 return 0;
433}
434
435template <typename IT, int ST> class TC {
436 public:
437 int dotest_lt(IT begin, IT end) {
438 // expected-note@+3 {{loop step is expected to be positive due to this condition}}
439 // expected-error@+2 {{increment expression must cause 'I' to increase on each iteration of OpenMP for loop}}
440 #pragma omp simd
441 for (IT I = begin; I < end; I = I + ST) {
442 ++I;
443 }
444 // expected-note@+3 {{loop step is expected to be positive due to this condition}}
445 // expected-error@+2 {{increment expression must cause 'I' to increase on each iteration of OpenMP for loop}}
446 #pragma omp simd
447 for (IT I = begin; I <= end; I += ST) {
448 ++I;
449 }
450 #pragma omp simd
451 for (IT I = begin; I < end; ++I) {
452 ++I;
453 }
454 }
455
456 static IT step() {
457 return IT(ST);
458 }
459};
460template <typename IT, int ST=0> int dotest_gt(IT begin, IT end) {
461 // expected-note@+3 2 {{loop step is expected to be negative due to this condition}}
462 // expected-error@+2 2 {{increment expression must cause 'I' to decrease on each iteration of OpenMP for loop}}
463 #pragma omp simd
464 for (IT I = begin; I >= end; I = I + ST) {
465 ++I;
466 }
467 // expected-note@+3 2 {{loop step is expected to be negative due to this condition}}
468 // expected-error@+2 2 {{increment expression must cause 'I' to decrease on each iteration of OpenMP for loop}}
469 #pragma omp simd
470 for (IT I = begin; I >= end; I += ST) {
471 ++I;
472 }
473
474 // expected-note@+3 {{loop step is expected to be negative due to this condition}}
475 // expected-error@+2 {{increment expression must cause 'I' to decrease on each iteration of OpenMP for loop}}
476 #pragma omp simd
477 for (IT I = begin; I >= end; ++I) {
478 ++I;
479 }
480
481 #pragma omp simd
482 for (IT I = begin; I < end; I+=TC<int,ST>::step()) {
483 ++I;
484 }
485}
486
487void test_with_template() {
488 GoodIter begin, end;
489 TC<GoodIter, 100> t1;
490 TC<GoodIter, -100> t2;
491 t1.dotest_lt(begin, end);
492 t2.dotest_lt(begin, end); // expected-note {{in instantiation of member function 'TC<GoodIter, -100>::dotest_lt' requested here}}
493 dotest_gt(begin, end); // expected-note {{in instantiation of function template specialization 'dotest_gt<GoodIter, 0>' requested here}}
494 dotest_gt<unsigned, -10>(0, 100); // expected-note {{in instantiation of function template specialization 'dotest_gt<unsigned int, -10>' requested here}}
495}
496
497void test_loop_break() {
498 const int N = 100;
499 float a[N], b[N], c[N];
500 #pragma omp simd
501 for (int i = 0; i < 10; i++) {
502 c[i] = a[i] + b[i];
503 for (int j = 0; j < 10; ++j) {
504 if (a[i] > b[j])
505 break; // OK in nested loop
506 }
507 switch(i) {
508 case 1:
509 b[i]++;
510 break;
511 default:
512 break;
513 }
514 if (c[i] > 10)
515 break; // expected-error {{'break' statement cannot be used in OpenMP for loop}}
516
517 if (c[i] > 11)
518 break; // expected-error {{'break' statement cannot be used in OpenMP for loop}}
519 }
520
521 #pragma omp simd
522 for (int i = 0; i < 10; i++) {
523 for (int j = 0; j < 10; j++) {
524 c[i] = a[i] + b[i];
525 if (c[i] > 10) {
526 if (c[i] < 20) {
527 break; // OK
528 }
529 }
530 }
531 }
532}
533
534void test_loop_eh() {
535 const int N = 100;
536 float a[N], b[N], c[N];
537 #pragma omp simd
538 for (int i = 0; i < 10; i++) {
539 c[i] = a[i] + b[i];
540 try { // expected-error {{'try' statement cannot be used in OpenMP simd region}}
541 for (int j = 0; j < 10; ++j) {
542 if (a[i] > b[j])
543 throw a[i]; // expected-error {{'throw' statement cannot be used in OpenMP simd region}}
544 }
545 throw a[i]; // expected-error {{'throw' statement cannot be used in OpenMP simd region}}
546 }
547 catch (float f) {
548 if (f > 0.1)
549 throw a[i]; // expected-error {{'throw' statement cannot be used in OpenMP simd region}}
550 return; // expected-error {{cannot return from OpenMP region}}
551 }
552 switch(i) {
553 case 1:
554 b[i]++;
555 break;
556 default:
557 break;
558 }
559 for (int j = 0; j < 10; j++) {
560 if (c[i] > 10)
561 throw c[i]; // expected-error {{'throw' statement cannot be used in OpenMP simd region}}
562 }
563 }
564 if (c[9] > 10)
565 throw c[9]; // OK
566
567 #pragma omp simd
568 for (int i = 0; i < 10; ++i) {
569 struct S {
570 void g() { throw 0; }
571 };
572 }
573}
574