1 /* SPDX-License-Identifier: GPL-2.0 */ 2 /* 3 * KUnit function redirection (static stubbing) API. 4 * 5 * Copyright (C) 2022, Google LLC. 6 * Author: David Gow <davidgow@google.com> 7 */ 8 #ifndef _KUNIT_STATIC_STUB_H 9 #define _KUNIT_STATIC_STUB_H 10 11 #if !IS_ENABLED(CONFIG_KUNIT) 12 13 /* If CONFIG_KUNIT is not enabled, these stubs quietly disappear. */ 14 #define KUNIT_STATIC_STUB_REDIRECT(real_fn_name, args...) do {} while (0) 15 16 #else 17 18 #include <kunit/test.h> 19 #include <kunit/test-bug.h> 20 21 #include <linux/compiler.h> /* for {un,}likely() */ 22 #include <linux/sched.h> /* for task_struct */ 23 24 25 /** 26 * KUNIT_STATIC_STUB_REDIRECT() - call a replacement 'static stub' if one exists 27 * @real_fn_name: The name of this function (as an identifier, not a string) 28 * @args: All of the arguments passed to this function 29 * 30 * This is a function prologue which is used to allow calls to the current 31 * function to be redirected by a KUnit test. KUnit tests can call 32 * kunit_activate_static_stub() to pass a replacement function in. The 33 * replacement function will be called by KUNIT_STATIC_STUB_REDIRECT(), which 34 * will then return from the function. If the caller is not in a KUnit context, 35 * the function will continue execution as normal. 36 * 37 * Example: 38 * 39 * .. code-block:: c 40 * 41 * int real_func(int n) 42 * { 43 * KUNIT_STATIC_STUB_REDIRECT(real_func, n); 44 * return 0; 45 * } 46 * 47 * int replacement_func(int n) 48 * { 49 * return 42; 50 * } 51 * 52 * void example_test(struct kunit *test) 53 * { 54 * kunit_activate_static_stub(test, real_func, replacement_func); 55 * KUNIT_EXPECT_EQ(test, real_func(1), 42); 56 * } 57 * 58 */ 59 #define KUNIT_STATIC_STUB_REDIRECT(real_fn_name, args...) \ 60 do { \ 61 typeof(&real_fn_name) replacement; \ 62 struct kunit *current_test = kunit_get_current_test(); \ 63 \ 64 if (likely(!current_test)) \ 65 break; \ 66 \ 67 replacement = kunit_hooks.get_static_stub_address(current_test, \ 68 &real_fn_name); \ 69 \ 70 if (unlikely(replacement)) \ 71 return replacement(args); \ 72 } while (0) 73 74 /* Helper function for kunit_activate_static_stub(). The macro does 75 * typechecking, so use it instead. 76 */ 77 void __kunit_activate_static_stub(struct kunit *test, 78 void *real_fn_addr, 79 void *replacement_addr); 80 81 /** 82 * kunit_activate_static_stub() - replace a function using static stubs. 83 * @test: A pointer to the 'struct kunit' test context for the current test. 84 * @real_fn_addr: The address of the function to replace. 85 * @replacement_addr: The address of the function to replace it with. 86 * 87 * When activated, calls to real_fn_addr from within this test (even if called 88 * indirectly) will instead call replacement_addr. The function pointed to by 89 * real_fn_addr must begin with the static stub prologue in 90 * KUNIT_STATIC_STUB_REDIRECT() for this to work. real_fn_addr and 91 * replacement_addr must have the same type. 92 * 93 * The redirection can be disabled again with kunit_deactivate_static_stub(). 94 */ 95 #define kunit_activate_static_stub(test, real_fn_addr, replacement_addr) do { \ 96 typecheck_fn(typeof(&real_fn_addr), replacement_addr); \ 97 __kunit_activate_static_stub(test, real_fn_addr, replacement_addr); \ 98 } while (0) 99 100 101 /** 102 * kunit_deactivate_static_stub() - disable a function redirection 103 * @test: A pointer to the 'struct kunit' test context for the current test. 104 * @real_fn_addr: The address of the function to no-longer redirect 105 * 106 * Deactivates a redirection configured with kunit_activate_static_stub(). After 107 * this function returns, calls to real_fn_addr() will execute the original 108 * real_fn, not any previously-configured replacement. 109 */ 110 void kunit_deactivate_static_stub(struct kunit *test, void *real_fn_addr); 111 112 #endif 113 #endif 114