blob: be688638973f6874fd8a9dd0a900bd9845dabf91 [file] [log] [blame]
/*
* 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 <signal.h>
#include "igt.h"
#include "drmtest.h"
#include "sw_sync.h"
IGT_TEST_DESCRIPTION("Tests that interrupt various atomic ioctls.");
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_require 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);
/* wait for the block to complete in block_plane */
sleep(1);
igt_fork(child, 1) {
signal(SIGCONT, SIG_IGN);
/* unblock after 5 seconds to allow the ioctl to complete,
* instead of failing with -EINTR.
*/
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_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->props[IGT_CRTC_MODE_ID],
plane->pipe->props[IGT_CRTC_ACTIVE],
/* connector: 1 prop */
output->props[IGT_CONNECTOR_CRTC_ID],
/* plane: remainder props */
plane->props[IGT_PLANE_CRTC_ID],
plane->props[IGT_PLANE_FB_ID],
plane->props[IGT_PLANE_SRC_W],
plane->props[IGT_PLANE_SRC_H],
plane->props[IGT_PLANE_CRTC_W],
plane->props[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_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->props[IGT_CONNECTOR_DPMS],
.connector_id = output->id,
};
do_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_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_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_ioctl(display->drm_fd, DRM_IOCTL_MODE_PAGE_FLIP, &pageflip);
drmHandleEvent(display->drm_fd, &drm_events);
break;
}
}
}
igt_waitchildren();
/* The mode is unset by the forked helper, force a refresh here */
if (test_type == test_legacy_modeset || test_type == test_atomic_modeset)
igt_pipe_refresh(display, pipe, true);
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_require(&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);
}
}