xref: /aosp_15_r20/external/igt-gpu-tools/tests/i915/i915_pm_sseu.c (revision d83cc019efdc2edc6c4b16e9034a3ceb8d35d77c)
1*d83cc019SAndroid Build Coastguard Worker /*
2*d83cc019SAndroid Build Coastguard Worker  * Copyright © 2015 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  * Authors:
24*d83cc019SAndroid Build Coastguard Worker  *    Jeff McGee <[email protected]>
25*d83cc019SAndroid Build Coastguard Worker  */
26*d83cc019SAndroid Build Coastguard Worker 
27*d83cc019SAndroid Build Coastguard Worker #include "igt.h"
28*d83cc019SAndroid Build Coastguard Worker #include <fcntl.h>
29*d83cc019SAndroid Build Coastguard Worker #include <unistd.h>
30*d83cc019SAndroid Build Coastguard Worker #include <string.h>
31*d83cc019SAndroid Build Coastguard Worker #include <errno.h>
32*d83cc019SAndroid Build Coastguard Worker #include <time.h>
33*d83cc019SAndroid Build Coastguard Worker #include "i915_drm.h"
34*d83cc019SAndroid Build Coastguard Worker #include "intel_bufmgr.h"
35*d83cc019SAndroid Build Coastguard Worker 
36*d83cc019SAndroid Build Coastguard Worker IGT_TEST_DESCRIPTION("Tests slice/subslice/EU power gating functionality.\n");
37*d83cc019SAndroid Build Coastguard Worker 
38*d83cc019SAndroid Build Coastguard Worker static struct {
39*d83cc019SAndroid Build Coastguard Worker 	int init;
40*d83cc019SAndroid Build Coastguard Worker 	int drm_fd;
41*d83cc019SAndroid Build Coastguard Worker 	int devid;
42*d83cc019SAndroid Build Coastguard Worker 	int gen;
43*d83cc019SAndroid Build Coastguard Worker 	drm_intel_bufmgr *bufmgr;
44*d83cc019SAndroid Build Coastguard Worker 	struct intel_batchbuffer *batch;
45*d83cc019SAndroid Build Coastguard Worker 	igt_media_spinfunc_t spinfunc;
46*d83cc019SAndroid Build Coastguard Worker 	struct igt_buf buf;
47*d83cc019SAndroid Build Coastguard Worker 	uint32_t spins_per_msec;
48*d83cc019SAndroid Build Coastguard Worker } gem;
49*d83cc019SAndroid Build Coastguard Worker 
50*d83cc019SAndroid Build Coastguard Worker static double
to_dt(const struct timespec * start,const struct timespec * end)51*d83cc019SAndroid Build Coastguard Worker to_dt(const struct timespec *start, const struct timespec *end)
52*d83cc019SAndroid Build Coastguard Worker {
53*d83cc019SAndroid Build Coastguard Worker 	double dt;
54*d83cc019SAndroid Build Coastguard Worker 
55*d83cc019SAndroid Build Coastguard Worker 	dt = (end->tv_sec - start->tv_sec) * 1e3;
56*d83cc019SAndroid Build Coastguard Worker 	dt += (end->tv_nsec - start->tv_nsec) * 1e-6;
57*d83cc019SAndroid Build Coastguard Worker 
58*d83cc019SAndroid Build Coastguard Worker 	return dt;
59*d83cc019SAndroid Build Coastguard Worker }
60*d83cc019SAndroid Build Coastguard Worker 
61*d83cc019SAndroid Build Coastguard Worker struct status {
62*d83cc019SAndroid Build Coastguard Worker 	struct {
63*d83cc019SAndroid Build Coastguard Worker 		int slice_total;
64*d83cc019SAndroid Build Coastguard Worker 		int subslice_total;
65*d83cc019SAndroid Build Coastguard Worker 		int subslice_per;
66*d83cc019SAndroid Build Coastguard Worker 		int eu_total;
67*d83cc019SAndroid Build Coastguard Worker 		int eu_per;
68*d83cc019SAndroid Build Coastguard Worker 		bool has_slice_pg;
69*d83cc019SAndroid Build Coastguard Worker 		bool has_subslice_pg;
70*d83cc019SAndroid Build Coastguard Worker 		bool has_eu_pg;
71*d83cc019SAndroid Build Coastguard Worker 	} info;
72*d83cc019SAndroid Build Coastguard Worker 	struct {
73*d83cc019SAndroid Build Coastguard Worker 		int slice_total;
74*d83cc019SAndroid Build Coastguard Worker 		int subslice_total;
75*d83cc019SAndroid Build Coastguard Worker 		int subslice_per;
76*d83cc019SAndroid Build Coastguard Worker 		int eu_total;
77*d83cc019SAndroid Build Coastguard Worker 		int eu_per;
78*d83cc019SAndroid Build Coastguard Worker 	} hw;
79*d83cc019SAndroid Build Coastguard Worker };
80*d83cc019SAndroid Build Coastguard Worker 
81*d83cc019SAndroid Build Coastguard Worker #define DBG_STATUS_BUF_SIZE 4096
82*d83cc019SAndroid Build Coastguard Worker 
83*d83cc019SAndroid Build Coastguard Worker struct {
84*d83cc019SAndroid Build Coastguard Worker 	int init;
85*d83cc019SAndroid Build Coastguard Worker 	int status_fd;
86*d83cc019SAndroid Build Coastguard Worker 	char status_buf[DBG_STATUS_BUF_SIZE];
87*d83cc019SAndroid Build Coastguard Worker } dbg;
88*d83cc019SAndroid Build Coastguard Worker 
89*d83cc019SAndroid Build Coastguard Worker static void
dbg_get_status_section(const char * title,char ** first,char ** last)90*d83cc019SAndroid Build Coastguard Worker dbg_get_status_section(const char *title, char **first, char **last)
91*d83cc019SAndroid Build Coastguard Worker {
92*d83cc019SAndroid Build Coastguard Worker 	char *pos;
93*d83cc019SAndroid Build Coastguard Worker 
94*d83cc019SAndroid Build Coastguard Worker 	*first = strstr(dbg.status_buf, title);
95*d83cc019SAndroid Build Coastguard Worker 	igt_assert(*first != NULL);
96*d83cc019SAndroid Build Coastguard Worker 
97*d83cc019SAndroid Build Coastguard Worker 	pos = *first;
98*d83cc019SAndroid Build Coastguard Worker 	do {
99*d83cc019SAndroid Build Coastguard Worker 		pos = strchr(pos, '\n');
100*d83cc019SAndroid Build Coastguard Worker 		igt_assert(pos != NULL);
101*d83cc019SAndroid Build Coastguard Worker 		pos++;
102*d83cc019SAndroid Build Coastguard Worker 	} while (*pos == ' '); /* lines in the section begin with a space */
103*d83cc019SAndroid Build Coastguard Worker 	*last = pos - 1;
104*d83cc019SAndroid Build Coastguard Worker }
105*d83cc019SAndroid Build Coastguard Worker 
106*d83cc019SAndroid Build Coastguard Worker static bool
dbg_has_line(const char * first,const char * last,const char * name)107*d83cc019SAndroid Build Coastguard Worker dbg_has_line(const char *first, const char *last, const char *name)
108*d83cc019SAndroid Build Coastguard Worker {
109*d83cc019SAndroid Build Coastguard Worker 	char *pos = strstr(first, name);
110*d83cc019SAndroid Build Coastguard Worker 
111*d83cc019SAndroid Build Coastguard Worker 	return pos != NULL && pos < last;
112*d83cc019SAndroid Build Coastguard Worker }
113*d83cc019SAndroid Build Coastguard Worker 
114*d83cc019SAndroid Build Coastguard Worker static int
dbg_get_int(const char * first,const char * last,const char * name)115*d83cc019SAndroid Build Coastguard Worker dbg_get_int(const char *first, const char *last, const char *name)
116*d83cc019SAndroid Build Coastguard Worker {
117*d83cc019SAndroid Build Coastguard Worker 	char *pos;
118*d83cc019SAndroid Build Coastguard Worker 
119*d83cc019SAndroid Build Coastguard Worker 	pos = strstr(first, name);
120*d83cc019SAndroid Build Coastguard Worker 	igt_assert(pos != NULL);
121*d83cc019SAndroid Build Coastguard Worker 	pos = strstr(pos, ":");
122*d83cc019SAndroid Build Coastguard Worker 	igt_assert(pos != NULL);
123*d83cc019SAndroid Build Coastguard Worker 	pos += 2;
124*d83cc019SAndroid Build Coastguard Worker 	igt_assert(pos != last);
125*d83cc019SAndroid Build Coastguard Worker 
126*d83cc019SAndroid Build Coastguard Worker 	return strtol(pos, &pos, 10);
127*d83cc019SAndroid Build Coastguard Worker }
128*d83cc019SAndroid Build Coastguard Worker 
129*d83cc019SAndroid Build Coastguard Worker static bool
dbg_get_bool(const char * first,const char * last,const char * name)130*d83cc019SAndroid Build Coastguard Worker dbg_get_bool(const char *first, const char *last, const char *name)
131*d83cc019SAndroid Build Coastguard Worker {
132*d83cc019SAndroid Build Coastguard Worker 	char *pos;
133*d83cc019SAndroid Build Coastguard Worker 
134*d83cc019SAndroid Build Coastguard Worker 	pos = strstr(first, name);
135*d83cc019SAndroid Build Coastguard Worker 	igt_assert(pos != NULL);
136*d83cc019SAndroid Build Coastguard Worker 	pos = strstr(pos, ":");
137*d83cc019SAndroid Build Coastguard Worker 	igt_assert(pos != NULL);
138*d83cc019SAndroid Build Coastguard Worker 	pos += 2;
139*d83cc019SAndroid Build Coastguard Worker 	igt_assert(pos < last);
140*d83cc019SAndroid Build Coastguard Worker 
141*d83cc019SAndroid Build Coastguard Worker 	if (*pos == 'y')
142*d83cc019SAndroid Build Coastguard Worker 		return true;
143*d83cc019SAndroid Build Coastguard Worker 	if (*pos == 'n')
144*d83cc019SAndroid Build Coastguard Worker 		return false;
145*d83cc019SAndroid Build Coastguard Worker 
146*d83cc019SAndroid Build Coastguard Worker 	igt_assert_f(false, "Could not read boolean value for %s.\n", name);
147*d83cc019SAndroid Build Coastguard Worker 	return false;
148*d83cc019SAndroid Build Coastguard Worker }
149*d83cc019SAndroid Build Coastguard Worker 
150*d83cc019SAndroid Build Coastguard Worker static void
dbg_get_status(struct status * stat)151*d83cc019SAndroid Build Coastguard Worker dbg_get_status(struct status *stat)
152*d83cc019SAndroid Build Coastguard Worker {
153*d83cc019SAndroid Build Coastguard Worker 	char *first, *last;
154*d83cc019SAndroid Build Coastguard Worker 	int nread;
155*d83cc019SAndroid Build Coastguard Worker 
156*d83cc019SAndroid Build Coastguard Worker 	lseek(dbg.status_fd, 0, SEEK_SET);
157*d83cc019SAndroid Build Coastguard Worker 	nread = read(dbg.status_fd, dbg.status_buf, DBG_STATUS_BUF_SIZE);
158*d83cc019SAndroid Build Coastguard Worker 	igt_assert_lt(nread, DBG_STATUS_BUF_SIZE);
159*d83cc019SAndroid Build Coastguard Worker 	dbg.status_buf[nread] = '\0';
160*d83cc019SAndroid Build Coastguard Worker 
161*d83cc019SAndroid Build Coastguard Worker 	memset(stat, 0, sizeof(*stat));
162*d83cc019SAndroid Build Coastguard Worker 
163*d83cc019SAndroid Build Coastguard Worker 	dbg_get_status_section("SSEU Device Info", &first, &last);
164*d83cc019SAndroid Build Coastguard Worker 	for (char *tmp = first; tmp < last; tmp++)
165*d83cc019SAndroid Build Coastguard Worker 		igt_debug("%c", *tmp);
166*d83cc019SAndroid Build Coastguard Worker 	igt_debug("\n");
167*d83cc019SAndroid Build Coastguard Worker 	stat->info.slice_total =
168*d83cc019SAndroid Build Coastguard Worker 		dbg_get_int(first, last, "Available Slice Total:");
169*d83cc019SAndroid Build Coastguard Worker 	stat->info.subslice_total =
170*d83cc019SAndroid Build Coastguard Worker 		dbg_get_int(first, last, "Available Subslice Total:");
171*d83cc019SAndroid Build Coastguard Worker 	/* Dealing with a change in 4.17. */
172*d83cc019SAndroid Build Coastguard Worker 	if (dbg_has_line(first, last, "Available Subslice Per Slice:")) {
173*d83cc019SAndroid Build Coastguard Worker 		stat->info.subslice_per =
174*d83cc019SAndroid Build Coastguard Worker 			dbg_get_int(first, last, "Available Subslice Per Slice:");
175*d83cc019SAndroid Build Coastguard Worker 	} else {
176*d83cc019SAndroid Build Coastguard Worker 		stat->info.subslice_per =
177*d83cc019SAndroid Build Coastguard Worker 			dbg_get_int(first, last, "Available Slice0 subslices:");
178*d83cc019SAndroid Build Coastguard Worker 	}
179*d83cc019SAndroid Build Coastguard Worker 	stat->info.eu_total =
180*d83cc019SAndroid Build Coastguard Worker 		dbg_get_int(first, last, "Available EU Total:");
181*d83cc019SAndroid Build Coastguard Worker 	stat->info.eu_per =
182*d83cc019SAndroid Build Coastguard Worker 		dbg_get_int(first, last, "Available EU Per Subslice:");
183*d83cc019SAndroid Build Coastguard Worker 	stat->info.has_slice_pg =
184*d83cc019SAndroid Build Coastguard Worker 		dbg_get_bool(first, last, "Has Slice Power Gating:");
185*d83cc019SAndroid Build Coastguard Worker 	stat->info.has_subslice_pg =
186*d83cc019SAndroid Build Coastguard Worker 		dbg_get_bool(first, last, "Has Subslice Power Gating:");
187*d83cc019SAndroid Build Coastguard Worker 	stat->info.has_eu_pg =
188*d83cc019SAndroid Build Coastguard Worker 		dbg_get_bool(first, last, "Has EU Power Gating:");
189*d83cc019SAndroid Build Coastguard Worker 
190*d83cc019SAndroid Build Coastguard Worker 	dbg_get_status_section("SSEU Device Status", &first, &last);
191*d83cc019SAndroid Build Coastguard Worker 	for (char *tmp = first; tmp < last; tmp++)
192*d83cc019SAndroid Build Coastguard Worker 		igt_debug("%c", *tmp);
193*d83cc019SAndroid Build Coastguard Worker 	igt_debug("\n");
194*d83cc019SAndroid Build Coastguard Worker 	stat->hw.slice_total =
195*d83cc019SAndroid Build Coastguard Worker 		dbg_get_int(first, last, "Enabled Slice Total:");
196*d83cc019SAndroid Build Coastguard Worker 	stat->hw.subslice_total =
197*d83cc019SAndroid Build Coastguard Worker 		dbg_get_int(first, last, "Enabled Subslice Total:");
198*d83cc019SAndroid Build Coastguard Worker 	/* Dealing with a change in 4.17. */
199*d83cc019SAndroid Build Coastguard Worker 	if (dbg_has_line(first, last, "Enabled Subslice Per Slice:")) {
200*d83cc019SAndroid Build Coastguard Worker 		stat->hw.subslice_per =
201*d83cc019SAndroid Build Coastguard Worker 			dbg_get_int(first, last, "Enabled Subslice Per Slice:");
202*d83cc019SAndroid Build Coastguard Worker 	} else if (dbg_has_line(first, last, "Enabled Slice0 subslices:")) {
203*d83cc019SAndroid Build Coastguard Worker 		stat->hw.subslice_per =
204*d83cc019SAndroid Build Coastguard Worker 			dbg_get_int(first, last, "Enabled Slice0 subslices:");
205*d83cc019SAndroid Build Coastguard Worker 	}
206*d83cc019SAndroid Build Coastguard Worker 	stat->hw.eu_total =
207*d83cc019SAndroid Build Coastguard Worker 		dbg_get_int(first, last, "Enabled EU Total:");
208*d83cc019SAndroid Build Coastguard Worker 	stat->hw.eu_per =
209*d83cc019SAndroid Build Coastguard Worker 		dbg_get_int(first, last, "Enabled EU Per Subslice:");
210*d83cc019SAndroid Build Coastguard Worker }
211*d83cc019SAndroid Build Coastguard Worker 
212*d83cc019SAndroid Build Coastguard Worker static void
dbg_init(void)213*d83cc019SAndroid Build Coastguard Worker dbg_init(void)
214*d83cc019SAndroid Build Coastguard Worker {
215*d83cc019SAndroid Build Coastguard Worker 	igt_assert(gem.init);
216*d83cc019SAndroid Build Coastguard Worker 	dbg.status_fd = igt_debugfs_open(gem.drm_fd, "i915_sseu_status", O_RDONLY);
217*d83cc019SAndroid Build Coastguard Worker 	igt_skip_on_f(dbg.status_fd == -1,
218*d83cc019SAndroid Build Coastguard Worker 		      "debugfs entry 'i915_sseu_status' not found\n");
219*d83cc019SAndroid Build Coastguard Worker 	dbg.init = 1;
220*d83cc019SAndroid Build Coastguard Worker }
221*d83cc019SAndroid Build Coastguard Worker 
222*d83cc019SAndroid Build Coastguard Worker static void
dbg_deinit(void)223*d83cc019SAndroid Build Coastguard Worker dbg_deinit(void)
224*d83cc019SAndroid Build Coastguard Worker {
225*d83cc019SAndroid Build Coastguard Worker 	switch (dbg.init)
226*d83cc019SAndroid Build Coastguard Worker 	{
227*d83cc019SAndroid Build Coastguard Worker 	case 1:
228*d83cc019SAndroid Build Coastguard Worker 		close(dbg.status_fd);
229*d83cc019SAndroid Build Coastguard Worker 	}
230*d83cc019SAndroid Build Coastguard Worker }
231*d83cc019SAndroid Build Coastguard Worker 
232*d83cc019SAndroid Build Coastguard Worker static void
gem_check_spin(uint32_t spins)233*d83cc019SAndroid Build Coastguard Worker gem_check_spin(uint32_t spins)
234*d83cc019SAndroid Build Coastguard Worker {
235*d83cc019SAndroid Build Coastguard Worker 	uint32_t *data;
236*d83cc019SAndroid Build Coastguard Worker 
237*d83cc019SAndroid Build Coastguard Worker 	data = (uint32_t*)gem.buf.bo->virtual;
238*d83cc019SAndroid Build Coastguard Worker 	igt_assert_eq_u32(*data, spins);
239*d83cc019SAndroid Build Coastguard Worker }
240*d83cc019SAndroid Build Coastguard Worker 
241*d83cc019SAndroid Build Coastguard Worker static uint32_t
gem_get_target_spins(double dt)242*d83cc019SAndroid Build Coastguard Worker gem_get_target_spins(double dt)
243*d83cc019SAndroid Build Coastguard Worker {
244*d83cc019SAndroid Build Coastguard Worker 	struct timespec tstart, tdone;
245*d83cc019SAndroid Build Coastguard Worker 	double prev_dt, cur_dt;
246*d83cc019SAndroid Build Coastguard Worker 	uint32_t spins;
247*d83cc019SAndroid Build Coastguard Worker 	int i, ret;
248*d83cc019SAndroid Build Coastguard Worker 
249*d83cc019SAndroid Build Coastguard Worker 	/* Double increments until we bound the target time */
250*d83cc019SAndroid Build Coastguard Worker 	prev_dt = 0.0;
251*d83cc019SAndroid Build Coastguard Worker 	for (i = 0; i < 32; i++) {
252*d83cc019SAndroid Build Coastguard Worker 		spins = 1 << i;
253*d83cc019SAndroid Build Coastguard Worker 		clock_gettime(CLOCK_MONOTONIC, &tstart);
254*d83cc019SAndroid Build Coastguard Worker 
255*d83cc019SAndroid Build Coastguard Worker 		gem.spinfunc(gem.batch, &gem.buf, spins);
256*d83cc019SAndroid Build Coastguard Worker 		ret = drm_intel_bo_map(gem.buf.bo, 0);
257*d83cc019SAndroid Build Coastguard Worker 		igt_assert_eq(ret, 0);
258*d83cc019SAndroid Build Coastguard Worker 		clock_gettime(CLOCK_MONOTONIC, &tdone);
259*d83cc019SAndroid Build Coastguard Worker 
260*d83cc019SAndroid Build Coastguard Worker 		gem_check_spin(spins);
261*d83cc019SAndroid Build Coastguard Worker 		drm_intel_bo_unmap(gem.buf.bo);
262*d83cc019SAndroid Build Coastguard Worker 
263*d83cc019SAndroid Build Coastguard Worker 		cur_dt = to_dt(&tstart, &tdone);
264*d83cc019SAndroid Build Coastguard Worker 		if (cur_dt > dt)
265*d83cc019SAndroid Build Coastguard Worker 			break;
266*d83cc019SAndroid Build Coastguard Worker 		prev_dt = cur_dt;
267*d83cc019SAndroid Build Coastguard Worker 	}
268*d83cc019SAndroid Build Coastguard Worker 	igt_assert_neq(i, 32);
269*d83cc019SAndroid Build Coastguard Worker 
270*d83cc019SAndroid Build Coastguard Worker 	/* Linearly interpolate between i and i-1 to get target increments */
271*d83cc019SAndroid Build Coastguard Worker 	spins = 1 << (i-1); /* lower bound spins */
272*d83cc019SAndroid Build Coastguard Worker 	spins += spins * (dt - prev_dt)/(cur_dt - prev_dt); /* target spins */
273*d83cc019SAndroid Build Coastguard Worker 
274*d83cc019SAndroid Build Coastguard Worker 	return spins;
275*d83cc019SAndroid Build Coastguard Worker }
276*d83cc019SAndroid Build Coastguard Worker 
277*d83cc019SAndroid Build Coastguard Worker static void
gem_init(void)278*d83cc019SAndroid Build Coastguard Worker gem_init(void)
279*d83cc019SAndroid Build Coastguard Worker {
280*d83cc019SAndroid Build Coastguard Worker 	gem.drm_fd = drm_open_driver(DRIVER_INTEL);
281*d83cc019SAndroid Build Coastguard Worker 	igt_require_gem(gem.drm_fd);
282*d83cc019SAndroid Build Coastguard Worker 	gem.init = 1;
283*d83cc019SAndroid Build Coastguard Worker 
284*d83cc019SAndroid Build Coastguard Worker 	gem.devid = intel_get_drm_devid(gem.drm_fd);
285*d83cc019SAndroid Build Coastguard Worker 	gem.gen = intel_gen(gem.devid);
286*d83cc019SAndroid Build Coastguard Worker 	igt_require_f(gem.gen >= 8,
287*d83cc019SAndroid Build Coastguard Worker 		      "SSEU power gating only relevant for Gen8+");
288*d83cc019SAndroid Build Coastguard Worker 
289*d83cc019SAndroid Build Coastguard Worker 	gem.spinfunc = igt_get_media_spinfunc(gem.devid);
290*d83cc019SAndroid Build Coastguard Worker 	igt_require(gem.spinfunc);
291*d83cc019SAndroid Build Coastguard Worker 
292*d83cc019SAndroid Build Coastguard Worker 	gem.bufmgr = drm_intel_bufmgr_gem_init(gem.drm_fd, 4096);
293*d83cc019SAndroid Build Coastguard Worker 	igt_assert(gem.bufmgr);
294*d83cc019SAndroid Build Coastguard Worker 	gem.init = 2;
295*d83cc019SAndroid Build Coastguard Worker 
296*d83cc019SAndroid Build Coastguard Worker 	drm_intel_bufmgr_gem_enable_reuse(gem.bufmgr);
297*d83cc019SAndroid Build Coastguard Worker 
298*d83cc019SAndroid Build Coastguard Worker 	gem.batch = intel_batchbuffer_alloc(gem.bufmgr, gem.devid);
299*d83cc019SAndroid Build Coastguard Worker 	igt_assert(gem.batch);
300*d83cc019SAndroid Build Coastguard Worker 	gem.init = 3;
301*d83cc019SAndroid Build Coastguard Worker 
302*d83cc019SAndroid Build Coastguard Worker 	gem.buf.stride = sizeof(uint32_t);
303*d83cc019SAndroid Build Coastguard Worker 	gem.buf.tiling = I915_TILING_NONE;
304*d83cc019SAndroid Build Coastguard Worker 	gem.buf.size = gem.buf.stride;
305*d83cc019SAndroid Build Coastguard Worker 	gem.buf.bo = drm_intel_bo_alloc(gem.bufmgr, "", gem.buf.size, 4096);
306*d83cc019SAndroid Build Coastguard Worker 	gem.buf.bpp = 32;
307*d83cc019SAndroid Build Coastguard Worker 	igt_assert(gem.buf.bo);
308*d83cc019SAndroid Build Coastguard Worker 	gem.init = 4;
309*d83cc019SAndroid Build Coastguard Worker 
310*d83cc019SAndroid Build Coastguard Worker 	gem.spins_per_msec = gem_get_target_spins(100) / 100;
311*d83cc019SAndroid Build Coastguard Worker }
312*d83cc019SAndroid Build Coastguard Worker 
313*d83cc019SAndroid Build Coastguard Worker static void
gem_deinit(void)314*d83cc019SAndroid Build Coastguard Worker gem_deinit(void)
315*d83cc019SAndroid Build Coastguard Worker {
316*d83cc019SAndroid Build Coastguard Worker 	switch (gem.init)
317*d83cc019SAndroid Build Coastguard Worker 	{
318*d83cc019SAndroid Build Coastguard Worker 	case 4:
319*d83cc019SAndroid Build Coastguard Worker 		drm_intel_bo_unmap(gem.buf.bo);
320*d83cc019SAndroid Build Coastguard Worker 		drm_intel_bo_unreference(gem.buf.bo);
321*d83cc019SAndroid Build Coastguard Worker 	case 3:
322*d83cc019SAndroid Build Coastguard Worker 		intel_batchbuffer_free(gem.batch);
323*d83cc019SAndroid Build Coastguard Worker 	case 2:
324*d83cc019SAndroid Build Coastguard Worker 		drm_intel_bufmgr_destroy(gem.bufmgr);
325*d83cc019SAndroid Build Coastguard Worker 	case 1:
326*d83cc019SAndroid Build Coastguard Worker 		close(gem.drm_fd);
327*d83cc019SAndroid Build Coastguard Worker 	}
328*d83cc019SAndroid Build Coastguard Worker }
329*d83cc019SAndroid Build Coastguard Worker 
330*d83cc019SAndroid Build Coastguard Worker static void
check_full_enable(struct status * stat)331*d83cc019SAndroid Build Coastguard Worker check_full_enable(struct status *stat)
332*d83cc019SAndroid Build Coastguard Worker {
333*d83cc019SAndroid Build Coastguard Worker 	igt_assert_eq(stat->hw.slice_total, stat->info.slice_total);
334*d83cc019SAndroid Build Coastguard Worker 	igt_assert_eq(stat->hw.subslice_total, stat->info.subslice_total);
335*d83cc019SAndroid Build Coastguard Worker 	igt_assert_eq(stat->hw.subslice_per, stat->info.subslice_per);
336*d83cc019SAndroid Build Coastguard Worker 
337*d83cc019SAndroid Build Coastguard Worker 	/*
338*d83cc019SAndroid Build Coastguard Worker 	 * EU are powered in pairs, but it is possible for one EU in the pair
339*d83cc019SAndroid Build Coastguard Worker 	 * to be non-functional due to fusing. The determination of enabled
340*d83cc019SAndroid Build Coastguard Worker 	 * EU does not account for this and can therefore actually exceed the
341*d83cc019SAndroid Build Coastguard Worker 	 * available count. Allow for this small discrepancy in our
342*d83cc019SAndroid Build Coastguard Worker 	 * comparison.
343*d83cc019SAndroid Build Coastguard Worker 	*/
344*d83cc019SAndroid Build Coastguard Worker 	igt_assert_lte(stat->info.eu_total, stat->hw.eu_total);
345*d83cc019SAndroid Build Coastguard Worker 	igt_assert_lte(stat->info.eu_per, stat->hw.eu_per);
346*d83cc019SAndroid Build Coastguard Worker }
347*d83cc019SAndroid Build Coastguard Worker 
348*d83cc019SAndroid Build Coastguard Worker static void
full_enable(void)349*d83cc019SAndroid Build Coastguard Worker full_enable(void)
350*d83cc019SAndroid Build Coastguard Worker {
351*d83cc019SAndroid Build Coastguard Worker 	struct status stat;
352*d83cc019SAndroid Build Coastguard Worker 	const int spin_msec = 10;
353*d83cc019SAndroid Build Coastguard Worker 	int ret, spins;
354*d83cc019SAndroid Build Coastguard Worker 
355*d83cc019SAndroid Build Coastguard Worker 	/* Simulation doesn't currently model slice/subslice/EU power gating. */
356*d83cc019SAndroid Build Coastguard Worker 	igt_skip_on_simulation();
357*d83cc019SAndroid Build Coastguard Worker 
358*d83cc019SAndroid Build Coastguard Worker 	/*
359*d83cc019SAndroid Build Coastguard Worker 	 * Gen9 SKL is the first case in which render power gating can leave
360*d83cc019SAndroid Build Coastguard Worker 	 * slice/subslice/EU in a partially enabled state upon resumption of
361*d83cc019SAndroid Build Coastguard Worker 	 * render work. So start checking that this is prevented as of Gen9.
362*d83cc019SAndroid Build Coastguard Worker 	*/
363*d83cc019SAndroid Build Coastguard Worker 	igt_require(gem.gen >= 9);
364*d83cc019SAndroid Build Coastguard Worker 
365*d83cc019SAndroid Build Coastguard Worker 	spins = spin_msec * gem.spins_per_msec;
366*d83cc019SAndroid Build Coastguard Worker 
367*d83cc019SAndroid Build Coastguard Worker 	gem.spinfunc(gem.batch, &gem.buf, spins);
368*d83cc019SAndroid Build Coastguard Worker 
369*d83cc019SAndroid Build Coastguard Worker 	usleep(2000); /* 2ms wait to make sure batch is running */
370*d83cc019SAndroid Build Coastguard Worker 	dbg_get_status(&stat);
371*d83cc019SAndroid Build Coastguard Worker 
372*d83cc019SAndroid Build Coastguard Worker 	ret = drm_intel_bo_map(gem.buf.bo, 0);
373*d83cc019SAndroid Build Coastguard Worker 	igt_assert_eq(ret, 0);
374*d83cc019SAndroid Build Coastguard Worker 
375*d83cc019SAndroid Build Coastguard Worker 	gem_check_spin(spins);
376*d83cc019SAndroid Build Coastguard Worker 	drm_intel_bo_unmap(gem.buf.bo);
377*d83cc019SAndroid Build Coastguard Worker 
378*d83cc019SAndroid Build Coastguard Worker 	check_full_enable(&stat);
379*d83cc019SAndroid Build Coastguard Worker }
380*d83cc019SAndroid Build Coastguard Worker 
381*d83cc019SAndroid Build Coastguard Worker static void
exit_handler(int sig)382*d83cc019SAndroid Build Coastguard Worker exit_handler(int sig)
383*d83cc019SAndroid Build Coastguard Worker {
384*d83cc019SAndroid Build Coastguard Worker 	dbg_deinit();
385*d83cc019SAndroid Build Coastguard Worker 	gem_deinit();
386*d83cc019SAndroid Build Coastguard Worker }
387*d83cc019SAndroid Build Coastguard Worker 
388*d83cc019SAndroid Build Coastguard Worker igt_main
389*d83cc019SAndroid Build Coastguard Worker {
390*d83cc019SAndroid Build Coastguard Worker 	igt_fixture {
391*d83cc019SAndroid Build Coastguard Worker 		igt_install_exit_handler(exit_handler);
392*d83cc019SAndroid Build Coastguard Worker 
393*d83cc019SAndroid Build Coastguard Worker 		gem_init();
394*d83cc019SAndroid Build Coastguard Worker 		dbg_init();
395*d83cc019SAndroid Build Coastguard Worker 	}
396*d83cc019SAndroid Build Coastguard Worker 
397*d83cc019SAndroid Build Coastguard Worker 	igt_subtest("full-enable")
398*d83cc019SAndroid Build Coastguard Worker 		full_enable();
399*d83cc019SAndroid Build Coastguard Worker }
400