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