1 // Tests UAF detection where Allocate/Deallocate/Use
2 // happen in separate threads.
3 // RUN: %clang_hwasan %s -o %t && not %run %t 2>&1 | FileCheck %s
4 // REQUIRES: stable-runtime
10 #include <sanitizer/hwasan_interface.h>
15 void *Allocate(void *arg
) {
16 x
= (char*)malloc(10);
17 __sync_fetch_and_add(&state
, 1);
18 while (__sync_fetch_and_add(&state
, 0) != 3) {}
21 void *Deallocate(void *arg
) {
23 __sync_fetch_and_add(&state
, 1);
24 while (__sync_fetch_and_add(&state
, 0) != 3) {}
28 void *Use(void *arg
) {
30 // CHECK: ERROR: HWAddressSanitizer: tag-mismatch on address
31 // CHECK: WRITE of size 1 {{.*}} in thread T3
32 // CHECK: thread-uaf.c:[[@LINE-3]]
33 // CHECK: Cause: use-after-free
34 // CHECK: freed by thread T2 here
35 // CHECK: in Deallocate
36 // CHECK: previously allocated here:
38 // CHECK-DAG: Thread: T2 0x
39 // CHECK-DAG: Thread: T3 0x
40 // CHECK-DAG: Thread: T0 0x
41 // CHECK-DAG: Thread: T1 0x
42 __sync_fetch_and_add(&state
, 1);
47 __hwasan_enable_allocator_tagging();
50 pthread_create(&t1
, NULL
, Allocate
, NULL
);
51 while (__sync_fetch_and_add(&state
, 0) != 1) {}
52 pthread_create(&t2
, NULL
, Deallocate
, NULL
);
53 while (__sync_fetch_and_add(&state
, 0) != 2) {}
54 pthread_create(&t3
, NULL
, Use
, NULL
);
56 pthread_join(t1
, NULL
);
57 pthread_join(t2
, NULL
);
58 pthread_join(t3
, NULL
);