diff mbox

[i-g-t] tests: Add kms_atomic_interruptible test, v2.

Message ID 20170906123028.13582-1-maarten.lankhorst@linux.intel.com (mailing list archive)
State New, archived
Headers show

Commit Message

Maarten Lankhorst Sept. 6, 2017, 12:30 p.m. UTC
This tests the various parts of atomic that I want to make
interruptible. Running with --debug shows the stats from
__igt_sigiter_continue, which can be used to make sure that
we don't fall over.

The default igt kms helpers use drmIoctl, which is not intercepted
by igt_while_interruptible. Only igt_ioctl is. This means we have
to call the ioctls manually here.

Changes since v1:
- Implement interruptible DPMS checking too.
- Use igt_ioctl + igt_while_interruptible, instead of the signal helper
  shotgun.

Signed-off-by: Maarten Lankhorst <maarten.lankhorst@linux.intel.com>
Cc: Daniel Stone <daniels@collabora.com>
---
 lib/igt_kms.c                    |   3 +-
 lib/igt_kms.h                    |   1 +
 tests/Makefile.sources           |   1 +
 tests/kms_atomic_interruptible.c | 319 +++++++++++++++++++++++++++++++++++++++
 4 files changed, 323 insertions(+), 1 deletion(-)
 create mode 100644 tests/kms_atomic_interruptible.c

Comments

