xref: /aosp_15_r20/external/igt-gpu-tools/tests/kms_concurrent.c (revision d83cc019efdc2edc6c4b16e9034a3ceb8d35d77c)
1*d83cc019SAndroid Build Coastguard Worker /*
2*d83cc019SAndroid Build Coastguard Worker  * Copyright © 2017 Intel Corporation
3*d83cc019SAndroid Build Coastguard Worker  *
4*d83cc019SAndroid Build Coastguard Worker  * Permission is hereby granted, free of charge, to any person obtaining a
5*d83cc019SAndroid Build Coastguard Worker  * copy of this software and associated documentation files (the "Software"),
6*d83cc019SAndroid Build Coastguard Worker  * to deal in the Software without restriction, including without limitation
7*d83cc019SAndroid Build Coastguard Worker  * the rights to use, copy, modify, merge, publish, distribute, sublicense,
8*d83cc019SAndroid Build Coastguard Worker  * and/or sell copies of the Software, and to permit persons to whom the
9*d83cc019SAndroid Build Coastguard Worker  * Software is furnished to do so, subject to the following conditions:
10*d83cc019SAndroid Build Coastguard Worker  *
11*d83cc019SAndroid Build Coastguard Worker  * The above copyright notice and this permission notice (including the next
12*d83cc019SAndroid Build Coastguard Worker  * paragraph) shall be included in all copies or substantial portions of the
13*d83cc019SAndroid Build Coastguard Worker  * Software.
14*d83cc019SAndroid Build Coastguard Worker  *
15*d83cc019SAndroid Build Coastguard Worker  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16*d83cc019SAndroid Build Coastguard Worker  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17*d83cc019SAndroid Build Coastguard Worker  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
18*d83cc019SAndroid Build Coastguard Worker  * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19*d83cc019SAndroid Build Coastguard Worker  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
20*d83cc019SAndroid Build Coastguard Worker  * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
21*d83cc019SAndroid Build Coastguard Worker  * IN THE SOFTWARE.
22*d83cc019SAndroid Build Coastguard Worker  *
23*d83cc019SAndroid Build Coastguard Worker  */
24*d83cc019SAndroid Build Coastguard Worker 
25*d83cc019SAndroid Build Coastguard Worker #include "igt.h"
26*d83cc019SAndroid Build Coastguard Worker #include "drmtest.h"
27*d83cc019SAndroid Build Coastguard Worker 
28*d83cc019SAndroid Build Coastguard Worker IGT_TEST_DESCRIPTION("Test atomic mode setting concurrently with multiple planes and screen resolution");
29*d83cc019SAndroid Build Coastguard Worker 
30*d83cc019SAndroid Build Coastguard Worker #define SIZE_PLANE      256
31*d83cc019SAndroid Build Coastguard Worker #define SIZE_CURSOR     128
32*d83cc019SAndroid Build Coastguard Worker #define LOOP_FOREVER     -1
33*d83cc019SAndroid Build Coastguard Worker 
34*d83cc019SAndroid Build Coastguard Worker typedef struct {
35*d83cc019SAndroid Build Coastguard Worker 	int drm_fd;
36*d83cc019SAndroid Build Coastguard Worker 	igt_display_t display;
37*d83cc019SAndroid Build Coastguard Worker 	igt_plane_t **plane;
38*d83cc019SAndroid Build Coastguard Worker 	struct igt_fb *fb;
39*d83cc019SAndroid Build Coastguard Worker } data_t;
40*d83cc019SAndroid Build Coastguard Worker 
41*d83cc019SAndroid Build Coastguard Worker /* Command line parameters. */
42*d83cc019SAndroid Build Coastguard Worker struct {
43*d83cc019SAndroid Build Coastguard Worker 	int iterations;
44*d83cc019SAndroid Build Coastguard Worker 	bool user_seed;
45*d83cc019SAndroid Build Coastguard Worker 	int seed;
46*d83cc019SAndroid Build Coastguard Worker 	bool run;
47*d83cc019SAndroid Build Coastguard Worker } opt = {
48*d83cc019SAndroid Build Coastguard Worker 	.iterations = 1,
49*d83cc019SAndroid Build Coastguard Worker 	.user_seed = false,
50*d83cc019SAndroid Build Coastguard Worker 	.seed = 1,
51*d83cc019SAndroid Build Coastguard Worker 	.run = true,
52*d83cc019SAndroid Build Coastguard Worker };
53*d83cc019SAndroid Build Coastguard Worker 
54*d83cc019SAndroid Build Coastguard Worker /*
55*d83cc019SAndroid Build Coastguard Worker  * Common code across all tests, acting on data_t
56*d83cc019SAndroid Build Coastguard Worker  */
test_init(data_t * data,enum pipe pipe,int n_planes,igt_output_t * output)57*d83cc019SAndroid Build Coastguard Worker static void test_init(data_t *data, enum pipe pipe, int n_planes,
58*d83cc019SAndroid Build Coastguard Worker 		      igt_output_t *output)
59*d83cc019SAndroid Build Coastguard Worker {
60*d83cc019SAndroid Build Coastguard Worker 	drmModeModeInfo *mode;
61*d83cc019SAndroid Build Coastguard Worker 	igt_plane_t *primary;
62*d83cc019SAndroid Build Coastguard Worker 	int ret;
63*d83cc019SAndroid Build Coastguard Worker 
64*d83cc019SAndroid Build Coastguard Worker 	data->plane = calloc(n_planes, sizeof(*data->plane));
65*d83cc019SAndroid Build Coastguard Worker 	igt_assert_f(data->plane != NULL, "Failed to allocate memory for planes\n");
66*d83cc019SAndroid Build Coastguard Worker 
67*d83cc019SAndroid Build Coastguard Worker 	data->fb = calloc(n_planes, sizeof(struct igt_fb));
68*d83cc019SAndroid Build Coastguard Worker 	igt_assert_f(data->fb != NULL, "Failed to allocate memory for FBs\n");
69*d83cc019SAndroid Build Coastguard Worker 
70*d83cc019SAndroid Build Coastguard Worker 	igt_output_set_pipe(output, pipe);
71*d83cc019SAndroid Build Coastguard Worker 
72*d83cc019SAndroid Build Coastguard Worker 	primary = igt_output_get_plane_type(output, DRM_PLANE_TYPE_PRIMARY);
73*d83cc019SAndroid Build Coastguard Worker 	data->plane[primary->index] = primary;
74*d83cc019SAndroid Build Coastguard Worker 
75*d83cc019SAndroid Build Coastguard Worker 	mode = igt_output_get_mode(output);
76*d83cc019SAndroid Build Coastguard Worker 
77*d83cc019SAndroid Build Coastguard Worker 	igt_create_color_fb(data->drm_fd, mode->hdisplay, mode->vdisplay,
78*d83cc019SAndroid Build Coastguard Worker 			    DRM_FORMAT_XRGB8888,
79*d83cc019SAndroid Build Coastguard Worker 			    LOCAL_I915_FORMAT_MOD_X_TILED,
80*d83cc019SAndroid Build Coastguard Worker 			    0.0f, 0.0f, 1.0f,
81*d83cc019SAndroid Build Coastguard Worker 			    &data->fb[primary->index]);
82*d83cc019SAndroid Build Coastguard Worker 
83*d83cc019SAndroid Build Coastguard Worker 	igt_plane_set_fb(data->plane[primary->index], &data->fb[primary->index]);
84*d83cc019SAndroid Build Coastguard Worker 
85*d83cc019SAndroid Build Coastguard Worker 	ret = igt_display_try_commit2(&data->display, COMMIT_ATOMIC);
86*d83cc019SAndroid Build Coastguard Worker 	igt_skip_on(ret != 0);
87*d83cc019SAndroid Build Coastguard Worker }
88*d83cc019SAndroid Build Coastguard Worker 
test_fini(data_t * data,enum pipe pipe,int n_planes,igt_output_t * output)89*d83cc019SAndroid Build Coastguard Worker static void test_fini(data_t *data, enum pipe pipe, int n_planes,
90*d83cc019SAndroid Build Coastguard Worker 		      igt_output_t *output)
91*d83cc019SAndroid Build Coastguard Worker {
92*d83cc019SAndroid Build Coastguard Worker 	int i;
93*d83cc019SAndroid Build Coastguard Worker 
94*d83cc019SAndroid Build Coastguard Worker 	for (i = 0; i < n_planes; i++) {
95*d83cc019SAndroid Build Coastguard Worker 		igt_plane_t *plane = data->plane[i];
96*d83cc019SAndroid Build Coastguard Worker 
97*d83cc019SAndroid Build Coastguard Worker 		if (!plane)
98*d83cc019SAndroid Build Coastguard Worker 			continue;
99*d83cc019SAndroid Build Coastguard Worker 
100*d83cc019SAndroid Build Coastguard Worker 		if (plane->type == DRM_PLANE_TYPE_PRIMARY)
101*d83cc019SAndroid Build Coastguard Worker 			continue;
102*d83cc019SAndroid Build Coastguard Worker 
103*d83cc019SAndroid Build Coastguard Worker 		igt_plane_set_fb(plane, NULL);
104*d83cc019SAndroid Build Coastguard Worker 
105*d83cc019SAndroid Build Coastguard Worker 		data->plane[i] = NULL;
106*d83cc019SAndroid Build Coastguard Worker 	}
107*d83cc019SAndroid Build Coastguard Worker 
108*d83cc019SAndroid Build Coastguard Worker 	/* reset the constraint on the pipe */
109*d83cc019SAndroid Build Coastguard Worker 	igt_output_set_pipe(output, PIPE_ANY);
110*d83cc019SAndroid Build Coastguard Worker 
111*d83cc019SAndroid Build Coastguard Worker 	free(data->plane);
112*d83cc019SAndroid Build Coastguard Worker 	data->plane = NULL;
113*d83cc019SAndroid Build Coastguard Worker 
114*d83cc019SAndroid Build Coastguard Worker 	free(data->fb);
115*d83cc019SAndroid Build Coastguard Worker 	data->fb = NULL;
116*d83cc019SAndroid Build Coastguard Worker }
117*d83cc019SAndroid Build Coastguard Worker 
118*d83cc019SAndroid Build Coastguard Worker static void
create_fb_for_mode_position(data_t * data,drmModeModeInfo * mode,int * rect_x,int * rect_y,int * rect_w,int * rect_h,uint64_t tiling,int max_planes,igt_output_t * output)119*d83cc019SAndroid Build Coastguard Worker create_fb_for_mode_position(data_t *data, drmModeModeInfo *mode,
120*d83cc019SAndroid Build Coastguard Worker 			    int *rect_x, int *rect_y,
121*d83cc019SAndroid Build Coastguard Worker 			    int *rect_w, int *rect_h,
122*d83cc019SAndroid Build Coastguard Worker 			    uint64_t tiling, int max_planes,
123*d83cc019SAndroid Build Coastguard Worker 			    igt_output_t *output)
124*d83cc019SAndroid Build Coastguard Worker {
125*d83cc019SAndroid Build Coastguard Worker 	unsigned int fb_id;
126*d83cc019SAndroid Build Coastguard Worker 	cairo_t *cr;
127*d83cc019SAndroid Build Coastguard Worker 	igt_plane_t *primary;
128*d83cc019SAndroid Build Coastguard Worker 
129*d83cc019SAndroid Build Coastguard Worker 	primary = igt_output_get_plane_type(output, DRM_PLANE_TYPE_PRIMARY);
130*d83cc019SAndroid Build Coastguard Worker 
131*d83cc019SAndroid Build Coastguard Worker 	fb_id = igt_create_fb(data->drm_fd,
132*d83cc019SAndroid Build Coastguard Worker 			      mode->hdisplay, mode->vdisplay,
133*d83cc019SAndroid Build Coastguard Worker 			      DRM_FORMAT_XRGB8888,
134*d83cc019SAndroid Build Coastguard Worker 			      tiling,
135*d83cc019SAndroid Build Coastguard Worker 			      &data->fb[primary->index]);
136*d83cc019SAndroid Build Coastguard Worker 	igt_assert(fb_id);
137*d83cc019SAndroid Build Coastguard Worker 
138*d83cc019SAndroid Build Coastguard Worker 	cr = igt_get_cairo_ctx(data->drm_fd, &data->fb[primary->index]);
139*d83cc019SAndroid Build Coastguard Worker 	igt_paint_color(cr, rect_x[0], rect_y[0],
140*d83cc019SAndroid Build Coastguard Worker 			mode->hdisplay, mode->vdisplay,
141*d83cc019SAndroid Build Coastguard Worker 			0.0f, 0.0f, 1.0f);
142*d83cc019SAndroid Build Coastguard Worker 
143*d83cc019SAndroid Build Coastguard Worker 	for (int i = 0; i < max_planes; i++) {
144*d83cc019SAndroid Build Coastguard Worker 		if (data->plane[i]->type == DRM_PLANE_TYPE_PRIMARY)
145*d83cc019SAndroid Build Coastguard Worker 			continue;
146*d83cc019SAndroid Build Coastguard Worker 
147*d83cc019SAndroid Build Coastguard Worker 		igt_paint_color(cr, rect_x[i], rect_y[i],
148*d83cc019SAndroid Build Coastguard Worker 				rect_w[i], rect_h[i], 0.0, 0.0, 0.0);
149*d83cc019SAndroid Build Coastguard Worker 	}
150*d83cc019SAndroid Build Coastguard Worker 
151*d83cc019SAndroid Build Coastguard Worker 	igt_put_cairo_ctx(data->drm_fd, &data->fb[primary->index], cr);
152*d83cc019SAndroid Build Coastguard Worker }
153*d83cc019SAndroid Build Coastguard Worker 
154*d83cc019SAndroid Build Coastguard Worker static void
prepare_planes(data_t * data,enum pipe pipe,int max_planes,igt_output_t * output)155*d83cc019SAndroid Build Coastguard Worker prepare_planes(data_t *data, enum pipe pipe, int max_planes,
156*d83cc019SAndroid Build Coastguard Worker 	       igt_output_t *output)
157*d83cc019SAndroid Build Coastguard Worker {
158*d83cc019SAndroid Build Coastguard Worker 	drmModeModeInfo *mode;
159*d83cc019SAndroid Build Coastguard Worker 	igt_pipe_t *p;
160*d83cc019SAndroid Build Coastguard Worker 	igt_plane_t *primary;
161*d83cc019SAndroid Build Coastguard Worker 	int *x;
162*d83cc019SAndroid Build Coastguard Worker 	int *y;
163*d83cc019SAndroid Build Coastguard Worker 	int *size;
164*d83cc019SAndroid Build Coastguard Worker 	int i;
165*d83cc019SAndroid Build Coastguard Worker 
166*d83cc019SAndroid Build Coastguard Worker 	igt_output_set_pipe(output, pipe);
167*d83cc019SAndroid Build Coastguard Worker 
168*d83cc019SAndroid Build Coastguard Worker 	primary = igt_output_get_plane_type(output, DRM_PLANE_TYPE_PRIMARY);
169*d83cc019SAndroid Build Coastguard Worker 	p = primary->pipe;
170*d83cc019SAndroid Build Coastguard Worker 
171*d83cc019SAndroid Build Coastguard Worker 	x = malloc(p->n_planes * sizeof(*x));
172*d83cc019SAndroid Build Coastguard Worker 	igt_assert_f(x, "Failed to allocate %ld bytes for variable x\n", (long int) (p->n_planes * sizeof(*x)));
173*d83cc019SAndroid Build Coastguard Worker 
174*d83cc019SAndroid Build Coastguard Worker 	y = malloc(p->n_planes * sizeof(*y));
175*d83cc019SAndroid Build Coastguard Worker 	igt_assert_f(y, "Failed to allocate %ld bytes for variable y\n", (long int) (p->n_planes * sizeof(*y)));
176*d83cc019SAndroid Build Coastguard Worker 
177*d83cc019SAndroid Build Coastguard Worker 	size = malloc(p->n_planes * sizeof(*size));
178*d83cc019SAndroid Build Coastguard Worker 	igt_assert_f(size, "Failed to allocate %ld bytes for variable size\n", (long int) (p->n_planes * sizeof(*size)));
179*d83cc019SAndroid Build Coastguard Worker 
180*d83cc019SAndroid Build Coastguard Worker 	mode = igt_output_get_mode(output);
181*d83cc019SAndroid Build Coastguard Worker 
182*d83cc019SAndroid Build Coastguard Worker 	/* planes with random positions */
183*d83cc019SAndroid Build Coastguard Worker 	x[primary->index] = 0;
184*d83cc019SAndroid Build Coastguard Worker 	y[primary->index] = 0;
185*d83cc019SAndroid Build Coastguard Worker 	for (i = 0; i < max_planes; i++) {
186*d83cc019SAndroid Build Coastguard Worker 		igt_plane_t *plane = igt_output_get_plane(output, i);
187*d83cc019SAndroid Build Coastguard Worker 		int ret;
188*d83cc019SAndroid Build Coastguard Worker 
189*d83cc019SAndroid Build Coastguard Worker 		if (plane->type == DRM_PLANE_TYPE_PRIMARY)
190*d83cc019SAndroid Build Coastguard Worker 			continue;
191*d83cc019SAndroid Build Coastguard Worker 		else if (plane->type == DRM_PLANE_TYPE_CURSOR)
192*d83cc019SAndroid Build Coastguard Worker 			size[i] = SIZE_CURSOR;
193*d83cc019SAndroid Build Coastguard Worker 		else
194*d83cc019SAndroid Build Coastguard Worker 			size[i] = SIZE_PLANE;
195*d83cc019SAndroid Build Coastguard Worker 
196*d83cc019SAndroid Build Coastguard Worker 		x[i] = rand() % (mode->hdisplay - size[i]);
197*d83cc019SAndroid Build Coastguard Worker 		y[i] = rand() % (mode->vdisplay - size[i]);
198*d83cc019SAndroid Build Coastguard Worker 
199*d83cc019SAndroid Build Coastguard Worker 		data->plane[i] = plane;
200*d83cc019SAndroid Build Coastguard Worker 
201*d83cc019SAndroid Build Coastguard Worker 		igt_create_color_fb(data->drm_fd,
202*d83cc019SAndroid Build Coastguard Worker 				    size[i], size[i],
203*d83cc019SAndroid Build Coastguard Worker 				    data->plane[i]->type == DRM_PLANE_TYPE_CURSOR ? DRM_FORMAT_ARGB8888 : DRM_FORMAT_XRGB8888,
204*d83cc019SAndroid Build Coastguard Worker 				    data->plane[i]->type == DRM_PLANE_TYPE_CURSOR ? LOCAL_DRM_FORMAT_MOD_NONE : LOCAL_I915_FORMAT_MOD_X_TILED,
205*d83cc019SAndroid Build Coastguard Worker 				    0.0f, 0.0f, 1.0f,
206*d83cc019SAndroid Build Coastguard Worker 				    &data->fb[i]);
207*d83cc019SAndroid Build Coastguard Worker 
208*d83cc019SAndroid Build Coastguard Worker 		igt_plane_set_position(data->plane[i], x[i], y[i]);
209*d83cc019SAndroid Build Coastguard Worker 		igt_plane_set_fb(data->plane[i], &data->fb[i]);
210*d83cc019SAndroid Build Coastguard Worker 
211*d83cc019SAndroid Build Coastguard Worker 		ret = igt_display_try_commit_atomic(&data->display, DRM_MODE_ATOMIC_TEST_ONLY, NULL);
212*d83cc019SAndroid Build Coastguard Worker 		if (ret) {
213*d83cc019SAndroid Build Coastguard Worker 			igt_plane_set_fb(data->plane[i], NULL);
214*d83cc019SAndroid Build Coastguard Worker 			igt_remove_fb(data->drm_fd, &data->fb[i]);
215*d83cc019SAndroid Build Coastguard Worker 			data->plane[i] = NULL;
216*d83cc019SAndroid Build Coastguard Worker 			break;
217*d83cc019SAndroid Build Coastguard Worker 		}
218*d83cc019SAndroid Build Coastguard Worker 	}
219*d83cc019SAndroid Build Coastguard Worker 	max_planes = i;
220*d83cc019SAndroid Build Coastguard Worker 
221*d83cc019SAndroid Build Coastguard Worker 	igt_assert_lt(0, max_planes);
222*d83cc019SAndroid Build Coastguard Worker 
223*d83cc019SAndroid Build Coastguard Worker 	/* primary plane */
224*d83cc019SAndroid Build Coastguard Worker 	data->plane[primary->index] = primary;
225*d83cc019SAndroid Build Coastguard Worker 	create_fb_for_mode_position(data, mode, x, y, size, size,
226*d83cc019SAndroid Build Coastguard Worker 				    LOCAL_I915_FORMAT_MOD_X_TILED,
227*d83cc019SAndroid Build Coastguard Worker 				    max_planes, output);
228*d83cc019SAndroid Build Coastguard Worker 
229*d83cc019SAndroid Build Coastguard Worker 	igt_plane_set_fb(data->plane[primary->index], &data->fb[primary->index]);
230*d83cc019SAndroid Build Coastguard Worker }
231*d83cc019SAndroid Build Coastguard Worker 
232*d83cc019SAndroid Build Coastguard Worker static void
test_plane_position_with_output(data_t * data,enum pipe pipe,igt_output_t * output)233*d83cc019SAndroid Build Coastguard Worker test_plane_position_with_output(data_t *data, enum pipe pipe, igt_output_t *output)
234*d83cc019SAndroid Build Coastguard Worker {
235*d83cc019SAndroid Build Coastguard Worker 	int i;
236*d83cc019SAndroid Build Coastguard Worker 	int iterations = opt.iterations < 1 ? 1 : opt.iterations;
237*d83cc019SAndroid Build Coastguard Worker 	bool loop_forever = opt.iterations == LOOP_FOREVER ? true : false;
238*d83cc019SAndroid Build Coastguard Worker 	int max_planes = data->display.pipes[pipe].n_planes;
239*d83cc019SAndroid Build Coastguard Worker 
240*d83cc019SAndroid Build Coastguard Worker 	igt_pipe_refresh(&data->display, pipe, true);
241*d83cc019SAndroid Build Coastguard Worker 
242*d83cc019SAndroid Build Coastguard Worker 	i = 0;
243*d83cc019SAndroid Build Coastguard Worker 	while (i < iterations || loop_forever) {
244*d83cc019SAndroid Build Coastguard Worker 		prepare_planes(data, pipe, max_planes, output);
245*d83cc019SAndroid Build Coastguard Worker 		igt_display_commit2(&data->display, COMMIT_ATOMIC);
246*d83cc019SAndroid Build Coastguard Worker 
247*d83cc019SAndroid Build Coastguard Worker 		i++;
248*d83cc019SAndroid Build Coastguard Worker 	}
249*d83cc019SAndroid Build Coastguard Worker }
250*d83cc019SAndroid Build Coastguard Worker 
251*d83cc019SAndroid Build Coastguard Worker static const drmModeModeInfo *
get_lowres_mode(data_t * data,const drmModeModeInfo * mode_default,igt_output_t * output)252*d83cc019SAndroid Build Coastguard Worker get_lowres_mode(data_t *data, const drmModeModeInfo *mode_default,
253*d83cc019SAndroid Build Coastguard Worker 		igt_output_t *output)
254*d83cc019SAndroid Build Coastguard Worker {
255*d83cc019SAndroid Build Coastguard Worker 	const drmModeModeInfo *mode = igt_std_1024_mode_get();
256*d83cc019SAndroid Build Coastguard Worker 	drmModeConnector *connector = output->config.connector;
257*d83cc019SAndroid Build Coastguard Worker 	int limit = mode_default->vdisplay - SIZE_PLANE;
258*d83cc019SAndroid Build Coastguard Worker 	bool found;
259*d83cc019SAndroid Build Coastguard Worker 
260*d83cc019SAndroid Build Coastguard Worker 	if (!connector)
261*d83cc019SAndroid Build Coastguard Worker 		return mode;
262*d83cc019SAndroid Build Coastguard Worker 
263*d83cc019SAndroid Build Coastguard Worker 	found = false;
264*d83cc019SAndroid Build Coastguard Worker 	for (int i = 0; i < connector->count_modes; i++) {
265*d83cc019SAndroid Build Coastguard Worker 		mode = &connector->modes[i];
266*d83cc019SAndroid Build Coastguard Worker 
267*d83cc019SAndroid Build Coastguard Worker 		if (mode->vdisplay < limit) {
268*d83cc019SAndroid Build Coastguard Worker 			found = true;
269*d83cc019SAndroid Build Coastguard Worker 			break;
270*d83cc019SAndroid Build Coastguard Worker 		}
271*d83cc019SAndroid Build Coastguard Worker 	}
272*d83cc019SAndroid Build Coastguard Worker 
273*d83cc019SAndroid Build Coastguard Worker 	if (!found)
274*d83cc019SAndroid Build Coastguard Worker 		mode = igt_std_1024_mode_get();
275*d83cc019SAndroid Build Coastguard Worker 
276*d83cc019SAndroid Build Coastguard Worker 	return mode;
277*d83cc019SAndroid Build Coastguard Worker }
278*d83cc019SAndroid Build Coastguard Worker 
279*d83cc019SAndroid Build Coastguard Worker static void
test_resolution_with_output(data_t * data,enum pipe pipe,igt_output_t * output)280*d83cc019SAndroid Build Coastguard Worker test_resolution_with_output(data_t *data, enum pipe pipe, igt_output_t *output)
281*d83cc019SAndroid Build Coastguard Worker {
282*d83cc019SAndroid Build Coastguard Worker 	const drmModeModeInfo *mode_hi, *mode_lo;
283*d83cc019SAndroid Build Coastguard Worker 	int iterations = opt.iterations < 1 ? 1 : opt.iterations;
284*d83cc019SAndroid Build Coastguard Worker 	bool loop_forever = opt.iterations == LOOP_FOREVER ? true : false;
285*d83cc019SAndroid Build Coastguard Worker 	int i;
286*d83cc019SAndroid Build Coastguard Worker 
287*d83cc019SAndroid Build Coastguard Worker 	i = 0;
288*d83cc019SAndroid Build Coastguard Worker 	while (i < iterations || loop_forever) {
289*d83cc019SAndroid Build Coastguard Worker 		mode_hi = igt_output_get_mode(output);
290*d83cc019SAndroid Build Coastguard Worker 		mode_lo = get_lowres_mode(data, mode_hi, output);
291*d83cc019SAndroid Build Coastguard Worker 
292*d83cc019SAndroid Build Coastguard Worker 		/* switch to lower resolution */
293*d83cc019SAndroid Build Coastguard Worker 		igt_output_override_mode(output, mode_lo);
294*d83cc019SAndroid Build Coastguard Worker 		igt_display_commit2(&data->display, COMMIT_ATOMIC);
295*d83cc019SAndroid Build Coastguard Worker 
296*d83cc019SAndroid Build Coastguard Worker 		/* switch back to higher resolution */
297*d83cc019SAndroid Build Coastguard Worker 		igt_output_override_mode(output, NULL);
298*d83cc019SAndroid Build Coastguard Worker 		igt_display_commit2(&data->display, COMMIT_ATOMIC);
299*d83cc019SAndroid Build Coastguard Worker 
300*d83cc019SAndroid Build Coastguard Worker 		i++;
301*d83cc019SAndroid Build Coastguard Worker 	}
302*d83cc019SAndroid Build Coastguard Worker }
303*d83cc019SAndroid Build Coastguard Worker 
304*d83cc019SAndroid Build Coastguard Worker static void
run_test(data_t * data,enum pipe pipe,igt_output_t * output)305*d83cc019SAndroid Build Coastguard Worker run_test(data_t *data, enum pipe pipe, igt_output_t *output)
306*d83cc019SAndroid Build Coastguard Worker {
307*d83cc019SAndroid Build Coastguard Worker 	int connected_outs;
308*d83cc019SAndroid Build Coastguard Worker 	int n_planes = data->display.pipes[pipe].n_planes;
309*d83cc019SAndroid Build Coastguard Worker 
310*d83cc019SAndroid Build Coastguard Worker 	if (!opt.user_seed)
311*d83cc019SAndroid Build Coastguard Worker 		opt.seed = time(NULL);
312*d83cc019SAndroid Build Coastguard Worker 
313*d83cc019SAndroid Build Coastguard Worker 	connected_outs = 0;
314*d83cc019SAndroid Build Coastguard Worker 	for_each_valid_output_on_pipe(&data->display, pipe, output) {
315*d83cc019SAndroid Build Coastguard Worker 		igt_info("Testing resolution with connector %s using pipe %s with seed %d\n",
316*d83cc019SAndroid Build Coastguard Worker 			 igt_output_name(output), kmstest_pipe_name(pipe), opt.seed);
317*d83cc019SAndroid Build Coastguard Worker 
318*d83cc019SAndroid Build Coastguard Worker 		test_init(data, pipe, n_planes, output);
319*d83cc019SAndroid Build Coastguard Worker 
320*d83cc019SAndroid Build Coastguard Worker 		igt_fork(child, 1) {
321*d83cc019SAndroid Build Coastguard Worker 			test_plane_position_with_output(data, pipe, output);
322*d83cc019SAndroid Build Coastguard Worker 		}
323*d83cc019SAndroid Build Coastguard Worker 
324*d83cc019SAndroid Build Coastguard Worker 		test_resolution_with_output(data, pipe, output);
325*d83cc019SAndroid Build Coastguard Worker 
326*d83cc019SAndroid Build Coastguard Worker 		igt_waitchildren();
327*d83cc019SAndroid Build Coastguard Worker 
328*d83cc019SAndroid Build Coastguard Worker 		test_fini(data, pipe, n_planes, output);
329*d83cc019SAndroid Build Coastguard Worker 
330*d83cc019SAndroid Build Coastguard Worker 		connected_outs++;
331*d83cc019SAndroid Build Coastguard Worker 	}
332*d83cc019SAndroid Build Coastguard Worker 
333*d83cc019SAndroid Build Coastguard Worker 	igt_skip_on(connected_outs == 0);
334*d83cc019SAndroid Build Coastguard Worker }
335*d83cc019SAndroid Build Coastguard Worker 
336*d83cc019SAndroid Build Coastguard Worker static void
run_tests_for_pipe(data_t * data,enum pipe pipe)337*d83cc019SAndroid Build Coastguard Worker run_tests_for_pipe(data_t *data, enum pipe pipe)
338*d83cc019SAndroid Build Coastguard Worker {
339*d83cc019SAndroid Build Coastguard Worker 	igt_output_t *output;
340*d83cc019SAndroid Build Coastguard Worker 
341*d83cc019SAndroid Build Coastguard Worker 	igt_fixture {
342*d83cc019SAndroid Build Coastguard Worker 		int valid_tests = 0;
343*d83cc019SAndroid Build Coastguard Worker 
344*d83cc019SAndroid Build Coastguard Worker 		igt_skip_on(pipe >= data->display.n_pipes);
345*d83cc019SAndroid Build Coastguard Worker 		igt_require(data->display.pipes[pipe].n_planes > 0);
346*d83cc019SAndroid Build Coastguard Worker 
347*d83cc019SAndroid Build Coastguard Worker 		for_each_valid_output_on_pipe(&data->display, pipe, output)
348*d83cc019SAndroid Build Coastguard Worker 			valid_tests++;
349*d83cc019SAndroid Build Coastguard Worker 
350*d83cc019SAndroid Build Coastguard Worker 		igt_require_f(valid_tests, "no valid crtc/connector combinations found\n");
351*d83cc019SAndroid Build Coastguard Worker 	}
352*d83cc019SAndroid Build Coastguard Worker 
353*d83cc019SAndroid Build Coastguard Worker 	igt_subtest_f("pipe-%s", kmstest_pipe_name(pipe))
354*d83cc019SAndroid Build Coastguard Worker 		for_each_valid_output_on_pipe(&data->display, pipe, output)
355*d83cc019SAndroid Build Coastguard Worker 			run_test(data, pipe, output);
356*d83cc019SAndroid Build Coastguard Worker }
357*d83cc019SAndroid Build Coastguard Worker 
opt_handler(int option,int option_index,void * input)358*d83cc019SAndroid Build Coastguard Worker static int opt_handler(int option, int option_index, void *input)
359*d83cc019SAndroid Build Coastguard Worker {
360*d83cc019SAndroid Build Coastguard Worker 	switch (option) {
361*d83cc019SAndroid Build Coastguard Worker 	case 'i':
362*d83cc019SAndroid Build Coastguard Worker 		opt.iterations = strtol(optarg, NULL, 0);
363*d83cc019SAndroid Build Coastguard Worker 
364*d83cc019SAndroid Build Coastguard Worker 		if (opt.iterations < LOOP_FOREVER || opt.iterations == 0) {
365*d83cc019SAndroid Build Coastguard Worker 			igt_info("incorrect number of iterations\n");
366*d83cc019SAndroid Build Coastguard Worker 			igt_assert(false);
367*d83cc019SAndroid Build Coastguard Worker 		}
368*d83cc019SAndroid Build Coastguard Worker 
369*d83cc019SAndroid Build Coastguard Worker 		break;
370*d83cc019SAndroid Build Coastguard Worker 	case 's':
371*d83cc019SAndroid Build Coastguard Worker 		opt.user_seed = true;
372*d83cc019SAndroid Build Coastguard Worker 		opt.seed = strtol(optarg, NULL, 0);
373*d83cc019SAndroid Build Coastguard Worker 		break;
374*d83cc019SAndroid Build Coastguard Worker 	default:
375*d83cc019SAndroid Build Coastguard Worker 		return IGT_OPT_HANDLER_ERROR;
376*d83cc019SAndroid Build Coastguard Worker 	}
377*d83cc019SAndroid Build Coastguard Worker 
378*d83cc019SAndroid Build Coastguard Worker 	return IGT_OPT_HANDLER_SUCCESS;
379*d83cc019SAndroid Build Coastguard Worker }
380*d83cc019SAndroid Build Coastguard Worker 
381*d83cc019SAndroid Build Coastguard Worker const char *help_str =
382*d83cc019SAndroid Build Coastguard Worker 	"  --iterations Number of iterations for test coverage. -1 loop forever, default 1 iteration\n"
383*d83cc019SAndroid Build Coastguard Worker 	"  --seed       Seed for random number generator\n";
384*d83cc019SAndroid Build Coastguard Worker struct option long_options[] = {
385*d83cc019SAndroid Build Coastguard Worker 	{ "iterations", required_argument, NULL, 'i'},
386*d83cc019SAndroid Build Coastguard Worker 	{ "seed",    required_argument, NULL, 's'},
387*d83cc019SAndroid Build Coastguard Worker 	{ 0, 0, 0, 0 }
388*d83cc019SAndroid Build Coastguard Worker };
389*d83cc019SAndroid Build Coastguard Worker 
390*d83cc019SAndroid Build Coastguard Worker static data_t data;
391*d83cc019SAndroid Build Coastguard Worker 
392*d83cc019SAndroid Build Coastguard Worker igt_main_args("", long_options, help_str, opt_handler, NULL)
393*d83cc019SAndroid Build Coastguard Worker {
394*d83cc019SAndroid Build Coastguard Worker 	enum pipe pipe;
395*d83cc019SAndroid Build Coastguard Worker 
396*d83cc019SAndroid Build Coastguard Worker 	igt_skip_on_simulation();
397*d83cc019SAndroid Build Coastguard Worker 
398*d83cc019SAndroid Build Coastguard Worker 	igt_fixture {
399*d83cc019SAndroid Build Coastguard Worker 		data.drm_fd = drm_open_driver_master(DRIVER_ANY);
400*d83cc019SAndroid Build Coastguard Worker 		kmstest_set_vt_graphics_mode();
401*d83cc019SAndroid Build Coastguard Worker 		igt_display_require(&data.display, data.drm_fd);
402*d83cc019SAndroid Build Coastguard Worker 		igt_require(data.display.is_atomic);
403*d83cc019SAndroid Build Coastguard Worker 	}
404*d83cc019SAndroid Build Coastguard Worker 
for_each_pipe_static(pipe)405*d83cc019SAndroid Build Coastguard Worker 	for_each_pipe_static(pipe) {
406*d83cc019SAndroid Build Coastguard Worker 		igt_subtest_group
407*d83cc019SAndroid Build Coastguard Worker 			run_tests_for_pipe(&data, pipe);
408*d83cc019SAndroid Build Coastguard Worker 	}
409*d83cc019SAndroid Build Coastguard Worker 
410*d83cc019SAndroid Build Coastguard Worker 	igt_fixture {
411*d83cc019SAndroid Build Coastguard Worker 		igt_display_fini(&data.display);
412*d83cc019SAndroid Build Coastguard Worker 		close(data.drm_fd);
413*d83cc019SAndroid Build Coastguard Worker 	}
414*d83cc019SAndroid Build Coastguard Worker }
415