10984b8deSDmitry Vyukov // RUN: %clang_tsan -O1 %s -o %t && %run %t 2>&1 | FileCheck %s
20984b8deSDmitry Vyukov
30984b8deSDmitry Vyukov // Test that mmap does not return unexpected addresses
40984b8deSDmitry Vyukov // (the check is in the interceptor).
50984b8deSDmitry Vyukov
60984b8deSDmitry Vyukov #include <fcntl.h>
70984b8deSDmitry Vyukov #include <stddef.h>
80984b8deSDmitry Vyukov #include <stdio.h>
90984b8deSDmitry Vyukov #include <stdlib.h>
100984b8deSDmitry Vyukov #include <sys/mman.h>
110984b8deSDmitry Vyukov #include <sys/stat.h>
120984b8deSDmitry Vyukov #include <sys/types.h>
130984b8deSDmitry Vyukov #include <unistd.h>
140984b8deSDmitry Vyukov
main()150984b8deSDmitry Vyukov int main() {
160984b8deSDmitry Vyukov int fd = open("/dev/zero", O_RDWR);
170984b8deSDmitry Vyukov if (fd == -1) perror("open(/dev/zero)"), exit(1);
180984b8deSDmitry Vyukov for (size_t mmap_size = 64ull << 30; mmap_size >= 4 << 10; mmap_size /= 2) {
190984b8deSDmitry Vyukov size_t allocated = 0;
200984b8deSDmitry Vyukov while (mmap(0, mmap_size, PROT_READ | PROT_WRITE, MAP_ANON | MAP_PRIVATE,
210984b8deSDmitry Vyukov fd, 0) != MAP_FAILED) {
220984b8deSDmitry Vyukov allocated += mmap_size;
230984b8deSDmitry Vyukov }
240984b8deSDmitry Vyukov fprintf(stderr, "allocated %zu with size %zu\n", allocated, mmap_size);
250984b8deSDmitry Vyukov }
260984b8deSDmitry Vyukov fprintf(stderr, "DONE\n");
27*fb19400dSDmitry Vyukov // If tsan runtime will try to allocate something during exit handling,
28*fb19400dSDmitry Vyukov // the allocation will fail because there is no VA whatsoever.
29*fb19400dSDmitry Vyukov // It's observed to fail with the following error in some cases:
30*fb19400dSDmitry Vyukov // failed to allocate 0x1000 (4096) bytes of DTLS_NextBlock.
31*fb19400dSDmitry Vyukov // So terminate the process immediately.
32*fb19400dSDmitry Vyukov _exit(0);
330984b8deSDmitry Vyukov }
340984b8deSDmitry Vyukov
350984b8deSDmitry Vyukov // CHECK: DONE
36