1 //===-- tsan_mman_test.cc -------------------------------------------------===//
2 //
3 // The LLVM Compiler Infrastructure
4 //
5 // This file is distributed under the University of Illinois Open Source
6 // License. See LICENSE.TXT for details.
7 //
8 //===----------------------------------------------------------------------===//
9 //
10 // This file is a part of ThreadSanitizer (TSan), a race detector.
11 //
12 //===----------------------------------------------------------------------===//
13 #include <limits>
14 #include <sanitizer/allocator_interface.h>
15 #include "tsan_mman.h"
16 #include "tsan_rtl.h"
17 #include "gtest/gtest.h"
18
19 namespace __tsan {
20
TEST(Mman,Internal)21 TEST(Mman, Internal) {
22 char *p = (char*)internal_alloc(MBlockScopedBuf, 10);
23 EXPECT_NE(p, (char*)0);
24 char *p2 = (char*)internal_alloc(MBlockScopedBuf, 20);
25 EXPECT_NE(p2, (char*)0);
26 EXPECT_NE(p2, p);
27 for (int i = 0; i < 10; i++) {
28 p[i] = 42;
29 }
30 for (int i = 0; i < 20; i++) {
31 ((char*)p2)[i] = 42;
32 }
33 internal_free(p);
34 internal_free(p2);
35 }
36
TEST(Mman,User)37 TEST(Mman, User) {
38 ThreadState *thr = cur_thread();
39 uptr pc = 0;
40 char *p = (char*)user_alloc(thr, pc, 10);
41 EXPECT_NE(p, (char*)0);
42 char *p2 = (char*)user_alloc(thr, pc, 20);
43 EXPECT_NE(p2, (char*)0);
44 EXPECT_NE(p2, p);
45 EXPECT_EQ(10U, user_alloc_usable_size(p));
46 EXPECT_EQ(20U, user_alloc_usable_size(p2));
47 user_free(thr, pc, p);
48 user_free(thr, pc, p2);
49 }
50
TEST(Mman,UserRealloc)51 TEST(Mman, UserRealloc) {
52 ThreadState *thr = cur_thread();
53 uptr pc = 0;
54 {
55 void *p = user_realloc(thr, pc, 0, 0);
56 // Strictly saying this is incorrect, realloc(NULL, N) is equivalent to
57 // malloc(N), thus must return non-NULL pointer.
58 EXPECT_EQ(p, (void*)0);
59 }
60 {
61 void *p = user_realloc(thr, pc, 0, 100);
62 EXPECT_NE(p, (void*)0);
63 memset(p, 0xde, 100);
64 user_free(thr, pc, p);
65 }
66 {
67 void *p = user_alloc(thr, pc, 100);
68 EXPECT_NE(p, (void*)0);
69 memset(p, 0xde, 100);
70 void *p2 = user_realloc(thr, pc, p, 0);
71 EXPECT_EQ(p2, (void*)0);
72 }
73 {
74 void *p = user_realloc(thr, pc, 0, 100);
75 EXPECT_NE(p, (void*)0);
76 memset(p, 0xde, 100);
77 void *p2 = user_realloc(thr, pc, p, 10000);
78 EXPECT_NE(p2, (void*)0);
79 for (int i = 0; i < 100; i++)
80 EXPECT_EQ(((char*)p2)[i], (char)0xde);
81 memset(p2, 0xde, 10000);
82 user_free(thr, pc, p2);
83 }
84 {
85 void *p = user_realloc(thr, pc, 0, 10000);
86 EXPECT_NE(p, (void*)0);
87 memset(p, 0xde, 10000);
88 void *p2 = user_realloc(thr, pc, p, 10);
89 EXPECT_NE(p2, (void*)0);
90 for (int i = 0; i < 10; i++)
91 EXPECT_EQ(((char*)p2)[i], (char)0xde);
92 user_free(thr, pc, p2);
93 }
94 }
95
TEST(Mman,UsableSize)96 TEST(Mman, UsableSize) {
97 ThreadState *thr = cur_thread();
98 uptr pc = 0;
99 char *p = (char*)user_alloc(thr, pc, 10);
100 char *p2 = (char*)user_alloc(thr, pc, 20);
101 EXPECT_EQ(0U, user_alloc_usable_size(NULL));
102 EXPECT_EQ(10U, user_alloc_usable_size(p));
103 EXPECT_EQ(20U, user_alloc_usable_size(p2));
104 user_free(thr, pc, p);
105 user_free(thr, pc, p2);
106 EXPECT_EQ(0U, user_alloc_usable_size((void*)0x4123));
107 }
108
TEST(Mman,Stats)109 TEST(Mman, Stats) {
110 ThreadState *thr = cur_thread();
111
112 uptr alloc0 = __sanitizer_get_current_allocated_bytes();
113 uptr heap0 = __sanitizer_get_heap_size();
114 uptr free0 = __sanitizer_get_free_bytes();
115 uptr unmapped0 = __sanitizer_get_unmapped_bytes();
116
117 EXPECT_EQ(10U, __sanitizer_get_estimated_allocated_size(10));
118 EXPECT_EQ(20U, __sanitizer_get_estimated_allocated_size(20));
119 EXPECT_EQ(100U, __sanitizer_get_estimated_allocated_size(100));
120
121 char *p = (char*)user_alloc(thr, 0, 10);
122 EXPECT_TRUE(__sanitizer_get_ownership(p));
123 EXPECT_EQ(10U, __sanitizer_get_allocated_size(p));
124
125 EXPECT_EQ(alloc0 + 16, __sanitizer_get_current_allocated_bytes());
126 EXPECT_GE(__sanitizer_get_heap_size(), heap0);
127 EXPECT_EQ(free0, __sanitizer_get_free_bytes());
128 EXPECT_EQ(unmapped0, __sanitizer_get_unmapped_bytes());
129
130 user_free(thr, 0, p);
131
132 EXPECT_EQ(alloc0, __sanitizer_get_current_allocated_bytes());
133 EXPECT_GE(__sanitizer_get_heap_size(), heap0);
134 EXPECT_EQ(free0, __sanitizer_get_free_bytes());
135 EXPECT_EQ(unmapped0, __sanitizer_get_unmapped_bytes());
136 }
137
TEST(Mman,CallocOverflow)138 TEST(Mman, CallocOverflow) {
139 #if SANITIZER_DEBUG
140 // EXPECT_DEATH clones a thread with 4K stack,
141 // which is overflown by tsan memory accesses functions in debug mode.
142 return;
143 #endif
144 ThreadState *thr = cur_thread();
145 uptr pc = 0;
146 size_t kArraySize = 4096;
147 volatile size_t kMaxSizeT = std::numeric_limits<size_t>::max();
148 volatile size_t kArraySize2 = kMaxSizeT / kArraySize + 10;
149 volatile void *p = NULL;
150 EXPECT_DEATH(p = user_calloc(thr, pc, kArraySize, kArraySize2),
151 "allocator is terminating the process instead of returning 0");
152 EXPECT_EQ(0L, p);
153 }
154
155 } // namespace __tsan
156