1bcaeed49SFangrui Song // RUN: %clangxx_tsan %s %link_libcxx_tsan -o %t
2bcaeed49SFangrui Song // RUN: %run %t 2>&1 | FileCheck %s --implicit-check-not='ThreadSanitizer'
3bcaeed49SFangrui Song
4bcaeed49SFangrui Song #include <dispatch/dispatch.h>
5bcaeed49SFangrui Song
6*a097c4ceSLouis Dionne #include <atomic>
7bcaeed49SFangrui Song #include <cstdio>
8*a097c4ceSLouis Dionne #include <memory>
9bcaeed49SFangrui Song
10*a097c4ceSLouis Dionne std::atomic<long> destructor_counter(0);
11bcaeed49SFangrui Song
12bcaeed49SFangrui Song struct MyStruct {
~MyStructMyStruct13bcaeed49SFangrui Song virtual ~MyStruct() {
14bcaeed49SFangrui Song usleep(10000);
15*a097c4ceSLouis Dionne std::atomic_fetch_add_explicit(&destructor_counter, 1, std::memory_order_relaxed);
16bcaeed49SFangrui Song }
17bcaeed49SFangrui Song };
18bcaeed49SFangrui Song
main(int argc,const char * argv[])19bcaeed49SFangrui Song int main(int argc, const char *argv[]) {
20*a097c4ceSLouis Dionne std::fprintf(stderr, "Hello world.\n");
21bcaeed49SFangrui Song
22bcaeed49SFangrui Song dispatch_queue_t q = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);
23bcaeed49SFangrui Song dispatch_group_t g = dispatch_group_create();
24bcaeed49SFangrui Song
25bcaeed49SFangrui Song for (int i = 0; i < 100; i++) {
26bcaeed49SFangrui Song std::shared_ptr<MyStruct> shared(new MyStruct());
27bcaeed49SFangrui Song
28bcaeed49SFangrui Song dispatch_group_async(g, q, ^{
29bcaeed49SFangrui Song shared.get(); // just to make sure the object is captured by the block
30bcaeed49SFangrui Song });
31bcaeed49SFangrui Song }
32bcaeed49SFangrui Song
33bcaeed49SFangrui Song dispatch_group_wait(g, DISPATCH_TIME_FOREVER);
34bcaeed49SFangrui Song
35bcaeed49SFangrui Song if (destructor_counter != 100) {
36*a097c4ceSLouis Dionne std::abort();
37bcaeed49SFangrui Song }
38bcaeed49SFangrui Song
39*a097c4ceSLouis Dionne std::fprintf(stderr, "Done.\n");
40bcaeed49SFangrui Song }
41bcaeed49SFangrui Song
42bcaeed49SFangrui Song // CHECK: Hello world.
43bcaeed49SFangrui Song // CHECK: Done.
44