aboutsummaryrefslogtreecommitdiff
path: root/compiler-rt/test/tsan/Darwin/write-interpose.c
blob: cbd9a0867c982f241834438e898726331452af84 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
// Test that dylibs interposing write, and then calling functions intercepted
// by TSan don't deadlock (self-lock)

// RUN: %clang_tsan %s -o %t
// RUN: %clang_tsan %s -o %t.dylib -fno-sanitize=thread -dynamiclib -DSHARED_LIB

// Note that running the below command with out `lock_during_write` should
// deadlock (self-lock)
// RUN: env DYLD_INSERT_LIBRARIES=%t.dylib TSAN_OPTIONS=verbosity=2:lock_during_write=disable_for_current_process %run %t 2>&1 | FileCheck %s

#include <stdio.h>

#if defined(SHARED_LIB)

// dylib implementation - interposes write() calls
#  include <os/lock.h>
#  include <unistd.h>

struct interpose_substitution {
  const void *replacement;
  const void *original;
};

#  define INTERPOSE(replacement, original)                                     \
    __attribute__((used)) static const struct interpose_substitution           \
        substitution_##original[]                                              \
        __attribute__((section("__DATA, __interpose"))) = {                    \
            {(const void *)(replacement), (const void *)(original)}}

static ssize_t my_write(int fd, const void *buf, size_t count) {
  struct os_unfair_lock_s lock = OS_UNFAIR_LOCK_INIT;
  os_unfair_lock_lock(&lock);
  printf("Interposed write called: fd=%d, count=%zu\n", fd, count);
  ssize_t res = write(fd, buf, count);
  os_unfair_lock_unlock(&lock);
  return res;
}
INTERPOSE(my_write, write);

#else // defined(SHARED_LIB)

int main() {
  printf("Write test completed\n");
  return 0;
}

#endif // defined(SHARED_LIB)

// CHECK: Interposed write called: fd={{[0-9]+}}, count={{[0-9]+}}
// CHECK: Write test completed