Home | History | Annotate | Line # | Download | only in dcn10
      1 /*	$NetBSD: amdgpu_rv1_clk_mgr.c,v 1.2 2021/12/18 23:45:02 riastradh Exp $	*/
      2 
      3 /*
      4  * Copyright 2018 Advanced Micro Devices, Inc.
      5  *
      6  * Permission is hereby granted, free of charge, to any person obtaining a
      7  * copy of this software and associated documentation files (the "Software"),
      8  * to deal in the Software without restriction, including without limitation
      9  * the rights to use, copy, modify, merge, publish, distribute, sublicense,
     10  * and/or sell copies of the Software, and to permit persons to whom the
     11  * Software is furnished to do so, subject to the following conditions:
     12  *
     13  * The above copyright notice and this permission notice shall be included in
     14  * all copies or substantial portions of the Software.
     15  *
     16  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
     17  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
     18  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
     19  * THE COPYRIGHT HOLDER(S) OR AUTHOR(S) BE LIABLE FOR ANY CLAIM, DAMAGES OR
     20  * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
     21  * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
     22  * OTHER DEALINGS IN THE SOFTWARE.
     23  *
     24  * Authors: AMD
     25  *
     26  */
     27 
     28 #include <sys/cdefs.h>
     29 __KERNEL_RCSID(0, "$NetBSD: amdgpu_rv1_clk_mgr.c,v 1.2 2021/12/18 23:45:02 riastradh Exp $");
     30 
     31 #include <linux/slab.h>
     32 
     33 #include "reg_helper.h"
     34 #include "core_types.h"
     35 #include "clk_mgr_internal.h"
     36 #include "rv1_clk_mgr.h"
     37 #include "dce100/dce_clk_mgr.h"
     38 #include "dce112/dce112_clk_mgr.h"
     39 #include "rv1_clk_mgr_vbios_smu.h"
     40 #include "rv1_clk_mgr_clk.h"
     41 
     42 void rv1_init_clocks(struct clk_mgr *clk_mgr)
     43 {
     44 	memset(&(clk_mgr->clks), 0, sizeof(struct dc_clocks));
     45 }
     46 
     47 static int rv1_determine_dppclk_threshold(struct clk_mgr_internal *clk_mgr, struct dc_clocks *new_clocks)
     48 {
     49 	bool request_dpp_div = new_clocks->dispclk_khz > new_clocks->dppclk_khz;
     50 	bool dispclk_increase = new_clocks->dispclk_khz > clk_mgr->base.clks.dispclk_khz;
     51 	int disp_clk_threshold = new_clocks->max_supported_dppclk_khz;
     52 	bool cur_dpp_div = clk_mgr->base.clks.dispclk_khz > clk_mgr->base.clks.dppclk_khz;
     53 
     54 	/* increase clock, looking for div is 0 for current, request div is 1*/
     55 	if (dispclk_increase) {
     56 		/* already divided by 2, no need to reach target clk with 2 steps*/
     57 		if (cur_dpp_div)
     58 			return new_clocks->dispclk_khz;
     59 
     60 		/* request disp clk is lower than maximum supported dpp clk,
     61 		 * no need to reach target clk with two steps.
     62 		 */
     63 		if (new_clocks->dispclk_khz <= disp_clk_threshold)
     64 			return new_clocks->dispclk_khz;
     65 
     66 		/* target dpp clk not request divided by 2, still within threshold */
     67 		if (!request_dpp_div)
     68 			return new_clocks->dispclk_khz;
     69 
     70 	} else {
     71 		/* decrease clock, looking for current dppclk divided by 2,
     72 		 * request dppclk not divided by 2.
     73 		 */
     74 
     75 		/* current dpp clk not divided by 2, no need to ramp*/
     76 		if (!cur_dpp_div)
     77 			return new_clocks->dispclk_khz;
     78 
     79 		/* current disp clk is lower than current maximum dpp clk,
     80 		 * no need to ramp
     81 		 */
     82 		if (clk_mgr->base.clks.dispclk_khz <= disp_clk_threshold)
     83 			return new_clocks->dispclk_khz;
     84 
     85 		/* request dpp clk need to be divided by 2 */
     86 		if (request_dpp_div)
     87 			return new_clocks->dispclk_khz;
     88 	}
     89 
     90 	return disp_clk_threshold;
     91 }
     92 
     93 static void ramp_up_dispclk_with_dpp(struct clk_mgr_internal *clk_mgr, struct dc *dc, struct dc_clocks *new_clocks)
     94 {
     95 	int i;
     96 	int dispclk_to_dpp_threshold = rv1_determine_dppclk_threshold(clk_mgr, new_clocks);
     97 	bool request_dpp_div = new_clocks->dispclk_khz > new_clocks->dppclk_khz;
     98 
     99 	/* set disp clk to dpp clk threshold */
    100 
    101 	clk_mgr->funcs->set_dispclk(clk_mgr, dispclk_to_dpp_threshold);
    102 	clk_mgr->funcs->set_dprefclk(clk_mgr);
    103 
    104 
    105 	/* update request dpp clk division option */
    106 	for (i = 0; i < dc->res_pool->pipe_count; i++) {
    107 		struct pipe_ctx *pipe_ctx = &dc->current_state->res_ctx.pipe_ctx[i];
    108 
    109 		if (!pipe_ctx->plane_state)
    110 			continue;
    111 
    112 		pipe_ctx->plane_res.dpp->funcs->dpp_dppclk_control(
    113 				pipe_ctx->plane_res.dpp,
    114 				request_dpp_div,
    115 				true);
    116 	}
    117 
    118 	/* If target clk not same as dppclk threshold, set to target clock */
    119 	if (dispclk_to_dpp_threshold != new_clocks->dispclk_khz) {
    120 		clk_mgr->funcs->set_dispclk(clk_mgr, new_clocks->dispclk_khz);
    121 		clk_mgr->funcs->set_dprefclk(clk_mgr);
    122 	}
    123 
    124 
    125 	clk_mgr->base.clks.dispclk_khz = new_clocks->dispclk_khz;
    126 	clk_mgr->base.clks.dppclk_khz = new_clocks->dppclk_khz;
    127 	clk_mgr->base.clks.max_supported_dppclk_khz = new_clocks->max_supported_dppclk_khz;
    128 }
    129 
    130 static void rv1_update_clocks(struct clk_mgr *clk_mgr_base,
    131 			struct dc_state *context,
    132 			bool safe_to_lower)
    133 {
    134 	struct clk_mgr_internal *clk_mgr = TO_CLK_MGR_INTERNAL(clk_mgr_base);
    135 	struct dc *dc = clk_mgr_base->ctx->dc;
    136 	struct dc_debug_options *debug = &dc->debug;
    137 	struct dc_clocks *new_clocks = &context->bw_ctx.bw.dcn.clk;
    138 	struct pp_smu_funcs_rv *pp_smu = NULL;
    139 	bool send_request_to_increase = false;
    140 	bool send_request_to_lower = false;
    141 	int display_count;
    142 
    143 	bool enter_display_off = false;
    144 
    145 	ASSERT(clk_mgr->pp_smu);
    146 
    147 	if (dc->work_arounds.skip_clock_update)
    148 		return;
    149 
    150 	pp_smu = &clk_mgr->pp_smu->rv_funcs;
    151 
    152 	display_count = clk_mgr_helper_get_active_display_cnt(dc, context);
    153 
    154 	if (display_count == 0)
    155 		enter_display_off = true;
    156 
    157 	if (enter_display_off == safe_to_lower) {
    158 		/*
    159 		 * Notify SMU active displays
    160 		 * if function pointer not set up, this message is
    161 		 * sent as part of pplib_apply_display_requirements.
    162 		 */
    163 		if (pp_smu->set_display_count)
    164 			pp_smu->set_display_count(&pp_smu->pp_smu, display_count);
    165 	}
    166 
    167 	if (new_clocks->dispclk_khz > clk_mgr_base->clks.dispclk_khz
    168 			|| new_clocks->phyclk_khz > clk_mgr_base->clks.phyclk_khz
    169 			|| new_clocks->fclk_khz > clk_mgr_base->clks.fclk_khz
    170 			|| new_clocks->dcfclk_khz > clk_mgr_base->clks.dcfclk_khz)
    171 		send_request_to_increase = true;
    172 
    173 	if (should_set_clock(safe_to_lower, new_clocks->phyclk_khz, clk_mgr_base->clks.phyclk_khz)) {
    174 		clk_mgr_base->clks.phyclk_khz = new_clocks->phyclk_khz;
    175 		send_request_to_lower = true;
    176 	}
    177 
    178 	// F Clock
    179 	if (debug->force_fclk_khz != 0)
    180 		new_clocks->fclk_khz = debug->force_fclk_khz;
    181 
    182 	if (should_set_clock(safe_to_lower, new_clocks->fclk_khz, clk_mgr_base->clks.fclk_khz)) {
    183 		clk_mgr_base->clks.fclk_khz = new_clocks->fclk_khz;
    184 		send_request_to_lower = true;
    185 	}
    186 
    187 	//DCF Clock
    188 	if (should_set_clock(safe_to_lower, new_clocks->dcfclk_khz, clk_mgr_base->clks.dcfclk_khz)) {
    189 		clk_mgr_base->clks.dcfclk_khz = new_clocks->dcfclk_khz;
    190 		send_request_to_lower = true;
    191 	}
    192 
    193 	if (should_set_clock(safe_to_lower,
    194 			new_clocks->dcfclk_deep_sleep_khz, clk_mgr_base->clks.dcfclk_deep_sleep_khz)) {
    195 		clk_mgr_base->clks.dcfclk_deep_sleep_khz = new_clocks->dcfclk_deep_sleep_khz;
    196 		send_request_to_lower = true;
    197 	}
    198 
    199 	/* make sure dcf clk is before dpp clk to
    200 	 * make sure we have enough voltage to run dpp clk
    201 	 */
    202 	if (send_request_to_increase) {
    203 		/*use dcfclk to request voltage*/
    204 		if (pp_smu->set_hard_min_fclk_by_freq &&
    205 				pp_smu->set_hard_min_dcfclk_by_freq &&
    206 				pp_smu->set_min_deep_sleep_dcfclk) {
    207 			pp_smu->set_hard_min_fclk_by_freq(&pp_smu->pp_smu, new_clocks->fclk_khz / 1000);
    208 			pp_smu->set_hard_min_dcfclk_by_freq(&pp_smu->pp_smu, new_clocks->dcfclk_khz / 1000);
    209 			pp_smu->set_min_deep_sleep_dcfclk(&pp_smu->pp_smu, (new_clocks->dcfclk_deep_sleep_khz + 999) / 1000);
    210 		}
    211 	}
    212 
    213 	/* dcn1 dppclk is tied to dispclk */
    214 	/* program dispclk on = as a w/a for sleep resume clock ramping issues */
    215 	if (should_set_clock(safe_to_lower, new_clocks->dispclk_khz, clk_mgr_base->clks.dispclk_khz)
    216 			|| new_clocks->dispclk_khz == clk_mgr_base->clks.dispclk_khz) {
    217 		ramp_up_dispclk_with_dpp(clk_mgr, dc, new_clocks);
    218 		clk_mgr_base->clks.dispclk_khz = new_clocks->dispclk_khz;
    219 		send_request_to_lower = true;
    220 	}
    221 
    222 	if (!send_request_to_increase && send_request_to_lower) {
    223 		/*use dcfclk to request voltage*/
    224 		if (pp_smu->set_hard_min_fclk_by_freq &&
    225 				pp_smu->set_hard_min_dcfclk_by_freq &&
    226 				pp_smu->set_min_deep_sleep_dcfclk) {
    227 			pp_smu->set_hard_min_fclk_by_freq(&pp_smu->pp_smu, new_clocks->fclk_khz / 1000);
    228 			pp_smu->set_hard_min_dcfclk_by_freq(&pp_smu->pp_smu, new_clocks->dcfclk_khz / 1000);
    229 			pp_smu->set_min_deep_sleep_dcfclk(&pp_smu->pp_smu, (new_clocks->dcfclk_deep_sleep_khz + 999) / 1000);
    230 		}
    231 	}
    232 }
    233 
    234 static void rv1_enable_pme_wa(struct clk_mgr *clk_mgr_base)
    235 {
    236 	struct clk_mgr_internal *clk_mgr = TO_CLK_MGR_INTERNAL(clk_mgr_base);
    237 	struct pp_smu_funcs_rv *pp_smu = NULL;
    238 
    239 	if (clk_mgr->pp_smu) {
    240 		pp_smu = &clk_mgr->pp_smu->rv_funcs;
    241 
    242 		if (pp_smu->set_pme_wa_enable)
    243 			pp_smu->set_pme_wa_enable(&pp_smu->pp_smu);
    244 	}
    245 }
    246 
    247 static struct clk_mgr_funcs rv1_clk_funcs = {
    248 	.init_clocks = rv1_init_clocks,
    249 	.get_dp_ref_clk_frequency = dce12_get_dp_ref_freq_khz,
    250 	.update_clocks = rv1_update_clocks,
    251 	.enable_pme_wa = rv1_enable_pme_wa,
    252 };
    253 
    254 static struct clk_mgr_internal_funcs rv1_clk_internal_funcs = {
    255 	.set_dispclk = rv1_vbios_smu_set_dispclk,
    256 	.set_dprefclk = dce112_set_dprefclk
    257 };
    258 
    259 void rv1_clk_mgr_construct(struct dc_context *ctx, struct clk_mgr_internal *clk_mgr, struct pp_smu_funcs *pp_smu)
    260 {
    261 	struct dc_debug_options *debug = &ctx->dc->debug;
    262 	struct dc_bios *bp = ctx->dc_bios;
    263 
    264 	clk_mgr->base.ctx = ctx;
    265 	clk_mgr->pp_smu = pp_smu;
    266 	clk_mgr->base.funcs = &rv1_clk_funcs;
    267 	clk_mgr->funcs = &rv1_clk_internal_funcs;
    268 
    269 	clk_mgr->dfs_bypass_disp_clk = 0;
    270 
    271 	clk_mgr->dprefclk_ss_percentage = 0;
    272 	clk_mgr->dprefclk_ss_divider = 1000;
    273 	clk_mgr->ss_on_dprefclk = false;
    274 	clk_mgr->base.dprefclk_khz = 600000;
    275 
    276 	if (bp->integrated_info)
    277 		clk_mgr->base.dentist_vco_freq_khz = bp->integrated_info->dentist_vco_freq;
    278 	if (bp->fw_info_valid && clk_mgr->base.dentist_vco_freq_khz == 0) {
    279 		clk_mgr->base.dentist_vco_freq_khz = bp->fw_info.smu_gpu_pll_output_freq;
    280 		if (clk_mgr->base.dentist_vco_freq_khz == 0)
    281 			clk_mgr->base.dentist_vco_freq_khz = 3600000;
    282 	}
    283 
    284 	if (!debug->disable_dfs_bypass && bp->integrated_info)
    285 		if (bp->integrated_info->gpu_cap_info & DFS_BYPASS_ENABLE)
    286 			clk_mgr->dfs_bypass_enabled = true;
    287 
    288 	dce_clock_read_ss_info(clk_mgr);
    289 }
    290 
    291 
    292