xref: /aosp_15_r20/external/libdrm/intel/test_decode.c (revision 7688df22e49036ff52a766b7101da3a49edadb8c)
1*7688df22SAndroid Build Coastguard Worker /*
2*7688df22SAndroid Build Coastguard Worker  * Copyright © 2011 Intel Corporation
3*7688df22SAndroid Build Coastguard Worker  *
4*7688df22SAndroid Build Coastguard Worker  * Permission is hereby granted, free of charge, to any person obtaining a
5*7688df22SAndroid Build Coastguard Worker  * copy of this software and associated documentation files (the "Software"),
6*7688df22SAndroid Build Coastguard Worker  * to deal in the Software without restriction, including without limitation
7*7688df22SAndroid Build Coastguard Worker  * the rights to use, copy, modify, merge, publish, distribute, sublicense,
8*7688df22SAndroid Build Coastguard Worker  * and/or sell copies of the Software, and to permit persons to whom the
9*7688df22SAndroid Build Coastguard Worker  * Software is furnished to do so, subject to the following conditions:
10*7688df22SAndroid Build Coastguard Worker  *
11*7688df22SAndroid Build Coastguard Worker  * The above copyright notice and this permission notice (including the next
12*7688df22SAndroid Build Coastguard Worker  * paragraph) shall be included in all copies or substantial portions of the
13*7688df22SAndroid Build Coastguard Worker  * Software.
14*7688df22SAndroid Build Coastguard Worker  *
15*7688df22SAndroid Build Coastguard Worker  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16*7688df22SAndroid Build Coastguard Worker  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17*7688df22SAndroid Build Coastguard Worker  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
18*7688df22SAndroid Build Coastguard Worker  * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19*7688df22SAndroid Build Coastguard Worker  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
20*7688df22SAndroid Build Coastguard Worker  * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
21*7688df22SAndroid Build Coastguard Worker  * IN THE SOFTWARE.
22*7688df22SAndroid Build Coastguard Worker  */
23*7688df22SAndroid Build Coastguard Worker 
24*7688df22SAndroid Build Coastguard Worker #include <string.h>
25*7688df22SAndroid Build Coastguard Worker #include <stdlib.h>
26*7688df22SAndroid Build Coastguard Worker #include <stdio.h>
27*7688df22SAndroid Build Coastguard Worker #include <unistd.h>
28*7688df22SAndroid Build Coastguard Worker #include <fcntl.h>
29*7688df22SAndroid Build Coastguard Worker #include <sys/types.h>
30*7688df22SAndroid Build Coastguard Worker #include <sys/stat.h>
31*7688df22SAndroid Build Coastguard Worker #include <err.h>
32*7688df22SAndroid Build Coastguard Worker 
33*7688df22SAndroid Build Coastguard Worker #include "libdrm_macros.h"
34*7688df22SAndroid Build Coastguard Worker #include "intel_bufmgr.h"
35*7688df22SAndroid Build Coastguard Worker #include "intel_chipset.h"
36*7688df22SAndroid Build Coastguard Worker 
37*7688df22SAndroid Build Coastguard Worker #define HW_OFFSET 0x12300000
38*7688df22SAndroid Build Coastguard Worker 
39*7688df22SAndroid Build Coastguard Worker static void
usage(void)40*7688df22SAndroid Build Coastguard Worker usage(void)
41*7688df22SAndroid Build Coastguard Worker {
42*7688df22SAndroid Build Coastguard Worker 	fprintf(stderr, "usage:\n");
43*7688df22SAndroid Build Coastguard Worker 	fprintf(stderr, "  test_decode <batch>\n");
44*7688df22SAndroid Build Coastguard Worker 	fprintf(stderr, "  test_decode <batch> -dump\n");
45*7688df22SAndroid Build Coastguard Worker 	exit(1);
46*7688df22SAndroid Build Coastguard Worker }
47*7688df22SAndroid Build Coastguard Worker 
48*7688df22SAndroid Build Coastguard Worker static void
read_file(const char * filename,void ** ptr,size_t * size)49*7688df22SAndroid Build Coastguard Worker read_file(const char *filename, void **ptr, size_t *size)
50*7688df22SAndroid Build Coastguard Worker {
51*7688df22SAndroid Build Coastguard Worker 	int fd, ret;
52*7688df22SAndroid Build Coastguard Worker 	struct stat st;
53*7688df22SAndroid Build Coastguard Worker 
54*7688df22SAndroid Build Coastguard Worker 	fd = open(filename, O_RDONLY);
55*7688df22SAndroid Build Coastguard Worker 	if (fd < 0)
56*7688df22SAndroid Build Coastguard Worker 		errx(1, "couldn't open `%s'", filename);
57*7688df22SAndroid Build Coastguard Worker 
58*7688df22SAndroid Build Coastguard Worker 	ret = fstat(fd, &st);
59*7688df22SAndroid Build Coastguard Worker 	if (ret)
60*7688df22SAndroid Build Coastguard Worker 		errx(1, "couldn't stat `%s'", filename);
61*7688df22SAndroid Build Coastguard Worker 
62*7688df22SAndroid Build Coastguard Worker 	*size = st.st_size;
63*7688df22SAndroid Build Coastguard Worker 	*ptr = drm_mmap(NULL, st.st_size, PROT_READ, MAP_SHARED, fd, 0);
64*7688df22SAndroid Build Coastguard Worker 	if (*ptr == MAP_FAILED)
65*7688df22SAndroid Build Coastguard Worker 		errx(1, "couldn't map `%s'", filename);
66*7688df22SAndroid Build Coastguard Worker 
67*7688df22SAndroid Build Coastguard Worker 	close(fd);
68*7688df22SAndroid Build Coastguard Worker }
69*7688df22SAndroid Build Coastguard Worker 
70*7688df22SAndroid Build Coastguard Worker static void
dump_batch(struct drm_intel_decode * ctx,const char * batch_filename)71*7688df22SAndroid Build Coastguard Worker dump_batch(struct drm_intel_decode *ctx, const char *batch_filename)
72*7688df22SAndroid Build Coastguard Worker {
73*7688df22SAndroid Build Coastguard Worker 	void *batch_ptr;
74*7688df22SAndroid Build Coastguard Worker 	size_t batch_size;
75*7688df22SAndroid Build Coastguard Worker 
76*7688df22SAndroid Build Coastguard Worker 	read_file(batch_filename, &batch_ptr, &batch_size);
77*7688df22SAndroid Build Coastguard Worker 
78*7688df22SAndroid Build Coastguard Worker 	drm_intel_decode_set_batch_pointer(ctx, batch_ptr, HW_OFFSET,
79*7688df22SAndroid Build Coastguard Worker 					   batch_size / 4);
80*7688df22SAndroid Build Coastguard Worker 	drm_intel_decode_set_output_file(ctx, stdout);
81*7688df22SAndroid Build Coastguard Worker 
82*7688df22SAndroid Build Coastguard Worker 	drm_intel_decode(ctx);
83*7688df22SAndroid Build Coastguard Worker }
84*7688df22SAndroid Build Coastguard Worker 
85*7688df22SAndroid Build Coastguard Worker static void
compare_batch(struct drm_intel_decode * ctx,const char * batch_filename)86*7688df22SAndroid Build Coastguard Worker compare_batch(struct drm_intel_decode *ctx, const char *batch_filename)
87*7688df22SAndroid Build Coastguard Worker {
88*7688df22SAndroid Build Coastguard Worker 	FILE *out = NULL;
89*7688df22SAndroid Build Coastguard Worker 	char *ptr;
90*7688df22SAndroid Build Coastguard Worker 	void *ref_ptr, *batch_ptr;
91*7688df22SAndroid Build Coastguard Worker #if HAVE_OPEN_MEMSTREAM
92*7688df22SAndroid Build Coastguard Worker 	size_t size;
93*7688df22SAndroid Build Coastguard Worker #endif
94*7688df22SAndroid Build Coastguard Worker 	size_t ref_size, batch_size;
95*7688df22SAndroid Build Coastguard Worker 	const char *ref_suffix = "-ref.txt";
96*7688df22SAndroid Build Coastguard Worker 	char *ref_filename;
97*7688df22SAndroid Build Coastguard Worker 
98*7688df22SAndroid Build Coastguard Worker 	ref_filename = malloc(strlen(batch_filename) + strlen(ref_suffix) + 1);
99*7688df22SAndroid Build Coastguard Worker 	sprintf(ref_filename, "%s%s", batch_filename, ref_suffix);
100*7688df22SAndroid Build Coastguard Worker 
101*7688df22SAndroid Build Coastguard Worker 	/* Read the batch and reference. */
102*7688df22SAndroid Build Coastguard Worker 	read_file(batch_filename, &batch_ptr, &batch_size);
103*7688df22SAndroid Build Coastguard Worker 	read_file(ref_filename, &ref_ptr, &ref_size);
104*7688df22SAndroid Build Coastguard Worker 
105*7688df22SAndroid Build Coastguard Worker 	/* Set up our decode output in memory, because I don't want to
106*7688df22SAndroid Build Coastguard Worker 	 * figure out how to output to a file in a safe and sane way
107*7688df22SAndroid Build Coastguard Worker 	 * inside of an automake project's test infrastructure.
108*7688df22SAndroid Build Coastguard Worker 	 */
109*7688df22SAndroid Build Coastguard Worker #if HAVE_OPEN_MEMSTREAM
110*7688df22SAndroid Build Coastguard Worker 	out = open_memstream(&ptr, &size);
111*7688df22SAndroid Build Coastguard Worker #else
112*7688df22SAndroid Build Coastguard Worker 	fprintf(stderr, "platform lacks open_memstream, skipping.\n");
113*7688df22SAndroid Build Coastguard Worker 	exit(77);
114*7688df22SAndroid Build Coastguard Worker #endif
115*7688df22SAndroid Build Coastguard Worker 
116*7688df22SAndroid Build Coastguard Worker 	drm_intel_decode_set_batch_pointer(ctx, batch_ptr, HW_OFFSET,
117*7688df22SAndroid Build Coastguard Worker 					   batch_size / 4);
118*7688df22SAndroid Build Coastguard Worker 	drm_intel_decode_set_output_file(ctx, out);
119*7688df22SAndroid Build Coastguard Worker 
120*7688df22SAndroid Build Coastguard Worker 	drm_intel_decode(ctx);
121*7688df22SAndroid Build Coastguard Worker 
122*7688df22SAndroid Build Coastguard Worker 	if (strcmp(ref_ptr, ptr) != 0) {
123*7688df22SAndroid Build Coastguard Worker 		fprintf(stderr, "Decode mismatch with reference `%s'.\n",
124*7688df22SAndroid Build Coastguard Worker 			ref_filename);
125*7688df22SAndroid Build Coastguard Worker 		fprintf(stderr, "You can dump the new output using:\n");
126*7688df22SAndroid Build Coastguard Worker 		fprintf(stderr, "  test_decode \"%s\" -dump\n", batch_filename);
127*7688df22SAndroid Build Coastguard Worker 		exit(1);
128*7688df22SAndroid Build Coastguard Worker 	}
129*7688df22SAndroid Build Coastguard Worker 
130*7688df22SAndroid Build Coastguard Worker 	fclose(out);
131*7688df22SAndroid Build Coastguard Worker 	free(ref_filename);
132*7688df22SAndroid Build Coastguard Worker 	free(ptr);
133*7688df22SAndroid Build Coastguard Worker }
134*7688df22SAndroid Build Coastguard Worker 
135*7688df22SAndroid Build Coastguard Worker static uint16_t
infer_devid(const char * batch_filename)136*7688df22SAndroid Build Coastguard Worker infer_devid(const char *batch_filename)
137*7688df22SAndroid Build Coastguard Worker {
138*7688df22SAndroid Build Coastguard Worker 	struct {
139*7688df22SAndroid Build Coastguard Worker 		const char *name;
140*7688df22SAndroid Build Coastguard Worker 		uint16_t devid;
141*7688df22SAndroid Build Coastguard Worker 	} chipsets[] = {
142*7688df22SAndroid Build Coastguard Worker 		{ "830",  0x3577},
143*7688df22SAndroid Build Coastguard Worker 		{ "855",  0x3582},
144*7688df22SAndroid Build Coastguard Worker 		{ "945",  0x2772},
145*7688df22SAndroid Build Coastguard Worker 		{ "gen4", 0x2a02 },
146*7688df22SAndroid Build Coastguard Worker 		{ "gm45", 0x2a42 },
147*7688df22SAndroid Build Coastguard Worker 		{ "gen5", PCI_CHIP_ILD_G },
148*7688df22SAndroid Build Coastguard Worker 		{ "gen6", PCI_CHIP_SANDYBRIDGE_GT2 },
149*7688df22SAndroid Build Coastguard Worker 		{ "gen7", PCI_CHIP_IVYBRIDGE_GT2 },
150*7688df22SAndroid Build Coastguard Worker 		{ "gen8", 0x1616 },
151*7688df22SAndroid Build Coastguard Worker 		{ NULL, 0 },
152*7688df22SAndroid Build Coastguard Worker 	};
153*7688df22SAndroid Build Coastguard Worker 	int i;
154*7688df22SAndroid Build Coastguard Worker 
155*7688df22SAndroid Build Coastguard Worker 	for (i = 0; chipsets[i].name != NULL; i++) {
156*7688df22SAndroid Build Coastguard Worker 		if (strstr(batch_filename, chipsets[i].name))
157*7688df22SAndroid Build Coastguard Worker 			return chipsets[i].devid;
158*7688df22SAndroid Build Coastguard Worker 	}
159*7688df22SAndroid Build Coastguard Worker 
160*7688df22SAndroid Build Coastguard Worker 	fprintf(stderr, "Couldn't guess chipset id from batch filename `%s'.\n",
161*7688df22SAndroid Build Coastguard Worker 		batch_filename);
162*7688df22SAndroid Build Coastguard Worker 	fprintf(stderr, "Must be contain one of:\n");
163*7688df22SAndroid Build Coastguard Worker 	for (i = 0; chipsets[i].name != NULL; i++) {
164*7688df22SAndroid Build Coastguard Worker 		fprintf(stderr, "  %s\n", chipsets[i].name);
165*7688df22SAndroid Build Coastguard Worker 	}
166*7688df22SAndroid Build Coastguard Worker 	exit(1);
167*7688df22SAndroid Build Coastguard Worker }
168*7688df22SAndroid Build Coastguard Worker 
169*7688df22SAndroid Build Coastguard Worker int
main(int argc,char ** argv)170*7688df22SAndroid Build Coastguard Worker main(int argc, char **argv)
171*7688df22SAndroid Build Coastguard Worker {
172*7688df22SAndroid Build Coastguard Worker 	uint16_t devid;
173*7688df22SAndroid Build Coastguard Worker 	struct drm_intel_decode *ctx;
174*7688df22SAndroid Build Coastguard Worker 
175*7688df22SAndroid Build Coastguard Worker 	if (argc < 2)
176*7688df22SAndroid Build Coastguard Worker 		usage();
177*7688df22SAndroid Build Coastguard Worker 
178*7688df22SAndroid Build Coastguard Worker 
179*7688df22SAndroid Build Coastguard Worker 	devid = infer_devid(argv[1]);
180*7688df22SAndroid Build Coastguard Worker 
181*7688df22SAndroid Build Coastguard Worker 	ctx = drm_intel_decode_context_alloc(devid);
182*7688df22SAndroid Build Coastguard Worker 
183*7688df22SAndroid Build Coastguard Worker 	if (argc == 3) {
184*7688df22SAndroid Build Coastguard Worker 		if (strcmp(argv[2], "-dump") == 0)
185*7688df22SAndroid Build Coastguard Worker 			dump_batch(ctx, argv[1]);
186*7688df22SAndroid Build Coastguard Worker 		else
187*7688df22SAndroid Build Coastguard Worker 			usage();
188*7688df22SAndroid Build Coastguard Worker 	} else {
189*7688df22SAndroid Build Coastguard Worker 		compare_batch(ctx, argv[1]);
190*7688df22SAndroid Build Coastguard Worker 	}
191*7688df22SAndroid Build Coastguard Worker 
192*7688df22SAndroid Build Coastguard Worker 	drm_intel_decode_context_free(ctx);
193*7688df22SAndroid Build Coastguard Worker 
194*7688df22SAndroid Build Coastguard Worker 	return 0;
195*7688df22SAndroid Build Coastguard Worker }
196