Loading...
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 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 115 | // SPDX-License-Identifier: GPL-2.0-only /* * Referrence tracker self test. * * Copyright (c) 2021 Eric Dumazet <edumazet@google.com> */ #include <linux/init.h> #include <linux/module.h> #include <linux/delay.h> #include <linux/ref_tracker.h> #include <linux/slab.h> #include <linux/timer.h> static struct ref_tracker_dir ref_dir; static struct ref_tracker *tracker[20]; #define TRT_ALLOC(X) static noinline void \ alloctest_ref_tracker_alloc##X(struct ref_tracker_dir *dir, \ struct ref_tracker **trackerp) \ { \ ref_tracker_alloc(dir, trackerp, GFP_KERNEL); \ } TRT_ALLOC(1) TRT_ALLOC(2) TRT_ALLOC(3) TRT_ALLOC(4) TRT_ALLOC(5) TRT_ALLOC(6) TRT_ALLOC(7) TRT_ALLOC(8) TRT_ALLOC(9) TRT_ALLOC(10) TRT_ALLOC(11) TRT_ALLOC(12) TRT_ALLOC(13) TRT_ALLOC(14) TRT_ALLOC(15) TRT_ALLOC(16) TRT_ALLOC(17) TRT_ALLOC(18) TRT_ALLOC(19) #undef TRT_ALLOC static noinline void alloctest_ref_tracker_free(struct ref_tracker_dir *dir, struct ref_tracker **trackerp) { ref_tracker_free(dir, trackerp); } static struct timer_list test_ref_tracker_timer; static atomic_t test_ref_timer_done = ATOMIC_INIT(0); static void test_ref_tracker_timer_func(struct timer_list *t) { ref_tracker_alloc(&ref_dir, &tracker[0], GFP_ATOMIC); atomic_set(&test_ref_timer_done, 1); } static int __init test_ref_tracker_init(void) { int i; ref_tracker_dir_init(&ref_dir, 100); timer_setup(&test_ref_tracker_timer, test_ref_tracker_timer_func, 0); mod_timer(&test_ref_tracker_timer, jiffies + 1); alloctest_ref_tracker_alloc1(&ref_dir, &tracker[1]); alloctest_ref_tracker_alloc2(&ref_dir, &tracker[2]); alloctest_ref_tracker_alloc3(&ref_dir, &tracker[3]); alloctest_ref_tracker_alloc4(&ref_dir, &tracker[4]); alloctest_ref_tracker_alloc5(&ref_dir, &tracker[5]); alloctest_ref_tracker_alloc6(&ref_dir, &tracker[6]); alloctest_ref_tracker_alloc7(&ref_dir, &tracker[7]); alloctest_ref_tracker_alloc8(&ref_dir, &tracker[8]); alloctest_ref_tracker_alloc9(&ref_dir, &tracker[9]); alloctest_ref_tracker_alloc10(&ref_dir, &tracker[10]); alloctest_ref_tracker_alloc11(&ref_dir, &tracker[11]); alloctest_ref_tracker_alloc12(&ref_dir, &tracker[12]); alloctest_ref_tracker_alloc13(&ref_dir, &tracker[13]); alloctest_ref_tracker_alloc14(&ref_dir, &tracker[14]); alloctest_ref_tracker_alloc15(&ref_dir, &tracker[15]); alloctest_ref_tracker_alloc16(&ref_dir, &tracker[16]); alloctest_ref_tracker_alloc17(&ref_dir, &tracker[17]); alloctest_ref_tracker_alloc18(&ref_dir, &tracker[18]); alloctest_ref_tracker_alloc19(&ref_dir, &tracker[19]); /* free all trackers but first 0 and 1. */ for (i = 2; i < ARRAY_SIZE(tracker); i++) alloctest_ref_tracker_free(&ref_dir, &tracker[i]); /* Attempt to free an already freed tracker. */ alloctest_ref_tracker_free(&ref_dir, &tracker[2]); while (!atomic_read(&test_ref_timer_done)) msleep(1); /* This should warn about tracker[0] & tracker[1] being not freed. */ ref_tracker_dir_exit(&ref_dir); return 0; } static void __exit test_ref_tracker_exit(void) { } module_init(test_ref_tracker_init); module_exit(test_ref_tracker_exit); MODULE_LICENSE("GPL v2"); |