123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316 |
- /* SPDX-License-Identifier: MIT */
- /*
- * Copyright © 2019 Intel Corporation
- * Copyright © 2021 Advanced Micro Devices, Inc.
- */
- #include <linux/slab.h>
- #include <linux/spinlock.h>
- #include <linux/dma-resv.h>
- #include "selftest.h"
- static struct spinlock fence_lock;
- static const char *fence_name(struct dma_fence *f)
- {
- return "selftest";
- }
- static const struct dma_fence_ops fence_ops = {
- .get_driver_name = fence_name,
- .get_timeline_name = fence_name,
- };
- static struct dma_fence *alloc_fence(void)
- {
- struct dma_fence *f;
- f = kmalloc(sizeof(*f), GFP_KERNEL);
- if (!f)
- return NULL;
- dma_fence_init(f, &fence_ops, &fence_lock, 0, 0);
- return f;
- }
- static int sanitycheck(void *arg)
- {
- struct dma_resv resv;
- struct dma_fence *f;
- int r;
- f = alloc_fence();
- if (!f)
- return -ENOMEM;
- dma_fence_enable_sw_signaling(f);
- dma_fence_signal(f);
- dma_fence_put(f);
- dma_resv_init(&resv);
- r = dma_resv_lock(&resv, NULL);
- if (r)
- pr_err("Resv locking failed\n");
- else
- dma_resv_unlock(&resv);
- dma_resv_fini(&resv);
- return r;
- }
- static int test_signaling(void *arg)
- {
- enum dma_resv_usage usage = (unsigned long)arg;
- struct dma_resv resv;
- struct dma_fence *f;
- int r;
- f = alloc_fence();
- if (!f)
- return -ENOMEM;
- dma_fence_enable_sw_signaling(f);
- dma_resv_init(&resv);
- r = dma_resv_lock(&resv, NULL);
- if (r) {
- pr_err("Resv locking failed\n");
- goto err_free;
- }
- r = dma_resv_reserve_fences(&resv, 1);
- if (r) {
- pr_err("Resv shared slot allocation failed\n");
- 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;
- goto err_unlock;
- }
- dma_fence_signal(f);
- if (!dma_resv_test_signaled(&resv, usage)) {
- pr_err("Resv not reporting signaled\n");
- r = -EINVAL;
- goto err_unlock;
- }
- err_unlock:
- dma_resv_unlock(&resv);
- err_free:
- dma_resv_fini(&resv);
- dma_fence_put(f);
- return r;
- }
- static int test_for_each(void *arg)
- {
- enum dma_resv_usage usage = (unsigned long)arg;
- struct dma_resv_iter cursor;
- struct dma_fence *f, *fence;
- struct dma_resv resv;
- int r;
- f = alloc_fence();
- if (!f)
- return -ENOMEM;
- dma_fence_enable_sw_signaling(f);
- dma_resv_init(&resv);
- r = dma_resv_lock(&resv, NULL);
- if (r) {
- pr_err("Resv locking failed\n");
- goto err_free;
- }
- r = dma_resv_reserve_fences(&resv, 1);
- if (r) {
- pr_err("Resv shared slot allocation failed\n");
- goto err_unlock;
- }
- dma_resv_add_fence(&resv, f, usage);
- r = -ENOENT;
- dma_resv_for_each_fence(&cursor, &resv, usage, fence) {
- if (!r) {
- pr_err("More than one fence found\n");
- r = -EINVAL;
- goto err_unlock;
- }
- if (f != fence) {
- pr_err("Unexpected fence\n");
- r = -EINVAL;
- goto err_unlock;
- }
- if (dma_resv_iter_usage(&cursor) != usage) {
- pr_err("Unexpected fence usage\n");
- r = -EINVAL;
- goto err_unlock;
- }
- r = 0;
- }
- if (r) {
- pr_err("No fence found\n");
- goto err_unlock;
- }
- dma_fence_signal(f);
- err_unlock:
- dma_resv_unlock(&resv);
- err_free:
- dma_resv_fini(&resv);
- dma_fence_put(f);
- return r;
- }
- static int test_for_each_unlocked(void *arg)
- {
- enum dma_resv_usage usage = (unsigned long)arg;
- struct dma_resv_iter cursor;
- struct dma_fence *f, *fence;
- struct dma_resv resv;
- int r;
- f = alloc_fence();
- if (!f)
- return -ENOMEM;
- dma_fence_enable_sw_signaling(f);
- dma_resv_init(&resv);
- r = dma_resv_lock(&resv, NULL);
- if (r) {
- pr_err("Resv locking failed\n");
- goto err_free;
- }
- r = dma_resv_reserve_fences(&resv, 1);
- if (r) {
- pr_err("Resv shared slot allocation failed\n");
- dma_resv_unlock(&resv);
- goto err_free;
- }
- dma_resv_add_fence(&resv, f, usage);
- dma_resv_unlock(&resv);
- r = -ENOENT;
- 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;
- goto err_iter_end;
- }
- if (!dma_resv_iter_is_restarted(&cursor)) {
- pr_err("No restart flag\n");
- goto err_iter_end;
- }
- if (f != fence) {
- pr_err("Unexpected fence\n");
- r = -EINVAL;
- goto err_iter_end;
- }
- if (dma_resv_iter_usage(&cursor) != usage) {
- pr_err("Unexpected fence usage\n");
- r = -EINVAL;
- goto err_iter_end;
- }
- /* We use r as state here */
- if (r == -ENOENT) {
- r = -EINVAL;
- /* That should trigger an restart */
- cursor.fences = (void*)~0;
- } else if (r == -EINVAL) {
- r = 0;
- }
- }
- if (r)
- pr_err("No fence found\n");
- 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)
- {
- enum dma_resv_usage usage = (unsigned long)arg;
- struct dma_fence *f, **fences = NULL;
- struct dma_resv resv;
- int r, i;
- f = alloc_fence();
- if (!f)
- return -ENOMEM;
- dma_fence_enable_sw_signaling(f);
- dma_resv_init(&resv);
- r = dma_resv_lock(&resv, NULL);
- if (r) {
- pr_err("Resv locking failed\n");
- goto err_resv;
- }
- r = dma_resv_reserve_fences(&resv, 1);
- if (r) {
- pr_err("Resv shared slot allocation failed\n");
- dma_resv_unlock(&resv);
- goto err_resv;
- }
- dma_resv_add_fence(&resv, f, usage);
- dma_resv_unlock(&resv);
- r = dma_resv_get_fences(&resv, usage, &i, &fences);
- if (r) {
- pr_err("get_fences failed\n");
- goto err_free;
- }
- if (i != 1 || fences[0] != f) {
- pr_err("get_fences returned unexpected fence\n");
- goto err_free;
- }
- dma_fence_signal(f);
- err_free:
- while (i--)
- dma_fence_put(fences[i]);
- kfree(fences);
- 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;
- 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;
- }
|