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
9 #include <sanitizer/hwasan_interface.h>
14 void *Allocate(void *arg
) {
15 x
= (char*)malloc(10);
16 __sync_fetch_and_add(&state
, 1);
17 while (__sync_fetch_and_add(&state
, 0) != 3) {}
20 void *Deallocate(void *arg
) {
22 __sync_fetch_and_add(&state
, 1);
23 while (__sync_fetch_and_add(&state
, 0) != 3) {}
27 void *Use(void *arg
) {
29 // CHECK: ERROR: HWAddressSanitizer: tag-mismatch on address
30 // CHECK: WRITE of size 1 {{.*}} in thread T3
31 // CHECK: thread-uaf.c:[[@LINE-3]]
32 // CHECK: Cause: use-after-free
33 // CHECK: freed by thread T2 here
34 // CHECK: in Deallocate
35 // CHECK: previously allocated by thread T1 here:
37 // CHECK-DAG: Thread: T2 0x
38 // CHECK-DAG: Thread: T3 0x
39 // CHECK-DAG: Thread: T0 0x
40 // CHECK-DAG: Thread: T1 0x
41 __sync_fetch_and_add(&state
, 1);
46 __hwasan_enable_allocator_tagging();
49 pthread_create(&t1
, NULL
, Allocate
, NULL
);
50 while (__sync_fetch_and_add(&state
, 0) != 1) {}
51 pthread_create(&t2
, NULL
, Deallocate
, NULL
);
52 while (__sync_fetch_and_add(&state
, 0) != 2) {}
53 pthread_create(&t3
, NULL
, Use
, NULL
);
55 pthread_join(t1
, NULL
);
56 pthread_join(t2
, NULL
);
57 pthread_join(t3
, NULL
);