// SPDX-License-Identifier: GPL-2.0-only #include #include #include #include #include /* a simple boot-time regression test */ #define TESTRL_INTERVAL (5 * HZ) static DEFINE_RATELIMIT_STATE(testrl, TESTRL_INTERVAL, 3); #define test_ratelimited(test, expected) \ KUNIT_ASSERT_EQ(test, ___ratelimit(&testrl, "test_ratelimit_smoke"), (expected)) static void test_ratelimit_smoke(struct kunit *test) { // Check settings. KUNIT_ASSERT_GE(test, TESTRL_INTERVAL, 100); // Test normal operation. test_ratelimited(test, true); test_ratelimited(test, true); test_ratelimited(test, true); test_ratelimited(test, false); schedule_timeout_idle(TESTRL_INTERVAL / 2); test_ratelimited(test, false); schedule_timeout_idle(TESTRL_INTERVAL * 3 / 4); test_ratelimited(test, true); schedule_timeout_idle(2 * TESTRL_INTERVAL); test_ratelimited(test, true); test_ratelimited(test, true); schedule_timeout_idle(TESTRL_INTERVAL / 2 ); test_ratelimited(test, true); schedule_timeout_idle(TESTRL_INTERVAL * 3 / 4); test_ratelimited(test, true); test_ratelimited(test, true); test_ratelimited(test, true); test_ratelimited(test, false); // Test disabling. testrl.burst = 0; test_ratelimited(test, false); testrl.burst = 2; testrl.interval = 0; test_ratelimited(test, true); test_ratelimited(test, true); test_ratelimited(test, true); test_ratelimited(test, true); test_ratelimited(test, true); test_ratelimited(test, true); test_ratelimited(test, true); // Testing re-enabling. testrl.interval = TESTRL_INTERVAL; test_ratelimited(test, true); test_ratelimited(test, true); test_ratelimited(test, false); test_ratelimited(test, false); } static struct ratelimit_state stressrl = RATELIMIT_STATE_INIT_FLAGS("stressrl", HZ / 10, 3, RATELIMIT_MSG_ON_RELEASE); static int doneflag; static const int stress_duration = 2 * HZ; struct stress_kthread { unsigned long nattempts; unsigned long nunlimited; unsigned long nlimited; unsigned long nmissed; struct task_struct *tp; }; static int test_ratelimit_stress_child(void *arg) { struct stress_kthread *sktp = arg; set_user_nice(current, MAX_NICE); WARN_ON_ONCE(!sktp->tp); while (!READ_ONCE(doneflag)) { sktp->nattempts++; if (___ratelimit(&stressrl, __func__)) sktp->nunlimited++; else sktp->nlimited++; cond_resched(); } sktp->nmissed = ratelimit_state_reset_miss(&stressrl); return 0; } static void test_ratelimit_stress(struct kunit *test) { int i; const int n_stress_kthread = cpumask_weight(cpu_online_mask); struct stress_kthread skt = { 0 }; struct stress_kthread *sktp = kcalloc(n_stress_kthread, sizeof(*sktp), GFP_KERNEL); KUNIT_EXPECT_NOT_NULL_MSG(test, sktp, "Memory allocation failure"); for (i = 0; i < n_stress_kthread; i++) { sktp[i].tp = kthread_run(test_ratelimit_stress_child, &sktp[i], "%s/%i", "test_ratelimit_stress_child", i); KUNIT_EXPECT_NOT_NULL_MSG(test, sktp, "kthread creation failure"); pr_alert("Spawned test_ratelimit_stress_child %d\n", i); } schedule_timeout_idle(stress_duration); WRITE_ONCE(doneflag, 1); for (i = 0; i < n_stress_kthread; i++) { kthread_stop(sktp[i].tp); skt.nattempts += sktp[i].nattempts; skt.nunlimited += sktp[i].nunlimited; skt.nlimited += sktp[i].nlimited; skt.nmissed += sktp[i].nmissed; } KUNIT_ASSERT_EQ_MSG(test, skt.nunlimited + skt.nlimited, skt.nattempts, "Outcomes not equal to attempts"); KUNIT_ASSERT_EQ_MSG(test, skt.nlimited, skt.nmissed, "Misses not equal to limits"); } static struct kunit_case ratelimit_test_cases[] = { KUNIT_CASE_SLOW(test_ratelimit_smoke), KUNIT_CASE_SLOW(test_ratelimit_stress), {} }; static struct kunit_suite ratelimit_test_suite = { .name = "lib_ratelimit", .test_cases = ratelimit_test_cases, }; kunit_test_suites(&ratelimit_test_suite); MODULE_DESCRIPTION("___ratelimit() KUnit test suite"); MODULE_LICENSE("GPL");