2022-03-28 17:41:43 +00:00
|
|
|
// SPDX-License-Identifier: GPL-2.0
|
|
|
|
/*
|
|
|
|
* KUnit resource API for test managed resources (allocations, etc.).
|
|
|
|
*
|
|
|
|
* Copyright (C) 2022, Google LLC.
|
|
|
|
* Author: Daniel Latypov <dlatypov@google.com>
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <kunit/resource.h>
|
|
|
|
#include <kunit/test.h>
|
|
|
|
#include <linux/kref.h>
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Used for static resources and when a kunit_resource * has been created by
|
|
|
|
* kunit_alloc_resource(). When an init function is supplied, @data is passed
|
|
|
|
* into the init function; otherwise, we simply set the resource data field to
|
kunit: Rework kunit_resource allocation policy
KUnit's test-managed resources can be created in two ways:
- Using the kunit_add_resource() family of functions, which accept a
struct kunit_resource pointer, typically allocated statically or on
the stack during the test.
- Using the kunit_alloc_resource() family of functions, which allocate a
struct kunit_resource using kzalloc() behind the scenes.
Both of these families of functions accept a 'free' function to be
called when the resource is finally disposed of.
At present, KUnit will kfree() the resource if this 'free' function is
specified, and will not if it is NULL. However, this can lead
kunit_alloc_resource() to leak memory (if no 'free' function is passed
in), or kunit_add_resource() to incorrectly kfree() memory which was
allocated by some other means (on the stack, as part of a larger
allocation, etc), if a 'free' function is provided.
Instead, always kfree() if the resource was allocated with
kunit_alloc_resource(), and never kfree() if it was passed into
kunit_add_resource() by the user. (If the user of kunit_add_resource()
wishes the resource be kfree()ed, they can call kfree() on the resource
from within the 'free' function.
This is implemented by adding a 'should_free' member to
struct kunit_resource and setting it appropriately. To facilitate this,
the various resource add/alloc functions have been refactored somewhat,
making them all call a __kunit_add_resource() helper after setting the
'should_free' member appropriately. In the process, all other functions
have been made static inline functions.
Signed-off-by: David Gow <davidgow@google.com>
Tested-by: Daniel Latypov <dlatypov@google.com>
Reviewed-by: Brendan Higgins <brendanhiggins@google.com>
Signed-off-by: Shuah Khan <skhan@linuxfoundation.org>
2022-04-02 04:35:30 +00:00
|
|
|
* the data value passed in. Doesn't initialize res->should_kfree.
|
2022-03-28 17:41:43 +00:00
|
|
|
*/
|
kunit: Rework kunit_resource allocation policy
KUnit's test-managed resources can be created in two ways:
- Using the kunit_add_resource() family of functions, which accept a
struct kunit_resource pointer, typically allocated statically or on
the stack during the test.
- Using the kunit_alloc_resource() family of functions, which allocate a
struct kunit_resource using kzalloc() behind the scenes.
Both of these families of functions accept a 'free' function to be
called when the resource is finally disposed of.
At present, KUnit will kfree() the resource if this 'free' function is
specified, and will not if it is NULL. However, this can lead
kunit_alloc_resource() to leak memory (if no 'free' function is passed
in), or kunit_add_resource() to incorrectly kfree() memory which was
allocated by some other means (on the stack, as part of a larger
allocation, etc), if a 'free' function is provided.
Instead, always kfree() if the resource was allocated with
kunit_alloc_resource(), and never kfree() if it was passed into
kunit_add_resource() by the user. (If the user of kunit_add_resource()
wishes the resource be kfree()ed, they can call kfree() on the resource
from within the 'free' function.
This is implemented by adding a 'should_free' member to
struct kunit_resource and setting it appropriately. To facilitate this,
the various resource add/alloc functions have been refactored somewhat,
making them all call a __kunit_add_resource() helper after setting the
'should_free' member appropriately. In the process, all other functions
have been made static inline functions.
Signed-off-by: David Gow <davidgow@google.com>
Tested-by: Daniel Latypov <dlatypov@google.com>
Reviewed-by: Brendan Higgins <brendanhiggins@google.com>
Signed-off-by: Shuah Khan <skhan@linuxfoundation.org>
2022-04-02 04:35:30 +00:00
|
|
|
int __kunit_add_resource(struct kunit *test,
|
|
|
|
kunit_resource_init_t init,
|
|
|
|
kunit_resource_free_t free,
|
|
|
|
struct kunit_resource *res,
|
|
|
|
void *data)
|
2022-03-28 17:41:43 +00:00
|
|
|
{
|
|
|
|
int ret = 0;
|
|
|
|
unsigned long flags;
|
|
|
|
|
|
|
|
res->free = free;
|
|
|
|
kref_init(&res->refcount);
|
|
|
|
|
|
|
|
if (init) {
|
|
|
|
ret = init(res, data);
|
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
} else {
|
|
|
|
res->data = data;
|
|
|
|
}
|
|
|
|
|
|
|
|
spin_lock_irqsave(&test->lock, flags);
|
|
|
|
list_add_tail(&res->node, &test->resources);
|
|
|
|
/* refcount for list is established by kref_init() */
|
|
|
|
spin_unlock_irqrestore(&test->lock, flags);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
kunit: Rework kunit_resource allocation policy
KUnit's test-managed resources can be created in two ways:
- Using the kunit_add_resource() family of functions, which accept a
struct kunit_resource pointer, typically allocated statically or on
the stack during the test.
- Using the kunit_alloc_resource() family of functions, which allocate a
struct kunit_resource using kzalloc() behind the scenes.
Both of these families of functions accept a 'free' function to be
called when the resource is finally disposed of.
At present, KUnit will kfree() the resource if this 'free' function is
specified, and will not if it is NULL. However, this can lead
kunit_alloc_resource() to leak memory (if no 'free' function is passed
in), or kunit_add_resource() to incorrectly kfree() memory which was
allocated by some other means (on the stack, as part of a larger
allocation, etc), if a 'free' function is provided.
Instead, always kfree() if the resource was allocated with
kunit_alloc_resource(), and never kfree() if it was passed into
kunit_add_resource() by the user. (If the user of kunit_add_resource()
wishes the resource be kfree()ed, they can call kfree() on the resource
from within the 'free' function.
This is implemented by adding a 'should_free' member to
struct kunit_resource and setting it appropriately. To facilitate this,
the various resource add/alloc functions have been refactored somewhat,
making them all call a __kunit_add_resource() helper after setting the
'should_free' member appropriately. In the process, all other functions
have been made static inline functions.
Signed-off-by: David Gow <davidgow@google.com>
Tested-by: Daniel Latypov <dlatypov@google.com>
Reviewed-by: Brendan Higgins <brendanhiggins@google.com>
Signed-off-by: Shuah Khan <skhan@linuxfoundation.org>
2022-04-02 04:35:30 +00:00
|
|
|
EXPORT_SYMBOL_GPL(__kunit_add_resource);
|
2022-03-28 17:41:43 +00:00
|
|
|
|
|
|
|
void kunit_remove_resource(struct kunit *test, struct kunit_resource *res)
|
|
|
|
{
|
|
|
|
unsigned long flags;
|
2022-04-02 04:35:29 +00:00
|
|
|
bool was_linked;
|
2022-03-28 17:41:43 +00:00
|
|
|
|
|
|
|
spin_lock_irqsave(&test->lock, flags);
|
2022-04-02 04:35:29 +00:00
|
|
|
was_linked = !list_empty(&res->node);
|
|
|
|
list_del_init(&res->node);
|
2022-03-28 17:41:43 +00:00
|
|
|
spin_unlock_irqrestore(&test->lock, flags);
|
2022-04-02 04:35:29 +00:00
|
|
|
|
|
|
|
if (was_linked)
|
|
|
|
kunit_put_resource(res);
|
2022-03-28 17:41:43 +00:00
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(kunit_remove_resource);
|
|
|
|
|
|
|
|
int kunit_destroy_resource(struct kunit *test, kunit_resource_match_t match,
|
|
|
|
void *match_data)
|
|
|
|
{
|
|
|
|
struct kunit_resource *res = kunit_find_resource(test, match,
|
|
|
|
match_data);
|
|
|
|
|
|
|
|
if (!res)
|
|
|
|
return -ENOENT;
|
|
|
|
|
|
|
|
kunit_remove_resource(test, res);
|
|
|
|
|
|
|
|
/* We have a reference also via _find(); drop it. */
|
|
|
|
kunit_put_resource(res);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(kunit_destroy_resource);
|
kunit: Add kunit_add_action() to defer a call until test exit
Many uses of the KUnit resource system are intended to simply defer
calling a function until the test exits (be it due to success or
failure). The existing kunit_alloc_resource() function is often used for
this, but was awkward to use (requiring passing NULL init functions, etc),
and returned a resource without incrementing its reference count, which
-- while okay for this use-case -- could cause problems in others.
Instead, introduce a simple kunit_add_action() API: a simple function
(returning nothing, accepting a single void* argument) can be scheduled
to be called when the test exits. Deferred actions are called in the
opposite order to that which they were registered.
This mimics the devres API, devm_add_action(), and also provides
kunit_remove_action(), to cancel a deferred action, and
kunit_release_action() to trigger one early.
This is implemented as a resource under the hood, so the ordering
between resource cleanup and deferred functions is maintained.
Reviewed-by: Benjamin Berg <benjamin.berg@intel.com>
Reviewed-by: Maxime Ripard <maxime@cerno.tech>
Tested-by: Maxime Ripard <maxime@cerno.tech>
Signed-off-by: David Gow <davidgow@google.com>
Signed-off-by: Shuah Khan <skhan@linuxfoundation.org>
2023-05-25 04:21:28 +00:00
|
|
|
|
|
|
|
struct kunit_action_ctx {
|
|
|
|
struct kunit_resource res;
|
|
|
|
kunit_action_t *func;
|
|
|
|
void *ctx;
|
|
|
|
};
|
|
|
|
|
|
|
|
static void __kunit_action_free(struct kunit_resource *res)
|
|
|
|
{
|
|
|
|
struct kunit_action_ctx *action_ctx = container_of(res, struct kunit_action_ctx, res);
|
|
|
|
|
|
|
|
action_ctx->func(action_ctx->ctx);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int kunit_add_action(struct kunit *test, void (*action)(void *), void *ctx)
|
|
|
|
{
|
|
|
|
struct kunit_action_ctx *action_ctx;
|
|
|
|
|
|
|
|
KUNIT_ASSERT_NOT_NULL_MSG(test, action, "Tried to action a NULL function!");
|
|
|
|
|
|
|
|
action_ctx = kzalloc(sizeof(*action_ctx), GFP_KERNEL);
|
|
|
|
if (!action_ctx)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
action_ctx->func = action;
|
|
|
|
action_ctx->ctx = ctx;
|
|
|
|
|
|
|
|
action_ctx->res.should_kfree = true;
|
|
|
|
/* As init is NULL, this cannot fail. */
|
|
|
|
__kunit_add_resource(test, NULL, __kunit_action_free, &action_ctx->res, action_ctx);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(kunit_add_action);
|
|
|
|
|
|
|
|
int kunit_add_action_or_reset(struct kunit *test, void (*action)(void *),
|
|
|
|
void *ctx)
|
|
|
|
{
|
|
|
|
int res = kunit_add_action(test, action, ctx);
|
|
|
|
|
|
|
|
if (res)
|
|
|
|
action(ctx);
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(kunit_add_action_or_reset);
|
|
|
|
|
|
|
|
static bool __kunit_action_match(struct kunit *test,
|
|
|
|
struct kunit_resource *res, void *match_data)
|
|
|
|
{
|
|
|
|
struct kunit_action_ctx *match_ctx = (struct kunit_action_ctx *)match_data;
|
|
|
|
struct kunit_action_ctx *res_ctx = container_of(res, struct kunit_action_ctx, res);
|
|
|
|
|
|
|
|
/* Make sure this is a free function. */
|
|
|
|
if (res->free != __kunit_action_free)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
/* Both the function and context data should match. */
|
|
|
|
return (match_ctx->func == res_ctx->func) && (match_ctx->ctx == res_ctx->ctx);
|
|
|
|
}
|
|
|
|
|
|
|
|
void kunit_remove_action(struct kunit *test,
|
|
|
|
kunit_action_t *action,
|
|
|
|
void *ctx)
|
|
|
|
{
|
|
|
|
struct kunit_action_ctx match_ctx;
|
|
|
|
struct kunit_resource *res;
|
|
|
|
|
|
|
|
match_ctx.func = action;
|
|
|
|
match_ctx.ctx = ctx;
|
|
|
|
|
|
|
|
res = kunit_find_resource(test, __kunit_action_match, &match_ctx);
|
|
|
|
if (res) {
|
|
|
|
/* Remove the free function so we don't run the action. */
|
|
|
|
res->free = NULL;
|
|
|
|
kunit_remove_resource(test, res);
|
|
|
|
kunit_put_resource(res);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(kunit_remove_action);
|
|
|
|
|
|
|
|
void kunit_release_action(struct kunit *test,
|
|
|
|
kunit_action_t *action,
|
|
|
|
void *ctx)
|
|
|
|
{
|
|
|
|
struct kunit_action_ctx match_ctx;
|
|
|
|
struct kunit_resource *res;
|
|
|
|
|
|
|
|
match_ctx.func = action;
|
|
|
|
match_ctx.ctx = ctx;
|
|
|
|
|
|
|
|
res = kunit_find_resource(test, __kunit_action_match, &match_ctx);
|
|
|
|
if (res) {
|
|
|
|
kunit_remove_resource(test, res);
|
|
|
|
/* We have to put() this here, else free won't be called. */
|
|
|
|
kunit_put_resource(res);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(kunit_release_action);
|