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.1.2 christos expect_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.1.2 christos expect_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.1.2 christos expect_false(saw_lock_error, "Unexpected witness lock error"); 125 1.1 christos witness_lock(&witness_tsdn, &c); 126 1.1.1.2 christos expect_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.1.2 christos expect_false(saw_lock_error, "Unexpected witness lock error"); 135 1.1 christos witness_lock(&witness_tsdn, &d); 136 1.1.1.2 christos expect_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.1.2 christos expect_false(saw_lock_error, "Unexpected witness lock error"); 166 1.1 christos witness_lock(&witness_tsdn, &a); 167 1.1.1.2 christos expect_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.1.2 christos expect_false(saw_lock_error, "Unexpected witness lock error"); 199 1.1.1.2 christos expect_false(saw_not_owner_error, "Unexpected witness not owner error"); 200 1.1 christos witness_lock(&witness_tsdn, &a); 201 1.1.1.2 christos expect_true(saw_lock_error, "Expected witness lock error"); 202 1.1.1.2 christos expect_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.1.2 christos expect_false(saw_owner_error, "Unexpected owner error"); 229 1.1 christos witness_unlock(&witness_tsdn, &a); 230 1.1.1.2 christos expect_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.1.2 christos expect_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.1.2 christos expect_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