2 * Copyright 2011 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.
22 * Authors: Alex Deucher
28 #include "rv770_dpm.h"
31 struct rv7xx_power_info
*rv770_get_pi(struct radeon_device
*rdev
);
33 u32
rv740_get_decoded_reference_divider(u32 encoded_ref
)
37 switch (encoded_ref
) {
60 DRM_ERROR("Invalid encoded Reference Divider\n");
68 struct dll_speed_setting
{
74 static struct dll_speed_setting dll_speed_table
[16] =
94 u32
rv740_get_dll_speed(bool is_gddr5
, u32 memory_clock
)
105 data_rate
= (u16
)(memory_clock
* factor
/ 1000);
107 if (data_rate
< dll_speed_table
[0].max
) {
108 for (i
= 0; i
< 16; i
++) {
109 if (data_rate
> dll_speed_table
[i
].min
&&
110 data_rate
<= dll_speed_table
[i
].max
)
111 return dll_speed_table
[i
].dll_speed
;
115 DRM_DEBUG_KMS("Target MCLK greater than largest MCLK in DLL speed table\n");
120 int rv740_populate_sclk_value(struct radeon_device
*rdev
, u32 engine_clock
,
121 RV770_SMC_SCLK_VALUE
*sclk
)
123 struct rv7xx_power_info
*pi
= rv770_get_pi(rdev
);
124 struct atom_clock_dividers dividers
;
125 u32 spll_func_cntl
= pi
->clk_regs
.rv770
.cg_spll_func_cntl
;
126 u32 spll_func_cntl_2
= pi
->clk_regs
.rv770
.cg_spll_func_cntl_2
;
127 u32 spll_func_cntl_3
= pi
->clk_regs
.rv770
.cg_spll_func_cntl_3
;
128 u32 cg_spll_spread_spectrum
= pi
->clk_regs
.rv770
.cg_spll_spread_spectrum
;
129 u32 cg_spll_spread_spectrum_2
= pi
->clk_regs
.rv770
.cg_spll_spread_spectrum_2
;
131 u32 reference_clock
= rdev
->clock
.spll
.reference_freq
;
132 u32 reference_divider
;
136 ret
= radeon_atom_get_clock_dividers(rdev
, COMPUTE_ENGINE_PLL_PARAM
,
137 engine_clock
, false, ÷rs
);
141 reference_divider
= 1 + dividers
.ref_div
;
143 tmp
= (u64
) engine_clock
* reference_divider
* dividers
.post_div
* 16384;
144 do_div(tmp
, reference_clock
);
147 spll_func_cntl
&= ~(SPLL_PDIV_A_MASK
| SPLL_REF_DIV_MASK
);
148 spll_func_cntl
|= SPLL_REF_DIV(dividers
.ref_div
);
149 spll_func_cntl
|= SPLL_PDIV_A(dividers
.post_div
);
151 spll_func_cntl_2
&= ~SCLK_MUX_SEL_MASK
;
152 spll_func_cntl_2
|= SCLK_MUX_SEL(2);
154 spll_func_cntl_3
&= ~SPLL_FB_DIV_MASK
;
155 spll_func_cntl_3
|= SPLL_FB_DIV(fbdiv
);
156 spll_func_cntl_3
|= SPLL_DITHEN
;
159 struct radeon_atom_ss ss
;
160 u32 vco_freq
= engine_clock
* dividers
.post_div
;
162 if (radeon_atombios_get_asic_ss_info(rdev
, &ss
,
163 ASIC_INTERNAL_ENGINE_SS
, vco_freq
)) {
164 u32 clk_s
= reference_clock
* 5 / (reference_divider
* ss
.rate
);
165 u32 clk_v
= 4 * ss
.percentage
* fbdiv
/ (clk_s
* 10000);
167 cg_spll_spread_spectrum
&= ~CLK_S_MASK
;
168 cg_spll_spread_spectrum
|= CLK_S(clk_s
);
169 cg_spll_spread_spectrum
|= SSEN
;
171 cg_spll_spread_spectrum_2
&= ~CLK_V_MASK
;
172 cg_spll_spread_spectrum_2
|= CLK_V(clk_v
);
176 sclk
->sclk_value
= cpu_to_be32(engine_clock
);
177 sclk
->vCG_SPLL_FUNC_CNTL
= cpu_to_be32(spll_func_cntl
);
178 sclk
->vCG_SPLL_FUNC_CNTL_2
= cpu_to_be32(spll_func_cntl_2
);
179 sclk
->vCG_SPLL_FUNC_CNTL_3
= cpu_to_be32(spll_func_cntl_3
);
180 sclk
->vCG_SPLL_SPREAD_SPECTRUM
= cpu_to_be32(cg_spll_spread_spectrum
);
181 sclk
->vCG_SPLL_SPREAD_SPECTRUM_2
= cpu_to_be32(cg_spll_spread_spectrum_2
);
186 int rv740_populate_mclk_value(struct radeon_device
*rdev
,
187 u32 engine_clock
, u32 memory_clock
,
188 RV7XX_SMC_MCLK_VALUE
*mclk
)
190 struct rv7xx_power_info
*pi
= rv770_get_pi(rdev
);
191 u32 mpll_ad_func_cntl
= pi
->clk_regs
.rv770
.mpll_ad_func_cntl
;
192 u32 mpll_ad_func_cntl_2
= pi
->clk_regs
.rv770
.mpll_ad_func_cntl_2
;
193 u32 mpll_dq_func_cntl
= pi
->clk_regs
.rv770
.mpll_dq_func_cntl
;
194 u32 mpll_dq_func_cntl_2
= pi
->clk_regs
.rv770
.mpll_dq_func_cntl_2
;
195 u32 mclk_pwrmgt_cntl
= pi
->clk_regs
.rv770
.mclk_pwrmgt_cntl
;
196 u32 dll_cntl
= pi
->clk_regs
.rv770
.dll_cntl
;
197 u32 mpll_ss1
= pi
->clk_regs
.rv770
.mpll_ss1
;
198 u32 mpll_ss2
= pi
->clk_regs
.rv770
.mpll_ss2
;
199 struct atom_clock_dividers dividers
;
204 ret
= radeon_atom_get_clock_dividers(rdev
, COMPUTE_MEMORY_PLL_PARAM
,
205 memory_clock
, false, ÷rs
);
209 ibias
= rv770_map_clkf_to_ibias(rdev
, dividers
.whole_fb_div
);
211 mpll_ad_func_cntl
&= ~(CLKR_MASK
|
216 mpll_ad_func_cntl
|= CLKR(dividers
.ref_div
);
217 mpll_ad_func_cntl
|= YCLK_POST_DIV(dividers
.post_div
);
218 mpll_ad_func_cntl
|= CLKF(dividers
.whole_fb_div
);
219 mpll_ad_func_cntl
|= CLKFRAC(dividers
.frac_fb_div
);
220 mpll_ad_func_cntl
|= IBIAS(ibias
);
222 if (dividers
.vco_mode
)
223 mpll_ad_func_cntl_2
|= VCO_MODE
;
225 mpll_ad_func_cntl_2
&= ~VCO_MODE
;
228 mpll_dq_func_cntl
&= ~(CLKR_MASK
|
233 mpll_dq_func_cntl
|= CLKR(dividers
.ref_div
);
234 mpll_dq_func_cntl
|= YCLK_POST_DIV(dividers
.post_div
);
235 mpll_dq_func_cntl
|= CLKF(dividers
.whole_fb_div
);
236 mpll_dq_func_cntl
|= CLKFRAC(dividers
.frac_fb_div
);
237 mpll_dq_func_cntl
|= IBIAS(ibias
);
239 if (dividers
.vco_mode
)
240 mpll_dq_func_cntl_2
|= VCO_MODE
;
242 mpll_dq_func_cntl_2
&= ~VCO_MODE
;
246 struct radeon_atom_ss ss
;
247 u32 vco_freq
= memory_clock
* dividers
.post_div
;
249 if (radeon_atombios_get_asic_ss_info(rdev
, &ss
,
250 ASIC_INTERNAL_MEMORY_SS
, vco_freq
)) {
251 u32 reference_clock
= rdev
->clock
.mpll
.reference_freq
;
252 u32 decoded_ref
= rv740_get_decoded_reference_divider(dividers
.ref_div
);
253 u32 clk_s
= reference_clock
* 5 / (decoded_ref
* ss
.rate
);
254 u32 clk_v
= 0x40000 * ss
.percentage
*
255 (dividers
.whole_fb_div
+ (dividers
.frac_fb_div
/ 8)) / (clk_s
* 10000);
257 mpll_ss1
&= ~CLKV_MASK
;
258 mpll_ss1
|= CLKV(clk_v
);
260 mpll_ss2
&= ~CLKS_MASK
;
261 mpll_ss2
|= CLKS(clk_s
);
265 dll_speed
= rv740_get_dll_speed(pi
->mem_gddr5
,
268 mclk_pwrmgt_cntl
&= ~DLL_SPEED_MASK
;
269 mclk_pwrmgt_cntl
|= DLL_SPEED(dll_speed
);
271 mclk
->mclk770
.mclk_value
= cpu_to_be32(memory_clock
);
272 mclk
->mclk770
.vMPLL_AD_FUNC_CNTL
= cpu_to_be32(mpll_ad_func_cntl
);
273 mclk
->mclk770
.vMPLL_AD_FUNC_CNTL_2
= cpu_to_be32(mpll_ad_func_cntl_2
);
274 mclk
->mclk770
.vMPLL_DQ_FUNC_CNTL
= cpu_to_be32(mpll_dq_func_cntl
);
275 mclk
->mclk770
.vMPLL_DQ_FUNC_CNTL_2
= cpu_to_be32(mpll_dq_func_cntl_2
);
276 mclk
->mclk770
.vMCLK_PWRMGT_CNTL
= cpu_to_be32(mclk_pwrmgt_cntl
);
277 mclk
->mclk770
.vDLL_CNTL
= cpu_to_be32(dll_cntl
);
278 mclk
->mclk770
.vMPLL_SS
= cpu_to_be32(mpll_ss1
);
279 mclk
->mclk770
.vMPLL_SS2
= cpu_to_be32(mpll_ss2
);
284 void rv740_read_clock_registers(struct radeon_device
*rdev
)
286 struct rv7xx_power_info
*pi
= rv770_get_pi(rdev
);
288 pi
->clk_regs
.rv770
.cg_spll_func_cntl
=
289 RREG32(CG_SPLL_FUNC_CNTL
);
290 pi
->clk_regs
.rv770
.cg_spll_func_cntl_2
=
291 RREG32(CG_SPLL_FUNC_CNTL_2
);
292 pi
->clk_regs
.rv770
.cg_spll_func_cntl_3
=
293 RREG32(CG_SPLL_FUNC_CNTL_3
);
294 pi
->clk_regs
.rv770
.cg_spll_spread_spectrum
=
295 RREG32(CG_SPLL_SPREAD_SPECTRUM
);
296 pi
->clk_regs
.rv770
.cg_spll_spread_spectrum_2
=
297 RREG32(CG_SPLL_SPREAD_SPECTRUM_2
);
299 pi
->clk_regs
.rv770
.mpll_ad_func_cntl
=
300 RREG32(MPLL_AD_FUNC_CNTL
);
301 pi
->clk_regs
.rv770
.mpll_ad_func_cntl_2
=
302 RREG32(MPLL_AD_FUNC_CNTL_2
);
303 pi
->clk_regs
.rv770
.mpll_dq_func_cntl
=
304 RREG32(MPLL_DQ_FUNC_CNTL
);
305 pi
->clk_regs
.rv770
.mpll_dq_func_cntl_2
=
306 RREG32(MPLL_DQ_FUNC_CNTL_2
);
307 pi
->clk_regs
.rv770
.mclk_pwrmgt_cntl
=
308 RREG32(MCLK_PWRMGT_CNTL
);
309 pi
->clk_regs
.rv770
.dll_cntl
= RREG32(DLL_CNTL
);
310 pi
->clk_regs
.rv770
.mpll_ss1
= RREG32(MPLL_SS1
);
311 pi
->clk_regs
.rv770
.mpll_ss2
= RREG32(MPLL_SS2
);
314 int rv740_populate_smc_acpi_state(struct radeon_device
*rdev
,
315 RV770_SMC_STATETABLE
*table
)
317 struct rv7xx_power_info
*pi
= rv770_get_pi(rdev
);
318 u32 mpll_ad_func_cntl
= pi
->clk_regs
.rv770
.mpll_ad_func_cntl
;
319 u32 mpll_ad_func_cntl_2
= pi
->clk_regs
.rv770
.mpll_ad_func_cntl_2
;
320 u32 mpll_dq_func_cntl
= pi
->clk_regs
.rv770
.mpll_dq_func_cntl
;
321 u32 mpll_dq_func_cntl_2
= pi
->clk_regs
.rv770
.mpll_dq_func_cntl_2
;
322 u32 spll_func_cntl
= pi
->clk_regs
.rv770
.cg_spll_func_cntl
;
323 u32 spll_func_cntl_2
= pi
->clk_regs
.rv770
.cg_spll_func_cntl_2
;
324 u32 spll_func_cntl_3
= pi
->clk_regs
.rv770
.cg_spll_func_cntl_3
;
325 u32 mclk_pwrmgt_cntl
= pi
->clk_regs
.rv770
.mclk_pwrmgt_cntl
;
326 u32 dll_cntl
= pi
->clk_regs
.rv770
.dll_cntl
;
328 table
->ACPIState
= table
->initialState
;
330 table
->ACPIState
.flags
&= ~PPSMC_SWSTATE_FLAG_DC
;
333 rv770_populate_vddc_value(rdev
, pi
->acpi_vddc
,
334 &table
->ACPIState
.levels
[0].vddc
);
335 table
->ACPIState
.levels
[0].gen2PCIE
=
337 pi
->acpi_pcie_gen2
: 0;
338 table
->ACPIState
.levels
[0].gen2XSP
=
341 rv770_populate_vddc_value(rdev
, pi
->min_vddc_in_table
,
342 &table
->ACPIState
.levels
[0].vddc
);
343 table
->ACPIState
.levels
[0].gen2PCIE
= 0;
346 mpll_ad_func_cntl_2
|= BIAS_GEN_PDNB
| RESET_EN
;
348 mpll_dq_func_cntl_2
|= BYPASS
| BIAS_GEN_PDNB
| RESET_EN
;
350 mclk_pwrmgt_cntl
|= (MRDCKA0_RESET
|
359 dll_cntl
|= (MRDCKA0_BYPASS
|
368 spll_func_cntl
|= SPLL_RESET
| SPLL_SLEEP
| SPLL_BYPASS_EN
;
370 spll_func_cntl_2
&= ~SCLK_MUX_SEL_MASK
;
371 spll_func_cntl_2
|= SCLK_MUX_SEL(4);
373 table
->ACPIState
.levels
[0].mclk
.mclk770
.vMPLL_AD_FUNC_CNTL
= cpu_to_be32(mpll_ad_func_cntl
);
374 table
->ACPIState
.levels
[0].mclk
.mclk770
.vMPLL_AD_FUNC_CNTL_2
= cpu_to_be32(mpll_ad_func_cntl_2
);
375 table
->ACPIState
.levels
[0].mclk
.mclk770
.vMPLL_DQ_FUNC_CNTL
= cpu_to_be32(mpll_dq_func_cntl
);
376 table
->ACPIState
.levels
[0].mclk
.mclk770
.vMPLL_DQ_FUNC_CNTL_2
= cpu_to_be32(mpll_dq_func_cntl_2
);
377 table
->ACPIState
.levels
[0].mclk
.mclk770
.vMCLK_PWRMGT_CNTL
= cpu_to_be32(mclk_pwrmgt_cntl
);
378 table
->ACPIState
.levels
[0].mclk
.mclk770
.vDLL_CNTL
= cpu_to_be32(dll_cntl
);
380 table
->ACPIState
.levels
[0].mclk
.mclk770
.mclk_value
= 0;
382 table
->ACPIState
.levels
[0].sclk
.vCG_SPLL_FUNC_CNTL
= cpu_to_be32(spll_func_cntl
);
383 table
->ACPIState
.levels
[0].sclk
.vCG_SPLL_FUNC_CNTL_2
= cpu_to_be32(spll_func_cntl_2
);
384 table
->ACPIState
.levels
[0].sclk
.vCG_SPLL_FUNC_CNTL_3
= cpu_to_be32(spll_func_cntl_3
);
386 table
->ACPIState
.levels
[0].sclk
.sclk_value
= 0;
388 table
->ACPIState
.levels
[1] = table
->ACPIState
.levels
[0];
389 table
->ACPIState
.levels
[2] = table
->ACPIState
.levels
[0];
391 rv770_populate_mvdd_value(rdev
, 0, &table
->ACPIState
.levels
[0].mvdd
);
396 void rv740_enable_mclk_spread_spectrum(struct radeon_device
*rdev
,
400 WREG32_P(MPLL_CNTL_MODE
, SS_SSEN
, ~SS_SSEN
);
402 WREG32_P(MPLL_CNTL_MODE
, 0, ~SS_SSEN
);
405 u8
rv740_get_mclk_frequency_ratio(u32 memory_clock
)
409 if ((memory_clock
< 10000) || (memory_clock
> 47500))
410 mc_para_index
= 0x00;
412 mc_para_index
= (u8
)((memory_clock
- 10000) / 2500);
414 return mc_para_index
;