blob: 2ce1f9450b3f39fa639c1d628a63ba308238abb2 [file] [log] [blame]
Alexey Bataevdb390212015-05-20 04:24:19 +00001// RUN: %clang_cc1 -verify -triple x86_64-apple-darwin10 -fopenmp -fexceptions -fcxx-exceptions -x c++ -emit-llvm %s -o - | FileCheck %s
2// RUN: %clang_cc1 -verify -triple x86_64-apple-darwin10 -fopenmp -fexceptions -fcxx-exceptions -gline-tables-only -x c++ -emit-llvm %s -o - | FileCheck %s --check-prefix=TERM_DEBUG
Alexey Bataev36bf0112015-03-10 05:15:26 +00003// expected-no-diagnostics
4
5int a;
Alexey Bataev10fec572015-03-11 04:48:56 +00006int b;
7
8struct St {
9 St() {}
10 ~St() {}
11 int &get() { return a; }
12};
13
14// CHECK-LABEL: parallel_atomic_ewc
15void parallel_atomic_ewc() {
16#pragma omp parallel
17 {
18 // CHECK: invoke void @_ZN2StC1Ev(%struct.St* [[TEMP_ST_ADDR:%.+]])
19 // CHECK: [[SCALAR_ADDR:%.+]] = invoke dereferenceable(4) i32* @_ZN2St3getEv(%struct.St* [[TEMP_ST_ADDR]])
20 // CHECK: [[SCALAR_VAL:%.+]] = load atomic i32, i32* [[SCALAR_ADDR]] monotonic
21 // CHECK: store i32 [[SCALAR_VAL]], i32* @b
22 // CHECK: invoke void @_ZN2StD1Ev(%struct.St* [[TEMP_ST_ADDR]])
23#pragma omp atomic read
24 b = St().get();
Alexey Bataev112a7bf2015-04-23 07:56:25 +000025 // CHECK-DAG: invoke void @_ZN2StC1Ev(%struct.St* [[TEMP_ST_ADDR:%.+]])
26 // CHECK-DAG: [[SCALAR_ADDR:%.+]] = invoke dereferenceable(4) i32* @_ZN2St3getEv(%struct.St* [[TEMP_ST_ADDR]])
27 // CHECK-DAG: [[B_VAL:%.+]] = load i32, i32* @b
Alexey Bataev10fec572015-03-11 04:48:56 +000028 // CHECK: store atomic i32 [[B_VAL]], i32* [[SCALAR_ADDR]] monotonic
29 // CHECK: invoke void @_ZN2StD1Ev(%struct.St* [[TEMP_ST_ADDR]])
30#pragma omp atomic write
31 St().get() = b;
Alexey Bataevb4505a72015-03-30 05:20:59 +000032 // CHECK: invoke void @_ZN2StC1Ev(%struct.St* [[TEMP_ST_ADDR:%.+]])
33 // CHECK: [[SCALAR_ADDR:%.+]] = invoke dereferenceable(4) i32* @_ZN2St3getEv(%struct.St* [[TEMP_ST_ADDR]])
34 // CHECK: [[B_VAL:%.+]] = load i32, i32* @b
35 // CHECK: [[OLD_VAL:%.+]] = load atomic i32, i32* [[SCALAR_ADDR]] monotonic,
36 // CHECK: br label %[[OMP_UPDATE:.+]]
37 // CHECK: [[OMP_UPDATE]]
38 // CHECK: [[OLD_PHI_VAL:%.+]] = phi i32 [ [[OLD_VAL]], %{{.+}} ], [ [[NEW_OLD_VAL:%.+]], %[[OMP_UPDATE]] ]
39 // CHECK: [[NEW_VAL:%.+]] = srem i32 [[OLD_PHI_VAL]], [[B_VAL]]
Alexey Bataevf0ab5532015-05-15 08:36:34 +000040 // CHECK: store i32 [[NEW_VAL]], i32* [[TEMP:%.+]],
41 // CHECK: [[NEW_VAL:%.+]] = load i32, i32* [[TEMP]],
Alexey Bataevb4505a72015-03-30 05:20:59 +000042 // CHECK: [[RES:%.+]] = cmpxchg i32* [[SCALAR_ADDR]], i32 [[OLD_PHI_VAL]], i32 [[NEW_VAL]] monotonic monotonic
43 // CHECK: [[NEW_OLD_VAL]] = extractvalue { i32, i1 } [[RES]], 0
44 // CHECK: [[COND:%.+]] = extractvalue { i32, i1 } [[RES]], 1
45 // CHECK: br i1 [[COND]], label %[[OMP_DONE:.+]], label %[[OMP_UPDATE]]
46 // CHECK: [[OMP_DONE]]
47 // CHECK: invoke void @_ZN2StD1Ev(%struct.St* [[TEMP_ST_ADDR]])
48#pragma omp atomic
49 St().get() %= b;
Alexey Bataev5e018f92015-04-23 06:35:10 +000050 // CHECK: invoke void @_ZN2StC1Ev(%struct.St* [[TEMP_ST_ADDR:%.+]])
51 // CHECK: [[SCALAR_ADDR:%.+]] = invoke dereferenceable(4) i32* @_ZN2St3getEv(%struct.St* [[TEMP_ST_ADDR]])
52 // CHECK: [[B_VAL:%.+]] = load i32, i32* @b
53 // CHECK: [[OLD_VAL:%.+]] = load atomic i32, i32* [[SCALAR_ADDR]] monotonic,
54 // CHECK: br label %[[OMP_UPDATE:.+]]
55 // CHECK: [[OMP_UPDATE]]
56 // CHECK: [[OLD_PHI_VAL:%.+]] = phi i32 [ [[OLD_VAL]], %{{.+}} ], [ [[NEW_OLD_VAL:%.+]], %[[OMP_UPDATE]] ]
Alexey Bataevf0ab5532015-05-15 08:36:34 +000057 // CHECK: [[NEW_CALC_VAL:%.+]] = srem i32 [[OLD_PHI_VAL]], [[B_VAL]]
58 // CHECK: store i32 [[NEW_CALC_VAL]], i32* [[TEMP:%.+]],
59 // CHECK: [[NEW_VAL:%.+]] = load i32, i32* [[TEMP]],
Alexey Bataev5e018f92015-04-23 06:35:10 +000060 // CHECK: [[RES:%.+]] = cmpxchg i32* [[SCALAR_ADDR]], i32 [[OLD_PHI_VAL]], i32 [[NEW_VAL]] monotonic monotonic
61 // CHECK: [[NEW_OLD_VAL]] = extractvalue { i32, i1 } [[RES]], 0
62 // CHECK: [[COND:%.+]] = extractvalue { i32, i1 } [[RES]], 1
63 // CHECK: br i1 [[COND]], label %[[OMP_DONE:.+]], label %[[OMP_UPDATE]]
64 // CHECK: [[OMP_DONE]]
Alexey Bataevf0ab5532015-05-15 08:36:34 +000065 // CHECK: store i32 [[NEW_CALC_VAL]], i32* @a,
Alexey Bataev5e018f92015-04-23 06:35:10 +000066 // CHECK: invoke void @_ZN2StD1Ev(%struct.St* [[TEMP_ST_ADDR]])
67#pragma omp atomic capture
68 a = St().get() %= b;
Alexey Bataev10fec572015-03-11 04:48:56 +000069 }
70}
71
Alexey Bataev36bf0112015-03-10 05:15:26 +000072int &foo() { return a; }
73
74// TERM_DEBUG-LABEL: parallel_atomic
75void parallel_atomic() {
76#pragma omp parallel
77 {
78#pragma omp atomic read
79 // TERM_DEBUG-NOT: __kmpc_global_thread_num
80 // TERM_DEBUG: invoke {{.*}}foo{{.*}}()
81 // TERM_DEBUG: unwind label %[[TERM_LPAD:.+]],
Alexey Bataev10fec572015-03-11 04:48:56 +000082 // TERM_DEBUG: load atomic i32, i32* @{{.+}} monotonic, {{.*}}!dbg [[READ_LOC:![0-9]+]]
Alexey Bataev36bf0112015-03-10 05:15:26 +000083 foo() = a;
84#pragma omp atomic write
85 // TERM_DEBUG-NOT: __kmpc_global_thread_num
86 // TERM_DEBUG: invoke {{.*}}foo{{.*}}()
87 // TERM_DEBUG: unwind label %[[TERM_LPAD:.+]],
88 // TERM_DEBUG-NOT: __kmpc_global_thread_num
89 // TERM_DEBUG: store atomic i32 {{%.+}}, i32* @{{.+}} monotonic, {{.*}}!dbg [[WRITE_LOC:![0-9]+]]
Alexey Bataev36bf0112015-03-10 05:15:26 +000090 a = foo();
Alexey Bataevb4505a72015-03-30 05:20:59 +000091#pragma omp atomic update
92 // TERM_DEBUG-NOT: __kmpc_global_thread_num
93 // TERM_DEBUG: invoke {{.*}}foo{{.*}}()
94 // TERM_DEBUG: unwind label %[[TERM_LPAD:.+]],
95 // TERM_DEBUG-NOT: __kmpc_global_thread_num
96 // TERM_DEBUG: atomicrmw add i32* @{{.+}}, i32 %{{.+}} monotonic, {{.*}}!dbg [[UPDATE_LOC:![0-9]+]]
97 a += foo();
Alexey Bataev5e018f92015-04-23 06:35:10 +000098#pragma omp atomic capture
99 // TERM_DEBUG-NOT: __kmpc_global_thread_num
100 // TERM_DEBUG: invoke {{.*}}foo{{.*}}()
101 // TERM_DEBUG: unwind label %[[TERM_LPAD:.+]],
102 // TERM_DEBUG-NOT: __kmpc_global_thread_num
103 // TERM_DEBUG: [[OLD_VAL:%.+]] = atomicrmw add i32* @{{.+}}, i32 %{{.+}} monotonic, {{.*}}!dbg [[CAPTURE_LOC:![0-9]+]]
104 // TERM_DEBUG: store i32 [[OLD_VAL]], i32* @b,
105 {b = a; a += foo(); }
Alexey Bataev36bf0112015-03-10 05:15:26 +0000106 }
Alexey Bataevb4505a72015-03-30 05:20:59 +0000107 // TERM_DEBUG: [[TERM_LPAD]]
108 // TERM_DEBUG: call void @__clang_call_terminate
109 // TERM_DEBUG: unreachable
Alexey Bataev36bf0112015-03-10 05:15:26 +0000110}
Duncan P. N. Exon Smith9dd4e4e2015-04-29 16:40:08 +0000111// TERM_DEBUG-DAG: [[READ_LOC]] = !DILocation(line: [[@LINE-33]],
112// TERM_DEBUG-DAG: [[WRITE_LOC]] = !DILocation(line: [[@LINE-28]],
113// TERM_DEBUG-DAG: [[UPDATE_LOC]] = !DILocation(line: [[@LINE-22]],
114// TERM_DEBUG-DAG: [[CAPTURE_LOC]] = !DILocation(line: [[@LINE-16]],