2015-11-20 01:28:46 +08:00
|
|
|
// RUN: %clangxx_tsan -O1 %s -o %t && %env_tsan_opts=atexit_sleep_ms=50 %run %t 2>&1 | FileCheck %s
|
2015-01-21 21:50:02 +08:00
|
|
|
#include "test.h"
|
2014-01-24 20:33:35 +08:00
|
|
|
#include <errno.h>
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/wait.h>
|
|
|
|
|
|
|
|
int counter;
|
|
|
|
|
|
|
|
static void *incrementer(void *p) {
|
|
|
|
for (;;)
|
|
|
|
__sync_fetch_and_add(&counter, 1);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void *watchdog(void *p) {
|
2015-01-21 21:50:02 +08:00
|
|
|
sleep(100); // is not intended to exit
|
2014-01-24 20:33:35 +08:00
|
|
|
fprintf(stderr, "timed out after 100 seconds\n");
|
|
|
|
exit(1);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int main() {
|
2015-01-21 21:50:02 +08:00
|
|
|
barrier_init(&barrier, 2);
|
2014-01-24 20:33:35 +08:00
|
|
|
pthread_t th1, th2;
|
|
|
|
pthread_create(&th1, 0, incrementer, 0);
|
|
|
|
pthread_create(&th2, 0, watchdog, 0);
|
|
|
|
for (int i = 0; i < 10; i++) {
|
|
|
|
switch (fork()) {
|
|
|
|
default: // parent
|
|
|
|
while (wait(0) < 0) {}
|
|
|
|
fprintf(stderr, ".");
|
|
|
|
break;
|
|
|
|
case 0: // child
|
|
|
|
__sync_fetch_and_add(&counter, 1);
|
|
|
|
exit(0);
|
|
|
|
break;
|
|
|
|
case -1: // error
|
|
|
|
fprintf(stderr, "failed to fork (%d)\n", errno);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
fprintf(stderr, "OK\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
// CHECK: OK
|
|
|
|
|