1*d83cc019SAndroid Build Coastguard Worker /*
2*d83cc019SAndroid Build Coastguard Worker * Copyright © 2019 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 #include "igt.h"
25*d83cc019SAndroid Build Coastguard Worker #include "igt_vgem.h"
26*d83cc019SAndroid Build Coastguard Worker
27*d83cc019SAndroid Build Coastguard Worker #include <sys/ioctl.h>
28*d83cc019SAndroid Build Coastguard Worker #include <sys/poll.h>
29*d83cc019SAndroid Build Coastguard Worker #include <time.h>
30*d83cc019SAndroid Build Coastguard Worker
31*d83cc019SAndroid Build Coastguard Worker struct crc_info {
32*d83cc019SAndroid Build Coastguard Worker igt_crc_t crc;
33*d83cc019SAndroid Build Coastguard Worker char *str;
34*d83cc019SAndroid Build Coastguard Worker const char *name;
35*d83cc019SAndroid Build Coastguard Worker };
36*d83cc019SAndroid Build Coastguard Worker
37*d83cc019SAndroid Build Coastguard Worker static struct {
38*d83cc019SAndroid Build Coastguard Worker double r, g, b;
39*d83cc019SAndroid Build Coastguard Worker uint32_t color;
40*d83cc019SAndroid Build Coastguard Worker struct crc_info prime_crc, direct_crc;
41*d83cc019SAndroid Build Coastguard Worker } colors[3] = {
42*d83cc019SAndroid Build Coastguard Worker { .r = 0.0, .g = 0.0, .b = 0.0, .color = 0xff000000 },
43*d83cc019SAndroid Build Coastguard Worker { .r = 1.0, .g = 1.0, .b = 1.0, .color = 0xffffffff },
44*d83cc019SAndroid Build Coastguard Worker { .r = 1.0, .g = 0.0, .b = 0.0, .color = 0xffff0000 },
45*d83cc019SAndroid Build Coastguard Worker };
46*d83cc019SAndroid Build Coastguard Worker
47*d83cc019SAndroid Build Coastguard Worker IGT_TEST_DESCRIPTION("Prime tests, focusing on KMS side");
48*d83cc019SAndroid Build Coastguard Worker
has_prime_import(int fd)49*d83cc019SAndroid Build Coastguard Worker static bool has_prime_import(int fd)
50*d83cc019SAndroid Build Coastguard Worker {
51*d83cc019SAndroid Build Coastguard Worker uint64_t value;
52*d83cc019SAndroid Build Coastguard Worker
53*d83cc019SAndroid Build Coastguard Worker if (drmGetCap(fd, DRM_CAP_PRIME, &value))
54*d83cc019SAndroid Build Coastguard Worker return false;
55*d83cc019SAndroid Build Coastguard Worker
56*d83cc019SAndroid Build Coastguard Worker return value & DRM_PRIME_CAP_IMPORT;
57*d83cc019SAndroid Build Coastguard Worker }
58*d83cc019SAndroid Build Coastguard Worker
has_prime_export(int fd)59*d83cc019SAndroid Build Coastguard Worker static bool has_prime_export(int fd)
60*d83cc019SAndroid Build Coastguard Worker {
61*d83cc019SAndroid Build Coastguard Worker uint64_t value;
62*d83cc019SAndroid Build Coastguard Worker
63*d83cc019SAndroid Build Coastguard Worker if (drmGetCap(fd, DRM_CAP_PRIME, &value))
64*d83cc019SAndroid Build Coastguard Worker return false;
65*d83cc019SAndroid Build Coastguard Worker
66*d83cc019SAndroid Build Coastguard Worker return value & DRM_PRIME_CAP_EXPORT;
67*d83cc019SAndroid Build Coastguard Worker }
68*d83cc019SAndroid Build Coastguard Worker
setup_display(int importer_fd,igt_display_t * display,enum pipe pipe)69*d83cc019SAndroid Build Coastguard Worker static igt_output_t *setup_display(int importer_fd, igt_display_t *display,
70*d83cc019SAndroid Build Coastguard Worker enum pipe pipe)
71*d83cc019SAndroid Build Coastguard Worker {
72*d83cc019SAndroid Build Coastguard Worker igt_display_require(display, importer_fd);
73*d83cc019SAndroid Build Coastguard Worker igt_skip_on(pipe >= display->n_pipes);
74*d83cc019SAndroid Build Coastguard Worker igt_output_t *output = igt_get_single_output_for_pipe(display, pipe);
75*d83cc019SAndroid Build Coastguard Worker
76*d83cc019SAndroid Build Coastguard Worker igt_require_f(output, "No connector found for pipe %s\n",
77*d83cc019SAndroid Build Coastguard Worker kmstest_pipe_name(pipe));
78*d83cc019SAndroid Build Coastguard Worker
79*d83cc019SAndroid Build Coastguard Worker igt_display_reset(display);
80*d83cc019SAndroid Build Coastguard Worker igt_output_set_pipe(output, pipe);
81*d83cc019SAndroid Build Coastguard Worker return output;
82*d83cc019SAndroid Build Coastguard Worker }
83*d83cc019SAndroid Build Coastguard Worker
prepare_scratch(int exporter_fd,struct vgem_bo * scratch,drmModeModeInfo * mode,uint32_t color)84*d83cc019SAndroid Build Coastguard Worker static void prepare_scratch(int exporter_fd, struct vgem_bo *scratch,
85*d83cc019SAndroid Build Coastguard Worker drmModeModeInfo *mode, uint32_t color)
86*d83cc019SAndroid Build Coastguard Worker {
87*d83cc019SAndroid Build Coastguard Worker uint32_t *ptr;
88*d83cc019SAndroid Build Coastguard Worker
89*d83cc019SAndroid Build Coastguard Worker scratch->width = mode->hdisplay;
90*d83cc019SAndroid Build Coastguard Worker scratch->height = mode->vdisplay;
91*d83cc019SAndroid Build Coastguard Worker scratch->bpp = 32;
92*d83cc019SAndroid Build Coastguard Worker vgem_create(exporter_fd, scratch);
93*d83cc019SAndroid Build Coastguard Worker
94*d83cc019SAndroid Build Coastguard Worker ptr = vgem_mmap(exporter_fd, scratch, PROT_WRITE);
95*d83cc019SAndroid Build Coastguard Worker for (size_t idx = 0; idx < scratch->size / sizeof(*ptr); ++idx)
96*d83cc019SAndroid Build Coastguard Worker ptr[idx] = color;
97*d83cc019SAndroid Build Coastguard Worker
98*d83cc019SAndroid Build Coastguard Worker munmap(ptr, scratch->size);
99*d83cc019SAndroid Build Coastguard Worker }
100*d83cc019SAndroid Build Coastguard Worker
prepare_fb(int importer_fd,struct vgem_bo * scratch,struct igt_fb * fb)101*d83cc019SAndroid Build Coastguard Worker static void prepare_fb(int importer_fd, struct vgem_bo *scratch, struct igt_fb *fb)
102*d83cc019SAndroid Build Coastguard Worker {
103*d83cc019SAndroid Build Coastguard Worker enum igt_color_encoding color_encoding = IGT_COLOR_YCBCR_BT709;
104*d83cc019SAndroid Build Coastguard Worker enum igt_color_range color_range = IGT_COLOR_YCBCR_LIMITED_RANGE;
105*d83cc019SAndroid Build Coastguard Worker
106*d83cc019SAndroid Build Coastguard Worker igt_init_fb(fb, importer_fd, scratch->width, scratch->height,
107*d83cc019SAndroid Build Coastguard Worker DRM_FORMAT_XRGB8888, LOCAL_DRM_FORMAT_MOD_NONE,
108*d83cc019SAndroid Build Coastguard Worker color_encoding, color_range);
109*d83cc019SAndroid Build Coastguard Worker }
110*d83cc019SAndroid Build Coastguard Worker
import_fb(int importer_fd,struct igt_fb * fb,int dmabuf_fd,uint32_t pitch)111*d83cc019SAndroid Build Coastguard Worker static void import_fb(int importer_fd, struct igt_fb *fb,
112*d83cc019SAndroid Build Coastguard Worker int dmabuf_fd, uint32_t pitch)
113*d83cc019SAndroid Build Coastguard Worker {
114*d83cc019SAndroid Build Coastguard Worker uint32_t offsets[4] = {}, pitches[4] = {}, handles[4] = {};
115*d83cc019SAndroid Build Coastguard Worker int ret;
116*d83cc019SAndroid Build Coastguard Worker
117*d83cc019SAndroid Build Coastguard Worker fb->gem_handle = prime_fd_to_handle(importer_fd, dmabuf_fd);
118*d83cc019SAndroid Build Coastguard Worker
119*d83cc019SAndroid Build Coastguard Worker handles[0] = fb->gem_handle;
120*d83cc019SAndroid Build Coastguard Worker pitches[0] = pitch;
121*d83cc019SAndroid Build Coastguard Worker offsets[0] = 0;
122*d83cc019SAndroid Build Coastguard Worker
123*d83cc019SAndroid Build Coastguard Worker ret = drmModeAddFB2(importer_fd, fb->width, fb->height,
124*d83cc019SAndroid Build Coastguard Worker DRM_FORMAT_XRGB8888,
125*d83cc019SAndroid Build Coastguard Worker handles, pitches, offsets,
126*d83cc019SAndroid Build Coastguard Worker &fb->fb_id, 0);
127*d83cc019SAndroid Build Coastguard Worker igt_assert(ret == 0);
128*d83cc019SAndroid Build Coastguard Worker }
129*d83cc019SAndroid Build Coastguard Worker
set_fb(struct igt_fb * fb,igt_display_t * display,igt_output_t * output)130*d83cc019SAndroid Build Coastguard Worker static void set_fb(struct igt_fb *fb,
131*d83cc019SAndroid Build Coastguard Worker igt_display_t *display,
132*d83cc019SAndroid Build Coastguard Worker igt_output_t *output)
133*d83cc019SAndroid Build Coastguard Worker {
134*d83cc019SAndroid Build Coastguard Worker igt_plane_t *primary;
135*d83cc019SAndroid Build Coastguard Worker int ret;
136*d83cc019SAndroid Build Coastguard Worker
137*d83cc019SAndroid Build Coastguard Worker primary = igt_output_get_plane_type(output, DRM_PLANE_TYPE_PRIMARY);
138*d83cc019SAndroid Build Coastguard Worker igt_assert(primary);
139*d83cc019SAndroid Build Coastguard Worker
140*d83cc019SAndroid Build Coastguard Worker igt_plane_set_fb(primary, fb);
141*d83cc019SAndroid Build Coastguard Worker ret = igt_display_commit(display);
142*d83cc019SAndroid Build Coastguard Worker
143*d83cc019SAndroid Build Coastguard Worker igt_assert(ret == 0);
144*d83cc019SAndroid Build Coastguard Worker }
145*d83cc019SAndroid Build Coastguard Worker
collect_crc_for_fb(int importer_fd,struct igt_fb * fb,igt_display_t * display,igt_output_t * output,igt_pipe_crc_t * pipe_crc,uint32_t color,struct crc_info * info)146*d83cc019SAndroid Build Coastguard Worker static void collect_crc_for_fb(int importer_fd, struct igt_fb *fb, igt_display_t *display,
147*d83cc019SAndroid Build Coastguard Worker igt_output_t *output, igt_pipe_crc_t *pipe_crc,
148*d83cc019SAndroid Build Coastguard Worker uint32_t color, struct crc_info *info)
149*d83cc019SAndroid Build Coastguard Worker {
150*d83cc019SAndroid Build Coastguard Worker set_fb(fb, display, output);
151*d83cc019SAndroid Build Coastguard Worker igt_pipe_crc_collect_crc(pipe_crc, &info->crc);
152*d83cc019SAndroid Build Coastguard Worker info->str = igt_crc_to_string(&info->crc);
153*d83cc019SAndroid Build Coastguard Worker igt_debug("CRC through '%s' method for %#08x is %s\n",
154*d83cc019SAndroid Build Coastguard Worker info->name, color, info->str);
155*d83cc019SAndroid Build Coastguard Worker igt_remove_fb(importer_fd, fb);
156*d83cc019SAndroid Build Coastguard Worker }
157*d83cc019SAndroid Build Coastguard Worker
test_crc(int exporter_fd,int importer_fd)158*d83cc019SAndroid Build Coastguard Worker static void test_crc(int exporter_fd, int importer_fd)
159*d83cc019SAndroid Build Coastguard Worker {
160*d83cc019SAndroid Build Coastguard Worker igt_display_t display;
161*d83cc019SAndroid Build Coastguard Worker igt_output_t *output;
162*d83cc019SAndroid Build Coastguard Worker igt_pipe_crc_t *pipe_crc;
163*d83cc019SAndroid Build Coastguard Worker enum pipe pipe = PIPE_A;
164*d83cc019SAndroid Build Coastguard Worker struct igt_fb fb;
165*d83cc019SAndroid Build Coastguard Worker int dmabuf_fd;
166*d83cc019SAndroid Build Coastguard Worker struct vgem_bo scratch = {}; /* despite the name, it suits for any
167*d83cc019SAndroid Build Coastguard Worker * gem-compatible device
168*d83cc019SAndroid Build Coastguard Worker * TODO: rename
169*d83cc019SAndroid Build Coastguard Worker */
170*d83cc019SAndroid Build Coastguard Worker int i, j;
171*d83cc019SAndroid Build Coastguard Worker drmModeModeInfo *mode;
172*d83cc019SAndroid Build Coastguard Worker
173*d83cc019SAndroid Build Coastguard Worker bool crc_equal = false;
174*d83cc019SAndroid Build Coastguard Worker
175*d83cc019SAndroid Build Coastguard Worker output = setup_display(importer_fd, &display, pipe);
176*d83cc019SAndroid Build Coastguard Worker
177*d83cc019SAndroid Build Coastguard Worker mode = igt_output_get_mode(output);
178*d83cc019SAndroid Build Coastguard Worker pipe_crc = igt_pipe_crc_new(importer_fd, pipe, INTEL_PIPE_CRC_SOURCE_AUTO);
179*d83cc019SAndroid Build Coastguard Worker
180*d83cc019SAndroid Build Coastguard Worker for (i = 0; i < ARRAY_SIZE(colors); i++) {
181*d83cc019SAndroid Build Coastguard Worker prepare_scratch(exporter_fd, &scratch, mode, colors[i].color);
182*d83cc019SAndroid Build Coastguard Worker dmabuf_fd = prime_handle_to_fd(exporter_fd, scratch.handle);
183*d83cc019SAndroid Build Coastguard Worker gem_close(exporter_fd, scratch.handle);
184*d83cc019SAndroid Build Coastguard Worker
185*d83cc019SAndroid Build Coastguard Worker prepare_fb(importer_fd, &scratch, &fb);
186*d83cc019SAndroid Build Coastguard Worker import_fb(importer_fd, &fb, dmabuf_fd, scratch.pitch);
187*d83cc019SAndroid Build Coastguard Worker close(dmabuf_fd);
188*d83cc019SAndroid Build Coastguard Worker
189*d83cc019SAndroid Build Coastguard Worker colors[i].prime_crc.name = "prime";
190*d83cc019SAndroid Build Coastguard Worker collect_crc_for_fb(importer_fd, &fb, &display, output,
191*d83cc019SAndroid Build Coastguard Worker pipe_crc, colors[i].color, &colors[i].prime_crc);
192*d83cc019SAndroid Build Coastguard Worker
193*d83cc019SAndroid Build Coastguard Worker igt_create_color_fb(importer_fd,
194*d83cc019SAndroid Build Coastguard Worker mode->hdisplay, mode->vdisplay,
195*d83cc019SAndroid Build Coastguard Worker DRM_FORMAT_XRGB8888, LOCAL_DRM_FORMAT_MOD_NONE,
196*d83cc019SAndroid Build Coastguard Worker colors[i].r, colors[i].g, colors[i].b,
197*d83cc019SAndroid Build Coastguard Worker &fb);
198*d83cc019SAndroid Build Coastguard Worker
199*d83cc019SAndroid Build Coastguard Worker colors[i].direct_crc.name = "direct";
200*d83cc019SAndroid Build Coastguard Worker collect_crc_for_fb(importer_fd, &fb, &display, output,
201*d83cc019SAndroid Build Coastguard Worker pipe_crc, colors[i].color, &colors[i].direct_crc);
202*d83cc019SAndroid Build Coastguard Worker }
203*d83cc019SAndroid Build Coastguard Worker igt_pipe_crc_free(pipe_crc);
204*d83cc019SAndroid Build Coastguard Worker
205*d83cc019SAndroid Build Coastguard Worker igt_debug("CRC table:\n");
206*d83cc019SAndroid Build Coastguard Worker igt_debug("Color\t\tPrime\t\tDirect\n");
207*d83cc019SAndroid Build Coastguard Worker for (i = 0; i < ARRAY_SIZE(colors); i++) {
208*d83cc019SAndroid Build Coastguard Worker igt_debug("%#08x\t%.8s\t%.8s\n", colors[i].color,
209*d83cc019SAndroid Build Coastguard Worker colors[i].prime_crc.str, colors[i].direct_crc.str);
210*d83cc019SAndroid Build Coastguard Worker free(colors[i].prime_crc.str);
211*d83cc019SAndroid Build Coastguard Worker free(colors[i].direct_crc.str);
212*d83cc019SAndroid Build Coastguard Worker }
213*d83cc019SAndroid Build Coastguard Worker
214*d83cc019SAndroid Build Coastguard Worker for (i = 0; i < ARRAY_SIZE(colors); i++) {
215*d83cc019SAndroid Build Coastguard Worker for (j = 0; j < ARRAY_SIZE(colors); j++) {
216*d83cc019SAndroid Build Coastguard Worker if (i == j) {
217*d83cc019SAndroid Build Coastguard Worker igt_assert_crc_equal(&colors[i].prime_crc.crc,
218*d83cc019SAndroid Build Coastguard Worker &colors[j].direct_crc.crc);
219*d83cc019SAndroid Build Coastguard Worker continue;
220*d83cc019SAndroid Build Coastguard Worker }
221*d83cc019SAndroid Build Coastguard Worker crc_equal = igt_check_crc_equal(&colors[i].prime_crc.crc,
222*d83cc019SAndroid Build Coastguard Worker &colors[j].direct_crc.crc);
223*d83cc019SAndroid Build Coastguard Worker igt_assert_f(!crc_equal, "CRC should be different");
224*d83cc019SAndroid Build Coastguard Worker }
225*d83cc019SAndroid Build Coastguard Worker }
226*d83cc019SAndroid Build Coastguard Worker igt_display_fini(&display);
227*d83cc019SAndroid Build Coastguard Worker }
228*d83cc019SAndroid Build Coastguard Worker
run_test_crc(int export_chipset,int import_chipset)229*d83cc019SAndroid Build Coastguard Worker static void run_test_crc(int export_chipset, int import_chipset)
230*d83cc019SAndroid Build Coastguard Worker {
231*d83cc019SAndroid Build Coastguard Worker int importer_fd = -1;
232*d83cc019SAndroid Build Coastguard Worker int exporter_fd = -1;
233*d83cc019SAndroid Build Coastguard Worker
234*d83cc019SAndroid Build Coastguard Worker exporter_fd = drm_open_driver(export_chipset);
235*d83cc019SAndroid Build Coastguard Worker importer_fd = drm_open_driver_master(import_chipset);
236*d83cc019SAndroid Build Coastguard Worker
237*d83cc019SAndroid Build Coastguard Worker igt_require(has_prime_export(exporter_fd));
238*d83cc019SAndroid Build Coastguard Worker igt_require(has_prime_import(importer_fd));
239*d83cc019SAndroid Build Coastguard Worker igt_require_pipe_crc(importer_fd);
240*d83cc019SAndroid Build Coastguard Worker
241*d83cc019SAndroid Build Coastguard Worker test_crc(exporter_fd, importer_fd);
242*d83cc019SAndroid Build Coastguard Worker close(importer_fd);
243*d83cc019SAndroid Build Coastguard Worker close(exporter_fd);
244*d83cc019SAndroid Build Coastguard Worker }
245*d83cc019SAndroid Build Coastguard Worker
246*d83cc019SAndroid Build Coastguard Worker igt_main
247*d83cc019SAndroid Build Coastguard Worker {
248*d83cc019SAndroid Build Coastguard Worker igt_fixture {
249*d83cc019SAndroid Build Coastguard Worker kmstest_set_vt_graphics_mode();
250*d83cc019SAndroid Build Coastguard Worker }
251*d83cc019SAndroid Build Coastguard Worker igt_describe("Make a dumb buffer inside vgem, fill it, export to another device and compare the CRC");
252*d83cc019SAndroid Build Coastguard Worker igt_subtest("basic-crc")
253*d83cc019SAndroid Build Coastguard Worker run_test_crc(DRIVER_VGEM, DRIVER_ANY);
254*d83cc019SAndroid Build Coastguard Worker }
255