Commit e44e81c5 authored by Sven Schnelle's avatar Sven Schnelle Committed by Steven Rostedt (VMware)

kprobes: convert tests to kunit

This converts the kprobes testcases to use the kunit framework.
It adds a dependency on CONFIG_KUNIT, and the output will change
to TAP:

TAP version 14
1..1
    # Subtest: kprobes_test
    1..4
random: crng init done
    ok 1 - test_kprobe
    ok 2 - test_kprobes
    ok 3 - test_kretprobe
    ok 4 - test_kretprobes
ok 1 - kprobes_test

Note that the kprobes testcases are no longer run immediately after
kprobes initialization, but as a late initcall when kunit is
initialized. kprobes itself is initialized with an early initcall,
so the order is still correct.
Signed-off-by: default avatarSven Schnelle <svens@linux.ibm.com>
Acked-by: default avatarMasami Hiramatsu <mhiramat@kernel.org>
Signed-off-by: default avatarMasami Hiramatsu <mhiramat@kernel.org>
Signed-off-by: default avatarSteven Rostedt (VMware) <rostedt@goodmis.org>
parent 8720aeec
...@@ -2581,9 +2581,6 @@ static int __init init_kprobes(void) ...@@ -2581,9 +2581,6 @@ static int __init init_kprobes(void)
err = register_module_notifier(&kprobe_module_nb); err = register_module_notifier(&kprobe_module_nb);
kprobes_initialized = (err == 0); kprobes_initialized = (err == 0);
if (!err)
init_test_probes();
return err; return err;
} }
early_initcall(init_kprobes); early_initcall(init_kprobes);
......
...@@ -5,18 +5,17 @@ ...@@ -5,18 +5,17 @@
* Copyright IBM Corp. 2008 * Copyright IBM Corp. 2008
*/ */
#define pr_fmt(fmt) "Kprobe smoke test: " fmt
#include <linux/kernel.h> #include <linux/kernel.h>
#include <linux/kprobes.h> #include <linux/kprobes.h>
#include <linux/random.h> #include <linux/random.h>
#include <kunit/test.h>
#define div_factor 3 #define div_factor 3
static u32 rand1, preh_val, posth_val; static u32 rand1, preh_val, posth_val;
static int errors, handler_errors, num_tests;
static u32 (*target)(u32 value); static u32 (*target)(u32 value);
static u32 (*target2)(u32 value); static u32 (*target2)(u32 value);
static struct kunit *current_test;
static noinline u32 kprobe_target(u32 value) static noinline u32 kprobe_target(u32 value)
{ {
...@@ -25,10 +24,7 @@ static noinline u32 kprobe_target(u32 value) ...@@ -25,10 +24,7 @@ static noinline u32 kprobe_target(u32 value)
static int kp_pre_handler(struct kprobe *p, struct pt_regs *regs) static int kp_pre_handler(struct kprobe *p, struct pt_regs *regs)
{ {
if (preemptible()) { KUNIT_EXPECT_FALSE(current_test, preemptible());
handler_errors++;
pr_err("pre-handler is preemptible\n");
}
preh_val = (rand1 / div_factor); preh_val = (rand1 / div_factor);
return 0; return 0;
} }
...@@ -36,14 +32,8 @@ static int kp_pre_handler(struct kprobe *p, struct pt_regs *regs) ...@@ -36,14 +32,8 @@ static int kp_pre_handler(struct kprobe *p, struct pt_regs *regs)
static void kp_post_handler(struct kprobe *p, struct pt_regs *regs, static void kp_post_handler(struct kprobe *p, struct pt_regs *regs,
unsigned long flags) unsigned long flags)
{ {
if (preemptible()) { KUNIT_EXPECT_FALSE(current_test, preemptible());
handler_errors++; KUNIT_EXPECT_EQ(current_test, preh_val, (rand1 / div_factor));
pr_err("post-handler is preemptible\n");
}
if (preh_val != (rand1 / div_factor)) {
handler_errors++;
pr_err("incorrect value in post_handler\n");
}
posth_val = preh_val + div_factor; posth_val = preh_val + div_factor;
} }
...@@ -53,30 +43,14 @@ static struct kprobe kp = { ...@@ -53,30 +43,14 @@ static struct kprobe kp = {
.post_handler = kp_post_handler .post_handler = kp_post_handler
}; };
static int test_kprobe(void) static void test_kprobe(struct kunit *test)
{ {
int ret; current_test = test;
KUNIT_EXPECT_EQ(test, 0, register_kprobe(&kp));
ret = register_kprobe(&kp); target(rand1);
if (ret < 0) {
pr_err("register_kprobe returned %d\n", ret);
return ret;
}
ret = target(rand1);
unregister_kprobe(&kp); unregister_kprobe(&kp);
KUNIT_EXPECT_NE(test, 0, preh_val);
if (preh_val == 0) { KUNIT_EXPECT_NE(test, 0, posth_val);
pr_err("kprobe pre_handler not called\n");
handler_errors++;
}
if (posth_val == 0) {
pr_err("kprobe post_handler not called\n");
handler_errors++;
}
return 0;
} }
static noinline u32 kprobe_target2(u32 value) static noinline u32 kprobe_target2(u32 value)
...@@ -93,10 +67,7 @@ static int kp_pre_handler2(struct kprobe *p, struct pt_regs *regs) ...@@ -93,10 +67,7 @@ static int kp_pre_handler2(struct kprobe *p, struct pt_regs *regs)
static void kp_post_handler2(struct kprobe *p, struct pt_regs *regs, static void kp_post_handler2(struct kprobe *p, struct pt_regs *regs,
unsigned long flags) unsigned long flags)
{ {
if (preh_val != (rand1 / div_factor) + 1) { KUNIT_EXPECT_EQ(current_test, preh_val, (rand1 / div_factor) + 1);
handler_errors++;
pr_err("incorrect value in post_handler2\n");
}
posth_val = preh_val + div_factor; posth_val = preh_val + div_factor;
} }
...@@ -106,51 +77,31 @@ static struct kprobe kp2 = { ...@@ -106,51 +77,31 @@ static struct kprobe kp2 = {
.post_handler = kp_post_handler2 .post_handler = kp_post_handler2
}; };
static int test_kprobes(void) static void test_kprobes(struct kunit *test)
{ {
int ret;
struct kprobe *kps[2] = {&kp, &kp2}; struct kprobe *kps[2] = {&kp, &kp2};
current_test = test;
/* addr and flags should be cleard for reusing kprobe. */ /* addr and flags should be cleard for reusing kprobe. */
kp.addr = NULL; kp.addr = NULL;
kp.flags = 0; kp.flags = 0;
ret = register_kprobes(kps, 2);
if (ret < 0) {
pr_err("register_kprobes returned %d\n", ret);
return ret;
}
KUNIT_EXPECT_EQ(test, 0, register_kprobes(kps, 2));
preh_val = 0; preh_val = 0;
posth_val = 0; posth_val = 0;
ret = target(rand1); target(rand1);
if (preh_val == 0) { KUNIT_EXPECT_NE(test, 0, preh_val);
pr_err("kprobe pre_handler not called\n"); KUNIT_EXPECT_NE(test, 0, posth_val);
handler_errors++;
}
if (posth_val == 0) {
pr_err("kprobe post_handler not called\n");
handler_errors++;
}
preh_val = 0; preh_val = 0;
posth_val = 0; posth_val = 0;
ret = target2(rand1); target2(rand1);
if (preh_val == 0) {
pr_err("kprobe pre_handler2 not called\n");
handler_errors++;
}
if (posth_val == 0) {
pr_err("kprobe post_handler2 not called\n");
handler_errors++;
}
KUNIT_EXPECT_NE(test, 0, preh_val);
KUNIT_EXPECT_NE(test, 0, posth_val);
unregister_kprobes(kps, 2); unregister_kprobes(kps, 2);
return 0;
} }
#ifdef CONFIG_KRETPROBES #ifdef CONFIG_KRETPROBES
...@@ -158,10 +109,7 @@ static u32 krph_val; ...@@ -158,10 +109,7 @@ static u32 krph_val;
static int entry_handler(struct kretprobe_instance *ri, struct pt_regs *regs) static int entry_handler(struct kretprobe_instance *ri, struct pt_regs *regs)
{ {
if (preemptible()) { KUNIT_EXPECT_FALSE(current_test, preemptible());
handler_errors++;
pr_err("kretprobe entry handler is preemptible\n");
}
krph_val = (rand1 / div_factor); krph_val = (rand1 / div_factor);
return 0; return 0;
} }
...@@ -170,19 +118,9 @@ static int return_handler(struct kretprobe_instance *ri, struct pt_regs *regs) ...@@ -170,19 +118,9 @@ static int return_handler(struct kretprobe_instance *ri, struct pt_regs *regs)
{ {
unsigned long ret = regs_return_value(regs); unsigned long ret = regs_return_value(regs);
if (preemptible()) { KUNIT_EXPECT_FALSE(current_test, preemptible());
handler_errors++; KUNIT_EXPECT_EQ(current_test, ret, rand1 / div_factor);
pr_err("kretprobe return handler is preemptible\n"); KUNIT_EXPECT_NE(current_test, krph_val, 0);
}
if (ret != (rand1 / div_factor)) {
handler_errors++;
pr_err("incorrect value in kretprobe handler\n");
}
if (krph_val == 0) {
handler_errors++;
pr_err("call to kretprobe entry handler failed\n");
}
krph_val = rand1; krph_val = rand1;
return 0; return 0;
} }
...@@ -193,39 +131,21 @@ static struct kretprobe rp = { ...@@ -193,39 +131,21 @@ static struct kretprobe rp = {
.kp.symbol_name = "kprobe_target" .kp.symbol_name = "kprobe_target"
}; };
static int test_kretprobe(void) static void test_kretprobe(struct kunit *test)
{ {
int ret; current_test = test;
KUNIT_EXPECT_EQ(test, 0, register_kretprobe(&rp));
ret = register_kretprobe(&rp); target(rand1);
if (ret < 0) {
pr_err("register_kretprobe returned %d\n", ret);
return ret;
}
ret = target(rand1);
unregister_kretprobe(&rp); unregister_kretprobe(&rp);
if (krph_val != rand1) { KUNIT_EXPECT_EQ(test, krph_val, rand1);
pr_err("kretprobe handler not called\n");
handler_errors++;
}
return 0;
} }
static int return_handler2(struct kretprobe_instance *ri, struct pt_regs *regs) static int return_handler2(struct kretprobe_instance *ri, struct pt_regs *regs)
{ {
unsigned long ret = regs_return_value(regs); unsigned long ret = regs_return_value(regs);
if (ret != (rand1 / div_factor) + 1) { KUNIT_EXPECT_EQ(current_test, ret, (rand1 / div_factor) + 1);
handler_errors++; KUNIT_EXPECT_NE(current_test, krph_val, 0);
pr_err("incorrect value in kretprobe handler2\n");
}
if (krph_val == 0) {
handler_errors++;
pr_err("call to kretprobe entry handler failed\n");
}
krph_val = rand1; krph_val = rand1;
return 0; return 0;
} }
...@@ -236,78 +156,54 @@ static struct kretprobe rp2 = { ...@@ -236,78 +156,54 @@ static struct kretprobe rp2 = {
.kp.symbol_name = "kprobe_target2" .kp.symbol_name = "kprobe_target2"
}; };
static int test_kretprobes(void) static void test_kretprobes(struct kunit *test)
{ {
int ret;
struct kretprobe *rps[2] = {&rp, &rp2}; struct kretprobe *rps[2] = {&rp, &rp2};
current_test = test;
/* addr and flags should be cleard for reusing kprobe. */ /* addr and flags should be cleard for reusing kprobe. */
rp.kp.addr = NULL; rp.kp.addr = NULL;
rp.kp.flags = 0; rp.kp.flags = 0;
ret = register_kretprobes(rps, 2); KUNIT_EXPECT_EQ(test, 0, register_kretprobes(rps, 2));
if (ret < 0) {
pr_err("register_kretprobe returned %d\n", ret);
return ret;
}
krph_val = 0; krph_val = 0;
ret = target(rand1); target(rand1);
if (krph_val != rand1) { KUNIT_EXPECT_EQ(test, krph_val, rand1);
pr_err("kretprobe handler not called\n");
handler_errors++;
}
krph_val = 0; krph_val = 0;
ret = target2(rand1); target2(rand1);
if (krph_val != rand1) { KUNIT_EXPECT_EQ(test, krph_val, rand1);
pr_err("kretprobe handler2 not called\n");
handler_errors++;
}
unregister_kretprobes(rps, 2); unregister_kretprobes(rps, 2);
return 0;
} }
#endif /* CONFIG_KRETPROBES */ #endif /* CONFIG_KRETPROBES */
int init_test_probes(void) static int kprobes_test_init(struct kunit *test)
{ {
int ret;
target = kprobe_target; target = kprobe_target;
target2 = kprobe_target2; target2 = kprobe_target2;
do { do {
rand1 = prandom_u32(); rand1 = prandom_u32();
} while (rand1 <= div_factor); } while (rand1 <= div_factor);
return 0;
}
pr_info("started\n"); static struct kunit_case kprobes_testcases[] = {
num_tests++; KUNIT_CASE(test_kprobe),
ret = test_kprobe(); KUNIT_CASE(test_kprobes),
if (ret < 0)
errors++;
num_tests++;
ret = test_kprobes();
if (ret < 0)
errors++;
#ifdef CONFIG_KRETPROBES #ifdef CONFIG_KRETPROBES
num_tests++; KUNIT_CASE(test_kretprobe),
ret = test_kretprobe(); KUNIT_CASE(test_kretprobes),
if (ret < 0) #endif
errors++; {}
};
num_tests++;
ret = test_kretprobes();
if (ret < 0)
errors++;
#endif /* CONFIG_KRETPROBES */
if (errors) static struct kunit_suite kprobes_test_suite = {
pr_err("BUG: %d out of %d tests failed\n", errors, num_tests); .name = "kprobes_test",
else if (handler_errors) .init = kprobes_test_init,
pr_err("BUG: %d error(s) running handlers\n", handler_errors); .test_cases = kprobes_testcases,
else };
pr_info("passed successfully\n");
return 0; kunit_test_suites(&kprobes_test_suite);
}
MODULE_LICENSE("GPL");
...@@ -2080,9 +2080,10 @@ config TEST_DIV64 ...@@ -2080,9 +2080,10 @@ config TEST_DIV64
If unsure, say N. If unsure, say N.
config KPROBES_SANITY_TEST config KPROBES_SANITY_TEST
bool "Kprobes sanity tests" tristate "Kprobes sanity tests"
depends on DEBUG_KERNEL depends on DEBUG_KERNEL
depends on KPROBES depends on KPROBES
depends on KUNIT
help help
This option provides for testing basic kprobes functionality on This option provides for testing basic kprobes functionality on
boot. Samples of kprobe and kretprobe are inserted and boot. Samples of kprobe and kretprobe are inserted and
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment