Linux Audio

Check our new training course

Yocto / OpenEmbedded training

Feb 10-13, 2025
Register
Loading...
Note: File does not exist in v6.9.4.
  1// SPDX-License-Identifier: MIT
  2/*
  3 * Copyright 2022 Advanced Micro Devices, Inc.
  4 *
  5 * Permission is hereby granted, free of charge, to any person obtaining a
  6 * copy of this software and associated documentation files (the "Software"),
  7 * to deal in the Software without restriction, including without limitation
  8 * the rights to use, copy, modify, merge, publish, distribute, sublicense,
  9 * and/or sell copies of the Software, and to permit persons to whom the
 10 * Software is furnished to do so, subject to the following conditions:
 11 *
 12 * The above copyright notice and this permission notice shall be included in
 13 * all copies or substantial portions of the Software.
 14 *
 15 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 16 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 17 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
 18 * THE COPYRIGHT HOLDER(S) OR AUTHOR(S) BE LIABLE FOR ANY CLAIM, DAMAGES OR
 19 * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
 20 * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
 21 * OTHER DEALINGS IN THE SOFTWARE.
 22 *
 23 * Authors: AMD
 24 *
 25 */
 26
 27
 28#include "dm_services.h"
 29#include "dm_helpers.h"
 30#include "core_types.h"
 31#include "resource.h"
 32#include "dccg.h"
 33#include "dce/dce_hwseq.h"
 34#include "clk_mgr.h"
 35#include "reg_helper.h"
 36#include "abm.h"
 37#include "hubp.h"
 38#include "dchubbub.h"
 39#include "timing_generator.h"
 40#include "opp.h"
 41#include "ipp.h"
 42#include "mpc.h"
 43#include "mcif_wb.h"
 44#include "dc_dmub_srv.h"
 45#include "dcn314_hwseq.h"
 46#include "link_hwss.h"
 47#include "dpcd_defs.h"
 48#include "dce/dmub_outbox.h"
 49#include "dc_link_dp.h"
 50#include "inc/dc_link_dp.h"
 51#include "inc/link_dpcd.h"
 52#include "dcn10/dcn10_hw_sequencer.h"
 53#include "inc/link_enc_cfg.h"
 54#include "dcn30/dcn30_vpg.h"
 55#include "dce/dce_i2c_hw.h"
 56#include "dsc.h"
 57#include "dcn20/dcn20_optc.h"
 58#include "dcn30/dcn30_cm_common.h"
 59
 60#define DC_LOGGER_INIT(logger)
 61
 62#define CTX \
 63	hws->ctx
 64#define REG(reg)\
 65	hws->regs->reg
 66#define DC_LOGGER \
 67		dc->ctx->logger
 68
 69
 70#undef FN
 71#define FN(reg_name, field_name) \
 72	hws->shifts->field_name, hws->masks->field_name
 73
 74static int calc_mpc_flow_ctrl_cnt(const struct dc_stream_state *stream,
 75		int opp_cnt)
 76{
 77	bool hblank_halved = optc2_is_two_pixels_per_containter(&stream->timing);
 78	int flow_ctrl_cnt;
 79
 80	if (opp_cnt >= 2)
 81		hblank_halved = true;
 82
 83	flow_ctrl_cnt = stream->timing.h_total - stream->timing.h_addressable -
 84			stream->timing.h_border_left -
 85			stream->timing.h_border_right;
 86
 87	if (hblank_halved)
 88		flow_ctrl_cnt /= 2;
 89
 90	/* ODM combine 4:1 case */
 91	if (opp_cnt == 4)
 92		flow_ctrl_cnt /= 2;
 93
 94	return flow_ctrl_cnt;
 95}
 96
 97static void update_dsc_on_stream(struct pipe_ctx *pipe_ctx, bool enable)
 98{
 99	struct display_stream_compressor *dsc = pipe_ctx->stream_res.dsc;
100	struct dc_stream_state *stream = pipe_ctx->stream;
101	struct pipe_ctx *odm_pipe;
102	int opp_cnt = 1;
103
104	ASSERT(dsc);
105	for (odm_pipe = pipe_ctx->next_odm_pipe; odm_pipe; odm_pipe = odm_pipe->next_odm_pipe)
106		opp_cnt++;
107
108	if (enable) {
109		struct dsc_config dsc_cfg;
110		struct dsc_optc_config dsc_optc_cfg;
111		enum optc_dsc_mode optc_dsc_mode;
112
113		/* Enable DSC hw block */
114		dsc_cfg.pic_width = (stream->timing.h_addressable + stream->timing.h_border_left + stream->timing.h_border_right) / opp_cnt;
115		dsc_cfg.pic_height = stream->timing.v_addressable + stream->timing.v_border_top + stream->timing.v_border_bottom;
116		dsc_cfg.pixel_encoding = stream->timing.pixel_encoding;
117		dsc_cfg.color_depth = stream->timing.display_color_depth;
118		dsc_cfg.is_odm = pipe_ctx->next_odm_pipe ? true : false;
119		dsc_cfg.dc_dsc_cfg = stream->timing.dsc_cfg;
120		ASSERT(dsc_cfg.dc_dsc_cfg.num_slices_h % opp_cnt == 0);
121		dsc_cfg.dc_dsc_cfg.num_slices_h /= opp_cnt;
122
123		dsc->funcs->dsc_set_config(dsc, &dsc_cfg, &dsc_optc_cfg);
124		dsc->funcs->dsc_enable(dsc, pipe_ctx->stream_res.opp->inst);
125		for (odm_pipe = pipe_ctx->next_odm_pipe; odm_pipe; odm_pipe = odm_pipe->next_odm_pipe) {
126			struct display_stream_compressor *odm_dsc = odm_pipe->stream_res.dsc;
127
128			ASSERT(odm_dsc);
129			odm_dsc->funcs->dsc_set_config(odm_dsc, &dsc_cfg, &dsc_optc_cfg);
130			odm_dsc->funcs->dsc_enable(odm_dsc, odm_pipe->stream_res.opp->inst);
131		}
132		dsc_cfg.dc_dsc_cfg.num_slices_h *= opp_cnt;
133		dsc_cfg.pic_width *= opp_cnt;
134
135		optc_dsc_mode = dsc_optc_cfg.is_pixel_format_444 ? OPTC_DSC_ENABLED_444 : OPTC_DSC_ENABLED_NATIVE_SUBSAMPLED;
136
137		/* Enable DSC in OPTC */
138		DC_LOG_DSC("Setting optc DSC config for tg instance %d:", pipe_ctx->stream_res.tg->inst);
139		pipe_ctx->stream_res.tg->funcs->set_dsc_config(pipe_ctx->stream_res.tg,
140							optc_dsc_mode,
141							dsc_optc_cfg.bytes_per_pixel,
142							dsc_optc_cfg.slice_width);
143	} else {
144		/* disable DSC in OPTC */
145		pipe_ctx->stream_res.tg->funcs->set_dsc_config(
146				pipe_ctx->stream_res.tg,
147				OPTC_DSC_DISABLED, 0, 0);
148
149		/* disable DSC block */
150		dsc->funcs->dsc_disable(pipe_ctx->stream_res.dsc);
151		for (odm_pipe = pipe_ctx->next_odm_pipe; odm_pipe; odm_pipe = odm_pipe->next_odm_pipe) {
152			ASSERT(odm_pipe->stream_res.dsc);
153			odm_pipe->stream_res.dsc->funcs->dsc_disable(odm_pipe->stream_res.dsc);
154		}
155	}
156}
157
158// Given any pipe_ctx, return the total ODM combine factor, and optionally return
159// the OPPids which are used
160static unsigned int get_odm_config(struct pipe_ctx *pipe_ctx, unsigned int *opp_instances)
161{
162	unsigned int opp_count = 1;
163	struct pipe_ctx *odm_pipe;
164
165	// First get to the top pipe
166	for (odm_pipe = pipe_ctx; odm_pipe->prev_odm_pipe; odm_pipe = odm_pipe->prev_odm_pipe)
167		;
168
169	// First pipe is always used
170	if (opp_instances)
171		opp_instances[0] = odm_pipe->stream_res.opp->inst;
172
173	// Find and count odm pipes, if any
174	for (odm_pipe = odm_pipe->next_odm_pipe; odm_pipe; odm_pipe = odm_pipe->next_odm_pipe) {
175		if (opp_instances)
176			opp_instances[opp_count] = odm_pipe->stream_res.opp->inst;
177		opp_count++;
178	}
179
180	return opp_count;
181}
182
183void dcn314_update_odm(struct dc *dc, struct dc_state *context, struct pipe_ctx *pipe_ctx)
184{
185	struct pipe_ctx *odm_pipe;
186	int opp_cnt = 0;
187	int opp_inst[MAX_PIPES] = {0};
188	bool rate_control_2x_pclk = (pipe_ctx->stream->timing.flags.INTERLACE || optc2_is_two_pixels_per_containter(&pipe_ctx->stream->timing));
189	struct mpc_dwb_flow_control flow_control;
190	struct mpc *mpc = dc->res_pool->mpc;
191	int i;
192
193	opp_cnt = get_odm_config(pipe_ctx, opp_inst);
194
195	if (opp_cnt > 1)
196		pipe_ctx->stream_res.tg->funcs->set_odm_combine(
197				pipe_ctx->stream_res.tg,
198				opp_inst, opp_cnt,
199				&pipe_ctx->stream->timing);
200	else
201		pipe_ctx->stream_res.tg->funcs->set_odm_bypass(
202				pipe_ctx->stream_res.tg, &pipe_ctx->stream->timing);
203
204	rate_control_2x_pclk = rate_control_2x_pclk || opp_cnt > 1;
205	flow_control.flow_ctrl_mode = 0;
206	flow_control.flow_ctrl_cnt0 = 0x80;
207	flow_control.flow_ctrl_cnt1 = calc_mpc_flow_ctrl_cnt(pipe_ctx->stream, opp_cnt);
208	if (mpc->funcs->set_out_rate_control) {
209		for (i = 0; i < opp_cnt; ++i) {
210			mpc->funcs->set_out_rate_control(
211					mpc, opp_inst[i],
212					true,
213					rate_control_2x_pclk,
214					&flow_control);
215		}
216	}
217
218	for (odm_pipe = pipe_ctx->next_odm_pipe; odm_pipe; odm_pipe = odm_pipe->next_odm_pipe) {
219		odm_pipe->stream_res.opp->funcs->opp_pipe_clock_control(
220				odm_pipe->stream_res.opp,
221				true);
222	}
223
224	if (pipe_ctx->stream_res.dsc) {
225		struct pipe_ctx *current_pipe_ctx = &dc->current_state->res_ctx.pipe_ctx[pipe_ctx->pipe_idx];
226
227		update_dsc_on_stream(pipe_ctx, pipe_ctx->stream->timing.flags.DSC);
228
229		/* Check if no longer using pipe for ODM, then need to disconnect DSC for that pipe */
230		if (!pipe_ctx->next_odm_pipe && current_pipe_ctx->next_odm_pipe &&
231				current_pipe_ctx->next_odm_pipe->stream_res.dsc) {
232			struct display_stream_compressor *dsc = current_pipe_ctx->next_odm_pipe->stream_res.dsc;
233			/* disconnect DSC block from stream */
234			dsc->funcs->dsc_disconnect(dsc);
235		}
236	}
237}
238
239void dcn314_dsc_pg_control(
240		struct dce_hwseq *hws,
241		unsigned int dsc_inst,
242		bool power_on)
243{
244	uint32_t power_gate = power_on ? 0 : 1;
245	uint32_t pwr_status = power_on ? 0 : 2;
246	uint32_t org_ip_request_cntl = 0;
247
248	if (hws->ctx->dc->debug.disable_dsc_power_gate)
249		return;
250
251	if (hws->ctx->dc->debug.root_clock_optimization.bits.dsc &&
252		hws->ctx->dc->res_pool->dccg->funcs->enable_dsc &&
253		power_on)
254		hws->ctx->dc->res_pool->dccg->funcs->enable_dsc(
255			hws->ctx->dc->res_pool->dccg, dsc_inst);
256
257	REG_GET(DC_IP_REQUEST_CNTL, IP_REQUEST_EN, &org_ip_request_cntl);
258	if (org_ip_request_cntl == 0)
259		REG_SET(DC_IP_REQUEST_CNTL, 0, IP_REQUEST_EN, 1);
260
261	switch (dsc_inst) {
262	case 0: /* DSC0 */
263		REG_UPDATE(DOMAIN16_PG_CONFIG,
264				DOMAIN_POWER_GATE, power_gate);
265
266		REG_WAIT(DOMAIN16_PG_STATUS,
267				DOMAIN_PGFSM_PWR_STATUS, pwr_status,
268				1, 1000);
269		break;
270	case 1: /* DSC1 */
271		REG_UPDATE(DOMAIN17_PG_CONFIG,
272				DOMAIN_POWER_GATE, power_gate);
273
274		REG_WAIT(DOMAIN17_PG_STATUS,
275				DOMAIN_PGFSM_PWR_STATUS, pwr_status,
276				1, 1000);
277		break;
278	case 2: /* DSC2 */
279		REG_UPDATE(DOMAIN18_PG_CONFIG,
280				DOMAIN_POWER_GATE, power_gate);
281
282		REG_WAIT(DOMAIN18_PG_STATUS,
283				DOMAIN_PGFSM_PWR_STATUS, pwr_status,
284				1, 1000);
285		break;
286	case 3: /* DSC3 */
287		REG_UPDATE(DOMAIN19_PG_CONFIG,
288				DOMAIN_POWER_GATE, power_gate);
289
290		REG_WAIT(DOMAIN19_PG_STATUS,
291				DOMAIN_PGFSM_PWR_STATUS, pwr_status,
292				1, 1000);
293		break;
294	default:
295		BREAK_TO_DEBUGGER();
296		break;
297	}
298
299	if (org_ip_request_cntl == 0)
300		REG_SET(DC_IP_REQUEST_CNTL, 0, IP_REQUEST_EN, 0);
301
302	if (hws->ctx->dc->debug.root_clock_optimization.bits.dsc) {
303		if (hws->ctx->dc->res_pool->dccg->funcs->disable_dsc && !power_on)
304			hws->ctx->dc->res_pool->dccg->funcs->disable_dsc(
305				hws->ctx->dc->res_pool->dccg, dsc_inst);
306	}
307
308}
309
310void dcn314_enable_power_gating_plane(struct dce_hwseq *hws, bool enable)
311{
312	bool force_on = true; /* disable power gating */
313	uint32_t org_ip_request_cntl = 0;
314
315	if (enable && !hws->ctx->dc->debug.disable_hubp_power_gate)
316		force_on = false;
317
318	REG_GET(DC_IP_REQUEST_CNTL, IP_REQUEST_EN, &org_ip_request_cntl);
319	if (org_ip_request_cntl == 0)
320		REG_SET(DC_IP_REQUEST_CNTL, 0, IP_REQUEST_EN, 1);
321	/* DCHUBP0/1/2/3/4/5 */
322	REG_UPDATE(DOMAIN0_PG_CONFIG, DOMAIN_POWER_FORCEON, force_on);
323	REG_UPDATE(DOMAIN2_PG_CONFIG, DOMAIN_POWER_FORCEON, force_on);
324	/* DPP0/1/2/3/4/5 */
325	REG_UPDATE(DOMAIN1_PG_CONFIG, DOMAIN_POWER_FORCEON, force_on);
326	REG_UPDATE(DOMAIN3_PG_CONFIG, DOMAIN_POWER_FORCEON, force_on);
327
328	force_on = true; /* disable power gating */
329	if (enable && !hws->ctx->dc->debug.disable_dsc_power_gate)
330		force_on = false;
331
332	/* DCS0/1/2/3/4 */
333	REG_UPDATE(DOMAIN16_PG_CONFIG, DOMAIN_POWER_FORCEON, force_on);
334	REG_UPDATE(DOMAIN17_PG_CONFIG, DOMAIN_POWER_FORCEON, force_on);
335	REG_UPDATE(DOMAIN18_PG_CONFIG, DOMAIN_POWER_FORCEON, force_on);
336	REG_UPDATE(DOMAIN19_PG_CONFIG, DOMAIN_POWER_FORCEON, force_on);
337
338	if (org_ip_request_cntl == 0)
339		REG_SET(DC_IP_REQUEST_CNTL, 0, IP_REQUEST_EN, 0);
340}
341
342unsigned int dcn314_calculate_dccg_k1_k2_values(struct pipe_ctx *pipe_ctx, unsigned int *k1_div, unsigned int *k2_div)
343{
344	struct dc_stream_state *stream = pipe_ctx->stream;
345	unsigned int odm_combine_factor = 0;
346	bool two_pix_per_container = false;
347
348	two_pix_per_container = optc2_is_two_pixels_per_containter(&stream->timing);
349	odm_combine_factor = get_odm_config(pipe_ctx, NULL);
350
351	if (is_dp_128b_132b_signal(pipe_ctx)) {
352		*k1_div = PIXEL_RATE_DIV_BY_1;
353		*k2_div = PIXEL_RATE_DIV_BY_1;
354	} else if (dc_is_hdmi_tmds_signal(pipe_ctx->stream->signal) || dc_is_dvi_signal(pipe_ctx->stream->signal)) {
355		*k1_div = PIXEL_RATE_DIV_BY_1;
356		if (stream->timing.pixel_encoding == PIXEL_ENCODING_YCBCR420)
357			*k2_div = PIXEL_RATE_DIV_BY_2;
358		else
359			*k2_div = PIXEL_RATE_DIV_BY_4;
360	} else if (dc_is_dp_signal(pipe_ctx->stream->signal) || dc_is_virtual_signal(pipe_ctx->stream->signal)) {
361		if (two_pix_per_container) {
362			*k1_div = PIXEL_RATE_DIV_BY_1;
363			*k2_div = PIXEL_RATE_DIV_BY_2;
364		} else {
365			*k1_div = PIXEL_RATE_DIV_BY_1;
366			*k2_div = PIXEL_RATE_DIV_BY_4;
367			if (odm_combine_factor == 2)
368				*k2_div = PIXEL_RATE_DIV_BY_2;
369		}
370	}
371
372	if ((*k1_div == PIXEL_RATE_DIV_NA) && (*k2_div == PIXEL_RATE_DIV_NA))
373		ASSERT(false);
374
375	return odm_combine_factor;
376}
377
378void dcn314_set_pixels_per_cycle(struct pipe_ctx *pipe_ctx)
379{
380	uint32_t pix_per_cycle = 1;
381	uint32_t odm_combine_factor = 1;
382
383	if (!pipe_ctx || !pipe_ctx->stream || !pipe_ctx->stream_res.stream_enc)
384		return;
385
386	odm_combine_factor = get_odm_config(pipe_ctx, NULL);
387	if (optc2_is_two_pixels_per_containter(&pipe_ctx->stream->timing) || odm_combine_factor > 1)
388		pix_per_cycle = 2;
389
390	if (pipe_ctx->stream_res.stream_enc->funcs->set_input_mode)
391		pipe_ctx->stream_res.stream_enc->funcs->set_input_mode(pipe_ctx->stream_res.stream_enc,
392				pix_per_cycle);
393}