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