Daniel Vetter Sept. 8, 2017, 7:08 a.m. UTC | #1
On Wed, Sep 06, 2017 at 02:30:28PM +0200, Maarten Lankhorst wrote:
> This tests the various parts of atomic that I want to make
> interruptible. Running with --debug shows the stats from
> __igt_sigiter_continue, which can be used to make sure that
> we don't fall over.
> 
> The default igt kms helpers use drmIoctl, which is not intercepted
> by igt_while_interruptible. Only igt_ioctl is. This means we have
> to call the ioctls manually here.
> 
> Changes since v1:
> - Implement interruptible DPMS checking too.
> - Use igt_ioctl + igt_while_interruptible, instead of the signal helper
>   shotgun.
> 
> Signed-off-by: Maarten Lankhorst <maarten.lankhorst@linux.intel.com>
> Cc: Daniel Stone <daniels@collabora.com>
> ---
>  lib/igt_kms.c                    |   3 +-
>  lib/igt_kms.h                    |   1 +
>  tests/Makefile.sources           |   1 +
>  tests/kms_atomic_interruptible.c | 319 +++++++++++++++++++++++++++++++++++++++
>  4 files changed, 323 insertions(+), 1 deletion(-)
>  create mode 100644 tests/kms_atomic_interruptible.c
> 
> diff --git a/lib/igt_kms.c b/lib/igt_kms.c
> index 14e2701c3afd..1f57e8981347 100644
> --- a/lib/igt_kms.c
> +++ b/lib/igt_kms.c
> @@ -186,7 +186,8 @@ const char *igt_crtc_prop_names[IGT_NUM_CRTC_PROPS] = {
>  
>  const char *igt_connector_prop_names[IGT_NUM_CONNECTOR_PROPS] = {
>  	"scaling mode",
> -	"CRTC_ID"
> +	"CRTC_ID",
> +	"DPMS"
>  };
>  
>  /*
> diff --git a/lib/igt_kms.h b/lib/igt_kms.h
> index e5dc329b161e..3d1061fa08c8 100644
> --- a/lib/igt_kms.h
> +++ b/lib/igt_kms.h
> @@ -114,6 +114,7 @@ extern const char *igt_crtc_prop_names[];
>  enum igt_atomic_connector_properties {
>         IGT_CONNECTOR_SCALING_MODE = 0,
>         IGT_CONNECTOR_CRTC_ID,
> +       IGT_CONNECTOR_DPMS,
>         IGT_NUM_CONNECTOR_PROPS
>  };
>  
> diff --git a/tests/Makefile.sources b/tests/Makefile.sources
> index 0f4e39af10a1..cf542df181a8 100644
> --- a/tests/Makefile.sources
> +++ b/tests/Makefile.sources
> @@ -172,6 +172,7 @@ TESTS_progs = \
>  	kms_3d \
>  	kms_addfb_basic \
>  	kms_atomic \
> +	kms_atomic_interruptible \
>  	kms_atomic_transition \
>  	kms_busy \
>  	kms_ccs \
> diff --git a/tests/kms_atomic_interruptible.c b/tests/kms_atomic_interruptible.c
> new file mode 100644
> index 000000000000..6ec7a666b995
> --- /dev/null
> +++ b/tests/kms_atomic_interruptible.c
> @@ -0,0 +1,319 @@
> +/*
> + * Copyright © 2016 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 "igt.h"
> +#include "drmtest.h"
> +#include "sw_sync.h"
> +
> +enum plane_test_type
> +{
> +	test_legacy_modeset,
> +	test_atomic_modeset,
> +	test_legacy_dpms,
> +	test_setplane,
> +	test_setcursor,
> +	test_pageflip
> +};
> +
> +static int block_plane(igt_display_t *display, igt_output_t *output, enum plane_test_type test_type, igt_plane_t *plane)
> +{
> +	int timeline = sw_sync_timeline_create();
> +
> +	igt_fork(child, 1) {
> +		/* Ignore the signal helper, we need to block indefinitely on the fence. */
> +		signal(SIGCONT, SIG_IGN);
> +
> +		if (test_type == test_legacy_modeset || test_type == test_atomic_modeset) {
> +			igt_output_set_pipe(output, PIPE_NONE);
> +			igt_plane_set_fb(plane, NULL);
> +		}
> +		igt_plane_set_fence_fd(plane, sw_sync_timeline_create_fence(timeline, 1));
> +
> +		igt_display_commit2(display, COMMIT_ATOMIC);
> +	}
> +
> +	return timeline;
> +}
> +
> +static void unblock(int block)
> +{
> +	sw_sync_timeline_inc(block, 1);
> +	close(block);
> +}
> +
> +static void ev_page_flip(int fd, unsigned seq, unsigned tv_sec, unsigned tv_usec, void *user_data)
> +{
> +	igt_debug("Retrieved vblank seq: %u on unk\n", seq);
> +}
> +
> +static drmEventContext drm_events = {
> +	.version = 2,
> +	.page_flip_handler = ev_page_flip
> +};
> +
> +static void run_plane_test(igt_display_t *display, enum pipe pipe, igt_output_t *output,
> +			   enum plane_test_type test_type, unsigned plane_type)
> +{
> +	drmModeModeInfo *mode;
> +	igt_fb_t fb, fb2;
> +	igt_plane_t *primary, *plane;
> +	int block;
> +
> +	/*
> +	 * Make sure we start with everything disabled to force a real modeset.
> +	 * igt_display_init only sets sw state, and assumes the first test doesn't care
> +	 * about hw state.
> +	 */
> +	igt_display_commit2(display, COMMIT_ATOMIC);
> +
> +	igt_output_set_pipe(output, pipe);
> +
> +	primary = igt_output_get_plane_type(output, DRM_PLANE_TYPE_PRIMARY);
> +	plane = igt_output_get_plane_type(output, plane_type);
> +	mode = igt_output_get_mode(output);
> +
> +	igt_create_fb(display->drm_fd, mode->hdisplay, mode->vdisplay,
> +		      DRM_FORMAT_XRGB8888, LOCAL_DRM_FORMAT_MOD_NONE, &fb);
> +
> +	switch (plane_type) {
> +	case DRM_PLANE_TYPE_PRIMARY:
> +		igt_create_fb(display->drm_fd, mode->hdisplay, mode->vdisplay,
> +			      DRM_FORMAT_XRGB8888, LOCAL_DRM_FORMAT_MOD_NONE, &fb2);
> +		break;
> +	case DRM_PLANE_TYPE_CURSOR:
> +		igt_create_fb(display->drm_fd, 64, 64,
> +		      DRM_FORMAT_ARGB8888, LOCAL_DRM_FORMAT_MOD_NONE, &fb2);
> +		break;
> +	}
> +
> +	if (test_type != test_legacy_modeset && test_type != test_atomic_modeset) {
> +		igt_plane_set_fb(primary, &fb);
> +		igt_display_commit2(display, COMMIT_ATOMIC);
> +	}
> +
> +	igt_plane_set_fb(plane, &fb2);
> +
> +	block = block_plane(display, output, test_type, plane);
> +	sleep(1);

Ugh. Oh well.

> +
> +	igt_fork(child, 1) {
> +		signal(SIGCONT, SIG_IGN);
> +		igt_assert(sleep(5) == 0);
> +
> +		unblock(block);
> +	}
> +
> +	/* run the test */
> +	igt_while_interruptible(true) {
> +		switch (test_type) {
> +		case test_legacy_modeset: {
> +			struct drm_mode_crtc crtc = {
> +				.set_connectors_ptr = (uint64_t)(uintptr_t)&output->id,
> +				.count_connectors = 1,
> +				.crtc_id = primary->pipe->crtc_id,
> +				.fb_id = fb2.fb_id,
> +				.mode_valid = 1,
> +				.mode = *(struct drm_mode_modeinfo*)mode,
> +			};
> +
> +			do_or_die(igt_ioctl(display->drm_fd, DRM_IOCTL_MODE_SETCRTC, &crtc));
> +			break;
> +		}
> +		case test_atomic_modeset: {

Why all these {} for each case statement? I've never seen that before,
looks funny and un-C.

> +			uint32_t objs[3] = { plane->pipe->crtc_id, output->id, plane->drm_plane->plane_id };
> +			uint32_t count_props[3] = { 2, 1, 6 };
> +			uint32_t props[] = {
> +				/* crtc: 2 props */
> +				plane->pipe->atomic_props_crtc[IGT_CRTC_MODE_ID],
> +				plane->pipe->atomic_props_crtc[IGT_CRTC_ACTIVE],
> +				/* connector: 1 prop */
> +				output->config.atomic_props_connector[IGT_CONNECTOR_CRTC_ID],
> +				/* plane: remainder props */
> +				plane->atomic_props_plane[IGT_PLANE_CRTC_ID],
> +				plane->atomic_props_plane[IGT_PLANE_FB_ID],
> +				plane->atomic_props_plane[IGT_PLANE_SRC_W],
> +				plane->atomic_props_plane[IGT_PLANE_SRC_H],
> +				plane->atomic_props_plane[IGT_PLANE_CRTC_W],
> +				plane->atomic_props_plane[IGT_PLANE_CRTC_H]
> +			};
> +			uint64_t prop_vals[] = {
> +				/* crtc */
> +				0, /* mode_id, filled in below */
> +				true,
> +				/* connector */
> +				plane->pipe->crtc_id,
> +				/* plane */
> +				plane->pipe->crtc_id,
> +				fb2.fb_id,
> +				IGT_FIXED(fb2.width, 0),
> +				IGT_FIXED(fb2.height, 0),
> +				fb2.width,
> +				fb2.height
> +			};
> +			uint32_t mode_blob;
> +
> +			struct drm_mode_atomic atm = {
> +				.flags = DRM_MODE_ATOMIC_ALLOW_MODESET,
> +				.count_objs = 3, /* crtc, connector, plane */
> +				.objs_ptr = (uint64_t)(uintptr_t)&objs,
> +				.count_props_ptr = (uint64_t)(uintptr_t)&count_props,
> +				.props_ptr = (uint64_t)(uintptr_t)&props,
> +				.prop_values_ptr = (uint64_t)(uintptr_t)&prop_vals,
> +			};
> +
> +			do_or_die(drmModeCreatePropertyBlob(display->drm_fd, mode, sizeof(*mode), &mode_blob));
> +			prop_vals[0] = mode_blob;
> +
> +			do_or_die(igt_ioctl(display->drm_fd, DRM_IOCTL_MODE_ATOMIC, &atm));
> +
> +			do_or_die(drmModeDestroyPropertyBlob(display->drm_fd, mode_blob));
> +			break;
> +		}
> +		case test_legacy_dpms: {
> +			struct drm_mode_connector_set_property prop = {
> +				.value = DRM_MODE_DPMS_OFF,
> +				.prop_id = output->config.atomic_props_connector[IGT_CONNECTOR_DPMS],
> +				.connector_id = output->id,
> +			};
> +
> +			do_or_die(igt_ioctl(display->drm_fd, DRM_IOCTL_MODE_SETPROPERTY, &prop));
> +			break;
> +		}
> +		case test_setcursor: {
> +			struct drm_mode_cursor cur = {
> +				.flags = DRM_MODE_CURSOR_BO,
> +				.crtc_id = plane->pipe->crtc_id,
> +				.width = fb2.width,
> +				.height = fb2.height,
> +				.handle = fb2.gem_handle,
> +			};
> +			do_or_die(igt_ioctl(display->drm_fd, DRM_IOCTL_MODE_CURSOR, &cur));
> +			break;
> +		}
> +		case test_setplane: {
> +			struct drm_mode_set_plane setplane = {
> +				.plane_id = plane->drm_plane->plane_id,
> +				.crtc_id = plane->pipe->crtc_id,
> +				.fb_id = fb2.fb_id,
> +				.crtc_w = fb2.width,
> +				.crtc_h = fb2.height,
> +				.src_w = IGT_FIXED(fb2.width, 0),
> +				.src_h = IGT_FIXED(fb2.height, 0),
> +			};
> +
> +			do_or_die(igt_ioctl(display->drm_fd, DRM_IOCTL_MODE_SETPLANE, &setplane));
> +			break;
> +		}
> +		case test_pageflip: {
> +			struct drm_mode_crtc_page_flip pageflip = {
> +				.crtc_id = plane->pipe->crtc_id,
> +				.fb_id = fb2.fb_id,
> +				.flags = DRM_MODE_PAGE_FLIP_EVENT,
> +			};
> +
> +			do_or_die(igt_ioctl(display->drm_fd, DRM_IOCTL_MODE_PAGE_FLIP, &pageflip));
> +
> +			drmHandleEvent(display->drm_fd, &drm_events);
> +			break;
> +		}
> +		}

Especially since it makes it look like you double-close the same block
here (but it's actually the case + switch you close). Please remove those
{} case blocks.

Otherwise lgtm. The time-based blocking isn't the most perfect testing
technique, but well, replacing all of libdrm's atomic stuff is a bit over
the top.

Acked-by: Daniel Vetter <daniel.vetter@ffwll.ch>

> +	}
> +
> +	igt_waitchildren();
> +
> +	igt_plane_set_fb(plane, NULL);
> +	igt_plane_set_fb(primary, NULL);
> +	igt_output_set_pipe(output, PIPE_NONE);
> +	igt_display_commit2(display, COMMIT_ATOMIC);
> +	igt_remove_fb(display->drm_fd, &fb);
> +}
> +
> +igt_main
> +{
> +	igt_display_t display;
> +	igt_output_t *output;
> +	enum pipe pipe;
> +
> +	igt_skip_on_simulation();
> +
> +	igt_fixture {
> +		display.drm_fd = drm_open_driver_master(DRIVER_ANY);
> +
> +		kmstest_set_vt_graphics_mode();
> +
> +		igt_display_init(&display, display.drm_fd);
> +
> +		igt_require(display.is_atomic);
> +
> +		igt_display_require_output(&display);
> +
> +		igt_require_sw_sync();
> +	}
> +
> +	igt_subtest("legacy-setmode")
> +		for_each_pipe_with_valid_output(&display, pipe, output) {
> +			run_plane_test(&display, pipe, output, test_legacy_modeset, DRM_PLANE_TYPE_PRIMARY);
> +			break;
> +		}
> +
> +	igt_subtest("atomic-setmode")
> +		for_each_pipe_with_valid_output(&display, pipe, output) {
> +			run_plane_test(&display, pipe, output, test_atomic_modeset, DRM_PLANE_TYPE_PRIMARY);
> +			break;
> +		}
> +
> +	igt_subtest("legacy-dpms")
> +		for_each_pipe_with_valid_output(&display, pipe, output) {
> +			run_plane_test(&display, pipe, output, test_legacy_dpms, DRM_PLANE_TYPE_PRIMARY);
> +			break;
> +		}
> +
> +	igt_subtest("legacy-pageflip")
> +		for_each_pipe_with_valid_output(&display, pipe, output) {
> +			run_plane_test(&display, pipe, output, test_pageflip, DRM_PLANE_TYPE_PRIMARY);
> +			break;
> +		}
> +
> +	igt_subtest("legacy-cursor")
> +		for_each_pipe_with_valid_output(&display, pipe, output) {
> +			run_plane_test(&display, pipe, output, test_setcursor, DRM_PLANE_TYPE_CURSOR);
> +			break;
> +		}
> +
> +	igt_subtest("universal-setplane-primary")
> +		for_each_pipe_with_valid_output(&display, pipe, output) {
> +			run_plane_test(&display, pipe, output, test_setplane, DRM_PLANE_TYPE_PRIMARY);
> +			break;
> +		}
> +
> +	igt_subtest("universal-setplane-cursor")
> +		for_each_pipe_with_valid_output(&display, pipe, output) {
> +			run_plane_test(&display, pipe, output, test_setplane, DRM_PLANE_TYPE_CURSOR);
> +			break;
> +		}
> +
> +	/* TODO: legacy gamma_set/get, object set/getprop, getcrtc, getconnector */
> +	igt_fixture {
> +		igt_display_fini(&display);
> +	}
> +}
> -- 
> 2.11.0
> 
> _______________________________________________
> Intel-gfx mailing list
> Intel-gfx@lists.freedesktop.org
> https://lists.freedesktop.org/mailman/listinfo/intel-gfx
Maarten Lankhorst Sept. 8, 2017, 7:39 a.m. UTC | #2
Op 08-09-17 om 09:08 schreef Daniel Vetter:
> On Wed, Sep 06, 2017 at 02:30:28PM +0200, Maarten Lankhorst wrote:
>> This tests the various parts of atomic that I want to make
>> interruptible. Running with --debug shows the stats from
>> __igt_sigiter_continue, which can be used to make sure that
>> we don't fall over.
>>
>> The default igt kms helpers use drmIoctl, which is not intercepted
>> by igt_while_interruptible. Only igt_ioctl is. This means we have
>> to call the ioctls manually here.
>>
>> Changes since v1:
>> - Implement interruptible DPMS checking too.
>> - Use igt_ioctl + igt_while_interruptible, instead of the signal helper
>>   shotgun.
>>
>> Signed-off-by: Maarten Lankhorst <maarten.lankhorst@linux.intel.com>
>> Cc: Daniel Stone <daniels@collabora.com>
>> ---
>>  lib/igt_kms.c                    |   3 +-
>>  lib/igt_kms.h                    |   1 +
>>  tests/Makefile.sources           |   1 +
>>  tests/kms_atomic_interruptible.c | 319 +++++++++++++++++++++++++++++++++++++++
>>  4 files changed, 323 insertions(+), 1 deletion(-)
>>  create mode 100644 tests/kms_atomic_interruptible.c
>>
>> diff --git a/lib/igt_kms.c b/lib/igt_kms.c
>> index 14e2701c3afd..1f57e8981347 100644
>> --- a/lib/igt_kms.c
>> +++ b/lib/igt_kms.c
>> @@ -186,7 +186,8 @@ const char *igt_crtc_prop_names[IGT_NUM_CRTC_PROPS] = {
>>  
>>  const char *igt_connector_prop_names[IGT_NUM_CONNECTOR_PROPS] = {
>>  	"scaling mode",
>> -	"CRTC_ID"
>> +	"CRTC_ID",
>> +	"DPMS"
>>  };
>>  
>>  /*
>> diff --git a/lib/igt_kms.h b/lib/igt_kms.h
>> index e5dc329b161e..3d1061fa08c8 100644
>> --- a/lib/igt_kms.h
>> +++ b/lib/igt_kms.h
>> @@ -114,6 +114,7 @@ extern const char *igt_crtc_prop_names[];
>>  enum igt_atomic_connector_properties {
>>         IGT_CONNECTOR_SCALING_MODE = 0,
>>         IGT_CONNECTOR_CRTC_ID,
>> +       IGT_CONNECTOR_DPMS,
>>         IGT_NUM_CONNECTOR_PROPS
>>  };
>>  
>> diff --git a/tests/Makefile.sources b/tests/Makefile.sources
>> index 0f4e39af10a1..cf542df181a8 100644
>> --- a/tests/Makefile.sources
>> +++ b/tests/Makefile.sources
>> @@ -172,6 +172,7 @@ TESTS_progs = \
>>  	kms_3d \
>>  	kms_addfb_basic \
>>  	kms_atomic \
>> +	kms_atomic_interruptible \
>>  	kms_atomic_transition \
>>  	kms_busy \
>>  	kms_ccs \
>> diff --git a/tests/kms_atomic_interruptible.c b/tests/kms_atomic_interruptible.c
>> new file mode 100644
>> index 000000000000..6ec7a666b995
>> --- /dev/null
>> +++ b/tests/kms_atomic_interruptible.c
>> @@ -0,0 +1,319 @@
>> +/*
>> + * Copyright © 2016 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 "igt.h"
>> +#include "drmtest.h"
>> +#include "sw_sync.h"
>> +
>> +enum plane_test_type
>> +{
>> +	test_legacy_modeset,
>> +	test_atomic_modeset,
>> +	test_legacy_dpms,
>> +	test_setplane,
>> +	test_setcursor,
>> +	test_pageflip
>> +};
>> +
>> +static int block_plane(igt_display_t *display, igt_output_t *output, enum plane_test_type test_type, igt_plane_t *plane)
>> +{
>> +	int timeline = sw_sync_timeline_create();
>> +
>> +	igt_fork(child, 1) {
>> +		/* Ignore the signal helper, we need to block indefinitely on the fence. */
>> +		signal(SIGCONT, SIG_IGN);
>> +
>> +		if (test_type == test_legacy_modeset || test_type == test_atomic_modeset) {
>> +			igt_output_set_pipe(output, PIPE_NONE);
>> +			igt_plane_set_fb(plane, NULL);
>> +		}
>> +		igt_plane_set_fence_fd(plane, sw_sync_timeline_create_fence(timeline, 1));
>> +
>> +		igt_display_commit2(display, COMMIT_ATOMIC);
>> +	}
>> +
>> +	return timeline;
>> +}
>> +
>> +static void unblock(int block)
>> +{
>> +	sw_sync_timeline_inc(block, 1);
>> +	close(block);
>> +}
>> +
>> +static void ev_page_flip(int fd, unsigned seq, unsigned tv_sec, unsigned tv_usec, void *user_data)
>> +{
>> +	igt_debug("Retrieved vblank seq: %u on unk\n", seq);
>> +}
>> +
>> +static drmEventContext drm_events = {
>> +	.version = 2,
>> +	.page_flip_handler = ev_page_flip
>> +};
>> +
>> +static void run_plane_test(igt_display_t *display, enum pipe pipe, igt_output_t *output,
>> +			   enum plane_test_type test_type, unsigned plane_type)
>> +{
>> +	drmModeModeInfo *mode;
>> +	igt_fb_t fb, fb2;
>> +	igt_plane_t *primary, *plane;
>> +	int block;
>> +
>> +	/*
>> +	 * Make sure we start with everything disabled to force a real modeset.
>> +	 * igt_display_init only sets sw state, and assumes the first test doesn't care
>> +	 * about hw state.
>> +	 */
>> +	igt_display_commit2(display, COMMIT_ATOMIC);
>> +
>> +	igt_output_set_pipe(output, pipe);
>> +
>> +	primary = igt_output_get_plane_type(output, DRM_PLANE_TYPE_PRIMARY);
>> +	plane = igt_output_get_plane_type(output, plane_type);
>> +	mode = igt_output_get_mode(output);
>> +
>> +	igt_create_fb(display->drm_fd, mode->hdisplay, mode->vdisplay,
>> +		      DRM_FORMAT_XRGB8888, LOCAL_DRM_FORMAT_MOD_NONE, &fb);
>> +
>> +	switch (plane_type) {
>> +	case DRM_PLANE_TYPE_PRIMARY:
>> +		igt_create_fb(display->drm_fd, mode->hdisplay, mode->vdisplay,
>> +			      DRM_FORMAT_XRGB8888, LOCAL_DRM_FORMAT_MOD_NONE, &fb2);
>> +		break;
>> +	case DRM_PLANE_TYPE_CURSOR:
>> +		igt_create_fb(display->drm_fd, 64, 64,
>> +		      DRM_FORMAT_ARGB8888, LOCAL_DRM_FORMAT_MOD_NONE, &fb2);
>> +		break;
>> +	}
>> +
>> +	if (test_type != test_legacy_modeset && test_type != test_atomic_modeset) {
>> +		igt_plane_set_fb(primary, &fb);
>> +		igt_display_commit2(display, COMMIT_ATOMIC);
>> +	}
>> +
>> +	igt_plane_set_fb(plane, &fb2);
>> +
>> +	block = block_plane(display, output, test_type, plane);
>> +	sleep(1);
> Ugh. Oh well.
>
>> +
>> +	igt_fork(child, 1) {
>> +		signal(SIGCONT, SIG_IGN);
>> +		igt_assert(sleep(5) == 0);
>> +
>> +		unblock(block);
>> +	}
>> +
>> +	/* run the test */
>> +	igt_while_interruptible(true) {
>> +		switch (test_type) {
>> +		case test_legacy_modeset: {
>> +			struct drm_mode_crtc crtc = {
>> +				.set_connectors_ptr = (uint64_t)(uintptr_t)&output->id,
>> +				.count_connectors = 1,
>> +				.crtc_id = primary->pipe->crtc_id,
>> +				.fb_id = fb2.fb_id,
>> +				.mode_valid = 1,
>> +				.mode = *(struct drm_mode_modeinfo*)mode,
>> +			};
>> +
>> +			do_or_die(igt_ioctl(display->drm_fd, DRM_IOCTL_MODE_SETCRTC, &crtc));
>> +			break;
>> +		}
>> +		case test_atomic_modeset: {
> Why all these {} for each case statement? I've never seen that before,
> looks funny and un-C.

drm_atomic_get_property uses it so you might have seen it, but never noticed.
It's required to declare the local variables in a case statement.

>
>> +			uint32_t objs[3] = { plane->pipe->crtc_id, output->id, plane->drm_plane->plane_id };
>> +			uint32_t count_props[3] = { 2, 1, 6 };
>> +			uint32_t props[] = {
>> +				/* crtc: 2 props */
>> +				plane->pipe->atomic_props_crtc[IGT_CRTC_MODE_ID],
>> +				plane->pipe->atomic_props_crtc[IGT_CRTC_ACTIVE],
>> +				/* connector: 1 prop */
>> +				output->config.atomic_props_connector[IGT_CONNECTOR_CRTC_ID],
>> +				/* plane: remainder props */
>> +				plane->atomic_props_plane[IGT_PLANE_CRTC_ID],
>> +				plane->atomic_props_plane[IGT_PLANE_FB_ID],
>> +				plane->atomic_props_plane[IGT_PLANE_SRC_W],
>> +				plane->atomic_props_plane[IGT_PLANE_SRC_H],
>> +				plane->atomic_props_plane[IGT_PLANE_CRTC_W],
>> +				plane->atomic_props_plane[IGT_PLANE_CRTC_H]
>> +			};
>> +			uint64_t prop_vals[] = {
>> +				/* crtc */
>> +				0, /* mode_id, filled in below */
>> +				true,
>> +				/* connector */
>> +				plane->pipe->crtc_id,
>> +				/* plane */
>> +				plane->pipe->crtc_id,
>> +				fb2.fb_id,
>> +				IGT_FIXED(fb2.width, 0),
>> +				IGT_FIXED(fb2.height, 0),
>> +				fb2.width,
>> +				fb2.height
>> +			};
>> +			uint32_t mode_blob;
>> +
>> +			struct drm_mode_atomic atm = {
>> +				.flags = DRM_MODE_ATOMIC_ALLOW_MODESET,
>> +				.count_objs = 3, /* crtc, connector, plane */
>> +				.objs_ptr = (uint64_t)(uintptr_t)&objs,
>> +				.count_props_ptr = (uint64_t)(uintptr_t)&count_props,
>> +				.props_ptr = (uint64_t)(uintptr_t)&props,
>> +				.prop_values_ptr = (uint64_t)(uintptr_t)&prop_vals,
>> +			};
>> +
>> +			do_or_die(drmModeCreatePropertyBlob(display->drm_fd, mode, sizeof(*mode), &mode_blob));
>> +			prop_vals[0] = mode_blob;
>> +
>> +			do_or_die(igt_ioctl(display->drm_fd, DRM_IOCTL_MODE_ATOMIC, &atm));
>> +
>> +			do_or_die(drmModeDestroyPropertyBlob(display->drm_fd, mode_blob));
>> +			break;
>> +		}
>> +		case test_legacy_dpms: {
>> +			struct drm_mode_connector_set_property prop = {
>> +				.value = DRM_MODE_DPMS_OFF,
>> +				.prop_id = output->config.atomic_props_connector[IGT_CONNECTOR_DPMS],
>> +				.connector_id = output->id,
>> +			};
>> +
>> +			do_or_die(igt_ioctl(display->drm_fd, DRM_IOCTL_MODE_SETPROPERTY, &prop));
>> +			break;
>> +		}
>> +		case test_setcursor: {
>> +			struct drm_mode_cursor cur = {
>> +				.flags = DRM_MODE_CURSOR_BO,
>> +				.crtc_id = plane->pipe->crtc_id,
>> +				.width = fb2.width,
>> +				.height = fb2.height,
>> +				.handle = fb2.gem_handle,
>> +			};
>> +			do_or_die(igt_ioctl(display->drm_fd, DRM_IOCTL_MODE_CURSOR, &cur));
>> +			break;
>> +		}
>> +		case test_setplane: {
>> +			struct drm_mode_set_plane setplane = {
>> +				.plane_id = plane->drm_plane->plane_id,
>> +				.crtc_id = plane->pipe->crtc_id,
>> +				.fb_id = fb2.fb_id,
>> +				.crtc_w = fb2.width,
>> +				.crtc_h = fb2.height,
>> +				.src_w = IGT_FIXED(fb2.width, 0),
>> +				.src_h = IGT_FIXED(fb2.height, 0),
>> +			};
>> +
>> +			do_or_die(igt_ioctl(display->drm_fd, DRM_IOCTL_MODE_SETPLANE, &setplane));
>> +			break;
>> +		}
>> +		case test_pageflip: {
>> +			struct drm_mode_crtc_page_flip pageflip = {
>> +				.crtc_id = plane->pipe->crtc_id,
>> +				.fb_id = fb2.fb_id,
>> +				.flags = DRM_MODE_PAGE_FLIP_EVENT,
>> +			};
>> +
>> +			do_or_die(igt_ioctl(display->drm_fd, DRM_IOCTL_MODE_PAGE_FLIP, &pageflip));
>> +
>> +			drmHandleEvent(display->drm_fd, &drm_events);
>> +			break;
>> +		}
>> +		}
> Especially since it makes it look like you double-close the same block
> here (but it's actually the case + switch you close). Please remove those
> {} case blocks.
It's required for declaring the local variables. I can increase
the indent of case by 1, which should make the apparent double close disappear. :)
> Otherwise lgtm. The time-based blocking isn't the most perfect testing
> technique, but well, replacing all of libdrm's atomic stuff is a bit over
> the top.
>
> Acked-by: Daniel Vetter <daniel.vetter@ffwll.ch>
Thanks for review.

~Maarten
diff mbox

Patch

diff --git a/lib/igt_kms.c b/lib/igt_kms.c
index 14e2701c3afd..1f57e8981347 100644
--- a/lib/igt_kms.c
+++ b/lib/igt_kms.c
@@ -186,7 +186,8 @@  const char *igt_crtc_prop_names[IGT_NUM_CRTC_PROPS] = {
 
 const char *igt_connector_prop_names[IGT_NUM_CONNECTOR_PROPS] = {
 	"scaling mode",
-	"CRTC_ID"
+	"CRTC_ID",
+	"DPMS"
 };
 
 /*
diff --git a/lib/igt_kms.h b/lib/igt_kms.h
index e5dc329b161e..3d1061fa08c8 100644
--- a/lib/igt_kms.h
+++ b/lib/igt_kms.h
@@ -114,6 +114,7 @@  extern const char *igt_crtc_prop_names[];
 enum igt_atomic_connector_properties {
        IGT_CONNECTOR_SCALING_MODE = 0,
        IGT_CONNECTOR_CRTC_ID,
+       IGT_CONNECTOR_DPMS,
        IGT_NUM_CONNECTOR_PROPS
 };
 
diff --git a/tests/Makefile.sources b/tests/Makefile.sources
index 0f4e39af10a1..cf542df181a8 100644
--- a/tests/Makefile.sources
+++ b/tests/Makefile.sources
@@ -172,6 +172,7 @@  TESTS_progs = \
 	kms_3d \
 	kms_addfb_basic \
 	kms_atomic \
+	kms_atomic_interruptible \
 	kms_atomic_transition \
 	kms_busy \
 	kms_ccs \
diff --git a/tests/kms_atomic_interruptible.c b/tests/kms_atomic_interruptible.c
new file mode 100644
index 000000000000..6ec7a666b995
--- /dev/null
+++ b/tests/kms_atomic_interruptible.c
@@ -0,0 +1,319 @@ 
+/*
+ * Copyright © 2016 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 "igt.h"
+#include "drmtest.h"
+#include "sw_sync.h"
+
+enum plane_test_type
+{
+	test_legacy_modeset,
+	test_atomic_modeset,
+	test_legacy_dpms,
+	test_setplane,
+	test_setcursor,
+	test_pageflip
+};
+
+static int block_plane(igt_display_t *display, igt_output_t *output, enum plane_test_type test_type, igt_plane_t *plane)
+{
+	int timeline = sw_sync_timeline_create();
+
+	igt_fork(child, 1) {
+		/* Ignore the signal helper, we need to block indefinitely on the fence. */
+		signal(SIGCONT, SIG_IGN);
+
+		if (test_type == test_legacy_modeset || test_type == test_atomic_modeset) {
+			igt_output_set_pipe(output, PIPE_NONE);
+			igt_plane_set_fb(plane, NULL);
+		}
+		igt_plane_set_fence_fd(plane, sw_sync_timeline_create_fence(timeline, 1));
+
+		igt_display_commit2(display, COMMIT_ATOMIC);
+	}
+
+	return timeline;
+}
+
+static void unblock(int block)
+{
+	sw_sync_timeline_inc(block, 1);
+	close(block);
+}
+
+static void ev_page_flip(int fd, unsigned seq, unsigned tv_sec, unsigned tv_usec, void *user_data)
+{
+	igt_debug("Retrieved vblank seq: %u on unk\n", seq);
+}
+
+static drmEventContext drm_events = {
+	.version = 2,
+	.page_flip_handler = ev_page_flip
+};
+
+static void run_plane_test(igt_display_t *display, enum pipe pipe, igt_output_t *output,
+			   enum plane_test_type test_type, unsigned plane_type)
+{
+	drmModeModeInfo *mode;
+	igt_fb_t fb, fb2;
+	igt_plane_t *primary, *plane;
+	int block;
+
+	/*
+	 * Make sure we start with everything disabled to force a real modeset.
+	 * igt_display_init only sets sw state, and assumes the first test doesn't care
+	 * about hw state.
+	 */
+	igt_display_commit2(display, COMMIT_ATOMIC);
+
+	igt_output_set_pipe(output, pipe);
+
+	primary = igt_output_get_plane_type(output, DRM_PLANE_TYPE_PRIMARY);
+	plane = igt_output_get_plane_type(output, plane_type);
+	mode = igt_output_get_mode(output);
+
+	igt_create_fb(display->drm_fd, mode->hdisplay, mode->vdisplay,
+		      DRM_FORMAT_XRGB8888, LOCAL_DRM_FORMAT_MOD_NONE, &fb);
+
+	switch (plane_type) {
+	case DRM_PLANE_TYPE_PRIMARY:
+		igt_create_fb(display->drm_fd, mode->hdisplay, mode->vdisplay,
+			      DRM_FORMAT_XRGB8888, LOCAL_DRM_FORMAT_MOD_NONE, &fb2);
+		break;
+	case DRM_PLANE_TYPE_CURSOR:
+		igt_create_fb(display->drm_fd, 64, 64,
+		      DRM_FORMAT_ARGB8888, LOCAL_DRM_FORMAT_MOD_NONE, &fb2);
+		break;
+	}
+
+	if (test_type != test_legacy_modeset && test_type != test_atomic_modeset) {
+		igt_plane_set_fb(primary, &fb);
+		igt_display_commit2(display, COMMIT_ATOMIC);
+	}
+
+	igt_plane_set_fb(plane, &fb2);
+
+	block = block_plane(display, output, test_type, plane);
+	sleep(1);
+
+	igt_fork(child, 1) {
+		signal(SIGCONT, SIG_IGN);
+		igt_assert(sleep(5) == 0);
+
+		unblock(block);
+	}
+
+	/* run the test */
+	igt_while_interruptible(true) {
+		switch (test_type) {
+		case test_legacy_modeset: {
+			struct drm_mode_crtc crtc = {
+				.set_connectors_ptr = (uint64_t)(uintptr_t)&output->id,
+				.count_connectors = 1,
+				.crtc_id = primary->pipe->crtc_id,
+				.fb_id = fb2.fb_id,
+				.mode_valid = 1,
+				.mode = *(struct drm_mode_modeinfo*)mode,
+			};
+
+			do_or_die(igt_ioctl(display->drm_fd, DRM_IOCTL_MODE_SETCRTC, &crtc));
+			break;
+		}
+		case test_atomic_modeset: {
+			uint32_t objs[3] = { plane->pipe->crtc_id, output->id, plane->drm_plane->plane_id };
+			uint32_t count_props[3] = { 2, 1, 6 };
+			uint32_t props[] = {
+				/* crtc: 2 props */
+				plane->pipe->atomic_props_crtc[IGT_CRTC_MODE_ID],
+				plane->pipe->atomic_props_crtc[IGT_CRTC_ACTIVE],
+				/* connector: 1 prop */
+				output->config.atomic_props_connector[IGT_CONNECTOR_CRTC_ID],
+				/* plane: remainder props */
+				plane->atomic_props_plane[IGT_PLANE_CRTC_ID],
+				plane->atomic_props_plane[IGT_PLANE_FB_ID],
+				plane->atomic_props_plane[IGT_PLANE_SRC_W],
+				plane->atomic_props_plane[IGT_PLANE_SRC_H],
+				plane->atomic_props_plane[IGT_PLANE_CRTC_W],
+				plane->atomic_props_plane[IGT_PLANE_CRTC_H]
+			};
+			uint64_t prop_vals[] = {
+				/* crtc */
+				0, /* mode_id, filled in below */
+				true,
+				/* connector */
+				plane->pipe->crtc_id,
+				/* plane */
+				plane->pipe->crtc_id,
+				fb2.fb_id,
+				IGT_FIXED(fb2.width, 0),
+				IGT_FIXED(fb2.height, 0),
+				fb2.width,
+				fb2.height
+			};
+			uint32_t mode_blob;
+
+			struct drm_mode_atomic atm = {
+				.flags = DRM_MODE_ATOMIC_ALLOW_MODESET,
+				.count_objs = 3, /* crtc, connector, plane */
+				.objs_ptr = (uint64_t)(uintptr_t)&objs,
+				.count_props_ptr = (uint64_t)(uintptr_t)&count_props,
+				.props_ptr = (uint64_t)(uintptr_t)&props,
+				.prop_values_ptr = (uint64_t)(uintptr_t)&prop_vals,
+			};
+
+			do_or_die(drmModeCreatePropertyBlob(display->drm_fd, mode, sizeof(*mode), &mode_blob));
+			prop_vals[0] = mode_blob;
+
+			do_or_die(igt_ioctl(display->drm_fd, DRM_IOCTL_MODE_ATOMIC, &atm));
+
+			do_or_die(drmModeDestroyPropertyBlob(display->drm_fd, mode_blob));
+			break;
+		}
+		case test_legacy_dpms: {
+			struct drm_mode_connector_set_property prop = {
+				.value = DRM_MODE_DPMS_OFF,
+				.prop_id = output->config.atomic_props_connector[IGT_CONNECTOR_DPMS],
+				.connector_id = output->id,
+			};
+
+			do_or_die(igt_ioctl(display->drm_fd, DRM_IOCTL_MODE_SETPROPERTY, &prop));
+			break;
+		}
+		case test_setcursor: {
+			struct drm_mode_cursor cur = {
+				.flags = DRM_MODE_CURSOR_BO,
+				.crtc_id = plane->pipe->crtc_id,
+				.width = fb2.width,
+				.height = fb2.height,
+				.handle = fb2.gem_handle,
+			};
+			do_or_die(igt_ioctl(display->drm_fd, DRM_IOCTL_MODE_CURSOR, &cur));
+			break;
+		}
+		case test_setplane: {
+			struct drm_mode_set_plane setplane = {
+				.plane_id = plane->drm_plane->plane_id,
+				.crtc_id = plane->pipe->crtc_id,
+				.fb_id = fb2.fb_id,
+				.crtc_w = fb2.width,
+				.crtc_h = fb2.height,
+				.src_w = IGT_FIXED(fb2.width, 0),
+				.src_h = IGT_FIXED(fb2.height, 0),
+			};
+
+			do_or_die(igt_ioctl(display->drm_fd, DRM_IOCTL_MODE_SETPLANE, &setplane));
+			break;
+		}
+		case test_pageflip: {
+			struct drm_mode_crtc_page_flip pageflip = {
+				.crtc_id = plane->pipe->crtc_id,
+				.fb_id = fb2.fb_id,
+				.flags = DRM_MODE_PAGE_FLIP_EVENT,
+			};
+
+			do_or_die(igt_ioctl(display->drm_fd, DRM_IOCTL_MODE_PAGE_FLIP, &pageflip));
+
+			drmHandleEvent(display->drm_fd, &drm_events);
+			break;
+		}
+		}
+	}
+
+	igt_waitchildren();
+
+	igt_plane_set_fb(plane, NULL);
+	igt_plane_set_fb(primary, NULL);
+	igt_output_set_pipe(output, PIPE_NONE);
+	igt_display_commit2(display, COMMIT_ATOMIC);
+	igt_remove_fb(display->drm_fd, &fb);
+}
+
+igt_main
+{
+	igt_display_t display;
+	igt_output_t *output;
+	enum pipe pipe;
+
+	igt_skip_on_simulation();
+
+	igt_fixture {
+		display.drm_fd = drm_open_driver_master(DRIVER_ANY);
+
+		kmstest_set_vt_graphics_mode();
+
+		igt_display_init(&display, display.drm_fd);
+
+		igt_require(display.is_atomic);
+
+		igt_display_require_output(&display);
+
+		igt_require_sw_sync();
+	}
+
+	igt_subtest("legacy-setmode")
+		for_each_pipe_with_valid_output(&display, pipe, output) {
+			run_plane_test(&display, pipe, output, test_legacy_modeset, DRM_PLANE_TYPE_PRIMARY);
+			break;
+		}
+
+	igt_subtest("atomic-setmode")
+		for_each_pipe_with_valid_output(&display, pipe, output) {
+			run_plane_test(&display, pipe, output, test_atomic_modeset, DRM_PLANE_TYPE_PRIMARY);
+			break;
+		}
+
+	igt_subtest("legacy-dpms")
+		for_each_pipe_with_valid_output(&display, pipe, output) {
+			run_plane_test(&display, pipe, output, test_legacy_dpms, DRM_PLANE_TYPE_PRIMARY);
+			break;
+		}
+
+	igt_subtest("legacy-pageflip")
+		for_each_pipe_with_valid_output(&display, pipe, output) {
+			run_plane_test(&display, pipe, output, test_pageflip, DRM_PLANE_TYPE_PRIMARY);
+			break;
+		}
+
+	igt_subtest("legacy-cursor")
+		for_each_pipe_with_valid_output(&display, pipe, output) {
+			run_plane_test(&display, pipe, output, test_setcursor, DRM_PLANE_TYPE_CURSOR);
+			break;
+		}
+
+	igt_subtest("universal-setplane-primary")
+		for_each_pipe_with_valid_output(&display, pipe, output) {
+			run_plane_test(&display, pipe, output, test_setplane, DRM_PLANE_TYPE_PRIMARY);
+			break;
+		}
+
+	igt_subtest("universal-setplane-cursor")
+		for_each_pipe_with_valid_output(&display, pipe, output) {
+			run_plane_test(&display, pipe, output, test_setplane, DRM_PLANE_TYPE_CURSOR);
+			break;
+		}
+
+	/* TODO: legacy gamma_set/get, object set/getprop, getcrtc, getconnector */
+	igt_fixture {
+		igt_display_fini(&display);
+	}
+}