| /* SPDX-License-Identifier: GPL-2.0 */ |
| /* |
| * KUnit function redirection (static stubbing) API. |
| * |
| * Copyright (C) 2022, Google LLC. |
| * Author: David Gow <davidgow@google.com> |
| */ |
| #ifndef _KUNIT_STATIC_STUB_H |
| #define _KUNIT_STATIC_STUB_H |
| |
| #if !IS_ENABLED(CONFIG_KUNIT) |
| |
| /* If CONFIG_KUNIT is not enabled, these stubs quietly disappear. */ |
| #define KUNIT_TRIGGER_STATIC_STUB(real_fn_name, args...) do {} while (0) |
| |
| #else |
| |
| #include <kunit/test.h> |
| #include <kunit/test-bug.h> |
| |
| #include <linux/compiler.h> /* for {un,}likely() */ |
| #include <linux/sched.h> /* for task_struct */ |
| |
| |
| /** |
| * KUNIT_STATIC_STUB_REDIRECT() - call a replacement 'static stub' if one exists |
| * @real_fn_name: The name of this function (as an identifier, not a string) |
| * @args: All of the arguments passed to this function |
| * |
| * This is a function prologue which is used to allow calls to the current |
| * function to be redirected by a KUnit test. KUnit tests can call |
| * kunit_activate_static_stub() to pass a replacement function in. The |
| * replacement function will be called by KUNIT_TRIGGER_STATIC_STUB(), which |
| * will then return from the function. If the caller is not in a KUnit context, |
| * the function will continue execution as normal. |
| * |
| * Example: |
| * |
| * .. code-block:: c |
| * |
| * int real_func(int n) |
| * { |
| * KUNIT_STATIC_STUB_REDIRECT(real_func, n); |
| * return 0; |
| * } |
| * |
| * int replacement_func(int n) |
| * { |
| * return 42; |
| * } |
| * |
| * void example_test(struct kunit *test) |
| * { |
| * kunit_activate_static_stub(test, real_func, replacement_func); |
| * KUNIT_EXPECT_EQ(test, real_func(1), 42); |
| * } |
| * |
| */ |
| #define KUNIT_STATIC_STUB_REDIRECT(real_fn_name, args...) \ |
| do { \ |
| typeof(&real_fn_name) replacement; \ |
| struct kunit *current_test = kunit_get_current_test(); \ |
| \ |
| if (likely(!current_test)) \ |
| break; \ |
| \ |
| replacement = kunit_hooks.get_static_stub_address(current_test, \ |
| &real_fn_name); \ |
| \ |
| if (unlikely(replacement)) \ |
| return replacement(args); \ |
| } while (0) |
| |
| /* Helper function for kunit_activate_static_stub(). The macro does |
| * typechecking, so use it instead. |
| */ |
| void __kunit_activate_static_stub(struct kunit *test, |
| void *real_fn_addr, |
| void *replacement_addr); |
| |
| /** |
| * kunit_activate_static_stub() - replace a function using static stubs. |
| * @test: A pointer to the 'struct kunit' test context for the current test. |
| * @real_fn_addr: The address of the function to replace. |
| * @replacement_addr: The address of the function to replace it with. |
| * |
| * When activated, calls to real_fn_addr from within this test (even if called |
| * indirectly) will instead call replacement_addr. The function pointed to by |
| * real_fn_addr must begin with the static stub prologue in |
| * KUNIT_TRIGGER_STATIC_STUB() for this to work. real_fn_addr and |
| * replacement_addr must have the same type. |
| * |
| * The redirection can be disabled again with kunit_deactivate_static_stub(). |
| */ |
| #define kunit_activate_static_stub(test, real_fn_addr, replacement_addr) do { \ |
| typecheck_fn(typeof(&real_fn_addr), replacement_addr); \ |
| __kunit_activate_static_stub(test, real_fn_addr, replacement_addr); \ |
| } while (0) |
| |
| |
| /** |
| * kunit_deactivate_static_stub() - disable a function redirection |
| * @test: A pointer to the 'struct kunit' test context for the current test. |
| * @real_fn_addr: The address of the function to no-longer redirect |
| * |
| * Deactivates a redirection configured with kunit_activate_static_stub(). After |
| * this function returns, calls to real_fn_addr() will execute the original |
| * real_fn, not any previously-configured replacement. |
| */ |
| void kunit_deactivate_static_stub(struct kunit *test, void *real_fn_addr); |
| |
| #endif |
| #endif |