| // RUN: %clang_tsan -O1 %s -o %t && %run %t 2>&1 | FileCheck %s |
| // CHECK-NOT: ThreadSanitizer: data race |
| // CHECK: DONE |
| |
| // pthread barriers are not available on OS X |
| // UNSUPPORTED: darwin |
| |
| #include <stdio.h> |
| #include <stdlib.h> |
| #include <pthread.h> |
| #include <unistd.h> |
| |
| const int kSize = 4; |
| volatile int kIter = 10; // prevent unwinding |
| int data[2][kSize]; |
| pthread_barrier_t barrier; |
| |
| void *thr(void *p) { |
| int idx = (int)(long)p; |
| for (int i = 0; i < kIter; i++) { |
| int *prev = data[i % 2]; |
| int *curr = data[(i + 1) % 2]; |
| int left = idx - 1 >= 0 ? prev[idx - 1] : 0; |
| int right = idx + 1 < kSize ? prev[idx + 1] : 0; |
| curr[idx] = (left + right) / 2; |
| pthread_barrier_wait(&barrier); |
| } |
| return 0; |
| } |
| |
| int main() { |
| pthread_barrier_init(&barrier, 0, kSize); |
| pthread_t th[kSize]; |
| for (int i = 0; i < kSize; i++) |
| pthread_create(&th[i], 0, thr, (void*)(long)i); |
| for (int i = 0; i < kSize; i++) |
| pthread_join(th[i], 0); |
| pthread_barrier_destroy(&barrier); |
| fprintf(stderr, "DONE\n"); |
| } |