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