witness.c revision 1.1 1 1.1 christos #include "test/jemalloc_test.h"
2 1.1 christos
3 1.1 christos static witness_lock_error_t *witness_lock_error_orig;
4 1.1 christos static witness_owner_error_t *witness_owner_error_orig;
5 1.1 christos static witness_not_owner_error_t *witness_not_owner_error_orig;
6 1.1 christos static witness_depth_error_t *witness_depth_error_orig;
7 1.1 christos
8 1.1 christos static bool saw_lock_error;
9 1.1 christos static bool saw_owner_error;
10 1.1 christos static bool saw_not_owner_error;
11 1.1 christos static bool saw_depth_error;
12 1.1 christos
13 1.1 christos static void
14 1.1 christos witness_lock_error_intercept(const witness_list_t *witnesses,
15 1.1 christos const witness_t *witness) {
16 1.1 christos saw_lock_error = true;
17 1.1 christos }
18 1.1 christos
19 1.1 christos static void
20 1.1 christos witness_owner_error_intercept(const witness_t *witness) {
21 1.1 christos saw_owner_error = true;
22 1.1 christos }
23 1.1 christos
24 1.1 christos static void
25 1.1 christos witness_not_owner_error_intercept(const witness_t *witness) {
26 1.1 christos saw_not_owner_error = true;
27 1.1 christos }
28 1.1 christos
29 1.1 christos static void
30 1.1 christos witness_depth_error_intercept(const witness_list_t *witnesses,
31 1.1 christos witness_rank_t rank_inclusive, unsigned depth) {
32 1.1 christos saw_depth_error = true;
33 1.1 christos }
34 1.1 christos
35 1.1 christos static int
36 1.1 christos witness_comp(const witness_t *a, void *oa, const witness_t *b, void *ob) {
37 1.1 christos assert_u_eq(a->rank, b->rank, "Witnesses should have equal rank");
38 1.1 christos
39 1.1 christos assert(oa == (void *)a);
40 1.1 christos assert(ob == (void *)b);
41 1.1 christos
42 1.1 christos return strcmp(a->name, b->name);
43 1.1 christos }
44 1.1 christos
45 1.1 christos static int
46 1.1 christos witness_comp_reverse(const witness_t *a, void *oa, const witness_t *b,
47 1.1 christos void *ob) {
48 1.1 christos assert_u_eq(a->rank, b->rank, "Witnesses should have equal rank");
49 1.1 christos
50 1.1 christos assert(oa == (void *)a);
51 1.1 christos assert(ob == (void *)b);
52 1.1 christos
53 1.1 christos return -strcmp(a->name, b->name);
54 1.1 christos }
55 1.1 christos
56 1.1 christos TEST_BEGIN(test_witness) {
57 1.1 christos witness_t a, b;
58 1.1 christos witness_tsdn_t witness_tsdn = { WITNESS_TSD_INITIALIZER };
59 1.1 christos
60 1.1 christos test_skip_if(!config_debug);
61 1.1 christos
62 1.1 christos witness_assert_lockless(&witness_tsdn);
63 1.1 christos witness_assert_depth(&witness_tsdn, 0);
64 1.1 christos witness_assert_depth_to_rank(&witness_tsdn, (witness_rank_t)1U, 0);
65 1.1 christos
66 1.1 christos witness_init(&a, "a", 1, NULL, NULL);
67 1.1 christos witness_assert_not_owner(&witness_tsdn, &a);
68 1.1 christos witness_lock(&witness_tsdn, &a);
69 1.1 christos witness_assert_owner(&witness_tsdn, &a);
70 1.1 christos witness_assert_depth(&witness_tsdn, 1);
71 1.1 christos witness_assert_depth_to_rank(&witness_tsdn, (witness_rank_t)1U, 1);
72 1.1 christos witness_assert_depth_to_rank(&witness_tsdn, (witness_rank_t)2U, 0);
73 1.1 christos
74 1.1 christos witness_init(&b, "b", 2, NULL, NULL);
75 1.1 christos witness_assert_not_owner(&witness_tsdn, &b);
76 1.1 christos witness_lock(&witness_tsdn, &b);
77 1.1 christos witness_assert_owner(&witness_tsdn, &b);
78 1.1 christos witness_assert_depth(&witness_tsdn, 2);
79 1.1 christos witness_assert_depth_to_rank(&witness_tsdn, (witness_rank_t)1U, 2);
80 1.1 christos witness_assert_depth_to_rank(&witness_tsdn, (witness_rank_t)2U, 1);
81 1.1 christos witness_assert_depth_to_rank(&witness_tsdn, (witness_rank_t)3U, 0);
82 1.1 christos
83 1.1 christos witness_unlock(&witness_tsdn, &a);
84 1.1 christos witness_assert_depth(&witness_tsdn, 1);
85 1.1 christos witness_assert_depth_to_rank(&witness_tsdn, (witness_rank_t)1U, 1);
86 1.1 christos witness_assert_depth_to_rank(&witness_tsdn, (witness_rank_t)2U, 1);
87 1.1 christos witness_assert_depth_to_rank(&witness_tsdn, (witness_rank_t)3U, 0);
88 1.1 christos witness_unlock(&witness_tsdn, &b);
89 1.1 christos
90 1.1 christos witness_assert_lockless(&witness_tsdn);
91 1.1 christos witness_assert_depth(&witness_tsdn, 0);
92 1.1 christos witness_assert_depth_to_rank(&witness_tsdn, (witness_rank_t)1U, 0);
93 1.1 christos }
94 1.1 christos TEST_END
95 1.1 christos
96 1.1 christos TEST_BEGIN(test_witness_comp) {
97 1.1 christos witness_t a, b, c, d;
98 1.1 christos witness_tsdn_t witness_tsdn = { WITNESS_TSD_INITIALIZER };
99 1.1 christos
100 1.1 christos test_skip_if(!config_debug);
101 1.1 christos
102 1.1 christos witness_assert_lockless(&witness_tsdn);
103 1.1 christos
104 1.1 christos witness_init(&a, "a", 1, witness_comp, &a);
105 1.1 christos witness_assert_not_owner(&witness_tsdn, &a);
106 1.1 christos witness_lock(&witness_tsdn, &a);
107 1.1 christos witness_assert_owner(&witness_tsdn, &a);
108 1.1 christos witness_assert_depth(&witness_tsdn, 1);
109 1.1 christos
110 1.1 christos witness_init(&b, "b", 1, witness_comp, &b);
111 1.1 christos witness_assert_not_owner(&witness_tsdn, &b);
112 1.1 christos witness_lock(&witness_tsdn, &b);
113 1.1 christos witness_assert_owner(&witness_tsdn, &b);
114 1.1 christos witness_assert_depth(&witness_tsdn, 2);
115 1.1 christos witness_unlock(&witness_tsdn, &b);
116 1.1 christos witness_assert_depth(&witness_tsdn, 1);
117 1.1 christos
118 1.1 christos witness_lock_error_orig = witness_lock_error;
119 1.1 christos witness_lock_error = witness_lock_error_intercept;
120 1.1 christos saw_lock_error = false;
121 1.1 christos
122 1.1 christos witness_init(&c, "c", 1, witness_comp_reverse, &c);
123 1.1 christos witness_assert_not_owner(&witness_tsdn, &c);
124 1.1 christos assert_false(saw_lock_error, "Unexpected witness lock error");
125 1.1 christos witness_lock(&witness_tsdn, &c);
126 1.1 christos assert_true(saw_lock_error, "Expected witness lock error");
127 1.1 christos witness_unlock(&witness_tsdn, &c);
128 1.1 christos witness_assert_depth(&witness_tsdn, 1);
129 1.1 christos
130 1.1 christos saw_lock_error = false;
131 1.1 christos
132 1.1 christos witness_init(&d, "d", 1, NULL, NULL);
133 1.1 christos witness_assert_not_owner(&witness_tsdn, &d);
134 1.1 christos assert_false(saw_lock_error, "Unexpected witness lock error");
135 1.1 christos witness_lock(&witness_tsdn, &d);
136 1.1 christos assert_true(saw_lock_error, "Expected witness lock error");
137 1.1 christos witness_unlock(&witness_tsdn, &d);
138 1.1 christos witness_assert_depth(&witness_tsdn, 1);
139 1.1 christos
140 1.1 christos witness_unlock(&witness_tsdn, &a);
141 1.1 christos
142 1.1 christos witness_assert_lockless(&witness_tsdn);
143 1.1 christos
144 1.1 christos witness_lock_error = witness_lock_error_orig;
145 1.1 christos }
146 1.1 christos TEST_END
147 1.1 christos
148 1.1 christos TEST_BEGIN(test_witness_reversal) {
149 1.1 christos witness_t a, b;
150 1.1 christos witness_tsdn_t witness_tsdn = { WITNESS_TSD_INITIALIZER };
151 1.1 christos
152 1.1 christos test_skip_if(!config_debug);
153 1.1 christos
154 1.1 christos witness_lock_error_orig = witness_lock_error;
155 1.1 christos witness_lock_error = witness_lock_error_intercept;
156 1.1 christos saw_lock_error = false;
157 1.1 christos
158 1.1 christos witness_assert_lockless(&witness_tsdn);
159 1.1 christos
160 1.1 christos witness_init(&a, "a", 1, NULL, NULL);
161 1.1 christos witness_init(&b, "b", 2, NULL, NULL);
162 1.1 christos
163 1.1 christos witness_lock(&witness_tsdn, &b);
164 1.1 christos witness_assert_depth(&witness_tsdn, 1);
165 1.1 christos assert_false(saw_lock_error, "Unexpected witness lock error");
166 1.1 christos witness_lock(&witness_tsdn, &a);
167 1.1 christos assert_true(saw_lock_error, "Expected witness lock error");
168 1.1 christos
169 1.1 christos witness_unlock(&witness_tsdn, &a);
170 1.1 christos witness_assert_depth(&witness_tsdn, 1);
171 1.1 christos witness_unlock(&witness_tsdn, &b);
172 1.1 christos
173 1.1 christos witness_assert_lockless(&witness_tsdn);
174 1.1 christos
175 1.1 christos witness_lock_error = witness_lock_error_orig;
176 1.1 christos }
177 1.1 christos TEST_END
178 1.1 christos
179 1.1 christos TEST_BEGIN(test_witness_recursive) {
180 1.1 christos witness_t a;
181 1.1 christos witness_tsdn_t witness_tsdn = { WITNESS_TSD_INITIALIZER };
182 1.1 christos
183 1.1 christos test_skip_if(!config_debug);
184 1.1 christos
185 1.1 christos witness_not_owner_error_orig = witness_not_owner_error;
186 1.1 christos witness_not_owner_error = witness_not_owner_error_intercept;
187 1.1 christos saw_not_owner_error = false;
188 1.1 christos
189 1.1 christos witness_lock_error_orig = witness_lock_error;
190 1.1 christos witness_lock_error = witness_lock_error_intercept;
191 1.1 christos saw_lock_error = false;
192 1.1 christos
193 1.1 christos witness_assert_lockless(&witness_tsdn);
194 1.1 christos
195 1.1 christos witness_init(&a, "a", 1, NULL, NULL);
196 1.1 christos
197 1.1 christos witness_lock(&witness_tsdn, &a);
198 1.1 christos assert_false(saw_lock_error, "Unexpected witness lock error");
199 1.1 christos assert_false(saw_not_owner_error, "Unexpected witness not owner error");
200 1.1 christos witness_lock(&witness_tsdn, &a);
201 1.1 christos assert_true(saw_lock_error, "Expected witness lock error");
202 1.1 christos assert_true(saw_not_owner_error, "Expected witness not owner error");
203 1.1 christos
204 1.1 christos witness_unlock(&witness_tsdn, &a);
205 1.1 christos
206 1.1 christos witness_assert_lockless(&witness_tsdn);
207 1.1 christos
208 1.1 christos witness_owner_error = witness_owner_error_orig;
209 1.1 christos witness_lock_error = witness_lock_error_orig;
210 1.1 christos
211 1.1 christos }
212 1.1 christos TEST_END
213 1.1 christos
214 1.1 christos TEST_BEGIN(test_witness_unlock_not_owned) {
215 1.1 christos witness_t a;
216 1.1 christos witness_tsdn_t witness_tsdn = { WITNESS_TSD_INITIALIZER };
217 1.1 christos
218 1.1 christos test_skip_if(!config_debug);
219 1.1 christos
220 1.1 christos witness_owner_error_orig = witness_owner_error;
221 1.1 christos witness_owner_error = witness_owner_error_intercept;
222 1.1 christos saw_owner_error = false;
223 1.1 christos
224 1.1 christos witness_assert_lockless(&witness_tsdn);
225 1.1 christos
226 1.1 christos witness_init(&a, "a", 1, NULL, NULL);
227 1.1 christos
228 1.1 christos assert_false(saw_owner_error, "Unexpected owner error");
229 1.1 christos witness_unlock(&witness_tsdn, &a);
230 1.1 christos assert_true(saw_owner_error, "Expected owner error");
231 1.1 christos
232 1.1 christos witness_assert_lockless(&witness_tsdn);
233 1.1 christos
234 1.1 christos witness_owner_error = witness_owner_error_orig;
235 1.1 christos }
236 1.1 christos TEST_END
237 1.1 christos
238 1.1 christos TEST_BEGIN(test_witness_depth) {
239 1.1 christos witness_t a;
240 1.1 christos witness_tsdn_t witness_tsdn = { WITNESS_TSD_INITIALIZER };
241 1.1 christos
242 1.1 christos test_skip_if(!config_debug);
243 1.1 christos
244 1.1 christos witness_depth_error_orig = witness_depth_error;
245 1.1 christos witness_depth_error = witness_depth_error_intercept;
246 1.1 christos saw_depth_error = false;
247 1.1 christos
248 1.1 christos witness_assert_lockless(&witness_tsdn);
249 1.1 christos witness_assert_depth(&witness_tsdn, 0);
250 1.1 christos
251 1.1 christos witness_init(&a, "a", 1, NULL, NULL);
252 1.1 christos
253 1.1 christos assert_false(saw_depth_error, "Unexpected depth error");
254 1.1 christos witness_assert_lockless(&witness_tsdn);
255 1.1 christos witness_assert_depth(&witness_tsdn, 0);
256 1.1 christos
257 1.1 christos witness_lock(&witness_tsdn, &a);
258 1.1 christos witness_assert_lockless(&witness_tsdn);
259 1.1 christos witness_assert_depth(&witness_tsdn, 0);
260 1.1 christos assert_true(saw_depth_error, "Expected depth error");
261 1.1 christos
262 1.1 christos witness_unlock(&witness_tsdn, &a);
263 1.1 christos
264 1.1 christos witness_assert_lockless(&witness_tsdn);
265 1.1 christos witness_assert_depth(&witness_tsdn, 0);
266 1.1 christos
267 1.1 christos witness_depth_error = witness_depth_error_orig;
268 1.1 christos }
269 1.1 christos TEST_END
270 1.1 christos
271 1.1 christos int
272 1.1 christos main(void) {
273 1.1 christos return test(
274 1.1 christos test_witness,
275 1.1 christos test_witness_comp,
276 1.1 christos test_witness_reversal,
277 1.1 christos test_witness_recursive,
278 1.1 christos test_witness_unlock_not_owned,
279 1.1 christos test_witness_depth);
280 1.1 christos }
281