Please use GitHub pull requests for new patches. Avoid migrating existing patches. Phabricator shutdown timeline
Changeset View
Changeset View
Standalone View
Standalone View
clang/test/OpenMP/atomic_codegen.cpp
Show All 19 Lines | struct St { | ||||
int &get() { return a; } | int &get() { return a; } | ||||
}; | }; | ||||
// CHECK-LABEL: parallel_atomic_ewc | // CHECK-LABEL: parallel_atomic_ewc | ||||
void parallel_atomic_ewc() { | void parallel_atomic_ewc() { | ||||
St s; | St s; | ||||
#pragma omp parallel | #pragma omp parallel | ||||
{ | { | ||||
// CHECK: invoke void @_ZN2StC1Ev(ptr {{[^,]*}} [[TEMP_ST_ADDR:%.+]]) | // CHECK: call unwindabort void @_ZN2StC1Ev(ptr {{[^,]*}} [[TEMP_ST_ADDR:%.+]]) | ||||
// CHECK: [[SCALAR_ADDR:%.+]] = invoke noundef nonnull align 4 dereferenceable(4) ptr @_ZN2St3getEv(ptr {{[^,]*}} [[TEMP_ST_ADDR]]) | // CHECK: [[SCALAR_ADDR:%.+]] = call unwindabort noundef nonnull align 4 dereferenceable(4) ptr @_ZN2St3getEv(ptr {{[^,]*}} [[TEMP_ST_ADDR]]) | ||||
// CHECK: [[SCALAR_VAL:%.+]] = load atomic i32, ptr [[SCALAR_ADDR]] monotonic, align 4 | // CHECK: [[SCALAR_VAL:%.+]] = load atomic i32, ptr [[SCALAR_ADDR]] monotonic, align 4 | ||||
// CHECK: store i32 [[SCALAR_VAL]], ptr @b | // CHECK: store i32 [[SCALAR_VAL]], ptr @b | ||||
// CHECK98: invoke void @_ZN2StD1Ev(ptr {{[^,]*}} [[TEMP_ST_ADDR]]) | // CHECK98: call unwindabort void @_ZN2StD1Ev(ptr {{[^,]*}} [[TEMP_ST_ADDR]]) | ||||
// CHECK11: call void @_ZN2StD1Ev(ptr {{[^,]*}} [[TEMP_ST_ADDR]]) | // CHECK11: call void @_ZN2StD1Ev(ptr {{[^,]*}} [[TEMP_ST_ADDR]]) | ||||
#pragma omp atomic read | #pragma omp atomic read | ||||
b = St().get(); | b = St().get(); | ||||
// CHECK-DAG: invoke void @_ZN2StC1Ev(ptr {{[^,]*}} [[TEMP_ST_ADDR:%.+]]) | // CHECK-DAG: call unwindabort void @_ZN2StC1Ev(ptr {{[^,]*}} [[TEMP_ST_ADDR:%.+]]) | ||||
// CHECK-DAG: [[SCALAR_ADDR:%.+]] = invoke noundef nonnull align 4 dereferenceable(4) ptr @_ZN2St3getEv(ptr {{[^,]*}} [[TEMP_ST_ADDR]]) | // CHECK-DAG: [[SCALAR_ADDR:%.+]] = call unwindabort noundef nonnull align 4 dereferenceable(4) ptr @_ZN2St3getEv(ptr {{[^,]*}} [[TEMP_ST_ADDR]]) | ||||
// CHECK-DAG: [[B_VAL:%.+]] = load i32, ptr @b | // CHECK-DAG: [[B_VAL:%.+]] = load i32, ptr @b | ||||
// CHECK: store atomic i32 [[B_VAL]], ptr [[SCALAR_ADDR]] monotonic, align 4 | // CHECK: store atomic i32 [[B_VAL]], ptr [[SCALAR_ADDR]] monotonic, align 4 | ||||
// CHECK: {{invoke|call}} void @_ZN2StD1Ev(ptr {{[^,]*}} [[TEMP_ST_ADDR]]) | // CHECK: {{call unwindabort|call}} void @_ZN2StD1Ev(ptr {{[^,]*}} [[TEMP_ST_ADDR]]) | ||||
#pragma omp atomic write | #pragma omp atomic write | ||||
St().get() = b; | St().get() = b; | ||||
// CHECK: invoke void @_ZN2StC1Ev(ptr {{[^,]*}} [[TEMP_ST_ADDR:%.+]]) | // CHECK: call unwindabort void @_ZN2StC1Ev(ptr {{[^,]*}} [[TEMP_ST_ADDR:%.+]]) | ||||
// CHECK: [[SCALAR_ADDR:%.+]] = invoke noundef nonnull align 4 dereferenceable(4) ptr @_ZN2St3getEv(ptr {{[^,]*}} [[TEMP_ST_ADDR]]) | // CHECK: [[SCALAR_ADDR:%.+]] = call unwindabort noundef nonnull align 4 dereferenceable(4) ptr @_ZN2St3getEv(ptr {{[^,]*}} [[TEMP_ST_ADDR]]) | ||||
// CHECK: [[B_VAL:%.+]] = load i32, ptr @b | // CHECK: [[B_VAL:%.+]] = load i32, ptr @b | ||||
// CHECK: [[OLD_VAL:%.+]] = load atomic i32, ptr [[SCALAR_ADDR]] monotonic, align 4 | // CHECK: [[OLD_VAL:%.+]] = load atomic i32, ptr [[SCALAR_ADDR]] monotonic, align 4 | ||||
// CHECK: br label %[[OMP_UPDATE:.+]] | // CHECK: br label %[[OMP_UPDATE:.+]] | ||||
// CHECK: [[OMP_UPDATE]] | // CHECK: [[OMP_UPDATE]] | ||||
// CHECK: [[OLD_PHI_VAL:%.+]] = phi i32 [ [[OLD_VAL]], %{{.+}} ], [ [[NEW_OLD_VAL:%.+]], %[[OMP_UPDATE]] ] | // CHECK: [[OLD_PHI_VAL:%.+]] = phi i32 [ [[OLD_VAL]], %{{.+}} ], [ [[NEW_OLD_VAL:%.+]], %[[OMP_UPDATE]] ] | ||||
// CHECK: [[NEW_VAL:%.+]] = srem i32 [[OLD_PHI_VAL]], [[B_VAL]] | // CHECK: [[NEW_VAL:%.+]] = srem i32 [[OLD_PHI_VAL]], [[B_VAL]] | ||||
// CHECK: store i32 [[NEW_VAL]], ptr [[TEMP:%.+]], | // CHECK: store i32 [[NEW_VAL]], ptr [[TEMP:%.+]], | ||||
// CHECK: [[NEW_VAL:%.+]] = load i32, ptr [[TEMP]], | // CHECK: [[NEW_VAL:%.+]] = load i32, ptr [[TEMP]], | ||||
// CHECK: [[RES:%.+]] = cmpxchg ptr [[SCALAR_ADDR]], i32 [[OLD_PHI_VAL]], i32 [[NEW_VAL]] monotonic monotonic, align 4 | // CHECK: [[RES:%.+]] = cmpxchg ptr [[SCALAR_ADDR]], i32 [[OLD_PHI_VAL]], i32 [[NEW_VAL]] monotonic monotonic, align 4 | ||||
// CHECK: [[NEW_OLD_VAL]] = extractvalue { i32, i1 } [[RES]], 0 | // CHECK: [[NEW_OLD_VAL]] = extractvalue { i32, i1 } [[RES]], 0 | ||||
// CHECK: [[COND:%.+]] = extractvalue { i32, i1 } [[RES]], 1 | // CHECK: [[COND:%.+]] = extractvalue { i32, i1 } [[RES]], 1 | ||||
// CHECK: br i1 [[COND]], label %[[OMP_DONE:.+]], label %[[OMP_UPDATE]] | // CHECK: br i1 [[COND]], label %[[OMP_DONE:.+]], label %[[OMP_UPDATE]] | ||||
// CHECK: [[OMP_DONE]] | // CHECK: [[OMP_DONE]] | ||||
// CHECK: {{invoke|call}} void @_ZN2StD1Ev(ptr {{[^,]*}} [[TEMP_ST_ADDR]]) | // CHECK: {{call unwindabort|call}} void @_ZN2StD1Ev(ptr {{[^,]*}} [[TEMP_ST_ADDR]]) | ||||
#pragma omp atomic | #pragma omp atomic | ||||
St().get() %= b; | St().get() %= b; | ||||
#pragma omp atomic hint(6) | #pragma omp atomic hint(6) | ||||
s.field++; | s.field++; | ||||
// CHECK: invoke void @_ZN2StC1Ev(ptr {{[^,]*}} [[TEMP_ST_ADDR:%.+]]) | // CHECK: call unwindabort void @_ZN2StC1Ev(ptr {{[^,]*}} [[TEMP_ST_ADDR:%.+]]) | ||||
// CHECK: [[SCALAR_ADDR:%.+]] = invoke noundef nonnull align 4 dereferenceable(4) ptr @_ZN2St3getEv(ptr {{[^,]*}} [[TEMP_ST_ADDR]]) | // CHECK: [[SCALAR_ADDR:%.+]] = call unwindabort noundef nonnull align 4 dereferenceable(4) ptr @_ZN2St3getEv(ptr {{[^,]*}} [[TEMP_ST_ADDR]]) | ||||
// CHECK: [[B_VAL:%.+]] = load i32, ptr @b | // CHECK: [[B_VAL:%.+]] = load i32, ptr @b | ||||
// CHECK: [[OLD_VAL:%.+]] = load atomic i32, ptr [[SCALAR_ADDR]] monotonic, align 4 | // CHECK: [[OLD_VAL:%.+]] = load atomic i32, ptr [[SCALAR_ADDR]] monotonic, align 4 | ||||
// CHECK: br label %[[OMP_UPDATE:.+]] | // CHECK: br label %[[OMP_UPDATE:.+]] | ||||
// CHECK: [[OMP_UPDATE]] | // CHECK: [[OMP_UPDATE]] | ||||
// CHECK: [[OLD_PHI_VAL:%.+]] = phi i32 [ [[OLD_VAL]], %{{.+}} ], [ [[NEW_OLD_VAL:%.+]], %[[OMP_UPDATE]] ] | // CHECK: [[OLD_PHI_VAL:%.+]] = phi i32 [ [[OLD_VAL]], %{{.+}} ], [ [[NEW_OLD_VAL:%.+]], %[[OMP_UPDATE]] ] | ||||
// CHECK: [[NEW_CALC_VAL:%.+]] = srem i32 [[OLD_PHI_VAL]], [[B_VAL]] | // CHECK: [[NEW_CALC_VAL:%.+]] = srem i32 [[OLD_PHI_VAL]], [[B_VAL]] | ||||
// CHECK: store i32 [[NEW_CALC_VAL]], ptr [[TEMP:%.+]], | // CHECK: store i32 [[NEW_CALC_VAL]], ptr [[TEMP:%.+]], | ||||
// CHECK: [[NEW_VAL:%.+]] = load i32, ptr [[TEMP]], | // CHECK: [[NEW_VAL:%.+]] = load i32, ptr [[TEMP]], | ||||
// CHECK: [[RES:%.+]] = cmpxchg ptr [[SCALAR_ADDR]], i32 [[OLD_PHI_VAL]], i32 [[NEW_VAL]] monotonic monotonic, align 4 | // CHECK: [[RES:%.+]] = cmpxchg ptr [[SCALAR_ADDR]], i32 [[OLD_PHI_VAL]], i32 [[NEW_VAL]] monotonic monotonic, align 4 | ||||
// CHECK: [[NEW_OLD_VAL]] = extractvalue { i32, i1 } [[RES]], 0 | // CHECK: [[NEW_OLD_VAL]] = extractvalue { i32, i1 } [[RES]], 0 | ||||
// CHECK: [[COND:%.+]] = extractvalue { i32, i1 } [[RES]], 1 | // CHECK: [[COND:%.+]] = extractvalue { i32, i1 } [[RES]], 1 | ||||
// CHECK: br i1 [[COND]], label %[[OMP_DONE:.+]], label %[[OMP_UPDATE]] | // CHECK: br i1 [[COND]], label %[[OMP_DONE:.+]], label %[[OMP_UPDATE]] | ||||
// CHECK: [[OMP_DONE]] | // CHECK: [[OMP_DONE]] | ||||
// CHECK: store i32 [[NEW_CALC_VAL]], ptr @a, | // CHECK: store i32 [[NEW_CALC_VAL]], ptr @a, | ||||
// CHECK: {{invoke|call}} void @_ZN2StD1Ev(ptr {{[^,]*}} [[TEMP_ST_ADDR]]) | // CHECK: {{call unwindabort|call}} void @_ZN2StD1Ev(ptr {{[^,]*}} [[TEMP_ST_ADDR]]) | ||||
#pragma omp atomic capture | #pragma omp atomic capture | ||||
a = St().get() %= b; | a = St().get() %= b; | ||||
} | } | ||||
} | } | ||||
int &foo() { extern void mayThrow(); mayThrow(); return a; } | int &foo() { extern void mayThrow(); mayThrow(); return a; } | ||||
// TERM_DEBUG-LABEL: parallel_atomic | // TERM_DEBUG-LABEL: parallel_atomic | ||||
void parallel_atomic() { | void parallel_atomic() { | ||||
#pragma omp parallel | #pragma omp parallel | ||||
{ | { | ||||
#pragma omp atomic read | #pragma omp atomic read | ||||
// TERM_DEBUG-NOT: __kmpc_global_thread_num | // TERM_DEBUG-NOT: __kmpc_global_thread_num | ||||
// TERM_DEBUG: invoke {{.*}}foo{{.*}}() | // TERM_DEBUG: call unwindabort {{.*}}foo{{.*}}() | ||||
// TERM_DEBUG: unwind label %[[TERM_LPAD:.+]], | |||||
// TERM_DEBUG: load atomic i32, ptr @{{.+}} monotonic, align 4, !dbg [[READ_LOC:![0-9]+]] | // TERM_DEBUG: load atomic i32, ptr @{{.+}} monotonic, align 4, !dbg [[READ_LOC:![0-9]+]] | ||||
foo() = a; | foo() = a; | ||||
#pragma omp atomic write | #pragma omp atomic write | ||||
// TERM_DEBUG-NOT: __kmpc_global_thread_num | // TERM_DEBUG-NOT: __kmpc_global_thread_num | ||||
// TERM_DEBUG: invoke {{.*}}foo{{.*}}() | // TERM_DEBUG: call unwindabort {{.*}}foo{{.*}}() | ||||
// TERM_DEBUG: unwind label %[[TERM_LPAD:.+]], | |||||
// TERM_DEBUG-NOT: __kmpc_global_thread_num | // TERM_DEBUG-NOT: __kmpc_global_thread_num | ||||
// TERM_DEBUG: store atomic i32 {{%.+}}, ptr @{{.+}} monotonic, align 4, !dbg [[WRITE_LOC:![0-9]+]] | // TERM_DEBUG: store atomic i32 {{%.+}}, ptr @{{.+}} monotonic, align 4, !dbg [[WRITE_LOC:![0-9]+]] | ||||
a = foo(); | a = foo(); | ||||
#pragma omp atomic update | #pragma omp atomic update | ||||
// TERM_DEBUG-NOT: __kmpc_global_thread_num | // TERM_DEBUG-NOT: __kmpc_global_thread_num | ||||
// TERM_DEBUG: invoke {{.*}}foo{{.*}}() | // TERM_DEBUG: call unwindabort {{.*}}foo{{.*}}() | ||||
// TERM_DEBUG: unwind label %[[TERM_LPAD:.+]], | |||||
// TERM_DEBUG-NOT: __kmpc_global_thread_num | // TERM_DEBUG-NOT: __kmpc_global_thread_num | ||||
// TERM_DEBUG: atomicrmw add ptr @{{.+}}, i32 %{{.+}} monotonic, align 4, !dbg [[UPDATE_LOC:![0-9]+]] | // TERM_DEBUG: atomicrmw add ptr @{{.+}}, i32 %{{.+}} monotonic, align 4, !dbg [[UPDATE_LOC:![0-9]+]] | ||||
a += foo(); | a += foo(); | ||||
#pragma omp atomic capture | #pragma omp atomic capture | ||||
// TERM_DEBUG-NOT: __kmpc_global_thread_num | // TERM_DEBUG-NOT: __kmpc_global_thread_num | ||||
// TERM_DEBUG: invoke {{.*}}foo{{.*}}() | // TERM_DEBUG: call unwindabort {{.*}}foo{{.*}}() | ||||
// TERM_DEBUG: unwind label %[[TERM_LPAD:.+]], | |||||
// TERM_DEBUG-NOT: __kmpc_global_thread_num | // TERM_DEBUG-NOT: __kmpc_global_thread_num | ||||
// TERM_DEBUG: [[OLD_VAL:%.+]] = atomicrmw add ptr @{{.+}}, i32 %{{.+}} monotonic, align 4, !dbg [[CAPTURE_LOC:![0-9]+]] | // TERM_DEBUG: [[OLD_VAL:%.+]] = atomicrmw add ptr @{{.+}}, i32 %{{.+}} monotonic, align 4, !dbg [[CAPTURE_LOC:![0-9]+]] | ||||
// TERM_DEBUG: store i32 [[OLD_VAL]], ptr @b, | // TERM_DEBUG: store i32 [[OLD_VAL]], ptr @b, | ||||
{b = a; a += foo(); } | {b = a; a += foo(); } | ||||
} | } | ||||
// TERM_DEBUG: [[TERM_LPAD]] | |||||
// TERM_DEBUG: call void @__clang_call_terminate | |||||
// TERM_DEBUG: unreachable | |||||
} | } | ||||
// TERM_DEBUG-DAG: [[READ_LOC]] = !DILocation(line: [[@LINE-28]], | // TERM_DEBUG-DAG: [[READ_LOC]] = !DILocation(line: [[@LINE-22]], | ||||
// TERM_DEBUG-DAG: [[WRITE_LOC]] = !DILocation(line: [[@LINE-22]], | // TERM_DEBUG-DAG: [[WRITE_LOC]] = !DILocation(line: [[@LINE-17]], | ||||
// TERM_DEBUG-DAG: [[UPDATE_LOC]] = !DILocation(line: [[@LINE-16]], | // TERM_DEBUG-DAG: [[UPDATE_LOC]] = !DILocation(line: [[@LINE-12]], | ||||
// TERM_DEBUG-DAG: [[CAPTURE_LOC]] = !DILocation(line: [[@LINE-9]], | // TERM_DEBUG-DAG: [[CAPTURE_LOC]] = !DILocation(line: [[@LINE-6]], |