/* * Copyright © 2014 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" IGT_TEST_DESCRIPTION("Check the debugfs force connector/edid features work" " correctly."); #define CHECK_MODE(m, h, w, r) \ igt_assert_eq(m.hdisplay, h); igt_assert_eq(m.vdisplay, w); \ igt_assert_eq(m.vrefresh, r); static void reset_connectors(void) { int drm_fd = 0; drmModeRes *res; drmModeConnector *connector = NULL; drm_fd = drm_open_driver_master(DRIVER_INTEL); res = drmModeGetResources(drm_fd); for (int i = 0; i < res->count_connectors; i++) { connector = drmModeGetConnectorCurrent(drm_fd, res->connectors[i]); kmstest_force_connector(drm_fd, connector, FORCE_CONNECTOR_UNSPECIFIED); kmstest_force_edid(drm_fd, connector, NULL, 0); drmModeFreeConnector(connector); } igt_set_module_param_int("load_detect_test", 0); } static int opt_handler(int opt, int opt_index, void *data) { switch (opt) { case 'r': reset_connectors(); exit(0); break; } return IGT_OPT_HANDLER_SUCCESS; } struct option long_opts[] = { {"reset", 0, 0, 'r'}, {0, 0, 0, 0} }; const char *help_str = " --reset\t\tReset all connector force states and edid.\n"; igt_main_args("", long_opts, help_str, opt_handler, NULL) { /* force the VGA output and test that it worked */ int drm_fd = 0; drmModeRes *res; drmModeConnector *vga_connector = NULL, *temp; int start_n_modes, start_connection; igt_fixture { unsigned vga_connector_id = 0; drm_fd = drm_open_driver_master(DRIVER_INTEL); res = drmModeGetResources(drm_fd); igt_require(res); /* find the vga connector */ for (int i = 0; i < res->count_connectors; i++) { vga_connector = drmModeGetConnectorCurrent(drm_fd, res->connectors[i]); if (vga_connector->connector_type == DRM_MODE_CONNECTOR_VGA) { /* Ensure that no override was left in place. */ kmstest_force_connector(drm_fd, vga_connector, FORCE_CONNECTOR_UNSPECIFIED); /* Only use the first VGA connector. */ if (!vga_connector_id) vga_connector_id = res->connectors[i]; } drmModeFreeConnector(vga_connector); } igt_require(vga_connector_id); /* Reacquire status after clearing any previous overrides */ vga_connector = drmModeGetConnector(drm_fd, vga_connector_id); start_n_modes = vga_connector->count_modes; start_connection = vga_connector->connection; } igt_subtest("force-load-detect") { int i, j, w = 64, h = 64; drmModePlaneRes *plane_resources; struct igt_fb xrgb_fb, argb_fb; igt_create_fb(drm_fd, w, h, DRM_FORMAT_XRGB8888, 0, &xrgb_fb); igt_create_fb(drm_fd, w, h, DRM_FORMAT_ARGB8888, 0, &argb_fb); igt_assert(drmSetClientCap(drm_fd, DRM_CLIENT_CAP_UNIVERSAL_PLANES, 1) == 0); /* * disable all outputs to make sure we have a * free crtc available for load detect */ kmstest_set_vt_graphics_mode(); kmstest_unset_all_crtcs(drm_fd, res); igt_set_module_param_int("load_detect_test", 1); plane_resources = drmModeGetPlaneResources(drm_fd); igt_assert(plane_resources); for (i = 0; i < plane_resources->count_planes; i++) { drmModePlane *drm_plane; bool found = false; uint32_t plane_id = plane_resources->planes[i]; drm_plane = drmModeGetPlane(drm_fd, plane_id); igt_assert(drm_plane); for (j = 0; j < drm_plane->count_formats; j++) { uint32_t format = drm_plane->formats[j]; uint32_t crtc = ffs(drm_plane->possible_crtcs) - 1; uint32_t crtc_id = res->crtcs[crtc]; if (format == DRM_FORMAT_XRGB8888) do_or_die(drmModeSetPlane(drm_fd, plane_id, crtc_id, xrgb_fb.fb_id, 0, 0, 0, w, h, 0, 0, IGT_FIXED(w, 0), IGT_FIXED(h, 0))); else if (format == DRM_FORMAT_ARGB8888) do_or_die(drmModeSetPlane(drm_fd, plane_id, crtc_id, argb_fb.fb_id, 0, 0, 0, w, h, 0, 0, IGT_FIXED(w, 0), IGT_FIXED(h, 0))); else continue; found = true; break; } drmModeFreePlane(drm_plane); igt_assert(found); } /* This can't use drmModeGetConnectorCurrent * because connector probing is the point of this test. */ temp = drmModeGetConnector(drm_fd, vga_connector->connector_id); igt_set_module_param_int("load_detect_test", 0); igt_assert(temp->connection != DRM_MODE_UNKNOWNCONNECTION); drmModeFreeConnector(temp); /* Look if planes are unmodified. */ for (i = 0; i < plane_resources->count_planes; i++) { drmModePlane *drm_plane; drm_plane = drmModeGetPlane(drm_fd, plane_resources->planes[i]); igt_assert(drm_plane); igt_assert(drm_plane->crtc_id); igt_assert(drm_plane->fb_id); if (drm_plane->fb_id != xrgb_fb.fb_id) igt_assert_eq(drm_plane->fb_id, argb_fb.fb_id); drmModeFreePlane(drm_plane); } } igt_subtest("force-connector-state") { igt_display_t display; /* force the connector on and check the reported values */ kmstest_force_connector(drm_fd, vga_connector, FORCE_CONNECTOR_ON); temp = drmModeGetConnectorCurrent(drm_fd, vga_connector->connector_id); igt_assert_eq(temp->connection, DRM_MODE_CONNECTED); igt_assert_lt(0, temp->count_modes); drmModeFreeConnector(temp); /* attempt to use the display */ kmstest_set_vt_graphics_mode(); igt_display_require(&display, drm_fd); igt_display_commit(&display); igt_display_fini(&display); /* force the connector off */ kmstest_force_connector(drm_fd, vga_connector, FORCE_CONNECTOR_OFF); temp = drmModeGetConnectorCurrent(drm_fd, vga_connector->connector_id); igt_assert_eq(temp->connection, DRM_MODE_DISCONNECTED); igt_assert_eq(0, temp->count_modes); drmModeFreeConnector(temp); /* check that the previous state is restored */ kmstest_force_connector(drm_fd, vga_connector, FORCE_CONNECTOR_UNSPECIFIED); temp = drmModeGetConnectorCurrent(drm_fd, vga_connector->connector_id); igt_assert_eq(temp->connection, start_connection); drmModeFreeConnector(temp); } igt_subtest("force-edid") { kmstest_force_connector(drm_fd, vga_connector, FORCE_CONNECTOR_ON); temp = drmModeGetConnectorCurrent(drm_fd, vga_connector->connector_id); drmModeFreeConnector(temp); /* test edid forcing */ kmstest_force_edid(drm_fd, vga_connector, igt_kms_get_base_edid(), EDID_LENGTH); temp = drmModeGetConnectorCurrent(drm_fd, vga_connector->connector_id); igt_debug("num_conn %i\n", temp->count_modes); CHECK_MODE(temp->modes[0], 1920, 1080, 60); /* Don't check non-preferred modes to avoid to tight coupling * with the in-kernel EDID parser. */ drmModeFreeConnector(temp); /* remove edid */ kmstest_force_edid(drm_fd, vga_connector, NULL, 0); kmstest_force_connector(drm_fd, vga_connector, FORCE_CONNECTOR_UNSPECIFIED); temp = drmModeGetConnectorCurrent(drm_fd, vga_connector->connector_id); /* the connector should now have the same number of modes that * it started with */ igt_assert_eq(temp->count_modes, start_n_modes); drmModeFreeConnector(temp); } igt_subtest("prune-stale-modes") { int i; kmstest_force_connector(drm_fd, vga_connector, FORCE_CONNECTOR_ON); /* test pruning of stale modes */ kmstest_force_edid(drm_fd, vga_connector, igt_kms_get_alt_edid(), EDID_LENGTH); temp = drmModeGetConnectorCurrent(drm_fd, vga_connector->connector_id); for (i = 0; i < temp->count_modes; i++) { if (temp->modes[i].hdisplay == 1400 && temp->modes[i].vdisplay == 1050) break; } igt_assert_f(i != temp->count_modes, "1400x1050 not on mode list\n"); drmModeFreeConnector(temp); kmstest_force_edid(drm_fd, vga_connector, igt_kms_get_base_edid(), EDID_LENGTH); temp = drmModeGetConnectorCurrent(drm_fd, vga_connector->connector_id); for (i = 0; i < temp->count_modes; i++) { if (temp->modes[i].hdisplay == 1400 && temp->modes[i].vdisplay == 1050) break; } igt_assert_f(i == temp->count_modes, "1400x1050 not pruned from mode list\n"); drmModeFreeConnector(temp); kmstest_force_edid(drm_fd, vga_connector, NULL, 0); kmstest_force_connector(drm_fd, vga_connector, FORCE_CONNECTOR_UNSPECIFIED); } igt_fixture { drmModeFreeConnector(vga_connector); close(drm_fd); reset_connectors(); } }