From: Jason Gunthorpe <jgg@nvidia.com>
To: David Airlie <airlied@gmail.com>,
Christian König <christian.koenig@amd.com>,
dri-devel@lists.freedesktop.org, intel-gfx@lists.freedesktop.org,
Jani Nikula <jani.nikula@linux.intel.com>,
Joonas Lahtinen <joonas.lahtinen@linux.intel.com>,
linaro-mm-sig@lists.linaro.org, linux-media@vger.kernel.org,
Rodrigo Vivi <rodrigo.vivi@intel.com>,
Simona Vetter <simona@ffwll.ch>,
Sumit Semwal <sumit.semwal@linaro.org>,
Tvrtko Ursulin <tursulin@ursulin.net>
Cc: patches@lists.linux.dev
Subject: [PATCH 1/5] dma-buf: Change st-dma-resv.c to use kunit
Date: Sun, 1 Mar 2026 14:57:53 -0400 [thread overview]
Message-ID: <1-v1-0a349a394eff+14110-dmabuf_kunit_jgg@nvidia.com> (raw)
In-Reply-To: <0-v1-0a349a394eff+14110-dmabuf_kunit_jgg@nvidia.com>
Modernize the open coded test framework by using kunit.
The kunit tool can be used to build a kernel and run it in a VM with:
$ tools/testing/kunit/kunit.py run --build_dir build_kunit_x86_64 --arch x86_64 --kunitconfig ./drivers/dma-buf/.kunitconfig
Along with the other ways to run kunits.
To make the kunit tool work like this the DMABUF_KUNIT_TEST kconfig must
select DMA_SHARED_BUFFER to get it turned on without building a driver
using it.
Signed-off-by: Jason Gunthorpe <jgg@nvidia.com>
---
drivers/dma-buf/.kunitconfig | 2 +
drivers/dma-buf/Kconfig | 8 ++
drivers/dma-buf/Makefile | 8 +-
drivers/dma-buf/selftests.h | 1 -
drivers/dma-buf/st-dma-resv.c | 145 +++++++++++++++++-----------------
5 files changed, 88 insertions(+), 76 deletions(-)
create mode 100644 drivers/dma-buf/.kunitconfig
diff --git a/drivers/dma-buf/.kunitconfig b/drivers/dma-buf/.kunitconfig
new file mode 100644
index 00000000000000..1ce5fb7e6cf9ff
--- /dev/null
+++ b/drivers/dma-buf/.kunitconfig
@@ -0,0 +1,2 @@
+CONFIG_KUNIT=y
+CONFIG_DMABUF_KUNIT_TEST=y
diff --git a/drivers/dma-buf/Kconfig b/drivers/dma-buf/Kconfig
index 8d4f2f89f24e3c..7d13c8f4484dd3 100644
--- a/drivers/dma-buf/Kconfig
+++ b/drivers/dma-buf/Kconfig
@@ -54,6 +54,14 @@ config DMABUF_SELFTESTS
default n
depends on DMA_SHARED_BUFFER
+config DMABUF_KUNIT_TEST
+ tristate "KUnit tests for DMA-BUF" if !KUNIT_ALL_TESTS
+ depends on KUNIT
+ select DMA_SHARED_BUFFER
+ default KUNIT_ALL_TESTS
+ help
+ Enable kunit tests for DMA-BUF
+
menuconfig DMABUF_HEAPS
bool "DMA-BUF Userland Memory Heaps"
select DMA_SHARED_BUFFER
diff --git a/drivers/dma-buf/Makefile b/drivers/dma-buf/Makefile
index 7a85565d906ba1..2e7a1453e2fe04 100644
--- a/drivers/dma-buf/Makefile
+++ b/drivers/dma-buf/Makefile
@@ -11,7 +11,11 @@ dmabuf_selftests-y := \
selftest.o \
st-dma-fence.o \
st-dma-fence-chain.o \
- st-dma-fence-unwrap.o \
- st-dma-resv.o
+ st-dma-fence-unwrap.o
obj-$(CONFIG_DMABUF_SELFTESTS) += dmabuf_selftests.o
+
+dmabuf_kunit-y := \
+ st-dma-resv.o
+
+obj-$(CONFIG_DMABUF_KUNIT_TEST) += dmabuf_kunit.o
diff --git a/drivers/dma-buf/selftests.h b/drivers/dma-buf/selftests.h
index 851965867d9c7f..2fdaca6b3e92e2 100644
--- a/drivers/dma-buf/selftests.h
+++ b/drivers/dma-buf/selftests.h
@@ -13,4 +13,3 @@ selftest(sanitycheck, __sanitycheck__) /* keep first (igt selfcheck) */
selftest(dma_fence, dma_fence)
selftest(dma_fence_chain, dma_fence_chain)
selftest(dma_fence_unwrap, dma_fence_unwrap)
-selftest(dma_resv, dma_resv)
diff --git a/drivers/dma-buf/st-dma-resv.c b/drivers/dma-buf/st-dma-resv.c
index ad4dfb49dcd9fa..95a4becdb8926d 100644
--- a/drivers/dma-buf/st-dma-resv.c
+++ b/drivers/dma-buf/st-dma-resv.c
@@ -5,13 +5,17 @@
* Copyright © 2021 Advanced Micro Devices, Inc.
*/
+#include <kunit/test.h>
#include <linux/slab.h>
#include <linux/spinlock.h>
#include <linux/dma-resv.h>
-#include "selftest.h"
+static DEFINE_SPINLOCK(fence_lock);
-static struct spinlock fence_lock;
+struct dma_resv_usage_param {
+ enum dma_resv_usage usage;
+ const char *desc;
+};
static const char *fence_name(struct dma_fence *f)
{
@@ -35,15 +39,14 @@ static struct dma_fence *alloc_fence(void)
return f;
}
-static int sanitycheck(void *arg)
+static void test_sanitycheck(struct kunit *test)
{
struct dma_resv resv;
struct dma_fence *f;
int r;
f = alloc_fence();
- if (!f)
- return -ENOMEM;
+ KUNIT_ASSERT_NOT_NULL(test, f);
dma_fence_enable_sw_signaling(f);
@@ -53,49 +56,46 @@ static int sanitycheck(void *arg)
dma_resv_init(&resv);
r = dma_resv_lock(&resv, NULL);
if (r)
- pr_err("Resv locking failed\n");
+ KUNIT_FAIL(test, "Resv locking failed\n");
else
dma_resv_unlock(&resv);
dma_resv_fini(&resv);
- return r;
}
-static int test_signaling(void *arg)
+static void test_signaling(struct kunit *test)
{
- enum dma_resv_usage usage = (unsigned long)arg;
+ const struct dma_resv_usage_param *param = test->param_value;
+ enum dma_resv_usage usage = param->usage;
struct dma_resv resv;
struct dma_fence *f;
int r;
f = alloc_fence();
- if (!f)
- return -ENOMEM;
+ KUNIT_ASSERT_NOT_NULL(test, f);
dma_fence_enable_sw_signaling(f);
dma_resv_init(&resv);
r = dma_resv_lock(&resv, NULL);
if (r) {
- pr_err("Resv locking failed\n");
+ KUNIT_FAIL(test, "Resv locking failed");
goto err_free;
}
r = dma_resv_reserve_fences(&resv, 1);
if (r) {
- pr_err("Resv shared slot allocation failed\n");
+ KUNIT_FAIL(test, "Resv shared slot allocation failed");
goto err_unlock;
}
dma_resv_add_fence(&resv, f, usage);
if (dma_resv_test_signaled(&resv, usage)) {
- pr_err("Resv unexpectedly signaled\n");
- r = -EINVAL;
+ KUNIT_FAIL(test, "Resv unexpectedly signaled");
goto err_unlock;
}
dma_fence_signal(f);
if (!dma_resv_test_signaled(&resv, usage)) {
- pr_err("Resv not reporting signaled\n");
- r = -EINVAL;
+ KUNIT_FAIL(test, "Resv not reporting signaled");
goto err_unlock;
}
err_unlock:
@@ -103,33 +103,32 @@ static int test_signaling(void *arg)
err_free:
dma_resv_fini(&resv);
dma_fence_put(f);
- return r;
}
-static int test_for_each(void *arg)
+static void test_for_each(struct kunit *test)
{
- enum dma_resv_usage usage = (unsigned long)arg;
+ const struct dma_resv_usage_param *param = test->param_value;
+ enum dma_resv_usage usage = param->usage;
struct dma_resv_iter cursor;
struct dma_fence *f, *fence;
struct dma_resv resv;
int r;
f = alloc_fence();
- if (!f)
- return -ENOMEM;
+ KUNIT_ASSERT_NOT_NULL(test, f);
dma_fence_enable_sw_signaling(f);
dma_resv_init(&resv);
r = dma_resv_lock(&resv, NULL);
if (r) {
- pr_err("Resv locking failed\n");
+ KUNIT_FAIL(test, "Resv locking failed");
goto err_free;
}
r = dma_resv_reserve_fences(&resv, 1);
if (r) {
- pr_err("Resv shared slot allocation failed\n");
+ KUNIT_FAIL(test, "Resv shared slot allocation failed");
goto err_unlock;
}
@@ -138,24 +137,23 @@ static int test_for_each(void *arg)
r = -ENOENT;
dma_resv_for_each_fence(&cursor, &resv, usage, fence) {
if (!r) {
- pr_err("More than one fence found\n");
- r = -EINVAL;
+ KUNIT_FAIL(test, "More than one fence found");
goto err_unlock;
}
if (f != fence) {
- pr_err("Unexpected fence\n");
+ KUNIT_FAIL(test, "Unexpected fence");
r = -EINVAL;
goto err_unlock;
}
if (dma_resv_iter_usage(&cursor) != usage) {
- pr_err("Unexpected fence usage\n");
+ KUNIT_FAIL(test, "Unexpected fence usage");
r = -EINVAL;
goto err_unlock;
}
r = 0;
}
if (r) {
- pr_err("No fence found\n");
+ KUNIT_FAIL(test, "No fence found");
goto err_unlock;
}
dma_fence_signal(f);
@@ -164,33 +162,32 @@ static int test_for_each(void *arg)
err_free:
dma_resv_fini(&resv);
dma_fence_put(f);
- return r;
}
-static int test_for_each_unlocked(void *arg)
+static void test_for_each_unlocked(struct kunit *test)
{
- enum dma_resv_usage usage = (unsigned long)arg;
+ const struct dma_resv_usage_param *param = test->param_value;
+ enum dma_resv_usage usage = param->usage;
struct dma_resv_iter cursor;
struct dma_fence *f, *fence;
struct dma_resv resv;
int r;
f = alloc_fence();
- if (!f)
- return -ENOMEM;
+ KUNIT_ASSERT_NOT_NULL(test, f);
dma_fence_enable_sw_signaling(f);
dma_resv_init(&resv);
r = dma_resv_lock(&resv, NULL);
if (r) {
- pr_err("Resv locking failed\n");
+ KUNIT_FAIL(test, "Resv locking failed");
goto err_free;
}
r = dma_resv_reserve_fences(&resv, 1);
if (r) {
- pr_err("Resv shared slot allocation failed\n");
+ KUNIT_FAIL(test, "Resv shared slot allocation failed");
dma_resv_unlock(&resv);
goto err_free;
}
@@ -202,21 +199,20 @@ static int test_for_each_unlocked(void *arg)
dma_resv_iter_begin(&cursor, &resv, usage);
dma_resv_for_each_fence_unlocked(&cursor, fence) {
if (!r) {
- pr_err("More than one fence found\n");
- r = -EINVAL;
+ KUNIT_FAIL(test, "More than one fence found");
goto err_iter_end;
}
if (!dma_resv_iter_is_restarted(&cursor)) {
- pr_err("No restart flag\n");
+ KUNIT_FAIL(test, "No restart flag");
goto err_iter_end;
}
if (f != fence) {
- pr_err("Unexpected fence\n");
+ KUNIT_FAIL(test, "Unexpected fence");
r = -EINVAL;
goto err_iter_end;
}
if (dma_resv_iter_usage(&cursor) != usage) {
- pr_err("Unexpected fence usage\n");
+ KUNIT_FAIL(test, "Unexpected fence usage");
r = -EINVAL;
goto err_iter_end;
}
@@ -230,40 +226,38 @@ static int test_for_each_unlocked(void *arg)
r = 0;
}
}
- if (r)
- pr_err("No fence found\n");
+ KUNIT_EXPECT_EQ(test, r, 0);
err_iter_end:
dma_resv_iter_end(&cursor);
dma_fence_signal(f);
err_free:
dma_resv_fini(&resv);
dma_fence_put(f);
- return r;
}
-static int test_get_fences(void *arg)
+static void test_get_fences(struct kunit *test)
{
- enum dma_resv_usage usage = (unsigned long)arg;
+ const struct dma_resv_usage_param *param = test->param_value;
+ enum dma_resv_usage usage = param->usage;
struct dma_fence *f, **fences = NULL;
struct dma_resv resv;
int r, i;
f = alloc_fence();
- if (!f)
- return -ENOMEM;
+ KUNIT_ASSERT_NOT_NULL(test, f);
dma_fence_enable_sw_signaling(f);
dma_resv_init(&resv);
r = dma_resv_lock(&resv, NULL);
if (r) {
- pr_err("Resv locking failed\n");
+ KUNIT_FAIL(test, "Resv locking failed");
goto err_resv;
}
r = dma_resv_reserve_fences(&resv, 1);
if (r) {
- pr_err("Resv shared slot allocation failed\n");
+ KUNIT_FAIL(test, "Resv shared slot allocation failed");
dma_resv_unlock(&resv);
goto err_resv;
}
@@ -273,12 +267,12 @@ static int test_get_fences(void *arg)
r = dma_resv_get_fences(&resv, usage, &i, &fences);
if (r) {
- pr_err("get_fences failed\n");
+ KUNIT_FAIL(test, "get_fences failed");
goto err_free;
}
if (i != 1 || fences[0] != f) {
- pr_err("get_fences returned unexpected fence\n");
+ KUNIT_FAIL(test, "get_fences returned unexpected fence");
goto err_free;
}
@@ -290,27 +284,32 @@ static int test_get_fences(void *arg)
err_resv:
dma_resv_fini(&resv);
dma_fence_put(f);
- return r;
}
-int dma_resv(void)
-{
- static const struct subtest tests[] = {
- SUBTEST(sanitycheck),
- SUBTEST(test_signaling),
- SUBTEST(test_for_each),
- SUBTEST(test_for_each_unlocked),
- SUBTEST(test_get_fences),
- };
- enum dma_resv_usage usage;
- int r;
+static const struct dma_resv_usage_param dma_resv_usage_params[] = {
+ { DMA_RESV_USAGE_KERNEL, "kernel" },
+ { DMA_RESV_USAGE_WRITE, "write" },
+ { DMA_RESV_USAGE_READ, "read" },
+ { DMA_RESV_USAGE_BOOKKEEP, "bookkeep" },
+};
- spin_lock_init(&fence_lock);
- for (usage = DMA_RESV_USAGE_KERNEL; usage <= DMA_RESV_USAGE_BOOKKEEP;
- ++usage) {
- r = subtests(tests, (void *)(unsigned long)usage);
- if (r)
- return r;
- }
- return 0;
-}
+KUNIT_ARRAY_PARAM_DESC(dma_resv_usage, dma_resv_usage_params, desc);
+
+static struct kunit_case dma_resv_cases[] = {
+ KUNIT_CASE(test_sanitycheck),
+ KUNIT_CASE_PARAM(test_signaling, dma_resv_usage_gen_params),
+ KUNIT_CASE_PARAM(test_for_each, dma_resv_usage_gen_params),
+ KUNIT_CASE_PARAM(test_for_each_unlocked, dma_resv_usage_gen_params),
+ KUNIT_CASE_PARAM(test_get_fences, dma_resv_usage_gen_params),
+ {}
+};
+
+static struct kunit_suite dma_resv_test_suite = {
+ .name = "dma-buf-resv",
+ .test_cases = dma_resv_cases,
+};
+
+kunit_test_suite(dma_resv_test_suite);
+
+MODULE_DESCRIPTION("KUnit tests for DMA-BUF");
+MODULE_LICENSE("GPL");
--
2.43.0
next prev parent reply other threads:[~2026-03-01 18:58 UTC|newest]
Thread overview: 15+ messages / expand[flat|nested] mbox.gz Atom feed top
2026-03-01 18:57 [PATCH 0/5] Replace the dmabuf custom test framework with kunit Jason Gunthorpe
2026-03-01 18:57 ` Jason Gunthorpe [this message]
2026-03-03 3:50 ` Claude review: dma-buf: Change st-dma-resv.c to use kunit Claude Code Review Bot
2026-03-01 18:57 ` [PATCH 2/5] dma-buf: Change st-dma-fence.c " Jason Gunthorpe
2026-03-03 3:50 ` Claude review: " Claude Code Review Bot
2026-03-01 18:57 ` [PATCH 3/5] dma-buf: Change st-dma-fence-unwrap.c " Jason Gunthorpe
2026-03-03 3:50 ` Claude review: " Claude Code Review Bot
2026-03-01 18:57 ` [PATCH 4/5] dma-buf: Change st-dma-fence-chain.c " Jason Gunthorpe
2026-03-03 3:50 ` Claude review: " Claude Code Review Bot
2026-03-01 18:57 ` [PATCH 5/5] dma-buf: Remove the old selftest Jason Gunthorpe
2026-03-03 3:50 ` Claude review: " Claude Code Review Bot
2026-03-02 11:43 ` [PATCH 0/5] Replace the dmabuf custom test framework with kunit Christian König
2026-03-02 13:01 ` Jason Gunthorpe
2026-03-02 13:58 ` Christian König
2026-03-03 3:50 ` Claude review: " Claude Code Review Bot
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=1-v1-0a349a394eff+14110-dmabuf_kunit_jgg@nvidia.com \
--to=jgg@nvidia.com \
--cc=airlied@gmail.com \
--cc=christian.koenig@amd.com \
--cc=dri-devel@lists.freedesktop.org \
--cc=intel-gfx@lists.freedesktop.org \
--cc=jani.nikula@linux.intel.com \
--cc=joonas.lahtinen@linux.intel.com \
--cc=linaro-mm-sig@lists.linaro.org \
--cc=linux-media@vger.kernel.org \
--cc=patches@lists.linux.dev \
--cc=rodrigo.vivi@intel.com \
--cc=simona@ffwll.ch \
--cc=sumit.semwal@linaro.org \
--cc=tursulin@ursulin.net \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox