1 | // RUN: %clangxx_tsan -O1 %s -o %t && %run %t |
---|---|
2 | |
3 | // Data race randomly triggered. |
4 | // UNSUPPORTED: target={{.*netbsd.*}} |
5 | |
6 | // Make sure TSan doesn't deadlock on a file stream lock at program shutdown. |
7 | // See https://github.com/google/sanitizers/issues/454 |
8 | |
9 | // https://github.com/google/sanitizers/issues/1733 |
10 | // UNSUPPORTED: glibc-2.38 |
11 | |
12 | #ifdef __FreeBSD__ |
13 | #define _WITH_GETLINE // to declare getline() |
14 | #endif |
15 | |
16 | #include <pthread.h> |
17 | #include <stdio.h> |
18 | #include <unistd.h> |
19 | |
20 | void *thread(void *unused) { |
21 | char *line = NULL; |
22 | size_t size; |
23 | int fd[2]; |
24 | pipe(pipedes: fd); |
25 | // Forge a non-standard stream to make sure it's not closed. |
26 | FILE *stream = fdopen(fd: fd[0], modes: "r"); |
27 | while (1) { |
28 | volatile int res = getline(lineptr: &line, n: &size, stream: stream); |
29 | (void)res; |
30 | } |
31 | return NULL; |
32 | } |
33 | |
34 | int main() { |
35 | pthread_t t; |
36 | pthread_attr_t a; |
37 | pthread_attr_init(attr: &a); |
38 | pthread_attr_setdetachstate(attr: &a, PTHREAD_CREATE_DETACHED); |
39 | pthread_create(newthread: &t, attr: &a, start_routine: thread, NULL); |
40 | pthread_attr_destroy(attr: &a); |
41 | fprintf(stderr, format: "DONE\n"); |
42 | return 0; |
43 | // ThreadSanitizer used to hang here because of a deadlock on a file stream. |
44 | } |
45 | |
46 | // CHECK: DONE |
47 |