2 * Copyright 2016 Advanced Micro Devices, Inc.
4 * Permission is hereby granted, free of charge, to any person obtaining a
5 * copy of this software and associated documentation files (the "Software"),
6 * to deal in the Software without restriction, including without limitation
7 * the rights to use, copy, modify, merge, publish, distribute, sublicense,
8 * and/or sell copies of the Software, and to permit persons to whom the
9 * Software is furnished to do so, subject to the following conditions:
11 * The above copyright notice and this permission notice shall be included in
12 * all copies or substantial portions of the Software.
14 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
15 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
16 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
17 * THE COPYRIGHT HOLDER(S) OR AUTHOR(S) BE LIABLE FOR ANY CLAIM, DAMAGES OR
18 * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
19 * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
20 * OTHER DEALINGS IN THE SOFTWARE.
26 #include "dm_services.h"
27 #include "dcn10_hubp.h"
28 #include "dcn10_hubbub.h"
29 #include "reg_helper.h"
34 hubbub1->base.ctx->logger
39 #define FN(reg_name, field_name) \
40 hubbub1->shifts->field_name, hubbub1->masks->field_name
42 void hubbub1_wm_read_state(struct hubbub
*hubbub
,
43 struct dcn_hubbub_wm
*wm
)
45 struct dcn10_hubbub
*hubbub1
= TO_DCN10_HUBBUB(hubbub
);
46 struct dcn_hubbub_wm_set
*s
;
48 memset(wm
, 0, sizeof(struct dcn_hubbub_wm
));
52 s
->data_urgent
= REG_READ(DCHUBBUB_ARB_DATA_URGENCY_WATERMARK_A
);
53 s
->pte_meta_urgent
= REG_READ(DCHUBBUB_ARB_PTE_META_URGENCY_WATERMARK_A
);
54 if (REG(DCHUBBUB_ARB_ALLOW_SR_ENTER_WATERMARK_A
)) {
55 s
->sr_enter
= REG_READ(DCHUBBUB_ARB_ALLOW_SR_ENTER_WATERMARK_A
);
56 s
->sr_exit
= REG_READ(DCHUBBUB_ARB_ALLOW_SR_EXIT_WATERMARK_A
);
58 s
->dram_clk_chanage
= REG_READ(DCHUBBUB_ARB_ALLOW_DRAM_CLK_CHANGE_WATERMARK_A
);
62 s
->data_urgent
= REG_READ(DCHUBBUB_ARB_DATA_URGENCY_WATERMARK_B
);
63 s
->pte_meta_urgent
= REG_READ(DCHUBBUB_ARB_PTE_META_URGENCY_WATERMARK_B
);
64 if (REG(DCHUBBUB_ARB_ALLOW_SR_ENTER_WATERMARK_B
)) {
65 s
->sr_enter
= REG_READ(DCHUBBUB_ARB_ALLOW_SR_ENTER_WATERMARK_B
);
66 s
->sr_exit
= REG_READ(DCHUBBUB_ARB_ALLOW_SR_EXIT_WATERMARK_B
);
68 s
->dram_clk_chanage
= REG_READ(DCHUBBUB_ARB_ALLOW_DRAM_CLK_CHANGE_WATERMARK_B
);
72 s
->data_urgent
= REG_READ(DCHUBBUB_ARB_DATA_URGENCY_WATERMARK_C
);
73 s
->pte_meta_urgent
= REG_READ(DCHUBBUB_ARB_PTE_META_URGENCY_WATERMARK_C
);
74 if (REG(DCHUBBUB_ARB_ALLOW_SR_ENTER_WATERMARK_C
)) {
75 s
->sr_enter
= REG_READ(DCHUBBUB_ARB_ALLOW_SR_ENTER_WATERMARK_C
);
76 s
->sr_exit
= REG_READ(DCHUBBUB_ARB_ALLOW_SR_EXIT_WATERMARK_C
);
78 s
->dram_clk_chanage
= REG_READ(DCHUBBUB_ARB_ALLOW_DRAM_CLK_CHANGE_WATERMARK_C
);
82 s
->data_urgent
= REG_READ(DCHUBBUB_ARB_DATA_URGENCY_WATERMARK_D
);
83 s
->pte_meta_urgent
= REG_READ(DCHUBBUB_ARB_PTE_META_URGENCY_WATERMARK_D
);
84 if (REG(DCHUBBUB_ARB_ALLOW_SR_ENTER_WATERMARK_D
)) {
85 s
->sr_enter
= REG_READ(DCHUBBUB_ARB_ALLOW_SR_ENTER_WATERMARK_D
);
86 s
->sr_exit
= REG_READ(DCHUBBUB_ARB_ALLOW_SR_EXIT_WATERMARK_D
);
88 s
->dram_clk_chanage
= REG_READ(DCHUBBUB_ARB_ALLOW_DRAM_CLK_CHANGE_WATERMARK_D
);
91 void hubbub1_disable_allow_self_refresh(struct hubbub
*hubbub
)
93 struct dcn10_hubbub
*hubbub1
= TO_DCN10_HUBBUB(hubbub
);
94 REG_UPDATE(DCHUBBUB_ARB_DRAM_STATE_CNTL
,
95 DCHUBBUB_ARB_ALLOW_SELF_REFRESH_FORCE_ENABLE
, 0);
98 bool hububu1_is_allow_self_refresh_enabled(struct hubbub
*hubbub
)
100 struct dcn10_hubbub
*hubbub1
= TO_DCN10_HUBBUB(hubbub
);
103 REG_GET(DCHUBBUB_ARB_DRAM_STATE_CNTL
,
104 DCHUBBUB_ARB_ALLOW_SELF_REFRESH_FORCE_ENABLE
, &enable
);
106 return enable
? true : false;
110 bool hubbub1_verify_allow_pstate_change_high(
111 struct hubbub
*hubbub
)
113 struct dcn10_hubbub
*hubbub1
= TO_DCN10_HUBBUB(hubbub
);
115 /* pstate latency is ~20us so if we wait over 40us and pstate allow
116 * still not asserted, we are probably stuck and going to hang
118 * TODO: Figure out why it takes ~100us on linux
119 * pstate takes around ~100us on linux. Unknown currently as to
120 * why it takes that long on linux
122 static unsigned int pstate_wait_timeout_us
= 200;
123 static unsigned int pstate_wait_expected_timeout_us
= 40;
124 static unsigned int max_sampled_pstate_wait_us
; /* data collection */
125 static bool forced_pstate_allow
; /* help with revert wa */
127 unsigned int debug_data
;
130 if (forced_pstate_allow
) {
131 /* we hacked to force pstate allow to prevent hang last time
132 * we verify_allow_pstate_change_high. so disable force
133 * here so we can check status
135 REG_UPDATE_2(DCHUBBUB_ARB_DRAM_STATE_CNTL
,
136 DCHUBBUB_ARB_ALLOW_PSTATE_CHANGE_FORCE_VALUE
, 0,
137 DCHUBBUB_ARB_ALLOW_PSTATE_CHANGE_FORCE_ENABLE
, 0);
138 forced_pstate_allow
= false;
142 * dchubbubdebugind, at: 0xB
144 * 0: Pipe0 Plane0 Allow Pstate Change
145 * 1: Pipe0 Plane1 Allow Pstate Change
146 * 2: Pipe0 Cursor0 Allow Pstate Change
147 * 3: Pipe0 Cursor1 Allow Pstate Change
148 * 4: Pipe1 Plane0 Allow Pstate Change
149 * 5: Pipe1 Plane1 Allow Pstate Change
150 * 6: Pipe1 Cursor0 Allow Pstate Change
151 * 7: Pipe1 Cursor1 Allow Pstate Change
152 * 8: Pipe2 Plane0 Allow Pstate Change
153 * 9: Pipe2 Plane1 Allow Pstate Change
154 * 10: Pipe2 Cursor0 Allow Pstate Change
155 * 11: Pipe2 Cursor1 Allow Pstate Change
156 * 12: Pipe3 Plane0 Allow Pstate Change
157 * 13: Pipe3 Plane1 Allow Pstate Change
158 * 14: Pipe3 Cursor0 Allow Pstate Change
159 * 15: Pipe3 Cursor1 Allow Pstate Change
160 * 16: Pipe4 Plane0 Allow Pstate Change
161 * 17: Pipe4 Plane1 Allow Pstate Change
162 * 18: Pipe4 Cursor0 Allow Pstate Change
163 * 19: Pipe4 Cursor1 Allow Pstate Change
164 * 20: Pipe5 Plane0 Allow Pstate Change
165 * 21: Pipe5 Plane1 Allow Pstate Change
166 * 22: Pipe5 Cursor0 Allow Pstate Change
167 * 23: Pipe5 Cursor1 Allow Pstate Change
168 * 24: Pipe6 Plane0 Allow Pstate Change
169 * 25: Pipe6 Plane1 Allow Pstate Change
170 * 26: Pipe6 Cursor0 Allow Pstate Change
171 * 27: Pipe6 Cursor1 Allow Pstate Change
172 * 28: WB0 Allow Pstate Change
173 * 29: WB1 Allow Pstate Change
174 * 30: Arbiter's allow_pstate_change
175 * 31: SOC pstate change request"
178 * dchubbubdebugind, at: 0x7
179 * description "3-0: Pipe0 cursor0 QOS
180 * 7-4: Pipe1 cursor0 QOS
181 * 11-8: Pipe2 cursor0 QOS
182 * 15-12: Pipe3 cursor0 QOS
183 * 16: Pipe0 Plane0 Allow Pstate Change
184 * 17: Pipe1 Plane0 Allow Pstate Change
185 * 18: Pipe2 Plane0 Allow Pstate Change
186 * 19: Pipe3 Plane0 Allow Pstate Change
187 * 20: Pipe0 Plane1 Allow Pstate Change
188 * 21: Pipe1 Plane1 Allow Pstate Change
189 * 22: Pipe2 Plane1 Allow Pstate Change
190 * 23: Pipe3 Plane1 Allow Pstate Change
191 * 24: Pipe0 cursor0 Allow Pstate Change
192 * 25: Pipe1 cursor0 Allow Pstate Change
193 * 26: Pipe2 cursor0 Allow Pstate Change
194 * 27: Pipe3 cursor0 Allow Pstate Change
195 * 28: WB0 Allow Pstate Change
196 * 29: WB1 Allow Pstate Change
197 * 30: Arbiter's allow_pstate_change
198 * 31: SOC pstate change request
201 REG_WRITE(DCHUBBUB_TEST_DEBUG_INDEX
, hubbub1
->debug_test_index_pstate
);
203 for (i
= 0; i
< pstate_wait_timeout_us
; i
++) {
204 debug_data
= REG_READ(DCHUBBUB_TEST_DEBUG_DATA
);
206 if (debug_data
& (1 << 30)) {
208 if (i
> pstate_wait_expected_timeout_us
)
209 DC_LOG_WARNING("pstate took longer than expected ~%dus\n",
214 if (max_sampled_pstate_wait_us
< i
)
215 max_sampled_pstate_wait_us
= i
;
220 /* force pstate allow to prevent system hang
221 * and break to debugger to investigate
223 REG_UPDATE_2(DCHUBBUB_ARB_DRAM_STATE_CNTL
,
224 DCHUBBUB_ARB_ALLOW_PSTATE_CHANGE_FORCE_VALUE
, 1,
225 DCHUBBUB_ARB_ALLOW_PSTATE_CHANGE_FORCE_ENABLE
, 1);
226 forced_pstate_allow
= true;
228 DC_LOG_WARNING("pstate TEST_DEBUG_DATA: 0x%X\n",
234 static uint32_t convert_and_clamp(
237 uint32_t clamp_value
)
239 uint32_t ret_val
= 0;
240 ret_val
= wm_ns
* refclk_mhz
;
243 if (ret_val
> clamp_value
)
244 ret_val
= clamp_value
;
250 void hubbub1_wm_change_req_wa(struct hubbub
*hubbub
)
252 struct dcn10_hubbub
*hubbub1
= TO_DCN10_HUBBUB(hubbub
);
254 REG_UPDATE_SEQ(DCHUBBUB_ARB_WATERMARK_CHANGE_CNTL
,
255 DCHUBBUB_ARB_WATERMARK_CHANGE_REQUEST
, 0, 1);
258 void hubbub1_program_watermarks(
259 struct hubbub
*hubbub
,
260 struct dcn_watermark_set
*watermarks
,
261 unsigned int refclk_mhz
,
264 struct dcn10_hubbub
*hubbub1
= TO_DCN10_HUBBUB(hubbub
);
266 uint32_t force_en
= hubbub1
->base
.ctx
->dc
->debug
.disable_stutter
? 1 : 0;
268 * Need to clamp to max of the register values (i.e. no wrap)
269 * for dcn1, all wm registers are 21-bit wide
271 uint32_t prog_wm_value
;
274 /* Repeat for water mark set A, B, C and D. */
276 if (safe_to_lower
|| watermarks
->a
.urgent_ns
> hubbub1
->watermarks
.a
.urgent_ns
) {
277 hubbub1
->watermarks
.a
.urgent_ns
= watermarks
->a
.urgent_ns
;
278 prog_wm_value
= convert_and_clamp(watermarks
->a
.urgent_ns
,
279 refclk_mhz
, 0x1fffff);
280 REG_WRITE(DCHUBBUB_ARB_DATA_URGENCY_WATERMARK_A
, prog_wm_value
);
282 DC_LOG_BANDWIDTH_CALCS("URGENCY_WATERMARK_A calculated =%d\n"
283 "HW register value = 0x%x\n",
284 watermarks
->a
.urgent_ns
, prog_wm_value
);
287 if (REG(DCHUBBUB_ARB_PTE_META_URGENCY_WATERMARK_A
)) {
288 if (safe_to_lower
|| watermarks
->a
.pte_meta_urgent_ns
> hubbub1
->watermarks
.a
.pte_meta_urgent_ns
) {
289 hubbub1
->watermarks
.a
.pte_meta_urgent_ns
= watermarks
->a
.pte_meta_urgent_ns
;
290 prog_wm_value
= convert_and_clamp(watermarks
->a
.pte_meta_urgent_ns
,
291 refclk_mhz
, 0x1fffff);
292 REG_WRITE(DCHUBBUB_ARB_PTE_META_URGENCY_WATERMARK_A
, prog_wm_value
);
293 DC_LOG_BANDWIDTH_CALCS("PTE_META_URGENCY_WATERMARK_A calculated =%d\n"
294 "HW register value = 0x%x\n",
295 watermarks
->a
.pte_meta_urgent_ns
, prog_wm_value
);
299 if (REG(DCHUBBUB_ARB_ALLOW_SR_ENTER_WATERMARK_A
)) {
300 if (safe_to_lower
|| watermarks
->a
.cstate_pstate
.cstate_enter_plus_exit_ns
301 > hubbub1
->watermarks
.a
.cstate_pstate
.cstate_enter_plus_exit_ns
) {
302 hubbub1
->watermarks
.a
.cstate_pstate
.cstate_enter_plus_exit_ns
=
303 watermarks
->a
.cstate_pstate
.cstate_enter_plus_exit_ns
;
304 prog_wm_value
= convert_and_clamp(
305 watermarks
->a
.cstate_pstate
.cstate_enter_plus_exit_ns
,
306 refclk_mhz
, 0x1fffff);
307 REG_WRITE(DCHUBBUB_ARB_ALLOW_SR_ENTER_WATERMARK_A
, prog_wm_value
);
308 DC_LOG_BANDWIDTH_CALCS("SR_ENTER_EXIT_WATERMARK_A calculated =%d\n"
309 "HW register value = 0x%x\n",
310 watermarks
->a
.cstate_pstate
.cstate_enter_plus_exit_ns
, prog_wm_value
);
313 if (safe_to_lower
|| watermarks
->a
.cstate_pstate
.cstate_exit_ns
314 > hubbub1
->watermarks
.a
.cstate_pstate
.cstate_exit_ns
) {
315 hubbub1
->watermarks
.a
.cstate_pstate
.cstate_exit_ns
=
316 watermarks
->a
.cstate_pstate
.cstate_exit_ns
;
317 prog_wm_value
= convert_and_clamp(
318 watermarks
->a
.cstate_pstate
.cstate_exit_ns
,
319 refclk_mhz
, 0x1fffff);
320 REG_WRITE(DCHUBBUB_ARB_ALLOW_SR_EXIT_WATERMARK_A
, prog_wm_value
);
321 DC_LOG_BANDWIDTH_CALCS("SR_EXIT_WATERMARK_A calculated =%d\n"
322 "HW register value = 0x%x\n",
323 watermarks
->a
.cstate_pstate
.cstate_exit_ns
, prog_wm_value
);
327 if (safe_to_lower
|| watermarks
->a
.cstate_pstate
.pstate_change_ns
328 > hubbub1
->watermarks
.a
.cstate_pstate
.pstate_change_ns
) {
329 hubbub1
->watermarks
.a
.cstate_pstate
.pstate_change_ns
=
330 watermarks
->a
.cstate_pstate
.pstate_change_ns
;
331 prog_wm_value
= convert_and_clamp(
332 watermarks
->a
.cstate_pstate
.pstate_change_ns
,
333 refclk_mhz
, 0x1fffff);
334 REG_WRITE(DCHUBBUB_ARB_ALLOW_DRAM_CLK_CHANGE_WATERMARK_A
, prog_wm_value
);
335 DC_LOG_BANDWIDTH_CALCS("DRAM_CLK_CHANGE_WATERMARK_A calculated =%d\n"
336 "HW register value = 0x%x\n\n",
337 watermarks
->a
.cstate_pstate
.pstate_change_ns
, prog_wm_value
);
341 if (safe_to_lower
|| watermarks
->b
.urgent_ns
> hubbub1
->watermarks
.b
.urgent_ns
) {
342 hubbub1
->watermarks
.b
.urgent_ns
= watermarks
->b
.urgent_ns
;
343 prog_wm_value
= convert_and_clamp(watermarks
->b
.urgent_ns
,
344 refclk_mhz
, 0x1fffff);
345 REG_WRITE(DCHUBBUB_ARB_DATA_URGENCY_WATERMARK_B
, prog_wm_value
);
347 DC_LOG_BANDWIDTH_CALCS("URGENCY_WATERMARK_B calculated =%d\n"
348 "HW register value = 0x%x\n",
349 watermarks
->b
.urgent_ns
, prog_wm_value
);
352 if (REG(DCHUBBUB_ARB_PTE_META_URGENCY_WATERMARK_B
)) {
353 if (safe_to_lower
|| watermarks
->b
.pte_meta_urgent_ns
> hubbub1
->watermarks
.b
.pte_meta_urgent_ns
) {
354 hubbub1
->watermarks
.b
.pte_meta_urgent_ns
= watermarks
->b
.pte_meta_urgent_ns
;
355 prog_wm_value
= convert_and_clamp(watermarks
->b
.pte_meta_urgent_ns
,
356 refclk_mhz
, 0x1fffff);
357 REG_WRITE(DCHUBBUB_ARB_PTE_META_URGENCY_WATERMARK_B
, prog_wm_value
);
358 DC_LOG_BANDWIDTH_CALCS("PTE_META_URGENCY_WATERMARK_B calculated =%d\n"
359 "HW register value = 0x%x\n",
360 watermarks
->b
.pte_meta_urgent_ns
, prog_wm_value
);
364 if (REG(DCHUBBUB_ARB_ALLOW_SR_ENTER_WATERMARK_B
)) {
365 if (safe_to_lower
|| watermarks
->b
.cstate_pstate
.cstate_enter_plus_exit_ns
366 > hubbub1
->watermarks
.b
.cstate_pstate
.cstate_enter_plus_exit_ns
) {
367 hubbub1
->watermarks
.b
.cstate_pstate
.cstate_enter_plus_exit_ns
=
368 watermarks
->b
.cstate_pstate
.cstate_enter_plus_exit_ns
;
369 prog_wm_value
= convert_and_clamp(
370 watermarks
->b
.cstate_pstate
.cstate_enter_plus_exit_ns
,
371 refclk_mhz
, 0x1fffff);
372 REG_WRITE(DCHUBBUB_ARB_ALLOW_SR_ENTER_WATERMARK_B
, prog_wm_value
);
373 DC_LOG_BANDWIDTH_CALCS("SR_ENTER_EXIT_WATERMARK_B calculated =%d\n"
374 "HW register value = 0x%x\n",
375 watermarks
->b
.cstate_pstate
.cstate_enter_plus_exit_ns
, prog_wm_value
);
378 if (safe_to_lower
|| watermarks
->b
.cstate_pstate
.cstate_exit_ns
379 > hubbub1
->watermarks
.b
.cstate_pstate
.cstate_exit_ns
) {
380 hubbub1
->watermarks
.b
.cstate_pstate
.cstate_exit_ns
=
381 watermarks
->b
.cstate_pstate
.cstate_exit_ns
;
382 prog_wm_value
= convert_and_clamp(
383 watermarks
->b
.cstate_pstate
.cstate_exit_ns
,
384 refclk_mhz
, 0x1fffff);
385 REG_WRITE(DCHUBBUB_ARB_ALLOW_SR_EXIT_WATERMARK_B
, prog_wm_value
);
386 DC_LOG_BANDWIDTH_CALCS("SR_EXIT_WATERMARK_B calculated =%d\n"
387 "HW register value = 0x%x\n",
388 watermarks
->b
.cstate_pstate
.cstate_exit_ns
, prog_wm_value
);
392 if (safe_to_lower
|| watermarks
->b
.cstate_pstate
.pstate_change_ns
393 > hubbub1
->watermarks
.b
.cstate_pstate
.pstate_change_ns
) {
394 hubbub1
->watermarks
.b
.cstate_pstate
.pstate_change_ns
=
395 watermarks
->b
.cstate_pstate
.pstate_change_ns
;
396 prog_wm_value
= convert_and_clamp(
397 watermarks
->b
.cstate_pstate
.pstate_change_ns
,
398 refclk_mhz
, 0x1fffff);
399 REG_WRITE(DCHUBBUB_ARB_ALLOW_DRAM_CLK_CHANGE_WATERMARK_B
, prog_wm_value
);
400 DC_LOG_BANDWIDTH_CALCS("DRAM_CLK_CHANGE_WATERMARK_B calculated =%d\n"
401 "HW register value = 0x%x\n\n",
402 watermarks
->b
.cstate_pstate
.pstate_change_ns
, prog_wm_value
);
406 if (safe_to_lower
|| watermarks
->c
.urgent_ns
> hubbub1
->watermarks
.c
.urgent_ns
) {
407 hubbub1
->watermarks
.c
.urgent_ns
= watermarks
->c
.urgent_ns
;
408 prog_wm_value
= convert_and_clamp(watermarks
->c
.urgent_ns
,
409 refclk_mhz
, 0x1fffff);
410 REG_WRITE(DCHUBBUB_ARB_DATA_URGENCY_WATERMARK_C
, prog_wm_value
);
412 DC_LOG_BANDWIDTH_CALCS("URGENCY_WATERMARK_C calculated =%d\n"
413 "HW register value = 0x%x\n",
414 watermarks
->c
.urgent_ns
, prog_wm_value
);
417 if (REG(DCHUBBUB_ARB_PTE_META_URGENCY_WATERMARK_C
)) {
418 if (safe_to_lower
|| watermarks
->c
.pte_meta_urgent_ns
> hubbub1
->watermarks
.c
.pte_meta_urgent_ns
) {
419 hubbub1
->watermarks
.c
.pte_meta_urgent_ns
= watermarks
->c
.pte_meta_urgent_ns
;
420 prog_wm_value
= convert_and_clamp(watermarks
->c
.pte_meta_urgent_ns
,
421 refclk_mhz
, 0x1fffff);
422 REG_WRITE(DCHUBBUB_ARB_PTE_META_URGENCY_WATERMARK_C
, prog_wm_value
);
423 DC_LOG_BANDWIDTH_CALCS("PTE_META_URGENCY_WATERMARK_C calculated =%d\n"
424 "HW register value = 0x%x\n",
425 watermarks
->c
.pte_meta_urgent_ns
, prog_wm_value
);
429 if (REG(DCHUBBUB_ARB_ALLOW_SR_ENTER_WATERMARK_C
)) {
430 if (safe_to_lower
|| watermarks
->c
.cstate_pstate
.cstate_enter_plus_exit_ns
431 > hubbub1
->watermarks
.c
.cstate_pstate
.cstate_enter_plus_exit_ns
) {
432 hubbub1
->watermarks
.c
.cstate_pstate
.cstate_enter_plus_exit_ns
=
433 watermarks
->c
.cstate_pstate
.cstate_enter_plus_exit_ns
;
434 prog_wm_value
= convert_and_clamp(
435 watermarks
->c
.cstate_pstate
.cstate_enter_plus_exit_ns
,
436 refclk_mhz
, 0x1fffff);
437 REG_WRITE(DCHUBBUB_ARB_ALLOW_SR_ENTER_WATERMARK_C
, prog_wm_value
);
438 DC_LOG_BANDWIDTH_CALCS("SR_ENTER_EXIT_WATERMARK_C calculated =%d\n"
439 "HW register value = 0x%x\n",
440 watermarks
->c
.cstate_pstate
.cstate_enter_plus_exit_ns
, prog_wm_value
);
443 if (safe_to_lower
|| watermarks
->c
.cstate_pstate
.cstate_exit_ns
444 > hubbub1
->watermarks
.c
.cstate_pstate
.cstate_exit_ns
) {
445 hubbub1
->watermarks
.c
.cstate_pstate
.cstate_exit_ns
=
446 watermarks
->c
.cstate_pstate
.cstate_exit_ns
;
447 prog_wm_value
= convert_and_clamp(
448 watermarks
->c
.cstate_pstate
.cstate_exit_ns
,
449 refclk_mhz
, 0x1fffff);
450 REG_WRITE(DCHUBBUB_ARB_ALLOW_SR_EXIT_WATERMARK_C
, prog_wm_value
);
451 DC_LOG_BANDWIDTH_CALCS("SR_EXIT_WATERMARK_C calculated =%d\n"
452 "HW register value = 0x%x\n",
453 watermarks
->c
.cstate_pstate
.cstate_exit_ns
, prog_wm_value
);
457 if (safe_to_lower
|| watermarks
->c
.cstate_pstate
.pstate_change_ns
458 > hubbub1
->watermarks
.c
.cstate_pstate
.pstate_change_ns
) {
459 hubbub1
->watermarks
.c
.cstate_pstate
.pstate_change_ns
=
460 watermarks
->c
.cstate_pstate
.pstate_change_ns
;
461 prog_wm_value
= convert_and_clamp(
462 watermarks
->c
.cstate_pstate
.pstate_change_ns
,
463 refclk_mhz
, 0x1fffff);
464 REG_WRITE(DCHUBBUB_ARB_ALLOW_DRAM_CLK_CHANGE_WATERMARK_C
, prog_wm_value
);
465 DC_LOG_BANDWIDTH_CALCS("DRAM_CLK_CHANGE_WATERMARK_C calculated =%d\n"
466 "HW register value = 0x%x\n\n",
467 watermarks
->c
.cstate_pstate
.pstate_change_ns
, prog_wm_value
);
471 if (safe_to_lower
|| watermarks
->d
.urgent_ns
> hubbub1
->watermarks
.d
.urgent_ns
) {
472 hubbub1
->watermarks
.d
.urgent_ns
= watermarks
->d
.urgent_ns
;
473 prog_wm_value
= convert_and_clamp(watermarks
->d
.urgent_ns
,
474 refclk_mhz
, 0x1fffff);
475 REG_WRITE(DCHUBBUB_ARB_DATA_URGENCY_WATERMARK_D
, prog_wm_value
);
477 DC_LOG_BANDWIDTH_CALCS("URGENCY_WATERMARK_D calculated =%d\n"
478 "HW register value = 0x%x\n",
479 watermarks
->d
.urgent_ns
, prog_wm_value
);
482 if (REG(DCHUBBUB_ARB_PTE_META_URGENCY_WATERMARK_D
)) {
483 if (safe_to_lower
|| watermarks
->d
.pte_meta_urgent_ns
> hubbub1
->watermarks
.d
.pte_meta_urgent_ns
) {
484 hubbub1
->watermarks
.d
.pte_meta_urgent_ns
= watermarks
->d
.pte_meta_urgent_ns
;
485 prog_wm_value
= convert_and_clamp(watermarks
->d
.pte_meta_urgent_ns
,
486 refclk_mhz
, 0x1fffff);
487 REG_WRITE(DCHUBBUB_ARB_PTE_META_URGENCY_WATERMARK_D
, prog_wm_value
);
488 DC_LOG_BANDWIDTH_CALCS("PTE_META_URGENCY_WATERMARK_D calculated =%d\n"
489 "HW register value = 0x%x\n",
490 watermarks
->d
.pte_meta_urgent_ns
, prog_wm_value
);
494 if (REG(DCHUBBUB_ARB_ALLOW_SR_ENTER_WATERMARK_D
)) {
495 if (safe_to_lower
|| watermarks
->d
.cstate_pstate
.cstate_enter_plus_exit_ns
496 > hubbub1
->watermarks
.d
.cstate_pstate
.cstate_enter_plus_exit_ns
) {
497 hubbub1
->watermarks
.d
.cstate_pstate
.cstate_enter_plus_exit_ns
=
498 watermarks
->d
.cstate_pstate
.cstate_enter_plus_exit_ns
;
499 prog_wm_value
= convert_and_clamp(
500 watermarks
->d
.cstate_pstate
.cstate_enter_plus_exit_ns
,
501 refclk_mhz
, 0x1fffff);
502 REG_WRITE(DCHUBBUB_ARB_ALLOW_SR_ENTER_WATERMARK_D
, prog_wm_value
);
503 DC_LOG_BANDWIDTH_CALCS("SR_ENTER_EXIT_WATERMARK_D calculated =%d\n"
504 "HW register value = 0x%x\n",
505 watermarks
->d
.cstate_pstate
.cstate_enter_plus_exit_ns
, prog_wm_value
);
508 if (safe_to_lower
|| watermarks
->d
.cstate_pstate
.cstate_exit_ns
509 > hubbub1
->watermarks
.d
.cstate_pstate
.cstate_exit_ns
) {
510 hubbub1
->watermarks
.d
.cstate_pstate
.cstate_exit_ns
=
511 watermarks
->d
.cstate_pstate
.cstate_exit_ns
;
512 prog_wm_value
= convert_and_clamp(
513 watermarks
->d
.cstate_pstate
.cstate_exit_ns
,
514 refclk_mhz
, 0x1fffff);
515 REG_WRITE(DCHUBBUB_ARB_ALLOW_SR_EXIT_WATERMARK_D
, prog_wm_value
);
516 DC_LOG_BANDWIDTH_CALCS("SR_EXIT_WATERMARK_D calculated =%d\n"
517 "HW register value = 0x%x\n",
518 watermarks
->d
.cstate_pstate
.cstate_exit_ns
, prog_wm_value
);
522 if (safe_to_lower
|| watermarks
->d
.cstate_pstate
.pstate_change_ns
523 > hubbub1
->watermarks
.d
.cstate_pstate
.pstate_change_ns
) {
524 hubbub1
->watermarks
.d
.cstate_pstate
.pstate_change_ns
=
525 watermarks
->d
.cstate_pstate
.pstate_change_ns
;
526 prog_wm_value
= convert_and_clamp(
527 watermarks
->d
.cstate_pstate
.pstate_change_ns
,
528 refclk_mhz
, 0x1fffff);
529 REG_WRITE(DCHUBBUB_ARB_ALLOW_DRAM_CLK_CHANGE_WATERMARK_D
, prog_wm_value
);
530 DC_LOG_BANDWIDTH_CALCS("DRAM_CLK_CHANGE_WATERMARK_D calculated =%d\n"
531 "HW register value = 0x%x\n\n",
532 watermarks
->d
.cstate_pstate
.pstate_change_ns
, prog_wm_value
);
535 REG_UPDATE(DCHUBBUB_ARB_SAT_LEVEL
,
536 DCHUBBUB_ARB_SAT_LEVEL
, 60 * refclk_mhz
);
537 REG_UPDATE(DCHUBBUB_ARB_DF_REQ_OUTSTAND
,
538 DCHUBBUB_ARB_MIN_REQ_OUTSTAND
, 68);
540 REG_UPDATE_2(DCHUBBUB_ARB_DRAM_STATE_CNTL
,
541 DCHUBBUB_ARB_ALLOW_SELF_REFRESH_FORCE_VALUE
, 0,
542 DCHUBBUB_ARB_ALLOW_SELF_REFRESH_FORCE_ENABLE
, force_en
);
545 REG_UPDATE_2(DCHUBBUB_ARB_WATERMARK_CHANGE_CNTL
,
546 DCHUBBUB_ARB_WATERMARK_CHANGE_DONE_INTERRUPT_DISABLE
, 1,
547 DCHUBBUB_ARB_WATERMARK_CHANGE_REQUEST
, 1);
551 void hubbub1_update_dchub(
552 struct hubbub
*hubbub
,
553 struct dchub_init_data
*dh_data
)
555 struct dcn10_hubbub
*hubbub1
= TO_DCN10_HUBBUB(hubbub
);
557 if (REG(DCHUBBUB_SDPIF_FB_TOP
) == 0) {
559 /*should not come here*/
562 /* TODO: port code from dal2 */
563 switch (dh_data
->fb_mode
) {
564 case FRAME_BUFFER_MODE_ZFB_ONLY
:
565 /*For ZFB case need to put DCHUB FB BASE and TOP upside down to indicate ZFB mode*/
566 REG_UPDATE(DCHUBBUB_SDPIF_FB_TOP
,
569 REG_UPDATE(DCHUBBUB_SDPIF_FB_BASE
,
570 SDPIF_FB_BASE
, 0x0FFFF);
572 REG_UPDATE(DCHUBBUB_SDPIF_AGP_BASE
,
573 SDPIF_AGP_BASE
, dh_data
->zfb_phys_addr_base
>> 22);
575 REG_UPDATE(DCHUBBUB_SDPIF_AGP_BOT
,
576 SDPIF_AGP_BOT
, dh_data
->zfb_mc_base_addr
>> 22);
578 REG_UPDATE(DCHUBBUB_SDPIF_AGP_TOP
,
579 SDPIF_AGP_TOP
, (dh_data
->zfb_mc_base_addr
+
580 dh_data
->zfb_size_in_byte
- 1) >> 22);
582 case FRAME_BUFFER_MODE_MIXED_ZFB_AND_LOCAL
:
583 /*Should not touch FB LOCATION (done by VBIOS on AsicInit table)*/
585 REG_UPDATE(DCHUBBUB_SDPIF_AGP_BASE
,
586 SDPIF_AGP_BASE
, dh_data
->zfb_phys_addr_base
>> 22);
588 REG_UPDATE(DCHUBBUB_SDPIF_AGP_BOT
,
589 SDPIF_AGP_BOT
, dh_data
->zfb_mc_base_addr
>> 22);
591 REG_UPDATE(DCHUBBUB_SDPIF_AGP_TOP
,
592 SDPIF_AGP_TOP
, (dh_data
->zfb_mc_base_addr
+
593 dh_data
->zfb_size_in_byte
- 1) >> 22);
595 case FRAME_BUFFER_MODE_LOCAL_ONLY
:
596 /*Should not touch FB LOCATION (done by VBIOS on AsicInit table)*/
597 REG_UPDATE(DCHUBBUB_SDPIF_AGP_BASE
,
600 REG_UPDATE(DCHUBBUB_SDPIF_AGP_BOT
,
601 SDPIF_AGP_BOT
, 0X03FFFF);
603 REG_UPDATE(DCHUBBUB_SDPIF_AGP_TOP
,
610 dh_data
->dchub_initialzied
= true;
611 dh_data
->dchub_info_valid
= false;
614 void hubbub1_toggle_watermark_change_req(struct hubbub
*hubbub
)
616 struct dcn10_hubbub
*hubbub1
= TO_DCN10_HUBBUB(hubbub
);
618 uint32_t watermark_change_req
;
620 REG_GET(DCHUBBUB_ARB_WATERMARK_CHANGE_CNTL
,
621 DCHUBBUB_ARB_WATERMARK_CHANGE_REQUEST
, &watermark_change_req
);
623 if (watermark_change_req
)
624 watermark_change_req
= 0;
626 watermark_change_req
= 1;
628 REG_UPDATE(DCHUBBUB_ARB_WATERMARK_CHANGE_CNTL
,
629 DCHUBBUB_ARB_WATERMARK_CHANGE_REQUEST
, watermark_change_req
);
632 void hubbub1_soft_reset(struct hubbub
*hubbub
, bool reset
)
634 struct dcn10_hubbub
*hubbub1
= TO_DCN10_HUBBUB(hubbub
);
636 uint32_t reset_en
= reset
? 1 : 0;
638 REG_UPDATE(DCHUBBUB_SOFT_RESET
,
639 DCHUBBUB_GLOBAL_SOFT_RESET
, reset_en
);
642 static bool hubbub1_dcc_support_swizzle(
643 enum swizzle_mode_values swizzle
,
644 unsigned int bytes_per_element
,
645 enum segment_order
*segment_order_horz
,
646 enum segment_order
*segment_order_vert
)
648 bool standard_swizzle
= false;
649 bool display_swizzle
= false;
658 standard_swizzle
= true;
666 display_swizzle
= true;
672 if (bytes_per_element
== 1 && standard_swizzle
) {
673 *segment_order_horz
= segment_order__contiguous
;
674 *segment_order_vert
= segment_order__na
;
677 if (bytes_per_element
== 2 && standard_swizzle
) {
678 *segment_order_horz
= segment_order__non_contiguous
;
679 *segment_order_vert
= segment_order__contiguous
;
682 if (bytes_per_element
== 4 && standard_swizzle
) {
683 *segment_order_horz
= segment_order__non_contiguous
;
684 *segment_order_vert
= segment_order__contiguous
;
687 if (bytes_per_element
== 8 && standard_swizzle
) {
688 *segment_order_horz
= segment_order__na
;
689 *segment_order_vert
= segment_order__contiguous
;
692 if (bytes_per_element
== 8 && display_swizzle
) {
693 *segment_order_horz
= segment_order__contiguous
;
694 *segment_order_vert
= segment_order__non_contiguous
;
701 static bool hubbub1_dcc_support_pixel_format(
702 enum surface_pixel_format format
,
703 unsigned int *bytes_per_element
)
705 /* DML: get_bytes_per_element */
707 case SURFACE_PIXEL_FORMAT_GRPH_ARGB1555
:
708 case SURFACE_PIXEL_FORMAT_GRPH_RGB565
:
709 *bytes_per_element
= 2;
711 case SURFACE_PIXEL_FORMAT_GRPH_ARGB8888
:
712 case SURFACE_PIXEL_FORMAT_GRPH_ABGR8888
:
713 case SURFACE_PIXEL_FORMAT_GRPH_ARGB2101010
:
714 case SURFACE_PIXEL_FORMAT_GRPH_ABGR2101010
:
715 *bytes_per_element
= 4;
717 case SURFACE_PIXEL_FORMAT_GRPH_ARGB16161616
:
718 case SURFACE_PIXEL_FORMAT_GRPH_ARGB16161616F
:
719 case SURFACE_PIXEL_FORMAT_GRPH_ABGR16161616F
:
720 *bytes_per_element
= 8;
727 static void hubbub1_get_blk256_size(unsigned int *blk256_width
, unsigned int *blk256_height
,
728 unsigned int bytes_per_element
)
730 /* copied from DML. might want to refactor DML to leverage from DML */
731 /* DML : get_blk256_size */
732 if (bytes_per_element
== 1) {
735 } else if (bytes_per_element
== 2) {
738 } else if (bytes_per_element
== 4) {
741 } else if (bytes_per_element
== 8) {
747 static void hubbub1_det_request_size(
751 bool *req128_horz_wc
,
752 bool *req128_vert_wc
)
754 unsigned int detile_buf_size
= 164 * 1024; /* 164KB for DCN1.0 */
756 unsigned int blk256_height
= 0;
757 unsigned int blk256_width
= 0;
758 unsigned int swath_bytes_horz_wc
, swath_bytes_vert_wc
;
760 hubbub1_get_blk256_size(&blk256_width
, &blk256_height
, bpe
);
762 swath_bytes_horz_wc
= height
* blk256_height
* bpe
;
763 swath_bytes_vert_wc
= width
* blk256_width
* bpe
;
765 *req128_horz_wc
= (2 * swath_bytes_horz_wc
<= detile_buf_size
) ?
766 false : /* full 256B request */
767 true; /* half 128b request */
769 *req128_vert_wc
= (2 * swath_bytes_vert_wc
<= detile_buf_size
) ?
770 false : /* full 256B request */
771 true; /* half 128b request */
774 static bool hubbub1_get_dcc_compression_cap(struct hubbub
*hubbub
,
775 const struct dc_dcc_surface_param
*input
,
776 struct dc_surface_dcc_cap
*output
)
778 struct dcn10_hubbub
*hubbub1
= TO_DCN10_HUBBUB(hubbub
);
779 struct dc
*dc
= hubbub1
->base
.ctx
->dc
;
781 /* implement section 1.6.2.1 of DCN1_Programming_Guide.docx */
782 enum dcc_control dcc_control
;
784 enum segment_order segment_order_horz
, segment_order_vert
;
785 bool req128_horz_wc
, req128_vert_wc
;
787 memset(output
, 0, sizeof(*output
));
789 if (dc
->debug
.disable_dcc
== DCC_DISABLE
)
792 if (!hubbub1
->base
.funcs
->dcc_support_pixel_format(input
->format
, &bpe
))
795 if (!hubbub1
->base
.funcs
->dcc_support_swizzle(input
->swizzle_mode
, bpe
,
796 &segment_order_horz
, &segment_order_vert
))
799 hubbub1_det_request_size(input
->surface_size
.height
, input
->surface_size
.width
,
800 bpe
, &req128_horz_wc
, &req128_vert_wc
);
802 if (!req128_horz_wc
&& !req128_vert_wc
) {
803 dcc_control
= dcc_control__256_256_xxx
;
804 } else if (input
->scan
== SCAN_DIRECTION_HORIZONTAL
) {
806 dcc_control
= dcc_control__256_256_xxx
;
807 else if (segment_order_horz
== segment_order__contiguous
)
808 dcc_control
= dcc_control__128_128_xxx
;
810 dcc_control
= dcc_control__256_64_64
;
811 } else if (input
->scan
== SCAN_DIRECTION_VERTICAL
) {
813 dcc_control
= dcc_control__256_256_xxx
;
814 else if (segment_order_vert
== segment_order__contiguous
)
815 dcc_control
= dcc_control__128_128_xxx
;
817 dcc_control
= dcc_control__256_64_64
;
819 if ((req128_horz_wc
&&
820 segment_order_horz
== segment_order__non_contiguous
) ||
822 segment_order_vert
== segment_order__non_contiguous
))
823 /* access_dir not known, must use most constraining */
824 dcc_control
= dcc_control__256_64_64
;
826 /* reg128 is true for either horz and vert
827 * but segment_order is contiguous
829 dcc_control
= dcc_control__128_128_xxx
;
832 if (dc
->debug
.disable_dcc
== DCC_HALF_REQ_DISALBE
&&
833 dcc_control
!= dcc_control__256_256_xxx
)
836 switch (dcc_control
) {
837 case dcc_control__256_256_xxx
:
838 output
->grph
.rgb
.max_uncompressed_blk_size
= 256;
839 output
->grph
.rgb
.max_compressed_blk_size
= 256;
840 output
->grph
.rgb
.independent_64b_blks
= false;
842 case dcc_control__128_128_xxx
:
843 output
->grph
.rgb
.max_uncompressed_blk_size
= 128;
844 output
->grph
.rgb
.max_compressed_blk_size
= 128;
845 output
->grph
.rgb
.independent_64b_blks
= false;
847 case dcc_control__256_64_64
:
848 output
->grph
.rgb
.max_uncompressed_blk_size
= 256;
849 output
->grph
.rgb
.max_compressed_blk_size
= 64;
850 output
->grph
.rgb
.independent_64b_blks
= true;
854 output
->capable
= true;
855 output
->const_color_support
= false;
860 static const struct hubbub_funcs hubbub1_funcs
= {
861 .update_dchub
= hubbub1_update_dchub
,
862 .dcc_support_swizzle
= hubbub1_dcc_support_swizzle
,
863 .dcc_support_pixel_format
= hubbub1_dcc_support_pixel_format
,
864 .get_dcc_compression_cap
= hubbub1_get_dcc_compression_cap
,
865 .wm_read_state
= hubbub1_wm_read_state
,
868 void hubbub1_construct(struct hubbub
*hubbub
,
869 struct dc_context
*ctx
,
870 const struct dcn_hubbub_registers
*hubbub_regs
,
871 const struct dcn_hubbub_shift
*hubbub_shift
,
872 const struct dcn_hubbub_mask
*hubbub_mask
)
874 struct dcn10_hubbub
*hubbub1
= TO_DCN10_HUBBUB(hubbub
);
876 hubbub1
->base
.ctx
= ctx
;
878 hubbub1
->base
.funcs
= &hubbub1_funcs
;
880 hubbub1
->regs
= hubbub_regs
;
881 hubbub1
->shifts
= hubbub_shift
;
882 hubbub1
->masks
= hubbub_mask
;
884 hubbub1
->debug_test_index_pstate
= 0x7;
885 #if defined(CONFIG_DRM_AMD_DC_DCN1_01)
886 if (ctx
->dce_version
== DCN_VERSION_1_01
)
887 hubbub1
->debug_test_index_pstate
= 0xB;