1bcaeed49SFangrui Song // RUN: %clangxx_tsan -O1 %s -o %t
2bcaeed49SFangrui Song // RUN: %deflake %run %t 2>&1 | FileCheck %s
3bcaeed49SFangrui Song 
4bcaeed49SFangrui Song #include <pthread.h>
5bcaeed49SFangrui Song #include <stdio.h>
6bcaeed49SFangrui Song #include <stdlib.h>
7bcaeed49SFangrui Song #include <string.h>
8bcaeed49SFangrui Song 
9bcaeed49SFangrui Song #include "../test.h"
10bcaeed49SFangrui Song 
11bcaeed49SFangrui Song extern "C" {
12bcaeed49SFangrui Song void __tsan_on_report(void *report);
13bcaeed49SFangrui Song int __tsan_get_report_loc(void *report, unsigned long idx, const char **type,
14bcaeed49SFangrui Song                           void **addr, void **start,
15bcaeed49SFangrui Song                           unsigned long *size, int *tid, int *fd,
16bcaeed49SFangrui Song                           int *suppressable, void **trace,
17bcaeed49SFangrui Song                           unsigned long trace_size);
18bcaeed49SFangrui Song int __tsan_get_report_loc_object_type(void *report, unsigned long idx,
19bcaeed49SFangrui Song                                       const char **object_type);
20bcaeed49SFangrui Song }
21bcaeed49SFangrui Song 
Thread(void * arg)22bcaeed49SFangrui Song void *Thread(void *arg) {
23bcaeed49SFangrui Song   barrier_wait(&barrier);
24bcaeed49SFangrui Song   *((long *)arg) = 42;
25bcaeed49SFangrui Song   return NULL;
26bcaeed49SFangrui Song }
27bcaeed49SFangrui Song 
main()28bcaeed49SFangrui Song int main() {
29bcaeed49SFangrui Song   barrier_init(&barrier, 2);
30bcaeed49SFangrui Song   void *tag = __tsan_external_register_tag("MyObject");
31bcaeed49SFangrui Song   long *obj = (long *)malloc(sizeof(long));
32bcaeed49SFangrui Song   fprintf(stderr, "obj = %p\n", obj);
33bcaeed49SFangrui Song   // CHECK: obj = [[ADDR:0x[0-9a-f]+]]
34bcaeed49SFangrui Song   __tsan_external_assign_tag(obj, tag);
35bcaeed49SFangrui Song 
36bcaeed49SFangrui Song   pthread_t t;
37bcaeed49SFangrui Song   pthread_create(&t, 0, Thread, obj);
38bcaeed49SFangrui Song   *obj = 41;
39bcaeed49SFangrui Song   barrier_wait(&barrier);
40bcaeed49SFangrui Song   pthread_join(t, 0);
41bcaeed49SFangrui Song   fprintf(stderr, "Done.\n");
42bcaeed49SFangrui Song   return 0;
43bcaeed49SFangrui Song }
44bcaeed49SFangrui Song 
45*e69d50d9SDmitry Vyukov __attribute__((disable_sanitizer_instrumentation)) void
__tsan_on_report(void * report)46*e69d50d9SDmitry Vyukov __tsan_on_report(void *report) {
47bcaeed49SFangrui Song   const char *type;
48bcaeed49SFangrui Song   void *addr;
49bcaeed49SFangrui Song   void *start;
50bcaeed49SFangrui Song   unsigned long size;
51bcaeed49SFangrui Song   int tid, fd, suppressable;
52bcaeed49SFangrui Song   void *trace[16] = {0};
53bcaeed49SFangrui Song   __tsan_get_report_loc(report, 0, &type, &addr, &start, &size, &tid, &fd,
54bcaeed49SFangrui Song                         &suppressable, trace, 16);
55bcaeed49SFangrui Song   fprintf(stderr, "type = %s, start = %p, size = %ld\n", type, start, size);
56bcaeed49SFangrui Song   // CHECK: type = heap, start = [[ADDR]], size = 8
57bcaeed49SFangrui Song 
58bcaeed49SFangrui Song   const char *object_type;
59bcaeed49SFangrui Song   __tsan_get_report_loc_object_type(report, 0, &object_type);
60bcaeed49SFangrui Song   fprintf(stderr, "object_type = %s\n", object_type);
61bcaeed49SFangrui Song   // CHECK: object_type = MyObject
62bcaeed49SFangrui Song }
63bcaeed49SFangrui Song 
64bcaeed49SFangrui Song // CHECK: Done.
65bcaeed49SFangrui Song // CHECK: ThreadSanitizer: reported 1 warnings
66