1 // SPDX-License-Identifier: GPL-2.0
3 * Test cases for the drm_plane_helper functions
6 #define pr_fmt(fmt) "drm_plane_helper: " fmt
8 #include <drm/drm_atomic_helper.h>
9 #include <drm/drm_plane_helper.h>
10 #include <drm/drm_modes.h>
12 #include "test-drm_modeset_common.h"
14 static void set_src(struct drm_plane_state
*plane_state
,
15 unsigned src_x
, unsigned src_y
,
16 unsigned src_w
, unsigned src_h
)
18 plane_state
->src_x
= src_x
;
19 plane_state
->src_y
= src_y
;
20 plane_state
->src_w
= src_w
;
21 plane_state
->src_h
= src_h
;
24 static bool check_src_eq(struct drm_plane_state
*plane_state
,
25 unsigned src_x
, unsigned src_y
,
26 unsigned src_w
, unsigned src_h
)
28 if (plane_state
->src
.x1
< 0) {
29 pr_err("src x coordinate %x should never be below 0.\n", plane_state
->src
.x1
);
30 drm_rect_debug_print("src: ", &plane_state
->src
, true);
33 if (plane_state
->src
.y1
< 0) {
34 pr_err("src y coordinate %x should never be below 0.\n", plane_state
->src
.y1
);
35 drm_rect_debug_print("src: ", &plane_state
->src
, true);
39 if (plane_state
->src
.x1
!= src_x
||
40 plane_state
->src
.y1
!= src_y
||
41 drm_rect_width(&plane_state
->src
) != src_w
||
42 drm_rect_height(&plane_state
->src
) != src_h
) {
43 drm_rect_debug_print("src: ", &plane_state
->src
, true);
50 static void set_crtc(struct drm_plane_state
*plane_state
,
51 int crtc_x
, int crtc_y
,
52 unsigned crtc_w
, unsigned crtc_h
)
54 plane_state
->crtc_x
= crtc_x
;
55 plane_state
->crtc_y
= crtc_y
;
56 plane_state
->crtc_w
= crtc_w
;
57 plane_state
->crtc_h
= crtc_h
;
60 static bool check_crtc_eq(struct drm_plane_state
*plane_state
,
61 int crtc_x
, int crtc_y
,
62 unsigned crtc_w
, unsigned crtc_h
)
64 if (plane_state
->dst
.x1
!= crtc_x
||
65 plane_state
->dst
.y1
!= crtc_y
||
66 drm_rect_width(&plane_state
->dst
) != crtc_w
||
67 drm_rect_height(&plane_state
->dst
) != crtc_h
) {
68 drm_rect_debug_print("dst: ", &plane_state
->dst
, false);
76 int igt_check_plane_state(void *ignored
)
80 const struct drm_crtc_state crtc_state
= {
81 .crtc
= ZERO_SIZE_PTR
,
85 DRM_MODE("1024x768", 0, 65000, 1024, 1048,
86 1184, 1344, 0, 768, 771, 777, 806, 0,
87 DRM_MODE_FLAG_NHSYNC
| DRM_MODE_FLAG_NVSYNC
)
90 struct drm_framebuffer fb
= {
94 struct drm_plane_state plane_state
= {
95 .crtc
= ZERO_SIZE_PTR
,
97 .rotation
= DRM_MODE_ROTATE_0
100 /* Simple clipping, no scaling. */
101 set_src(&plane_state
, 0, 0, fb
.width
<< 16, fb
.height
<< 16);
102 set_crtc(&plane_state
, 0, 0, fb
.width
, fb
.height
);
103 ret
= drm_atomic_helper_check_plane_state(&plane_state
, &crtc_state
,
104 DRM_PLANE_HELPER_NO_SCALING
,
105 DRM_PLANE_HELPER_NO_SCALING
,
107 FAIL(ret
< 0, "Simple clipping check should pass\n");
108 FAIL_ON(!plane_state
.visible
);
109 FAIL_ON(!check_src_eq(&plane_state
, 0, 0, 1024 << 16, 768 << 16));
110 FAIL_ON(!check_crtc_eq(&plane_state
, 0, 0, 1024, 768));
112 /* Rotated clipping + reflection, no scaling. */
113 plane_state
.rotation
= DRM_MODE_ROTATE_90
| DRM_MODE_REFLECT_X
;
114 ret
= drm_atomic_helper_check_plane_state(&plane_state
, &crtc_state
,
115 DRM_PLANE_HELPER_NO_SCALING
,
116 DRM_PLANE_HELPER_NO_SCALING
,
118 FAIL(ret
< 0, "Rotated clipping check should pass\n");
119 FAIL_ON(!plane_state
.visible
);
120 FAIL_ON(!check_src_eq(&plane_state
, 0, 0, 768 << 16, 1024 << 16));
121 FAIL_ON(!check_crtc_eq(&plane_state
, 0, 0, 1024, 768));
122 plane_state
.rotation
= DRM_MODE_ROTATE_0
;
124 /* Check whether positioning works correctly. */
125 set_src(&plane_state
, 0, 0, 1023 << 16, 767 << 16);
126 set_crtc(&plane_state
, 0, 0, 1023, 767);
127 ret
= drm_atomic_helper_check_plane_state(&plane_state
, &crtc_state
,
128 DRM_PLANE_HELPER_NO_SCALING
,
129 DRM_PLANE_HELPER_NO_SCALING
,
131 FAIL(!ret
, "Should not be able to position on the crtc with can_position=false\n");
133 ret
= drm_atomic_helper_check_plane_state(&plane_state
, &crtc_state
,
134 DRM_PLANE_HELPER_NO_SCALING
,
135 DRM_PLANE_HELPER_NO_SCALING
,
137 FAIL(ret
< 0, "Simple positioning should work\n");
138 FAIL_ON(!plane_state
.visible
);
139 FAIL_ON(!check_src_eq(&plane_state
, 0, 0, 1023 << 16, 767 << 16));
140 FAIL_ON(!check_crtc_eq(&plane_state
, 0, 0, 1023, 767));
142 /* Simple scaling tests. */
143 set_src(&plane_state
, 0, 0, 512 << 16, 384 << 16);
144 set_crtc(&plane_state
, 0, 0, 1024, 768);
145 ret
= drm_atomic_helper_check_plane_state(&plane_state
, &crtc_state
,
147 DRM_PLANE_HELPER_NO_SCALING
,
149 FAIL(!ret
, "Upscaling out of range should fail.\n");
150 ret
= drm_atomic_helper_check_plane_state(&plane_state
, &crtc_state
,
152 DRM_PLANE_HELPER_NO_SCALING
,
154 FAIL(ret
< 0, "Upscaling exactly 2x should work\n");
155 FAIL_ON(!plane_state
.visible
);
156 FAIL_ON(!check_src_eq(&plane_state
, 0, 0, 512 << 16, 384 << 16));
157 FAIL_ON(!check_crtc_eq(&plane_state
, 0, 0, 1024, 768));
159 set_src(&plane_state
, 0, 0, 2048 << 16, 1536 << 16);
160 ret
= drm_atomic_helper_check_plane_state(&plane_state
, &crtc_state
,
161 DRM_PLANE_HELPER_NO_SCALING
,
162 0x1ffff, false, false);
163 FAIL(!ret
, "Downscaling out of range should fail.\n");
164 ret
= drm_atomic_helper_check_plane_state(&plane_state
, &crtc_state
,
165 DRM_PLANE_HELPER_NO_SCALING
,
166 0x20000, false, false);
167 FAIL(ret
< 0, "Should succeed with exact scaling limit\n");
168 FAIL_ON(!plane_state
.visible
);
169 FAIL_ON(!check_src_eq(&plane_state
, 0, 0, 2048 << 16, 1536 << 16));
170 FAIL_ON(!check_crtc_eq(&plane_state
, 0, 0, 1024, 768));
172 /* Testing rounding errors. */
173 set_src(&plane_state
, 0, 0, 0x40001, 0x40001);
174 set_crtc(&plane_state
, 1022, 766, 4, 4);
175 ret
= drm_atomic_helper_check_plane_state(&plane_state
, &crtc_state
,
176 DRM_PLANE_HELPER_NO_SCALING
,
179 FAIL(ret
< 0, "Should succeed by clipping to exact multiple");
180 FAIL_ON(!plane_state
.visible
);
181 FAIL_ON(!check_src_eq(&plane_state
, 0, 0, 2 << 16, 2 << 16));
182 FAIL_ON(!check_crtc_eq(&plane_state
, 1022, 766, 2, 2));
184 set_src(&plane_state
, 0x20001, 0x20001, 0x4040001, 0x3040001);
185 set_crtc(&plane_state
, -2, -2, 1028, 772);
186 ret
= drm_atomic_helper_check_plane_state(&plane_state
, &crtc_state
,
187 DRM_PLANE_HELPER_NO_SCALING
,
190 FAIL(ret
< 0, "Should succeed by clipping to exact multiple");
191 FAIL_ON(!plane_state
.visible
);
192 FAIL_ON(!check_src_eq(&plane_state
, 0x40002, 0x40002, 1024 << 16, 768 << 16));
193 FAIL_ON(!check_crtc_eq(&plane_state
, 0, 0, 1024, 768));
195 set_src(&plane_state
, 0, 0, 0x3ffff, 0x3ffff);
196 set_crtc(&plane_state
, 1022, 766, 4, 4);
197 ret
= drm_atomic_helper_check_plane_state(&plane_state
, &crtc_state
,
199 DRM_PLANE_HELPER_NO_SCALING
,
201 FAIL(ret
< 0, "Should succeed by clipping to exact multiple");
202 FAIL_ON(!plane_state
.visible
);
203 /* Should not be rounded to 0x20001, which would be upscaling. */
204 FAIL_ON(!check_src_eq(&plane_state
, 0, 0, 2 << 16, 2 << 16));
205 FAIL_ON(!check_crtc_eq(&plane_state
, 1022, 766, 2, 2));
207 set_src(&plane_state
, 0x1ffff, 0x1ffff, 0x403ffff, 0x303ffff);
208 set_crtc(&plane_state
, -2, -2, 1028, 772);
209 ret
= drm_atomic_helper_check_plane_state(&plane_state
, &crtc_state
,
211 DRM_PLANE_HELPER_NO_SCALING
,
213 FAIL(ret
< 0, "Should succeed by clipping to exact multiple");
214 FAIL_ON(!plane_state
.visible
);
215 FAIL_ON(!check_src_eq(&plane_state
, 0x3fffe, 0x3fffe, 1024 << 16, 768 << 16));
216 FAIL_ON(!check_crtc_eq(&plane_state
, 0, 0, 1024, 768));