Blame view
lib/test_bitmap.c
10.6 KB
09c434b8a treewide: Add SPD... |
1 |
// SPDX-License-Identifier: GPL-2.0-only |
5fd003f56 test_bitmap: unit... |
2 3 4 5 6 7 8 9 10 11 12 13 14 |
/* * Test cases for printf facility. */ #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt #include <linux/bitmap.h> #include <linux/init.h> #include <linux/kernel.h> #include <linux/module.h> #include <linux/printk.h> #include <linux/slab.h> #include <linux/string.h> |
6ea86bdfc lib/test_bitmap: ... |
15 |
#include <linux/uaccess.h> |
5fd003f56 test_bitmap: unit... |
16 |
|
6b1a4d5b1 lib: Use new ksel... |
17 |
#include "../tools/testing/selftests/kselftest_module.h" |
5fd003f56 test_bitmap: unit... |
18 19 20 21 22 23 24 25 26 27 28 |
static unsigned total_tests __initdata; static unsigned failed_tests __initdata; static char pbl_buffer[PAGE_SIZE] __initdata; static bool __init __check_eq_uint(const char *srcfile, unsigned int line, const unsigned int exp_uint, unsigned int x) { if (exp_uint != x) { |
3aa56885e bitmap: replace b... |
29 30 |
pr_err("[%s:%u] expected %u, got %u ", |
5fd003f56 test_bitmap: unit... |
31 32 33 34 35 36 37 38 39 |
srcfile, line, exp_uint, x); return false; } return true; } static bool __init __check_eq_bitmap(const char *srcfile, unsigned int line, |
3aa56885e bitmap: replace b... |
40 41 |
const unsigned long *exp_bmap, const unsigned long *bmap, unsigned int nbits) |
5fd003f56 test_bitmap: unit... |
42 |
{ |
5fd003f56 test_bitmap: unit... |
43 44 45 46 |
if (!bitmap_equal(exp_bmap, bmap, nbits)) { pr_warn("[%s:%u] bitmaps contents differ: expected \"%*pbl\", got \"%*pbl\" ", srcfile, line, |
3aa56885e bitmap: replace b... |
47 |
nbits, exp_bmap, nbits, bmap); |
5fd003f56 test_bitmap: unit... |
48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 |
return false; } return true; } static bool __init __check_eq_pbl(const char *srcfile, unsigned int line, const char *expected_pbl, const unsigned long *bitmap, unsigned int nbits) { snprintf(pbl_buffer, sizeof(pbl_buffer), "%*pbl", nbits, bitmap); if (strcmp(expected_pbl, pbl_buffer)) { pr_warn("[%s:%u] expected \"%s\", got \"%s\" ", srcfile, line, expected_pbl, pbl_buffer); return false; } return true; } static bool __init __check_eq_u32_array(const char *srcfile, unsigned int line, const u32 *exp_arr, unsigned int exp_len, |
3aa56885e bitmap: replace b... |
72 73 74 75 |
const u32 *arr, unsigned int len) __used; static bool __init __check_eq_u32_array(const char *srcfile, unsigned int line, const u32 *exp_arr, unsigned int exp_len, |
5fd003f56 test_bitmap: unit... |
76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 |
const u32 *arr, unsigned int len) { if (exp_len != len) { pr_warn("[%s:%u] array length differ: expected %u, got %u ", srcfile, line, exp_len, len); return false; } if (memcmp(exp_arr, arr, len*sizeof(*arr))) { pr_warn("[%s:%u] array contents differ ", srcfile, line); print_hex_dump(KERN_WARNING, " exp: ", DUMP_PREFIX_OFFSET, 32, 4, exp_arr, exp_len*sizeof(*exp_arr), false); print_hex_dump(KERN_WARNING, " got: ", DUMP_PREFIX_OFFSET, 32, 4, arr, len*sizeof(*arr), false); return false; } return true; } #define __expect_eq(suffix, ...) \ ({ \ int result = 0; \ total_tests++; \ if (!__check_eq_ ## suffix(__FILE__, __LINE__, \ ##__VA_ARGS__)) { \ failed_tests++; \ result = 1; \ } \ result; \ }) #define expect_eq_uint(...) __expect_eq(uint, ##__VA_ARGS__) #define expect_eq_bitmap(...) __expect_eq(bitmap, ##__VA_ARGS__) #define expect_eq_pbl(...) __expect_eq(pbl, ##__VA_ARGS__) #define expect_eq_u32_array(...) __expect_eq(u32_array, ##__VA_ARGS__) |
ee3527bd5 lib/test_bitmap.c... |
115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 |
static void __init test_zero_clear(void) { DECLARE_BITMAP(bmap, 1024); /* Known way to set all bits */ memset(bmap, 0xff, 128); expect_eq_pbl("0-22", bmap, 23); expect_eq_pbl("0-1023", bmap, 1024); /* single-word bitmaps */ bitmap_clear(bmap, 0, 9); expect_eq_pbl("9-1023", bmap, 1024); bitmap_zero(bmap, 35); expect_eq_pbl("64-1023", bmap, 1024); /* cross boundaries operations */ bitmap_clear(bmap, 79, 19); expect_eq_pbl("64-78,98-1023", bmap, 1024); bitmap_zero(bmap, 115); expect_eq_pbl("128-1023", bmap, 1024); /* Zeroing entire area */ bitmap_zero(bmap, 1024); expect_eq_pbl("", bmap, 1024); } |
978f369c5 lib/test_bitmap.c... |
143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 |
static void __init test_fill_set(void) { DECLARE_BITMAP(bmap, 1024); /* Known way to clear all bits */ memset(bmap, 0x00, 128); expect_eq_pbl("", bmap, 23); expect_eq_pbl("", bmap, 1024); /* single-word bitmaps */ bitmap_set(bmap, 0, 9); expect_eq_pbl("0-8", bmap, 1024); bitmap_fill(bmap, 35); expect_eq_pbl("0-63", bmap, 1024); /* cross boundaries operations */ bitmap_set(bmap, 79, 19); expect_eq_pbl("0-63,79-97", bmap, 1024); bitmap_fill(bmap, 115); expect_eq_pbl("0-127", bmap, 1024); /* Zeroing entire area */ bitmap_fill(bmap, 1024); expect_eq_pbl("0-1023", bmap, 1024); } |
fe81814c3 lib/test_bitmap.c... |
171 |
static void __init test_copy(void) |
5fd003f56 test_bitmap: unit... |
172 173 174 175 176 177 178 179 |
{ DECLARE_BITMAP(bmap1, 1024); DECLARE_BITMAP(bmap2, 1024); bitmap_zero(bmap1, 1024); bitmap_zero(bmap2, 1024); /* single-word bitmaps */ |
fe81814c3 lib/test_bitmap.c... |
180 |
bitmap_set(bmap1, 0, 19); |
5fd003f56 test_bitmap: unit... |
181 182 |
bitmap_copy(bmap2, bmap1, 23); expect_eq_pbl("0-18", bmap2, 1024); |
fe81814c3 lib/test_bitmap.c... |
183 |
bitmap_set(bmap2, 0, 23); |
5fd003f56 test_bitmap: unit... |
184 185 |
bitmap_copy(bmap2, bmap1, 23); expect_eq_pbl("0-18", bmap2, 1024); |
5fd003f56 test_bitmap: unit... |
186 |
/* multi-word bitmaps */ |
fe81814c3 lib/test_bitmap.c... |
187 |
bitmap_set(bmap1, 0, 109); |
5fd003f56 test_bitmap: unit... |
188 189 190 191 |
bitmap_copy(bmap2, bmap1, 1024); expect_eq_pbl("0-108", bmap2, 1024); bitmap_fill(bmap2, 1024); |
5fd003f56 test_bitmap: unit... |
192 193 194 195 196 197 198 199 200 201 202 203 204 205 |
bitmap_copy(bmap2, bmap1, 1024); expect_eq_pbl("0-108", bmap2, 1024); /* the following tests assume a 32- or 64-bit arch (even 128b * if we care) */ bitmap_fill(bmap2, 1024); bitmap_copy(bmap2, bmap1, 109); /* ... but 0-padded til word length */ expect_eq_pbl("0-108,128-1023", bmap2, 1024); bitmap_fill(bmap2, 1024); bitmap_copy(bmap2, bmap1, 97); /* ... but aligned on word length */ expect_eq_pbl("0-108,128-1023", bmap2, 1024); |
5fd003f56 test_bitmap: unit... |
206 |
} |
6df0d464d lib/test_bitmap.c... |
207 208 209 210 211 212 213 214 215 |
#define PARSE_TIME 0x1 struct test_bitmap_parselist{ const int errno; const char *in; const unsigned long *expected; const int nbits; const int flags; }; |
60ef69001 bitmap: introduce... |
216 217 218 219 220 221 222 223 224 225 226 |
static const unsigned long exp[] __initconst = { BITMAP_FROM_U64(1), BITMAP_FROM_U64(2), BITMAP_FROM_U64(0x0000ffff), BITMAP_FROM_U64(0xffff0000), BITMAP_FROM_U64(0x55555555), BITMAP_FROM_U64(0xaaaaaaaa), BITMAP_FROM_U64(0x11111111), BITMAP_FROM_U64(0x22222222), BITMAP_FROM_U64(0xffffffff), BITMAP_FROM_U64(0xfffffffe), |
8185f5708 lib/test_bitmap.c... |
227 |
BITMAP_FROM_U64(0x3333333311111111ULL), |
a4ab50509 lib/test_bitmap: ... |
228 229 |
BITMAP_FROM_U64(0xffffffff77777777ULL), BITMAP_FROM_U64(0), |
60ef69001 bitmap: introduce... |
230 231 232 |
}; static const unsigned long exp2[] __initconst = { |
8185f5708 lib/test_bitmap.c... |
233 234 |
BITMAP_FROM_U64(0x3333333311111111ULL), BITMAP_FROM_U64(0xffffffff77777777ULL) |
60ef69001 bitmap: introduce... |
235 |
}; |
6df0d464d lib/test_bitmap.c... |
236 237 |
static const struct test_bitmap_parselist parselist_tests[] __initconst = { |
60ef69001 bitmap: introduce... |
238 |
#define step (sizeof(u64) / sizeof(unsigned long)) |
6df0d464d lib/test_bitmap.c... |
239 |
{0, "0", &exp[0], 8, 0}, |
60ef69001 bitmap: introduce... |
240 241 242 243 244 245 246 247 248 249 250 |
{0, "1", &exp[1 * step], 8, 0}, {0, "0-15", &exp[2 * step], 32, 0}, {0, "16-31", &exp[3 * step], 32, 0}, {0, "0-31:1/2", &exp[4 * step], 32, 0}, {0, "1-31:1/2", &exp[5 * step], 32, 0}, {0, "0-31:1/4", &exp[6 * step], 32, 0}, {0, "1-31:1/4", &exp[7 * step], 32, 0}, {0, "0-31:4/4", &exp[8 * step], 32, 0}, {0, "1-31:4/4", &exp[9 * step], 32, 0}, {0, "0-31:1/4,32-63:2/4", &exp[10 * step], 64, 0}, {0, "0-31:3/4,32-63:4/4", &exp[11 * step], 64, 0}, |
a4ab50509 lib/test_bitmap: ... |
251 |
{0, " ,, 0-31:3/4 ,, 32-63:4/4 ,, ", &exp[11 * step], 64, 0}, |
6df0d464d lib/test_bitmap.c... |
252 253 254 255 |
{0, "0-31:1/4,32-63:2/4,64-95:3/4,96-127:4/4", exp2, 128, 0}, {0, "0-2047:128/256", NULL, 2048, PARSE_TIME}, |
a4ab50509 lib/test_bitmap: ... |
256 257 258 259 260 261 262 |
{0, "", &exp[12 * step], 8, 0}, {0, " ", &exp[12 * step], 8, 0}, {0, ",, ,, , , ,", &exp[12 * step], 8, 0}, {0, " , ,, , , ", &exp[12 * step], 8, 0}, {0, " , ,, , , ", &exp[12 * step], 8, 0}, |
6df0d464d lib/test_bitmap.c... |
263 264 265 |
{-EINVAL, "-1", NULL, 8, 0}, {-EINVAL, "-0", NULL, 8, 0}, {-EINVAL, "10-1", NULL, 8, 0}, |
8351760ff lib: fix stall in... |
266 267 |
{-EINVAL, "0-31:", NULL, 8, 0}, {-EINVAL, "0-31:0", NULL, 8, 0}, |
a4ab50509 lib/test_bitmap: ... |
268 |
{-EINVAL, "0-31:0/", NULL, 8, 0}, |
8351760ff lib: fix stall in... |
269 270 |
{-EINVAL, "0-31:0/0", NULL, 8, 0}, {-EINVAL, "0-31:1/0", NULL, 8, 0}, |
6df0d464d lib/test_bitmap.c... |
271 |
{-EINVAL, "0-31:10/1", NULL, 8, 0}, |
a4ab50509 lib/test_bitmap: ... |
272 273 274 275 276 277 278 279 |
{-EOVERFLOW, "0-98765432123456789:10/1", NULL, 8, 0}, {-EINVAL, "a-31", NULL, 8, 0}, {-EINVAL, "0-a1", NULL, 8, 0}, {-EINVAL, "a-31:10/1", NULL, 8, 0}, {-EINVAL, "0-31:a/1", NULL, 8, 0}, {-EINVAL, "0- ", NULL, 8, 0}, |
6df0d464d lib/test_bitmap.c... |
280 |
}; |
6ea86bdfc lib/test_bitmap: ... |
281 |
static void __init __test_bitmap_parselist(int is_user) |
6df0d464d lib/test_bitmap.c... |
282 283 284 |
{ int i; int err; |
0c2111a5c lib/test_bitmap: ... |
285 |
ktime_t time; |
6df0d464d lib/test_bitmap.c... |
286 |
DECLARE_BITMAP(bmap, 2048); |
6ea86bdfc lib/test_bitmap: ... |
287 |
char *mode = is_user ? "_user" : ""; |
6df0d464d lib/test_bitmap.c... |
288 289 290 |
for (i = 0; i < ARRAY_SIZE(parselist_tests); i++) { #define ptest parselist_tests[i] |
6ea86bdfc lib/test_bitmap: ... |
291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 |
if (is_user) { mm_segment_t orig_fs = get_fs(); size_t len = strlen(ptest.in); set_fs(KERNEL_DS); time = ktime_get(); err = bitmap_parselist_user(ptest.in, len, bmap, ptest.nbits); time = ktime_get() - time; set_fs(orig_fs); } else { time = ktime_get(); err = bitmap_parselist(ptest.in, bmap, ptest.nbits); time = ktime_get() - time; } |
6df0d464d lib/test_bitmap.c... |
306 307 |
if (err != ptest.errno) { |
6ea86bdfc lib/test_bitmap: ... |
308 309 310 |
pr_err("parselist%s: %d: input is %s, errno is %d, expected %d ", mode, i, ptest.in, err, ptest.errno); |
6df0d464d lib/test_bitmap.c... |
311 312 313 314 315 |
continue; } if (!err && ptest.expected && !__bitmap_equal(bmap, ptest.expected, ptest.nbits)) { |
6ea86bdfc lib/test_bitmap: ... |
316 317 318 319 |
pr_err("parselist%s: %d: input is %s, result is 0x%lx, expected 0x%lx ", mode, i, ptest.in, bmap[0], *ptest.expected); |
6df0d464d lib/test_bitmap.c... |
320 321 322 323 |
continue; } if (ptest.flags & PARSE_TIME) |
6ea86bdfc lib/test_bitmap: ... |
324 325 326 |
pr_err("parselist%s: %d: input is '%s' OK, Time: %llu ", mode, i, ptest.in, time); |
6df0d464d lib/test_bitmap.c... |
327 328 |
} } |
6ea86bdfc lib/test_bitmap: ... |
329 330 331 332 333 334 335 336 337 |
static void __init test_bitmap_parselist(void) { __test_bitmap_parselist(0); } static void __init test_bitmap_parselist_user(void) { __test_bitmap_parselist(1); } |
f6f66c1bf lib/test_bitmap.c... |
338 |
#define EXP_BYTES (sizeof(exp) * 8) |
3aa56885e bitmap: replace b... |
339 |
static void __init test_bitmap_arr32(void) |
5fd003f56 test_bitmap: unit... |
340 |
{ |
f6f66c1bf lib/test_bitmap.c... |
341 |
unsigned int nbits, next_bit; |
3aa56885e bitmap: replace b... |
342 |
u32 arr[sizeof(exp) / 4]; |
f6f66c1bf lib/test_bitmap.c... |
343 |
DECLARE_BITMAP(bmap2, EXP_BYTES); |
3aa56885e bitmap: replace b... |
344 345 |
memset(arr, 0xa5, sizeof(arr)); |
f6f66c1bf lib/test_bitmap.c... |
346 |
for (nbits = 0; nbits < EXP_BYTES; ++nbits) { |
3aa56885e bitmap: replace b... |
347 348 349 350 351 352 353 354 355 356 357 |
bitmap_to_arr32(arr, exp, nbits); bitmap_from_arr32(bmap2, arr, nbits); expect_eq_bitmap(bmap2, exp, nbits); next_bit = find_next_bit(bmap2, round_up(nbits, BITS_PER_LONG), nbits); if (next_bit < round_up(nbits, BITS_PER_LONG)) pr_err("bitmap_copy_arr32(nbits == %d:" " tail is not safely cleared: %d ", nbits, next_bit); |
f6f66c1bf lib/test_bitmap.c... |
358 |
if (nbits < EXP_BYTES - 32) |
3aa56885e bitmap: replace b... |
359 360 |
expect_eq_uint(arr[DIV_ROUND_UP(nbits, 32)], 0xa5a5a5a5); |
5fd003f56 test_bitmap: unit... |
361 362 |
} } |
3cc78125a lib/test_bitmap.c... |
363 364 365 366 367 368 369 |
static void noinline __init test_mem_optimisations(void) { DECLARE_BITMAP(bmap1, 1024); DECLARE_BITMAP(bmap2, 1024); unsigned int start, nbits; for (start = 0; start < 1024; start += 8) { |
3cc78125a lib/test_bitmap.c... |
370 |
for (nbits = 0; nbits < 1024 - start; nbits += 8) { |
1e3054b98 lib/test_bitmap.c... |
371 372 |
memset(bmap1, 0x5a, sizeof(bmap1)); memset(bmap2, 0x5a, sizeof(bmap2)); |
3cc78125a lib/test_bitmap.c... |
373 374 |
bitmap_set(bmap1, start, nbits); __bitmap_set(bmap2, start, nbits); |
1e3054b98 lib/test_bitmap.c... |
375 |
if (!bitmap_equal(bmap1, bmap2, 1024)) { |
3cc78125a lib/test_bitmap.c... |
376 377 |
printk("set not equal %d %d ", start, nbits); |
1e3054b98 lib/test_bitmap.c... |
378 379 380 |
failed_tests++; } if (!__bitmap_equal(bmap1, bmap2, 1024)) { |
3cc78125a lib/test_bitmap.c... |
381 382 |
printk("set not __equal %d %d ", start, nbits); |
1e3054b98 lib/test_bitmap.c... |
383 384 |
failed_tests++; } |
3cc78125a lib/test_bitmap.c... |
385 386 387 |
bitmap_clear(bmap1, start, nbits); __bitmap_clear(bmap2, start, nbits); |
1e3054b98 lib/test_bitmap.c... |
388 |
if (!bitmap_equal(bmap1, bmap2, 1024)) { |
3cc78125a lib/test_bitmap.c... |
389 390 |
printk("clear not equal %d %d ", start, nbits); |
1e3054b98 lib/test_bitmap.c... |
391 392 393 |
failed_tests++; } if (!__bitmap_equal(bmap1, bmap2, 1024)) { |
3cc78125a lib/test_bitmap.c... |
394 395 396 |
printk("clear not __equal %d %d ", start, nbits); |
1e3054b98 lib/test_bitmap.c... |
397 398 |
failed_tests++; } |
3cc78125a lib/test_bitmap.c... |
399 400 401 |
} } } |
6b1a4d5b1 lib: Use new ksel... |
402 |
static void __init selftest(void) |
5fd003f56 test_bitmap: unit... |
403 |
{ |
ee3527bd5 lib/test_bitmap.c... |
404 |
test_zero_clear(); |
978f369c5 lib/test_bitmap.c... |
405 |
test_fill_set(); |
fe81814c3 lib/test_bitmap.c... |
406 |
test_copy(); |
3aa56885e bitmap: replace b... |
407 |
test_bitmap_arr32(); |
6df0d464d lib/test_bitmap.c... |
408 |
test_bitmap_parselist(); |
6ea86bdfc lib/test_bitmap: ... |
409 |
test_bitmap_parselist_user(); |
3cc78125a lib/test_bitmap.c... |
410 |
test_mem_optimisations(); |
5fd003f56 test_bitmap: unit... |
411 |
} |
6b1a4d5b1 lib: Use new ksel... |
412 |
KSTM_MODULE_LOADERS(test_bitmap); |
5fd003f56 test_bitmap: unit... |
413 414 |
MODULE_AUTHOR("david decotigny <david.decotigny@googlers.com>"); MODULE_LICENSE("GPL"); |