d4cdd146d0
In its original form, the kunit resources API - consisting the struct kunit_resource and associated functions - was focused on adding allocated resources during test operation that would be automatically cleaned up on test completion. The recent RFC patch proposing converting KASAN tests to KUnit [1] showed another potential model - where outside of test context, but with a pointer to the test state, we wish to access/update test-related data, but expressly want to avoid allocations. It turns out we can generalize the kunit_resource to support static resources where the struct kunit_resource * is passed in and initialized for us. As part of this work, we also change the "allocation" field to the more general "data" name, as instead of associating an allocation, we can associate a pointer to static data. Static data is distinguished by a NULL free functions. A test is added to cover using kunit_add_resource() with a static resource and data. Finally we also make use of the kernel's krefcount interfaces to manage reference counting of KUnit resources. The motivation for this is simple; if we have kernel threads accessing and using resources (say via kunit_find_resource()) we need to ensure we do not remove said resources (or indeed free them if they were dynamically allocated) until the reference count reaches zero. A new function - kunit_put_resource() - is added to handle this, and it should be called after a thread using kunit_find_resource() is finished with the retrieved resource. We ensure that the functions needed to look up, use and drop reference count are "static inline"-defined so that they can be used by builtin code as well as modules in the case that KUnit is built as a module. A cosmetic change here also; I've tried moving to kunit_[action]_resource() as the format of function names for consistency and readability. [1] https://lkml.org/lkml/2020/2/26/1286 Signed-off-by: Alan Maguire <alan.maguire@oracle.com> Reviewed-by: Brendan Higgins <brendanhiggins@google.com> Signed-off-by: Shuah Khan <skhan@linuxfoundation.org>
217 lines
4.7 KiB
C
217 lines
4.7 KiB
C
// SPDX-License-Identifier: GPL-2.0
|
|
/*
|
|
* C++ stream style string builder used in KUnit for building messages.
|
|
*
|
|
* Copyright (C) 2019, Google LLC.
|
|
* Author: Brendan Higgins <brendanhiggins@google.com>
|
|
*/
|
|
|
|
#include <kunit/test.h>
|
|
#include <linux/list.h>
|
|
#include <linux/slab.h>
|
|
|
|
#include "string-stream.h"
|
|
|
|
struct string_stream_fragment_alloc_context {
|
|
struct kunit *test;
|
|
int len;
|
|
gfp_t gfp;
|
|
};
|
|
|
|
static int string_stream_fragment_init(struct kunit_resource *res,
|
|
void *context)
|
|
{
|
|
struct string_stream_fragment_alloc_context *ctx = context;
|
|
struct string_stream_fragment *frag;
|
|
|
|
frag = kunit_kzalloc(ctx->test, sizeof(*frag), ctx->gfp);
|
|
if (!frag)
|
|
return -ENOMEM;
|
|
|
|
frag->test = ctx->test;
|
|
frag->fragment = kunit_kmalloc(ctx->test, ctx->len, ctx->gfp);
|
|
if (!frag->fragment)
|
|
return -ENOMEM;
|
|
|
|
res->data = frag;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void string_stream_fragment_free(struct kunit_resource *res)
|
|
{
|
|
struct string_stream_fragment *frag = res->data;
|
|
|
|
list_del(&frag->node);
|
|
kunit_kfree(frag->test, frag->fragment);
|
|
kunit_kfree(frag->test, frag);
|
|
}
|
|
|
|
static struct string_stream_fragment *alloc_string_stream_fragment(
|
|
struct kunit *test, int len, gfp_t gfp)
|
|
{
|
|
struct string_stream_fragment_alloc_context context = {
|
|
.test = test,
|
|
.len = len,
|
|
.gfp = gfp
|
|
};
|
|
|
|
return kunit_alloc_resource(test,
|
|
string_stream_fragment_init,
|
|
string_stream_fragment_free,
|
|
gfp,
|
|
&context);
|
|
}
|
|
|
|
static int string_stream_fragment_destroy(struct string_stream_fragment *frag)
|
|
{
|
|
return kunit_destroy_resource(frag->test,
|
|
kunit_resource_instance_match,
|
|
frag);
|
|
}
|
|
|
|
int string_stream_vadd(struct string_stream *stream,
|
|
const char *fmt,
|
|
va_list args)
|
|
{
|
|
struct string_stream_fragment *frag_container;
|
|
int len;
|
|
va_list args_for_counting;
|
|
|
|
/* Make a copy because `vsnprintf` could change it */
|
|
va_copy(args_for_counting, args);
|
|
|
|
/* Need space for null byte. */
|
|
len = vsnprintf(NULL, 0, fmt, args_for_counting) + 1;
|
|
|
|
va_end(args_for_counting);
|
|
|
|
frag_container = alloc_string_stream_fragment(stream->test,
|
|
len,
|
|
stream->gfp);
|
|
if (!frag_container)
|
|
return -ENOMEM;
|
|
|
|
len = vsnprintf(frag_container->fragment, len, fmt, args);
|
|
spin_lock(&stream->lock);
|
|
stream->length += len;
|
|
list_add_tail(&frag_container->node, &stream->fragments);
|
|
spin_unlock(&stream->lock);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int string_stream_add(struct string_stream *stream, const char *fmt, ...)
|
|
{
|
|
va_list args;
|
|
int result;
|
|
|
|
va_start(args, fmt);
|
|
result = string_stream_vadd(stream, fmt, args);
|
|
va_end(args);
|
|
|
|
return result;
|
|
}
|
|
|
|
static void string_stream_clear(struct string_stream *stream)
|
|
{
|
|
struct string_stream_fragment *frag_container, *frag_container_safe;
|
|
|
|
spin_lock(&stream->lock);
|
|
list_for_each_entry_safe(frag_container,
|
|
frag_container_safe,
|
|
&stream->fragments,
|
|
node) {
|
|
string_stream_fragment_destroy(frag_container);
|
|
}
|
|
stream->length = 0;
|
|
spin_unlock(&stream->lock);
|
|
}
|
|
|
|
char *string_stream_get_string(struct string_stream *stream)
|
|
{
|
|
struct string_stream_fragment *frag_container;
|
|
size_t buf_len = stream->length + 1; /* +1 for null byte. */
|
|
char *buf;
|
|
|
|
buf = kunit_kzalloc(stream->test, buf_len, stream->gfp);
|
|
if (!buf)
|
|
return NULL;
|
|
|
|
spin_lock(&stream->lock);
|
|
list_for_each_entry(frag_container, &stream->fragments, node)
|
|
strlcat(buf, frag_container->fragment, buf_len);
|
|
spin_unlock(&stream->lock);
|
|
|
|
return buf;
|
|
}
|
|
|
|
int string_stream_append(struct string_stream *stream,
|
|
struct string_stream *other)
|
|
{
|
|
const char *other_content;
|
|
|
|
other_content = string_stream_get_string(other);
|
|
|
|
if (!other_content)
|
|
return -ENOMEM;
|
|
|
|
return string_stream_add(stream, other_content);
|
|
}
|
|
|
|
bool string_stream_is_empty(struct string_stream *stream)
|
|
{
|
|
return list_empty(&stream->fragments);
|
|
}
|
|
|
|
struct string_stream_alloc_context {
|
|
struct kunit *test;
|
|
gfp_t gfp;
|
|
};
|
|
|
|
static int string_stream_init(struct kunit_resource *res, void *context)
|
|
{
|
|
struct string_stream *stream;
|
|
struct string_stream_alloc_context *ctx = context;
|
|
|
|
stream = kunit_kzalloc(ctx->test, sizeof(*stream), ctx->gfp);
|
|
if (!stream)
|
|
return -ENOMEM;
|
|
|
|
res->data = stream;
|
|
stream->gfp = ctx->gfp;
|
|
stream->test = ctx->test;
|
|
INIT_LIST_HEAD(&stream->fragments);
|
|
spin_lock_init(&stream->lock);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void string_stream_free(struct kunit_resource *res)
|
|
{
|
|
struct string_stream *stream = res->data;
|
|
|
|
string_stream_clear(stream);
|
|
}
|
|
|
|
struct string_stream *alloc_string_stream(struct kunit *test, gfp_t gfp)
|
|
{
|
|
struct string_stream_alloc_context context = {
|
|
.test = test,
|
|
.gfp = gfp
|
|
};
|
|
|
|
return kunit_alloc_resource(test,
|
|
string_stream_init,
|
|
string_stream_free,
|
|
gfp,
|
|
&context);
|
|
}
|
|
|
|
int string_stream_destroy(struct string_stream *stream)
|
|
{
|
|
return kunit_destroy_resource(stream->test,
|
|
kunit_resource_instance_match,
|
|
stream);
|
|
}
|