| // RUN: %clang_tsan -O1 %s -o %t && %run %t 2>&1 | FileCheck %s |
| // CHECK-NOT: WARNING: ThreadSanitizer: data race |
| // CHECK-NOT: ThreadSanitizer WARNING: double lock |
| // CHECK-NOT: ThreadSanitizer WARNING: mutex unlock by another thread |
| // CHECK: OK |
| |
| #include <stdio.h> |
| #include <stdlib.h> |
| #include <pthread.h> |
| |
| pthread_mutex_t m; |
| pthread_cond_t c; |
| int x; |
| |
| void *thr1(void *p) { |
| int i; |
| |
| for (i = 0; i < 10; i += 2) { |
| pthread_mutex_lock(&m); |
| while (x != i) |
| pthread_cond_wait(&c, &m); |
| x = i + 1; |
| pthread_cond_signal(&c); |
| pthread_mutex_unlock(&m); |
| } |
| return 0; |
| } |
| |
| void *thr2(void *p) { |
| int i; |
| |
| for (i = 1; i < 10; i += 2) { |
| pthread_mutex_lock(&m); |
| while (x != i) |
| pthread_cond_wait(&c, &m); |
| x = i + 1; |
| pthread_mutex_unlock(&m); |
| pthread_cond_broadcast(&c); |
| } |
| return 0; |
| } |
| |
| int main() { |
| pthread_t th1, th2; |
| |
| pthread_mutex_init(&m, 0); |
| pthread_cond_init(&c, 0); |
| pthread_create(&th1, 0, thr1, 0); |
| pthread_create(&th2, 0, thr2, 0); |
| pthread_join(th1, 0); |
| pthread_join(th2, 0); |
| fprintf(stderr, "OK\n"); |
| } |