1*67e74705SXin Li // RUN: %clang_cc1 -verify -fopenmp -ast-print %s | FileCheck %s
2*67e74705SXin Li // RUN: %clang_cc1 -fopenmp -x c++ -std=c++11 -emit-pch -o %t %s
3*67e74705SXin Li // RUN: %clang_cc1 -fopenmp -std=c++11 -include-pch %t -fsyntax-only -verify %s -ast-print | FileCheck %s
4*67e74705SXin Li // expected-no-diagnostics
5*67e74705SXin Li
6*67e74705SXin Li #ifndef HEADER
7*67e74705SXin Li #define HEADER
8*67e74705SXin Li
9*67e74705SXin Li struct SS {
SSSS10*67e74705SXin Li SS(): a(0) {}
SSSS11*67e74705SXin Li SS(int v) : a(v) {}
12*67e74705SXin Li int a;
13*67e74705SXin Li typedef int type;
14*67e74705SXin Li };
15*67e74705SXin Li
16*67e74705SXin Li template <typename T>
17*67e74705SXin Li class S7 : public T {
18*67e74705SXin Li protected:
19*67e74705SXin Li T *a;
20*67e74705SXin Li T b[2];
S7()21*67e74705SXin Li S7() : a(0) {}
22*67e74705SXin Li
23*67e74705SXin Li public:
S7(typename T::type & v)24*67e74705SXin Li S7(typename T::type &v) : a((T*)&v) {
25*67e74705SXin Li #pragma omp simd aligned(a)
26*67e74705SXin Li for (int k = 0; k < a->a; ++k)
27*67e74705SXin Li ++this->a->a;
28*67e74705SXin Li }
operator =(S7 & s)29*67e74705SXin Li S7 &operator=(S7 &s) {
30*67e74705SXin Li #pragma omp simd aligned(this->b : 8)
31*67e74705SXin Li for (int k = 0; k < s.a->a; ++k)
32*67e74705SXin Li ++s.a->a;
33*67e74705SXin Li return *this;
34*67e74705SXin Li }
35*67e74705SXin Li };
36*67e74705SXin Li
37*67e74705SXin Li // CHECK: #pragma omp simd aligned(this->a)
38*67e74705SXin Li // CHECK: #pragma omp simd aligned(this->a)
39*67e74705SXin Li // CHECK: #pragma omp simd aligned(this->b: 8)
40*67e74705SXin Li
41*67e74705SXin Li class S8 : public S7<SS> {
S8()42*67e74705SXin Li S8() {}
43*67e74705SXin Li
44*67e74705SXin Li public:
S8(int v)45*67e74705SXin Li S8(int v) : S7<SS>(v){
46*67e74705SXin Li #pragma omp simd aligned(S7<SS>::a)
47*67e74705SXin Li for (int k = 0; k < a->a; ++k)
48*67e74705SXin Li ++this->a->a;
49*67e74705SXin Li }
operator =(S8 & s)50*67e74705SXin Li S8 &operator=(S8 &s) {
51*67e74705SXin Li #pragma omp simd aligned(this->b: 4)
52*67e74705SXin Li for (int k = 0; k < s.a->a; ++k)
53*67e74705SXin Li ++s.a->a;
54*67e74705SXin Li return *this;
55*67e74705SXin Li }
56*67e74705SXin Li };
57*67e74705SXin Li
58*67e74705SXin Li // CHECK: #pragma omp simd aligned(this->S7<SS>::a)
59*67e74705SXin Li // CHECK: #pragma omp simd aligned(this->b: 4)
60*67e74705SXin Li
foo()61*67e74705SXin Li void foo() {}
62*67e74705SXin Li int g_ind = 1;
reduct(T * arr,N num)63*67e74705SXin Li template<class T, class N> T reduct(T* arr, N num) {
64*67e74705SXin Li N i;
65*67e74705SXin Li N ind;
66*67e74705SXin Li N myind;
67*67e74705SXin Li N &ref = i;
68*67e74705SXin Li T sum = (T)0;
69*67e74705SXin Li // CHECK: T sum = (T)0;
70*67e74705SXin Li #pragma omp simd private(myind, g_ind), linear(ind), aligned(arr), linear(uval(ref))
71*67e74705SXin Li // CHECK-NEXT: #pragma omp simd private(myind,g_ind) linear(ind) aligned(arr) linear(uval(ref))
72*67e74705SXin Li for (i = 0; i < num; ++i) {
73*67e74705SXin Li myind = ind;
74*67e74705SXin Li T cur = arr[myind];
75*67e74705SXin Li ind += g_ind;
76*67e74705SXin Li sum += cur;
77*67e74705SXin Li }
78*67e74705SXin Li }
79*67e74705SXin Li
80*67e74705SXin Li template<class T> struct S {
SS81*67e74705SXin Li S(const T &a)
82*67e74705SXin Li :m_a(a)
83*67e74705SXin Li {}
resultS84*67e74705SXin Li T result(T *v) const {
85*67e74705SXin Li T res;
86*67e74705SXin Li T val;
87*67e74705SXin Li T lin = 0;
88*67e74705SXin Li T &ref = res;
89*67e74705SXin Li // CHECK: T res;
90*67e74705SXin Li // CHECK: T val;
91*67e74705SXin Li // CHECK: T lin = 0;
92*67e74705SXin Li // CHECK: T &ref = res;
93*67e74705SXin Li #pragma omp simd private(val) safelen(7) linear(lin : -5) lastprivate(res) simdlen(5) linear(ref(ref))
94*67e74705SXin Li // CHECK-NEXT: #pragma omp simd private(val) safelen(7) linear(lin: -5) lastprivate(res) simdlen(5) linear(ref(ref))
95*67e74705SXin Li for (T i = 7; i < m_a; ++i) {
96*67e74705SXin Li val = v[i-7] + m_a;
97*67e74705SXin Li res = val;
98*67e74705SXin Li lin -= 5;
99*67e74705SXin Li }
100*67e74705SXin Li const T clen = 3;
101*67e74705SXin Li // CHECK: T clen = 3;
102*67e74705SXin Li #pragma omp simd safelen(clen-1) simdlen(clen-1)
103*67e74705SXin Li // CHECK-NEXT: #pragma omp simd safelen(clen - 1) simdlen(clen - 1)
104*67e74705SXin Li for(T i = clen+2; i < 20; ++i) {
105*67e74705SXin Li // CHECK-NEXT: for (T i = clen + 2; i < 20; ++i) {
106*67e74705SXin Li v[i] = v[v-clen] + 1;
107*67e74705SXin Li // CHECK-NEXT: v[i] = v[v - clen] + 1;
108*67e74705SXin Li }
109*67e74705SXin Li // CHECK-NEXT: }
110*67e74705SXin Li return res;
111*67e74705SXin Li }
~SS112*67e74705SXin Li ~S()
113*67e74705SXin Li {}
114*67e74705SXin Li T m_a;
115*67e74705SXin Li };
116*67e74705SXin Li
117*67e74705SXin Li template<int LEN> struct S2 {
funcS2118*67e74705SXin Li static void func(int n, float *a, float *b, float *c) {
119*67e74705SXin Li int k1 = 0, k2 = 0;
120*67e74705SXin Li #pragma omp simd safelen(LEN) linear(k1,k2:LEN) aligned(a:LEN) simdlen(LEN)
121*67e74705SXin Li for(int i = 0; i < n; i++) {
122*67e74705SXin Li c[i] = a[i] + b[i];
123*67e74705SXin Li c[k1] = a[k1] + b[k1];
124*67e74705SXin Li c[k2] = a[k2] + b[k2];
125*67e74705SXin Li k1 = k1 + LEN;
126*67e74705SXin Li k2 = k2 + LEN;
127*67e74705SXin Li }
128*67e74705SXin Li }
129*67e74705SXin Li };
130*67e74705SXin Li
131*67e74705SXin Li // S2<4>::func is called below in main.
132*67e74705SXin Li // CHECK: template <int LEN = 4> struct S2 {
133*67e74705SXin Li // CHECK-NEXT: static void func(int n, float *a, float *b, float *c) {
134*67e74705SXin Li // CHECK-NEXT: int k1 = 0, k2 = 0;
135*67e74705SXin Li // CHECK-NEXT: #pragma omp simd safelen(4) linear(k1,k2: 4) aligned(a: 4) simdlen(4)
136*67e74705SXin Li // CHECK-NEXT: for (int i = 0; i < n; i++) {
137*67e74705SXin Li // CHECK-NEXT: c[i] = a[i] + b[i];
138*67e74705SXin Li // CHECK-NEXT: c[k1] = a[k1] + b[k1];
139*67e74705SXin Li // CHECK-NEXT: c[k2] = a[k2] + b[k2];
140*67e74705SXin Li // CHECK-NEXT: k1 = k1 + 4;
141*67e74705SXin Li // CHECK-NEXT: k2 = k2 + 4;
142*67e74705SXin Li // CHECK-NEXT: }
143*67e74705SXin Li // CHECK-NEXT: }
144*67e74705SXin Li
main(int argc,char ** argv)145*67e74705SXin Li int main (int argc, char **argv) {
146*67e74705SXin Li int b = argc, c, d, e, f, g;
147*67e74705SXin Li int k1=0,k2=0;
148*67e74705SXin Li int &ref = b;
149*67e74705SXin Li static int *a;
150*67e74705SXin Li // CHECK: static int *a;
151*67e74705SXin Li #pragma omp simd
152*67e74705SXin Li // CHECK-NEXT: #pragma omp simd
153*67e74705SXin Li for (int i=0; i < 2; ++i)*a=2;
154*67e74705SXin Li // CHECK-NEXT: for (int i = 0; i < 2; ++i)
155*67e74705SXin Li // CHECK-NEXT: *a = 2;
156*67e74705SXin Li #pragma omp simd private(argc, b),lastprivate(d,f) collapse(2) aligned(a : 4)
157*67e74705SXin Li for (int i = 0; i < 10; ++i)
158*67e74705SXin Li for (int j = 0; j < 10; ++j) {foo(); k1 += 8; k2 += 8;}
159*67e74705SXin Li // CHECK-NEXT: #pragma omp simd private(argc,b) lastprivate(d,f) collapse(2) aligned(a: 4)
160*67e74705SXin Li // CHECK-NEXT: for (int i = 0; i < 10; ++i)
161*67e74705SXin Li // CHECK-NEXT: for (int j = 0; j < 10; ++j) {
162*67e74705SXin Li // CHECK-NEXT: foo();
163*67e74705SXin Li // CHECK-NEXT: k1 += 8;
164*67e74705SXin Li // CHECK-NEXT: k2 += 8;
165*67e74705SXin Li // CHECK-NEXT: }
166*67e74705SXin Li for (int i = 0; i < 10; ++i)foo();
167*67e74705SXin Li // CHECK-NEXT: for (int i = 0; i < 10; ++i)
168*67e74705SXin Li // CHECK-NEXT: foo();
169*67e74705SXin Li const int CLEN = 4;
170*67e74705SXin Li // CHECK-NEXT: const int CLEN = 4;
171*67e74705SXin Li #pragma omp simd aligned(a:CLEN) linear(a:CLEN) safelen(CLEN) collapse( 1 ) simdlen(CLEN) linear(val(ref): CLEN)
172*67e74705SXin Li // CHECK-NEXT: #pragma omp simd aligned(a: CLEN) linear(a: CLEN) safelen(CLEN) collapse(1) simdlen(CLEN) linear(val(ref): CLEN)
173*67e74705SXin Li for (int i = 0; i < 10; ++i)foo();
174*67e74705SXin Li // CHECK-NEXT: for (int i = 0; i < 10; ++i)
175*67e74705SXin Li // CHECK-NEXT: foo();
176*67e74705SXin Li
177*67e74705SXin Li float arr[16];
178*67e74705SXin Li S2<4>::func(0,arr,arr,arr);
179*67e74705SXin Li return (0);
180*67e74705SXin Li }
181*67e74705SXin Li
182*67e74705SXin Li #endif
183