blob: f9869ada5dcda255ccb9d459c7a98850959b6baa [file] [log] [blame]
// SPDX-License-Identifier: GPL-2.0
/*
* C++ stream style string formatter and printer used in KUnit for outputting
* KUnit messages.
*
* Copyright (C) 2018, Google LLC.
* Author: Brendan Higgins <brendanhiggins@google.com>
*/
#include <test/test.h>
#include <test/test-stream.h>
#include <test/string-stream.h>
static void test_stream_set_level(struct test_stream *this,
const char *level)
{
this->level = level;
}
static void test_stream_add(struct test_stream *this, const char *fmt, ...)
{
va_list args;
struct string_stream *stream = this->internal_stream;
va_start(args, fmt);
if (stream->vadd(stream, fmt, args) < 0)
test_err(this->test, "Failed to allocate fragment: %s", fmt);
va_end(args);
}
static void test_stream_append(struct test_stream *this,
struct test_stream *other)
{
struct string_stream *other_stream = other->internal_stream;
const char *other_content;
other_content = other_stream->get_string(other_stream);
if (!other_content) {
test_err(this->test,
"Failed to get string from second argument for appending.");
return;
}
this->add(this, other_content);
}
static void test_stream_clear(struct test_stream *this)
{
this->internal_stream->clear(this->internal_stream);
}
static void test_stream_commit(struct test_stream *this)
{
char *buf;
struct string_stream_fragment *fragment;
struct string_stream *stream = this->internal_stream;
if (!this->level) {
test_err(this->test,
"Stream was committed without a specified log level.");
this->set_level(this, KERN_ERR);
}
buf = stream->get_string(stream);
if (!buf) {
test_err(this->test,
"Could not allocate buffer, dumping stream:");
list_for_each_entry(fragment, &stream->fragments, node) {
test_err(this->test, fragment->fragment);
}
goto cleanup;
}
test_printk(this->level, this->test, buf);
kfree(buf);
cleanup:
this->clear(this);
}
static int test_stream_init(struct test_resource *res, void *context)
{
struct test_stream *stream;
struct test *test = context;
stream = kzalloc(sizeof(*stream), GFP_KERNEL);
if (!stream)
return -ENOMEM;
res->allocation = stream;
stream->test = test;
stream->level = NULL;
stream->internal_stream = new_string_stream();
if (!stream->internal_stream)
return -ENOMEM;
stream->set_level = test_stream_set_level;
stream->add = test_stream_add;
stream->append = test_stream_append;
stream->commit = test_stream_commit;
stream->clear = test_stream_clear;
return 0;
}
static void test_stream_free(struct test_resource *res)
{
struct test_stream *stream = res->allocation;
if (!stream->internal_stream->is_empty(stream->internal_stream)) {
test_err(stream->test,
"End of test case reached with uncommitted stream entries.");
stream->commit(stream);
}
destroy_string_stream(stream->internal_stream);
kfree(stream);
}
struct test_stream *test_new_stream(struct test *test)
{
struct test_resource *res;
res = test_alloc_resource(test,
test_stream_init,
test_stream_free,
test);
if (res)
return res->allocation;
else
return NULL;
}