1*67e74705SXin Li // RUN: %clang_cc1 -verify -triple x86_64-apple-darwin10 -fopenmp -fexceptions -fcxx-exceptions -x c++ -emit-llvm %s -o - | FileCheck %s
2*67e74705SXin Li // RUN: %clang_cc1 -verify -triple x86_64-apple-darwin10 -fopenmp -fexceptions -fcxx-exceptions -debug-info-kind=line-tables-only -x c++ -emit-llvm %s -o - | FileCheck %s --check-prefix=TERM_DEBUG
3*67e74705SXin Li // expected-no-diagnostics
4*67e74705SXin Li
5*67e74705SXin Li int a;
6*67e74705SXin Li int b;
7*67e74705SXin Li
8*67e74705SXin Li struct St {
9*67e74705SXin Li unsigned long field;
StSt10*67e74705SXin Li St() {}
~StSt11*67e74705SXin Li ~St() {}
getSt12*67e74705SXin Li int &get() { return a; }
13*67e74705SXin Li };
14*67e74705SXin Li
15*67e74705SXin Li // CHECK-LABEL: parallel_atomic_ewc
parallel_atomic_ewc()16*67e74705SXin Li void parallel_atomic_ewc() {
17*67e74705SXin Li St s;
18*67e74705SXin Li #pragma omp parallel
19*67e74705SXin Li {
20*67e74705SXin Li // CHECK: invoke void @_ZN2StC1Ev(%struct.St* [[TEMP_ST_ADDR:%.+]])
21*67e74705SXin Li // CHECK: [[SCALAR_ADDR:%.+]] = invoke dereferenceable(4) i32* @_ZN2St3getEv(%struct.St* [[TEMP_ST_ADDR]])
22*67e74705SXin Li // CHECK: [[SCALAR_VAL:%.+]] = load atomic i32, i32* [[SCALAR_ADDR]] monotonic
23*67e74705SXin Li // CHECK: store i32 [[SCALAR_VAL]], i32* @b
24*67e74705SXin Li // CHECK: invoke void @_ZN2StD1Ev(%struct.St* [[TEMP_ST_ADDR]])
25*67e74705SXin Li #pragma omp atomic read
26*67e74705SXin Li b = St().get();
27*67e74705SXin Li // CHECK-DAG: invoke void @_ZN2StC1Ev(%struct.St* [[TEMP_ST_ADDR:%.+]])
28*67e74705SXin Li // CHECK-DAG: [[SCALAR_ADDR:%.+]] = invoke dereferenceable(4) i32* @_ZN2St3getEv(%struct.St* [[TEMP_ST_ADDR]])
29*67e74705SXin Li // CHECK-DAG: [[B_VAL:%.+]] = load i32, i32* @b
30*67e74705SXin Li // CHECK: store atomic i32 [[B_VAL]], i32* [[SCALAR_ADDR]] monotonic
31*67e74705SXin Li // CHECK: invoke void @_ZN2StD1Ev(%struct.St* [[TEMP_ST_ADDR]])
32*67e74705SXin Li #pragma omp atomic write
33*67e74705SXin Li St().get() = b;
34*67e74705SXin Li // CHECK: invoke void @_ZN2StC1Ev(%struct.St* [[TEMP_ST_ADDR:%.+]])
35*67e74705SXin Li // CHECK: [[SCALAR_ADDR:%.+]] = invoke dereferenceable(4) i32* @_ZN2St3getEv(%struct.St* [[TEMP_ST_ADDR]])
36*67e74705SXin Li // CHECK: [[B_VAL:%.+]] = load i32, i32* @b
37*67e74705SXin Li // CHECK: [[OLD_VAL:%.+]] = load atomic i32, i32* [[SCALAR_ADDR]] monotonic,
38*67e74705SXin Li // CHECK: br label %[[OMP_UPDATE:.+]]
39*67e74705SXin Li // CHECK: [[OMP_UPDATE]]
40*67e74705SXin Li // CHECK: [[OLD_PHI_VAL:%.+]] = phi i32 [ [[OLD_VAL]], %{{.+}} ], [ [[NEW_OLD_VAL:%.+]], %[[OMP_UPDATE]] ]
41*67e74705SXin Li // CHECK: [[NEW_VAL:%.+]] = srem i32 [[OLD_PHI_VAL]], [[B_VAL]]
42*67e74705SXin Li // CHECK: store i32 [[NEW_VAL]], i32* [[TEMP:%.+]],
43*67e74705SXin Li // CHECK: [[NEW_VAL:%.+]] = load i32, i32* [[TEMP]],
44*67e74705SXin Li // CHECK: [[RES:%.+]] = cmpxchg i32* [[SCALAR_ADDR]], i32 [[OLD_PHI_VAL]], i32 [[NEW_VAL]] monotonic monotonic
45*67e74705SXin Li // CHECK: [[NEW_OLD_VAL]] = extractvalue { i32, i1 } [[RES]], 0
46*67e74705SXin Li // CHECK: [[COND:%.+]] = extractvalue { i32, i1 } [[RES]], 1
47*67e74705SXin Li // CHECK: br i1 [[COND]], label %[[OMP_DONE:.+]], label %[[OMP_UPDATE]]
48*67e74705SXin Li // CHECK: [[OMP_DONE]]
49*67e74705SXin Li // CHECK: invoke void @_ZN2StD1Ev(%struct.St* [[TEMP_ST_ADDR]])
50*67e74705SXin Li #pragma omp atomic
51*67e74705SXin Li St().get() %= b;
52*67e74705SXin Li #pragma omp atomic
53*67e74705SXin Li s.field++;
54*67e74705SXin Li // CHECK: invoke void @_ZN2StC1Ev(%struct.St* [[TEMP_ST_ADDR:%.+]])
55*67e74705SXin Li // CHECK: [[SCALAR_ADDR:%.+]] = invoke dereferenceable(4) i32* @_ZN2St3getEv(%struct.St* [[TEMP_ST_ADDR]])
56*67e74705SXin Li // CHECK: [[B_VAL:%.+]] = load i32, i32* @b
57*67e74705SXin Li // CHECK: [[OLD_VAL:%.+]] = load atomic i32, i32* [[SCALAR_ADDR]] monotonic,
58*67e74705SXin Li // CHECK: br label %[[OMP_UPDATE:.+]]
59*67e74705SXin Li // CHECK: [[OMP_UPDATE]]
60*67e74705SXin Li // CHECK: [[OLD_PHI_VAL:%.+]] = phi i32 [ [[OLD_VAL]], %{{.+}} ], [ [[NEW_OLD_VAL:%.+]], %[[OMP_UPDATE]] ]
61*67e74705SXin Li // CHECK: [[NEW_CALC_VAL:%.+]] = srem i32 [[OLD_PHI_VAL]], [[B_VAL]]
62*67e74705SXin Li // CHECK: store i32 [[NEW_CALC_VAL]], i32* [[TEMP:%.+]],
63*67e74705SXin Li // CHECK: [[NEW_VAL:%.+]] = load i32, i32* [[TEMP]],
64*67e74705SXin Li // CHECK: [[RES:%.+]] = cmpxchg i32* [[SCALAR_ADDR]], i32 [[OLD_PHI_VAL]], i32 [[NEW_VAL]] monotonic monotonic
65*67e74705SXin Li // CHECK: [[NEW_OLD_VAL]] = extractvalue { i32, i1 } [[RES]], 0
66*67e74705SXin Li // CHECK: [[COND:%.+]] = extractvalue { i32, i1 } [[RES]], 1
67*67e74705SXin Li // CHECK: br i1 [[COND]], label %[[OMP_DONE:.+]], label %[[OMP_UPDATE]]
68*67e74705SXin Li // CHECK: [[OMP_DONE]]
69*67e74705SXin Li // CHECK: store i32 [[NEW_CALC_VAL]], i32* @a,
70*67e74705SXin Li // CHECK: invoke void @_ZN2StD1Ev(%struct.St* [[TEMP_ST_ADDR]])
71*67e74705SXin Li #pragma omp atomic capture
72*67e74705SXin Li a = St().get() %= b;
73*67e74705SXin Li }
74*67e74705SXin Li }
75*67e74705SXin Li
foo()76*67e74705SXin Li int &foo() { return a; }
77*67e74705SXin Li
78*67e74705SXin Li // TERM_DEBUG-LABEL: parallel_atomic
parallel_atomic()79*67e74705SXin Li void parallel_atomic() {
80*67e74705SXin Li #pragma omp parallel
81*67e74705SXin Li {
82*67e74705SXin Li #pragma omp atomic read
83*67e74705SXin Li // TERM_DEBUG-NOT: __kmpc_global_thread_num
84*67e74705SXin Li // TERM_DEBUG: invoke {{.*}}foo{{.*}}()
85*67e74705SXin Li // TERM_DEBUG: unwind label %[[TERM_LPAD:.+]],
86*67e74705SXin Li // TERM_DEBUG: load atomic i32, i32* @{{.+}} monotonic, {{.*}}!dbg [[READ_LOC:![0-9]+]]
87*67e74705SXin Li foo() = a;
88*67e74705SXin Li #pragma omp atomic write
89*67e74705SXin Li // TERM_DEBUG-NOT: __kmpc_global_thread_num
90*67e74705SXin Li // TERM_DEBUG: invoke {{.*}}foo{{.*}}()
91*67e74705SXin Li // TERM_DEBUG: unwind label %[[TERM_LPAD:.+]],
92*67e74705SXin Li // TERM_DEBUG-NOT: __kmpc_global_thread_num
93*67e74705SXin Li // TERM_DEBUG: store atomic i32 {{%.+}}, i32* @{{.+}} monotonic, {{.*}}!dbg [[WRITE_LOC:![0-9]+]]
94*67e74705SXin Li a = foo();
95*67e74705SXin Li #pragma omp atomic update
96*67e74705SXin Li // TERM_DEBUG-NOT: __kmpc_global_thread_num
97*67e74705SXin Li // TERM_DEBUG: invoke {{.*}}foo{{.*}}()
98*67e74705SXin Li // TERM_DEBUG: unwind label %[[TERM_LPAD:.+]],
99*67e74705SXin Li // TERM_DEBUG-NOT: __kmpc_global_thread_num
100*67e74705SXin Li // TERM_DEBUG: atomicrmw add i32* @{{.+}}, i32 %{{.+}} monotonic, {{.*}}!dbg [[UPDATE_LOC:![0-9]+]]
101*67e74705SXin Li a += foo();
102*67e74705SXin Li #pragma omp atomic capture
103*67e74705SXin Li // TERM_DEBUG-NOT: __kmpc_global_thread_num
104*67e74705SXin Li // TERM_DEBUG: invoke {{.*}}foo{{.*}}()
105*67e74705SXin Li // TERM_DEBUG: unwind label %[[TERM_LPAD:.+]],
106*67e74705SXin Li // TERM_DEBUG-NOT: __kmpc_global_thread_num
107*67e74705SXin Li // TERM_DEBUG: [[OLD_VAL:%.+]] = atomicrmw add i32* @{{.+}}, i32 %{{.+}} monotonic, {{.*}}!dbg [[CAPTURE_LOC:![0-9]+]]
108*67e74705SXin Li // TERM_DEBUG: store i32 [[OLD_VAL]], i32* @b,
109*67e74705SXin Li {b = a; a += foo(); }
110*67e74705SXin Li }
111*67e74705SXin Li // TERM_DEBUG: [[TERM_LPAD]]
112*67e74705SXin Li // TERM_DEBUG: call void @__clang_call_terminate
113*67e74705SXin Li // TERM_DEBUG: unreachable
114*67e74705SXin Li }
115*67e74705SXin Li // TERM_DEBUG-DAG: [[READ_LOC]] = !DILocation(line: [[@LINE-28]],
116*67e74705SXin Li // TERM_DEBUG-DAG: [[WRITE_LOC]] = !DILocation(line: [[@LINE-22]],
117*67e74705SXin Li // TERM_DEBUG-DAG: [[UPDATE_LOC]] = !DILocation(line: [[@LINE-16]],
118*67e74705SXin Li // TERM_DEBUG-DAG: [[CAPTURE_LOC]] = !DILocation(line: [[@LINE-9]],
119