/* * Copyright 2019 Advanced Micro Devices, Inc. * * 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 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 COPYRIGHT HOLDER(S) OR AUTHOR(S) 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" /* Common test data */ typedef struct data { struct igt_fb pattern_fb_info; int fd; igt_display_t display; igt_plane_t *primary; igt_output_t *output; igt_pipe_t *pipe; enum pipe pipe_id; bool use_virtual_connector; } data_t; /* Video modes indexed by VIC */ static drmModeModeInfo test_modes[] = { [1] = { 25175, 640, 656, 752, 800, 0, 480, 489, 492, 525, 0, 60, 0xa, 0x40, "640x480", /* VIC 1 */ }, [96] = { 594000, 3840, 4896, 4984, 5280, 0, 2160, 2168, 2178, 2250, 0, 50, 0x5|DRM_MODE_FLAG_PIC_AR_16_9, 0x40, "3840x2160", /* VIC 96 */ }, [97] = { 594000, 3840, 4016, 4104, 4400, 0, 2160, 2168, 2178, 2250, 0, 60, 0x5|DRM_MODE_FLAG_PIC_AR_16_9, 0x40, "3840x2160", /* VIC 97 */ }, [101] = { 594000, 4096, 5064, 5152, 5280, 0, 2160, 2168, 2178, 2250, 0, 50, 0x5|DRM_MODE_FLAG_PIC_AR_256_135, 0x40, "4096x2160", /* VIC 101 */ }, [102] = { 594000, 4096, 4184, 4272, 4400, 0, 2160, 2168, 2178, 2250, 0, 60, 0x5|DRM_MODE_FLAG_PIC_AR_256_135, 0x40, "4096x2160", /* VIC 102 */ }, [106] = { 594000, 3840, 4896, 4984, 5280, 0, 2160, 2168, 2178, 2250, 0, 50, 0x5|DRM_MODE_FLAG_PIC_AR_64_27, 0x40, "3840x2160", /* VIC 106 */ }, [107] = { 594000, 3840, 4016, 4104, 4400, 0, 2160, 2168, 2178, 2250, 0, 60, 0x5|DRM_MODE_FLAG_PIC_AR_64_27, 0x40, "3840x2160", /* VIC 107 */ }, }; /* Common test setup. */ static void test_init(data_t *data) { igt_display_t *display = &data->display; data->pipe_id = PIPE_A; data->pipe = &data->display.pipes[data->pipe_id]; igt_display_reset(display); /* find a connected HDMI output */ data->output = NULL; for (int i=0; i < data->display.n_outputs; ++i) { drmModeConnector *connector = data->display.outputs[i].config.connector; if (connector->connection == DRM_MODE_CONNECTED && (connector->connector_type == DRM_MODE_CONNECTOR_HDMIA || (data->use_virtual_connector && connector->connector_type == DRM_MODE_CONNECTOR_VIRTUAL))) { data->output = &data->display.outputs[i]; } } igt_require(data->output); data->primary = igt_pipe_get_plane_type(data->pipe, DRM_PLANE_TYPE_PRIMARY); igt_output_set_pipe(data->output, data->pipe_id); } /* Common test cleanup. */ static void test_fini(data_t *data) { igt_display_reset(&data->display); } static void wait_for_keypress(void) { while (getchar() != '\n') ; } static void test_vic_mode(data_t *data, int vic) { igt_display_t *display = &data->display; drmModeModeInfo *mode; igt_fb_t afb; test_init(data); mode = &test_modes[vic]; igt_output_override_mode(data->output, mode); igt_create_pattern_fb(data->fd, mode->hdisplay, mode->vdisplay, DRM_FORMAT_XRGB8888, 0, &afb); igt_plane_set_fb(data->primary, &afb); igt_display_commit_atomic(display, DRM_MODE_ATOMIC_ALLOW_MODESET, NULL); igt_info("Press [Enter] to finish\n"); wait_for_keypress(); test_fini(data); } const char *optstr = "hvt:"; static void usage(const char *name) { igt_info("Usage: %s [options]\n", name); igt_info("-h Show help\n"); igt_info("-t vic Select video mode based on VIC\n"); igt_info("-v Test on 'Virtual' connector as well, for debugging.\n"); } int main(int argc, char **argv) { data_t data; int c; int vic = 1; /* default to VIC 1 (640x480) */ memset(&data, 0, sizeof(data)); while((c = getopt(argc, argv, optstr)) != -1) { switch(c) { case 't': vic = atoi(optarg); break; case 'v': data.use_virtual_connector = true; break; default: case 'h': usage(argv[0]); exit(EXIT_SUCCESS); } } if (vic < 1 || vic > ARRAY_SIZE(test_modes) || !test_modes[vic].name[0]) { igt_warn("VIC %d is not supported\n", vic); exit(EXIT_FAILURE); } data.fd = drm_open_driver_master(DRIVER_ANY); kmstest_set_vt_graphics_mode(); igt_display_require(&data.display, data.fd); igt_require(data.display.is_atomic); igt_display_require_output(&data.display); test_vic_mode(&data, vic); igt_display_fini(&data.display); }