xref: /aosp_15_r20/external/igt-gpu-tools/lib/igt_psr.c (revision d83cc019efdc2edc6c4b16e9034a3ceb8d35d77c)
1*d83cc019SAndroid Build Coastguard Worker /*
2*d83cc019SAndroid Build Coastguard Worker  * Copyright © 2018 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_psr.h"
25*d83cc019SAndroid Build Coastguard Worker #include "igt_sysfs.h"
26*d83cc019SAndroid Build Coastguard Worker #include <errno.h>
27*d83cc019SAndroid Build Coastguard Worker 
psr_active_check(int debugfs_fd,enum psr_mode mode)28*d83cc019SAndroid Build Coastguard Worker static bool psr_active_check(int debugfs_fd, enum psr_mode mode)
29*d83cc019SAndroid Build Coastguard Worker {
30*d83cc019SAndroid Build Coastguard Worker 	char buf[PSR_STATUS_MAX_LEN];
31*d83cc019SAndroid Build Coastguard Worker 	const char *state = mode == PSR_MODE_1 ? "SRDENT" : "DEEP_SLEEP";
32*d83cc019SAndroid Build Coastguard Worker 
33*d83cc019SAndroid Build Coastguard Worker 	igt_debugfs_simple_read(debugfs_fd, "i915_edp_psr_status", buf,
34*d83cc019SAndroid Build Coastguard Worker 				sizeof(buf));
35*d83cc019SAndroid Build Coastguard Worker 
36*d83cc019SAndroid Build Coastguard Worker 	return strstr(buf, state);
37*d83cc019SAndroid Build Coastguard Worker }
38*d83cc019SAndroid Build Coastguard Worker 
psr_active_state_get(enum psr_mode mode)39*d83cc019SAndroid Build Coastguard Worker static inline const char *psr_active_state_get(enum psr_mode mode)
40*d83cc019SAndroid Build Coastguard Worker {
41*d83cc019SAndroid Build Coastguard Worker 	return mode == PSR_MODE_1 ? "SRDENT" : "DEEP_SLEEP";
42*d83cc019SAndroid Build Coastguard Worker }
43*d83cc019SAndroid Build Coastguard Worker 
44*d83cc019SAndroid Build Coastguard Worker /*
45*d83cc019SAndroid Build Coastguard Worker  * For PSR1, we wait until PSR is active. We wait until DEEP_SLEEP for PSR2.
46*d83cc019SAndroid Build Coastguard Worker  */
psr_wait_entry(int debugfs_fd,enum psr_mode mode)47*d83cc019SAndroid Build Coastguard Worker bool psr_wait_entry(int debugfs_fd, enum psr_mode mode)
48*d83cc019SAndroid Build Coastguard Worker {
49*d83cc019SAndroid Build Coastguard Worker 	return igt_wait(psr_active_check(debugfs_fd, mode), 500, 20);
50*d83cc019SAndroid Build Coastguard Worker }
51*d83cc019SAndroid Build Coastguard Worker 
psr_wait_update(int debugfs_fd,enum psr_mode mode)52*d83cc019SAndroid Build Coastguard Worker bool psr_wait_update(int debugfs_fd, enum psr_mode mode)
53*d83cc019SAndroid Build Coastguard Worker {
54*d83cc019SAndroid Build Coastguard Worker 	return igt_wait(!psr_active_check(debugfs_fd, mode), 40, 10);
55*d83cc019SAndroid Build Coastguard Worker }
56*d83cc019SAndroid Build Coastguard Worker 
psr_long_wait_update(int debugfs_fd,enum psr_mode mode)57*d83cc019SAndroid Build Coastguard Worker bool psr_long_wait_update(int debugfs_fd, enum psr_mode mode)
58*d83cc019SAndroid Build Coastguard Worker {
59*d83cc019SAndroid Build Coastguard Worker 	return igt_wait(!psr_active_check(debugfs_fd, mode), 500, 10);
60*d83cc019SAndroid Build Coastguard Worker }
61*d83cc019SAndroid Build Coastguard Worker 
psr_write(int debugfs_fd,const char * buf)62*d83cc019SAndroid Build Coastguard Worker static ssize_t psr_write(int debugfs_fd, const char *buf)
63*d83cc019SAndroid Build Coastguard Worker {
64*d83cc019SAndroid Build Coastguard Worker 	return igt_sysfs_write(debugfs_fd, "i915_edp_psr_debug", buf,
65*d83cc019SAndroid Build Coastguard Worker 			       strlen(buf));
66*d83cc019SAndroid Build Coastguard Worker }
67*d83cc019SAndroid Build Coastguard Worker 
has_psr_debugfs(int debugfs_fd)68*d83cc019SAndroid Build Coastguard Worker static int has_psr_debugfs(int debugfs_fd)
69*d83cc019SAndroid Build Coastguard Worker {
70*d83cc019SAndroid Build Coastguard Worker 	int ret;
71*d83cc019SAndroid Build Coastguard Worker 
72*d83cc019SAndroid Build Coastguard Worker 	/*
73*d83cc019SAndroid Build Coastguard Worker 	 * Check if new PSR debugfs api is usable by writing an invalid value.
74*d83cc019SAndroid Build Coastguard Worker 	 * Legacy mode will return OK here, debugfs api will return -EINVAL.
75*d83cc019SAndroid Build Coastguard Worker 	 * -ENODEV is returned when PSR is unavailable.
76*d83cc019SAndroid Build Coastguard Worker 	 */
77*d83cc019SAndroid Build Coastguard Worker 	ret = psr_write(debugfs_fd, "0xf");
78*d83cc019SAndroid Build Coastguard Worker 	if (ret == -EINVAL)
79*d83cc019SAndroid Build Coastguard Worker 		return 0;
80*d83cc019SAndroid Build Coastguard Worker 	else if (ret < 0)
81*d83cc019SAndroid Build Coastguard Worker 		return ret;
82*d83cc019SAndroid Build Coastguard Worker 
83*d83cc019SAndroid Build Coastguard Worker 	/* legacy debugfs api, we enabled irqs by writing, disable them. */
84*d83cc019SAndroid Build Coastguard Worker 	psr_write(debugfs_fd, "0");
85*d83cc019SAndroid Build Coastguard Worker 	return -EINVAL;
86*d83cc019SAndroid Build Coastguard Worker }
87*d83cc019SAndroid Build Coastguard Worker 
psr_modparam_set(int val)88*d83cc019SAndroid Build Coastguard Worker static bool psr_modparam_set(int val)
89*d83cc019SAndroid Build Coastguard Worker {
90*d83cc019SAndroid Build Coastguard Worker 	static int oldval = -1;
91*d83cc019SAndroid Build Coastguard Worker 
92*d83cc019SAndroid Build Coastguard Worker 	igt_set_module_param_int("enable_psr", val);
93*d83cc019SAndroid Build Coastguard Worker 
94*d83cc019SAndroid Build Coastguard Worker 	if (val == oldval)
95*d83cc019SAndroid Build Coastguard Worker 		return false;
96*d83cc019SAndroid Build Coastguard Worker 
97*d83cc019SAndroid Build Coastguard Worker 	oldval = val;
98*d83cc019SAndroid Build Coastguard Worker 	return true;
99*d83cc019SAndroid Build Coastguard Worker }
100*d83cc019SAndroid Build Coastguard Worker 
101*d83cc019SAndroid Build Coastguard Worker static int psr_restore_debugfs_fd = -1;
102*d83cc019SAndroid Build Coastguard Worker 
restore_psr_debugfs(int sig)103*d83cc019SAndroid Build Coastguard Worker static void restore_psr_debugfs(int sig)
104*d83cc019SAndroid Build Coastguard Worker {
105*d83cc019SAndroid Build Coastguard Worker 	psr_write(psr_restore_debugfs_fd, "0");
106*d83cc019SAndroid Build Coastguard Worker }
107*d83cc019SAndroid Build Coastguard Worker 
psr_set(int debugfs_fd,int mode)108*d83cc019SAndroid Build Coastguard Worker static bool psr_set(int debugfs_fd, int mode)
109*d83cc019SAndroid Build Coastguard Worker {
110*d83cc019SAndroid Build Coastguard Worker 	int ret;
111*d83cc019SAndroid Build Coastguard Worker 
112*d83cc019SAndroid Build Coastguard Worker 	ret = has_psr_debugfs(debugfs_fd);
113*d83cc019SAndroid Build Coastguard Worker 	if (ret == -ENODEV) {
114*d83cc019SAndroid Build Coastguard Worker 		igt_skip("PSR not available\n");
115*d83cc019SAndroid Build Coastguard Worker 		return false;
116*d83cc019SAndroid Build Coastguard Worker 	}
117*d83cc019SAndroid Build Coastguard Worker 
118*d83cc019SAndroid Build Coastguard Worker 	if (ret == -EINVAL) {
119*d83cc019SAndroid Build Coastguard Worker 		/*
120*d83cc019SAndroid Build Coastguard Worker 		 * We can not control what PSR version is going to be enabled
121*d83cc019SAndroid Build Coastguard Worker 		 * by setting enable_psr parameter, when unmatched the PSR
122*d83cc019SAndroid Build Coastguard Worker 		 * version enabled and the PSR version of the test, it will
123*d83cc019SAndroid Build Coastguard Worker 		 * fail in the first psr_wait_entry() of the test.
124*d83cc019SAndroid Build Coastguard Worker 		 */
125*d83cc019SAndroid Build Coastguard Worker 		ret = psr_modparam_set(mode >= PSR_MODE_1);
126*d83cc019SAndroid Build Coastguard Worker 	} else {
127*d83cc019SAndroid Build Coastguard Worker 		const char *debug_val;
128*d83cc019SAndroid Build Coastguard Worker 
129*d83cc019SAndroid Build Coastguard Worker 		switch (mode) {
130*d83cc019SAndroid Build Coastguard Worker 		case PSR_MODE_1:
131*d83cc019SAndroid Build Coastguard Worker 			debug_val = "0x3";
132*d83cc019SAndroid Build Coastguard Worker 			break;
133*d83cc019SAndroid Build Coastguard Worker 		case PSR_MODE_2:
134*d83cc019SAndroid Build Coastguard Worker 			debug_val = "0x2";
135*d83cc019SAndroid Build Coastguard Worker 			break;
136*d83cc019SAndroid Build Coastguard Worker 		default:
137*d83cc019SAndroid Build Coastguard Worker 			/* Disables PSR */
138*d83cc019SAndroid Build Coastguard Worker 			debug_val = "0x1";
139*d83cc019SAndroid Build Coastguard Worker 		}
140*d83cc019SAndroid Build Coastguard Worker 
141*d83cc019SAndroid Build Coastguard Worker 		ret = psr_write(debugfs_fd, debug_val);
142*d83cc019SAndroid Build Coastguard Worker 		igt_assert(ret > 0);
143*d83cc019SAndroid Build Coastguard Worker 	}
144*d83cc019SAndroid Build Coastguard Worker 
145*d83cc019SAndroid Build Coastguard Worker 	/* Restore original value on exit */
146*d83cc019SAndroid Build Coastguard Worker 	if (psr_restore_debugfs_fd == -1) {
147*d83cc019SAndroid Build Coastguard Worker 		psr_restore_debugfs_fd = dup(debugfs_fd);
148*d83cc019SAndroid Build Coastguard Worker 		igt_assert(psr_restore_debugfs_fd >= 0);
149*d83cc019SAndroid Build Coastguard Worker 		igt_install_exit_handler(restore_psr_debugfs);
150*d83cc019SAndroid Build Coastguard Worker 	}
151*d83cc019SAndroid Build Coastguard Worker 
152*d83cc019SAndroid Build Coastguard Worker 	return ret;
153*d83cc019SAndroid Build Coastguard Worker }
154*d83cc019SAndroid Build Coastguard Worker 
psr_enable(int debugfs_fd,enum psr_mode mode)155*d83cc019SAndroid Build Coastguard Worker bool psr_enable(int debugfs_fd, enum psr_mode mode)
156*d83cc019SAndroid Build Coastguard Worker {
157*d83cc019SAndroid Build Coastguard Worker 	return psr_set(debugfs_fd, mode);
158*d83cc019SAndroid Build Coastguard Worker }
159*d83cc019SAndroid Build Coastguard Worker 
psr_disable(int debugfs_fd)160*d83cc019SAndroid Build Coastguard Worker bool psr_disable(int debugfs_fd)
161*d83cc019SAndroid Build Coastguard Worker {
162*d83cc019SAndroid Build Coastguard Worker 	/* Any mode different than PSR_MODE_1/2 will disable PSR */
163*d83cc019SAndroid Build Coastguard Worker 	return psr_set(debugfs_fd, -1);
164*d83cc019SAndroid Build Coastguard Worker }
165*d83cc019SAndroid Build Coastguard Worker 
psr_sink_support(int debugfs_fd,enum psr_mode mode)166*d83cc019SAndroid Build Coastguard Worker bool psr_sink_support(int debugfs_fd, enum psr_mode mode)
167*d83cc019SAndroid Build Coastguard Worker {
168*d83cc019SAndroid Build Coastguard Worker 	char buf[PSR_STATUS_MAX_LEN];
169*d83cc019SAndroid Build Coastguard Worker 	int ret;
170*d83cc019SAndroid Build Coastguard Worker 
171*d83cc019SAndroid Build Coastguard Worker 	ret = igt_debugfs_simple_read(debugfs_fd, "i915_edp_psr_status", buf,
172*d83cc019SAndroid Build Coastguard Worker 				      sizeof(buf));
173*d83cc019SAndroid Build Coastguard Worker 	if (ret < 1)
174*d83cc019SAndroid Build Coastguard Worker 		return false;
175*d83cc019SAndroid Build Coastguard Worker 
176*d83cc019SAndroid Build Coastguard Worker 	if (mode == PSR_MODE_1)
177*d83cc019SAndroid Build Coastguard Worker 		return strstr(buf, "Sink_Support: yes\n") ||
178*d83cc019SAndroid Build Coastguard Worker 		       strstr(buf, "Sink support: yes");
179*d83cc019SAndroid Build Coastguard Worker 	else
180*d83cc019SAndroid Build Coastguard Worker 		/*
181*d83cc019SAndroid Build Coastguard Worker 		 * i915 requires PSR version 0x03 that is PSR2 + SU with
182*d83cc019SAndroid Build Coastguard Worker 		 * Y-coordinate to support PSR2
183*d83cc019SAndroid Build Coastguard Worker 		 */
184*d83cc019SAndroid Build Coastguard Worker 		return strstr(buf, "Sink support: yes [0x03]");
185*d83cc019SAndroid Build Coastguard Worker }
186*d83cc019SAndroid Build Coastguard Worker 
187*d83cc019SAndroid Build Coastguard Worker #define PSR2_SU_BLOCK_STR_LOOKUP "PSR2 SU blocks:\n0\t"
188*d83cc019SAndroid Build Coastguard Worker 
189*d83cc019SAndroid Build Coastguard Worker static bool
psr2_read_last_num_su_blocks_val(int debugfs_fd,uint16_t * num_su_blocks)190*d83cc019SAndroid Build Coastguard Worker psr2_read_last_num_su_blocks_val(int debugfs_fd, uint16_t *num_su_blocks)
191*d83cc019SAndroid Build Coastguard Worker {
192*d83cc019SAndroid Build Coastguard Worker 	char buf[PSR_STATUS_MAX_LEN];
193*d83cc019SAndroid Build Coastguard Worker 	char *str;
194*d83cc019SAndroid Build Coastguard Worker 	int ret;
195*d83cc019SAndroid Build Coastguard Worker 
196*d83cc019SAndroid Build Coastguard Worker 	ret = igt_debugfs_simple_read(debugfs_fd, "i915_edp_psr_status", buf,
197*d83cc019SAndroid Build Coastguard Worker 				      sizeof(buf));
198*d83cc019SAndroid Build Coastguard Worker 	if (ret < 0)
199*d83cc019SAndroid Build Coastguard Worker 		return false;
200*d83cc019SAndroid Build Coastguard Worker 
201*d83cc019SAndroid Build Coastguard Worker 	str = strstr(buf, PSR2_SU_BLOCK_STR_LOOKUP);
202*d83cc019SAndroid Build Coastguard Worker 	if (!str)
203*d83cc019SAndroid Build Coastguard Worker 		return false;
204*d83cc019SAndroid Build Coastguard Worker 
205*d83cc019SAndroid Build Coastguard Worker 	str = &str[strlen(PSR2_SU_BLOCK_STR_LOOKUP)];
206*d83cc019SAndroid Build Coastguard Worker 	*num_su_blocks = (uint16_t)strtol(str, NULL, 10);
207*d83cc019SAndroid Build Coastguard Worker 
208*d83cc019SAndroid Build Coastguard Worker 	return true;
209*d83cc019SAndroid Build Coastguard Worker }
210*d83cc019SAndroid Build Coastguard Worker 
psr2_wait_su(int debugfs_fd,uint16_t * num_su_blocks)211*d83cc019SAndroid Build Coastguard Worker bool psr2_wait_su(int debugfs_fd, uint16_t *num_su_blocks)
212*d83cc019SAndroid Build Coastguard Worker {
213*d83cc019SAndroid Build Coastguard Worker 	return igt_wait(psr2_read_last_num_su_blocks_val(debugfs_fd, num_su_blocks), 40, 1);
214*d83cc019SAndroid Build Coastguard Worker }
215