[Intel-gfx] [PATCH v2] drm/i915: Import the kfence selftests for i915_sw_fence
Mika Kuoppala
mika.kuoppala at linux.intel.com
Tue May 16 14:49:45 UTC 2017
Chris Wilson <chris at chris-wilson.co.uk> writes:
> A long time ago, I wrote some selftests for the struct kfence idea. Now
> that we have infrastructure in i915/igt for running kselftests, include
> some for i915_sw_fence.
>
> Signed-off-by: Chris Wilson <chris at chris-wilson.co.uk>
Only minor thing that caught my eye was that I would
have preferred:
err_free_C: type of gotos instead of just err_C. In here
tho everything was straightfoward teardown of allocations
so it was clear.
Reviewed-by: Mika Kuoppala <mika.kuoppala at intel.com>
> ---
> drivers/gpu/drm/i915/Kconfig.debug | 12 +
> drivers/gpu/drm/i915/i915_sw_fence.c | 7 +-
> .../gpu/drm/i915/selftests/i915_mock_selftests.h | 1 +
> drivers/gpu/drm/i915/selftests/i915_sw_fence.c | 576 +++++++++++++++++++++
> 4 files changed, 595 insertions(+), 1 deletion(-)
> create mode 100644 drivers/gpu/drm/i915/selftests/i915_sw_fence.c
>
> diff --git a/drivers/gpu/drm/i915/Kconfig.debug b/drivers/gpu/drm/i915/Kconfig.debug
> index b00edd3b8800..78c5c049a347 100644
> --- a/drivers/gpu/drm/i915/Kconfig.debug
> +++ b/drivers/gpu/drm/i915/Kconfig.debug
> @@ -61,6 +61,18 @@ config DRM_I915_SW_FENCE_DEBUG_OBJECTS
>
> If in doubt, say "N".
>
> +config DRM_I915_SW_FENCE_CHECK_DAG
> + bool "Enable additional driver debugging for detecting dependency cycles"
> + depends on DRM_I915
> + default n
> + help
> + Choose this option to turn on extra driver debugging that may affect
> + performance but will catch some internal issues.
> +
> + Recommended for driver developers only.
> +
> + If in doubt, say "N".
> +
> config DRM_I915_SELFTEST
> bool "Enable selftests upon driver load"
> depends on DRM_I915
> diff --git a/drivers/gpu/drm/i915/i915_sw_fence.c b/drivers/gpu/drm/i915/i915_sw_fence.c
> index a0a690d6627e..474d23c0c0ce 100644
> --- a/drivers/gpu/drm/i915/i915_sw_fence.c
> +++ b/drivers/gpu/drm/i915/i915_sw_fence.c
> @@ -12,6 +12,7 @@
> #include <linux/reservation.h>
>
> #include "i915_sw_fence.h"
> +#include "i915_selftest.h"
>
> #define I915_SW_FENCE_FLAG_ALLOC BIT(3) /* after WQ_FLAG_* for safety */
>
> @@ -274,7 +275,7 @@ static bool i915_sw_fence_check_if_after(struct i915_sw_fence *fence,
> unsigned long flags;
> bool err;
>
> - if (!IS_ENABLED(CONFIG_I915_SW_FENCE_CHECK_DAG))
> + if (!IS_ENABLED(CONFIG_DRM_I915_SW_FENCE_CHECK_DAG))
> return false;
>
> spin_lock_irqsave(&i915_sw_fence_lock, flags);
> @@ -490,3 +491,7 @@ int i915_sw_fence_await_reservation(struct i915_sw_fence *fence,
>
> return ret;
> }
> +
> +#if IS_ENABLED(CONFIG_DRM_I915_SELFTEST)
> +#include "selftests/i915_sw_fence.c"
> +#endif
> diff --git a/drivers/gpu/drm/i915/selftests/i915_mock_selftests.h b/drivers/gpu/drm/i915/selftests/i915_mock_selftests.h
> index 76c1f149a0a0..fc74687501ba 100644
> --- a/drivers/gpu/drm/i915/selftests/i915_mock_selftests.h
> +++ b/drivers/gpu/drm/i915/selftests/i915_mock_selftests.h
> @@ -9,6 +9,7 @@
> * Tests are executed in order by igt/drv_selftest
> */
> selftest(sanitycheck, i915_mock_sanitycheck) /* keep first (igt selfcheck) */
> +selftest(fence, i915_sw_fence_mock_selftests)
> selftest(scatterlist, scatterlist_mock_selftests)
> selftest(syncmap, i915_syncmap_mock_selftests)
> selftest(uncore, intel_uncore_mock_selftests)
> diff --git a/drivers/gpu/drm/i915/selftests/i915_sw_fence.c b/drivers/gpu/drm/i915/selftests/i915_sw_fence.c
> new file mode 100644
> index 000000000000..c552c23eecff
> --- /dev/null
> +++ b/drivers/gpu/drm/i915/selftests/i915_sw_fence.c
> @@ -0,0 +1,576 @@
> +/*
> + * Copyright © 2017 Intel Corporation
> + *
> + * Permission is hereby granted, free of charge, to any person obtaining a
> + * copy of this software and associated documentation files (the "Software"),
> + * to deal in the Software without restriction, including without limitation
> + * the rights to use, copy, modify, merge, publish, distribute, sublicense,
> + * and/or sell copies of the Software, and to permit persons to whom the
> + * Software is furnished to do so, subject to the following conditions:
> + *
> + * The above copyright notice and this permission notice (including the next
> + * paragraph) shall be included in all copies or substantial portions of the
> + * Software.
> + *
> + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
> + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
> + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
> + * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
> + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
> + * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
> + * IN THE SOFTWARE.
> + *
> + */
> +
> +#include <linux/completion.h>
> +#include <linux/delay.h>
> +
> +#include "../i915_selftest.h"
> +
> +static int __i915_sw_fence_call
> +fence_notify(struct i915_sw_fence *fence, enum i915_sw_fence_notify state)
> +{
> + switch (state) {
> + case FENCE_COMPLETE:
> + break;
> +
> + case FENCE_FREE:
> + /* Leave the fence for the caller to free it after testing */
> + break;
> + }
> +
> + return NOTIFY_DONE;
> +}
> +
> +static struct i915_sw_fence *alloc_fence(void)
> +{
> + struct i915_sw_fence *fence;
> +
> + fence = kmalloc(sizeof(*fence), GFP_KERNEL);
> + if (!fence)
> + return NULL;
> +
> + i915_sw_fence_init(fence, fence_notify);
> + return fence;
> +}
> +
> +static void free_fence(struct i915_sw_fence *fence)
> +{
> + i915_sw_fence_fini(fence);
> + kfree(fence);
> +}
> +
> +static int __test_self(struct i915_sw_fence *fence)
> +{
> + if (i915_sw_fence_done(fence))
> + return -EINVAL;
> +
> + i915_sw_fence_commit(fence);
> + if (!i915_sw_fence_done(fence))
> + return -EINVAL;
> +
> + i915_sw_fence_wait(fence);
> + if (!i915_sw_fence_done(fence))
> + return -EINVAL;
> +
> + return 0;
> +}
> +
> +static int test_self(void *arg)
> +{
> + struct i915_sw_fence *fence;
> + int ret;
> +
> + /* Test i915_sw_fence signaling and completion testing */
> + fence = alloc_fence();
> + if (!fence)
> + return -ENOMEM;
> +
> + ret = __test_self(fence);
> +
> + free_fence(fence);
> + return ret;
> +}
> +
> +static int test_dag(void *arg)
> +{
> + struct i915_sw_fence *A, *B, *C;
> + int ret = -EINVAL;
> +
> + /* Test detection of cycles within the i915_sw_fence graphs */
> + if (!IS_ENABLED(CONFIG_DRM_I915_SW_FENCE_CHECK_DAG))
> + return 0;
> +
> + A = alloc_fence();
> + if (!A)
> + return -ENOMEM;
> +
> + if (i915_sw_fence_await_sw_fence_gfp(A, A, GFP_KERNEL) != -EINVAL) {
> + pr_err("recursive cycle not detected (AA)\n");
> + goto err_A;
> + }
> +
> + B = alloc_fence();
> + if (!B) {
> + ret = -ENOMEM;
> + goto err_A;
> + }
> +
> + i915_sw_fence_await_sw_fence_gfp(A, B, GFP_KERNEL);
> + if (i915_sw_fence_await_sw_fence_gfp(B, A, GFP_KERNEL) != -EINVAL) {
> + pr_err("single depth cycle not detected (BAB)\n");
> + goto err_B;
> + }
> +
> + C = alloc_fence();
> + if (i915_sw_fence_await_sw_fence_gfp(B, C, GFP_KERNEL) == -EINVAL) {
> + pr_err("invalid cycle detected\n");
> + goto err_C;
> + }
> + if (i915_sw_fence_await_sw_fence_gfp(C, B, GFP_KERNEL) != -EINVAL) {
> + pr_err("single depth cycle not detected (CBC)\n");
> + goto err_C;
> + }
> + if (i915_sw_fence_await_sw_fence_gfp(C, A, GFP_KERNEL) != -EINVAL) {
> + pr_err("cycle not detected (BA, CB, AC)\n");
> + goto err_C;
> + }
> + if (i915_sw_fence_await_sw_fence_gfp(A, C, GFP_KERNEL) == -EINVAL) {
> + pr_err("invalid cycle detected\n");
> + goto err_C;
> + }
> +
> + i915_sw_fence_commit(A);
> + i915_sw_fence_commit(B);
> + i915_sw_fence_commit(C);
> +
> + ret = 0;
> + if (!i915_sw_fence_done(C)) {
> + pr_err("fence C not done\n");
> + ret = -EINVAL;
> + }
> + if (!i915_sw_fence_done(B)) {
> + pr_err("fence B not done\n");
> + ret = -EINVAL;
> + }
> + if (!i915_sw_fence_done(A)) {
> + pr_err("fence A not done\n");
> + ret = -EINVAL;
> + }
> +err_C:
> + free_fence(C);
> +err_B:
> + free_fence(B);
> +err_A:
> + free_fence(A);
> + return ret;
> +}
> +
> +static int test_AB(void *arg)
> +{
> + struct i915_sw_fence *A, *B;
> + int ret;
> +
> + /* Test i915_sw_fence (A) waiting on an event source (B) */
> + A = alloc_fence();
> + if (!A)
> + return -ENOMEM;
> + B = alloc_fence();
> + if (!B) {
> + ret = -ENOMEM;
> + goto err_A;
> + }
> +
> + ret = i915_sw_fence_await_sw_fence_gfp(A, B, GFP_KERNEL);
> + if (ret < 0)
> + goto err_B;
> + if (ret == 0) {
> + pr_err("Incorrectly reported fence A was complete before await\n");
> + ret = -EINVAL;
> + goto err_B;
> + }
> +
> + ret = -EINVAL;
> + i915_sw_fence_commit(A);
> + if (i915_sw_fence_done(A))
> + goto err_B;
> +
> + i915_sw_fence_commit(B);
> + if (!i915_sw_fence_done(B)) {
> + pr_err("Fence B is not done\n");
> + goto err_B;
> + }
> +
> + if (!i915_sw_fence_done(A)) {
> + pr_err("Fence A is not done\n");
> + goto err_B;
> + }
> +
> + ret = 0;
> +err_B:
> + free_fence(B);
> +err_A:
> + free_fence(A);
> + return ret;
> +}
> +
> +static int test_ABC(void *arg)
> +{
> + struct i915_sw_fence *A, *B, *C;
> + int ret;
> +
> + /* Test a chain of fences, A waits on B who waits on C */
> + A = alloc_fence();
> + if (!A)
> + return -ENOMEM;
> +
> + B = alloc_fence();
> + if (!B) {
> + ret = -ENOMEM;
> + goto err_A;
> + }
> +
> + C = alloc_fence();
> + if (!C) {
> + ret = -ENOMEM;
> + goto err_B;
> + }
> +
> + ret = i915_sw_fence_await_sw_fence_gfp(A, B, GFP_KERNEL);
> + if (ret < 0)
> + goto err_C;
> + if (ret == 0) {
> + pr_err("Incorrectly reported fence B was complete before await\n");
> + goto err_C;
> + }
> +
> + ret = i915_sw_fence_await_sw_fence_gfp(B, C, GFP_KERNEL);
> + if (ret < 0)
> + goto err_C;
> + if (ret == 0) {
> + pr_err("Incorrectly reported fence C was complete before await\n");
> + goto err_C;
> + }
> +
> + ret = -EINVAL;
> + i915_sw_fence_commit(A);
> + if (i915_sw_fence_done(A)) {
> + pr_err("Fence A completed early\n");
> + goto err_C;
> + }
> +
> + i915_sw_fence_commit(B);
> + if (i915_sw_fence_done(B)) {
> + pr_err("Fence B completed early\n");
> + goto err_C;
> + }
> +
> + if (i915_sw_fence_done(A)) {
> + pr_err("Fence A completed early (after signaling B)\n");
> + goto err_C;
> + }
> +
> + i915_sw_fence_commit(C);
> +
> + ret = 0;
> + if (!i915_sw_fence_done(C)) {
> + pr_err("Fence C not done\n");
> + ret = -EINVAL;
> + }
> + if (!i915_sw_fence_done(B)) {
> + pr_err("Fence B not done\n");
> + ret = -EINVAL;
> + }
> + if (!i915_sw_fence_done(A)) {
> + pr_err("Fence A not done\n");
> + ret = -EINVAL;
> + }
> +err_C:
> + free_fence(C);
> +err_B:
> + free_fence(B);
> +err_A:
> + free_fence(A);
> + return ret;
> +}
> +
> +static int test_AB_C(void *arg)
> +{
> + struct i915_sw_fence *A, *B, *C;
> + int ret = -EINVAL;
> +
> + /* Test multiple fences (AB) waiting on a single event (C) */
> + A = alloc_fence();
> + if (!A)
> + return -ENOMEM;
> +
> + B = alloc_fence();
> + if (!B) {
> + ret = -ENOMEM;
> + goto err_A;
> + }
> +
> + C = alloc_fence();
> + if (!B) {
> + ret = -ENOMEM;
> + goto err_B;
> + }
> +
> + ret = i915_sw_fence_await_sw_fence_gfp(A, C, GFP_KERNEL);
> + if (ret < 0)
> + goto err_C;
> + if (ret == 0) {
> + ret = -EINVAL;
> + goto err_C;
> + }
> +
> + ret = i915_sw_fence_await_sw_fence_gfp(B, C, GFP_KERNEL);
> + if (ret < 0)
> + goto err_C;
> + if (ret == 0) {
> + ret = -EINVAL;
> + goto err_C;
> + }
> +
> + i915_sw_fence_commit(A);
> + i915_sw_fence_commit(B);
> +
> + ret = 0;
> + if (i915_sw_fence_done(A)) {
> + pr_err("Fence A completed early\n");
> + ret = -EINVAL;
> + }
> +
> + if (i915_sw_fence_done(B)) {
> + pr_err("Fence B completed early\n");
> + ret = -EINVAL;
> + }
> +
> + i915_sw_fence_commit(C);
> + if (!i915_sw_fence_done(C)) {
> + pr_err("Fence C not done\n");
> + ret = -EINVAL;
> + }
> +
> + if (!i915_sw_fence_done(B)) {
> + pr_err("Fence B not done\n");
> + ret = -EINVAL;
> + }
> +
> + if (!i915_sw_fence_done(A)) {
> + pr_err("Fence A not done\n");
> + ret = -EINVAL;
> + }
> +
> +err_C:
> + free_fence(C);
> +err_B:
> + free_fence(B);
> +err_A:
> + free_fence(A);
> + return ret;
> +}
> +
> +static int test_C_AB(void *arg)
> +{
> + struct i915_sw_fence *A, *B, *C;
> + int ret;
> +
> + /* Test multiple event sources (A,B) for a single fence (C) */
> + A = alloc_fence();
> + if (!A)
> + return -ENOMEM;
> +
> + B = alloc_fence();
> + if (!B) {
> + ret = -ENOMEM;
> + goto err_A;
> + }
> +
> + C = alloc_fence();
> + if (!B) {
> + ret = -ENOMEM;
> + goto err_B;
> + }
> +
> + ret = i915_sw_fence_await_sw_fence_gfp(C, A, GFP_KERNEL);
> + if (ret < 0)
> + goto err_C;
> + if (ret == 0) {
> + ret = -EINVAL;
> + goto err_C;
> + }
> +
> + ret = i915_sw_fence_await_sw_fence_gfp(C, B, GFP_KERNEL);
> + if (ret < 0)
> + goto err_C;
> + if (ret == 0) {
> + ret = -EINVAL;
> + goto err_C;
> + }
> +
> + ret = 0;
> + i915_sw_fence_commit(C);
> + if (i915_sw_fence_done(C))
> + ret = -EINVAL;
> +
> + i915_sw_fence_commit(A);
> + i915_sw_fence_commit(B);
> +
> + if (!i915_sw_fence_done(A)) {
> + pr_err("Fence A not done\n");
> + ret = -EINVAL;
> + }
> +
> + if (!i915_sw_fence_done(B)) {
> + pr_err("Fence B not done\n");
> + ret = -EINVAL;
> + }
> +
> + if (!i915_sw_fence_done(C)) {
> + pr_err("Fence C not done\n");
> + ret = -EINVAL;
> + }
> +
> +err_C:
> + free_fence(C);
> +err_B:
> + free_fence(B);
> +err_A:
> + free_fence(A);
> + return ret;
> +}
> +
> +static int test_chain(void *arg)
> +{
> + int nfences = 4096;
> + struct i915_sw_fence **fences;
> + int ret, i;
> +
> + /* Test a long chain of fences */
> + fences = kmalloc_array(nfences, sizeof(*fences), GFP_KERNEL);
> + if (!fences)
> + return -ENOMEM;
> +
> + for (i = 0; i < nfences; i++) {
> + fences[i] = alloc_fence();
> + if (!fences[i]) {
> + nfences = i;
> + ret = -ENOMEM;
> + goto err;
> + }
> +
> + if (i > 0) {
> + ret = i915_sw_fence_await_sw_fence_gfp(fences[i],
> + fences[i - 1],
> + GFP_KERNEL);
> + if (ret < 0) {
> + nfences = i + 1;
> + goto err;
> + }
> +
> + i915_sw_fence_commit(fences[i]);
> + }
> + }
> +
> + ret = 0;
> + for (i = nfences; --i; ) {
> + if (i915_sw_fence_done(fences[i])) {
> + if (ret == 0)
> + pr_err("Fence[%d] completed early\n", i);
> + ret = -EINVAL;
> + }
> + }
> + i915_sw_fence_commit(fences[0]);
> + for (i = 0; ret == 0 && i < nfences; i++) {
> + if (!i915_sw_fence_done(fences[i])) {
> + pr_err("Fence[%d] is not done\n", i);
> + ret = -EINVAL;
> + }
> + }
> +
> +err:
> + for (i = 0; i < nfences; i++)
> + free_fence(fences[i]);
> + kfree(fences);
> + return ret;
> +}
> +
> +struct task_ipc {
> + struct work_struct work;
> + struct completion started;
> + struct i915_sw_fence *in, *out;
> + int value;
> +};
> +
> +static void task_ipc(struct work_struct *work)
> +{
> + struct task_ipc *ipc = container_of(work, typeof(*ipc), work);
> +
> + complete(&ipc->started);
> +
> + i915_sw_fence_wait(ipc->in);
> + smp_store_mb(ipc->value, 1);
> + i915_sw_fence_commit(ipc->out);
> +}
> +
> +static int test_ipc(void *arg)
> +{
> + struct task_ipc ipc;
> + int ret = 0;
> +
> + /* Test use of i915_sw_fence as an interprocess signaling mechanism */
> + ipc.in = alloc_fence();
> + if (!ipc.in)
> + return -ENOMEM;
> + ipc.out = alloc_fence();
> + if (!ipc.out) {
> + ret = -ENOMEM;
> + goto err_in;
> + }
> +
> + /* use a completion to avoid chicken-and-egg testing */
> + init_completion(&ipc.started);
> +
> + ipc.value = 0;
> + INIT_WORK(&ipc.work, task_ipc);
> + schedule_work(&ipc.work);
> +
> + wait_for_completion(&ipc.started);
> +
> + usleep_range(1000, 2000);
> + if (READ_ONCE(ipc.value)) {
> + pr_err("worker updated value before i915_sw_fence was signaled\n");
> + ret = -EINVAL;
> + }
> +
> + i915_sw_fence_commit(ipc.in);
> + i915_sw_fence_wait(ipc.out);
> +
> + if (!READ_ONCE(ipc.value)) {
> + pr_err("worker signaled i915_sw_fence before value was posted\n");
> + ret = -EINVAL;
> + }
> +
> + flush_work(&ipc.work);
> + free_fence(ipc.out);
> +err_in:
> + free_fence(ipc.in);
> + return ret;
> +}
> +
> +int i915_sw_fence_mock_selftests(void)
> +{
> + static const struct i915_subtest tests[] = {
> + SUBTEST(test_self),
> + SUBTEST(test_dag),
> + SUBTEST(test_AB),
> + SUBTEST(test_ABC),
> + SUBTEST(test_AB_C),
> + SUBTEST(test_C_AB),
> + SUBTEST(test_chain),
> + SUBTEST(test_ipc),
> + };
> +
> + return i915_subtests(tests, NULL);
> +}
> --
> 2.11.0
>
> _______________________________________________
> Intel-gfx mailing list
> Intel-gfx at lists.freedesktop.org
> https://lists.freedesktop.org/mailman/listinfo/intel-gfx
More information about the Intel-gfx
mailing list