xref: /openbsd-src/sys/dev/pci/drm/amd/display/dc/link/link_validation.c (revision a874959911b123f19729c77bb95100a33795ab12)
1f005ef32Sjsg /*
2f005ef32Sjsg  * Copyright 2023 Advanced Micro Devices, Inc.
3f005ef32Sjsg  *
4f005ef32Sjsg  * Permission is hereby granted, free of charge, to any person obtaining a
5f005ef32Sjsg  * copy of this software and associated documentation files (the "Software"),
6f005ef32Sjsg  * to deal in the Software without restriction, including without limitation
7f005ef32Sjsg  * the rights to use, copy, modify, merge, publish, distribute, sublicense,
8f005ef32Sjsg  * and/or sell copies of the Software, and to permit persons to whom the
9f005ef32Sjsg  * Software is furnished to do so, subject to the following conditions:
10f005ef32Sjsg  *
11f005ef32Sjsg  * The above copyright notice and this permission notice shall be included in
12f005ef32Sjsg  * all copies or substantial portions of the Software.
13f005ef32Sjsg  *
14f005ef32Sjsg  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
15f005ef32Sjsg  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
16f005ef32Sjsg  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
17f005ef32Sjsg  * THE COPYRIGHT HOLDER(S) OR AUTHOR(S) BE LIABLE FOR ANY CLAIM, DAMAGES OR
18f005ef32Sjsg  * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
19f005ef32Sjsg  * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
20f005ef32Sjsg  * OTHER DEALINGS IN THE SOFTWARE.
21f005ef32Sjsg  *
22f005ef32Sjsg  * Authors: AMD
23f005ef32Sjsg  *
24f005ef32Sjsg  */
25f005ef32Sjsg 
26f005ef32Sjsg /* FILE POLICY AND INTENDED USAGE:
27f005ef32Sjsg  * This file owns timing validation against various link limitations. (ex.
28f005ef32Sjsg  * link bandwidth, receiver capability or our hardware capability) It also
29f005ef32Sjsg  * provides helper functions exposing bandwidth formulas used in validation.
30f005ef32Sjsg  */
31f005ef32Sjsg #include "link_validation.h"
32f005ef32Sjsg #include "protocols/link_dp_capability.h"
33f005ef32Sjsg #include "protocols/link_dp_dpia_bw.h"
34f005ef32Sjsg #include "resource.h"
35f005ef32Sjsg 
36f005ef32Sjsg #define DC_LOGGER_INIT(logger)
37f005ef32Sjsg 
get_tmds_output_pixel_clock_100hz(const struct dc_crtc_timing * timing)38f005ef32Sjsg static uint32_t get_tmds_output_pixel_clock_100hz(const struct dc_crtc_timing *timing)
39f005ef32Sjsg {
40f005ef32Sjsg 
41f005ef32Sjsg 	uint32_t pxl_clk = timing->pix_clk_100hz;
42f005ef32Sjsg 
43f005ef32Sjsg 	if (timing->pixel_encoding == PIXEL_ENCODING_YCBCR420)
44f005ef32Sjsg 		pxl_clk /= 2;
45f005ef32Sjsg 	else if (timing->pixel_encoding == PIXEL_ENCODING_YCBCR422)
46f005ef32Sjsg 		pxl_clk = pxl_clk * 2 / 3;
47f005ef32Sjsg 
48f005ef32Sjsg 	if (timing->display_color_depth == COLOR_DEPTH_101010)
49f005ef32Sjsg 		pxl_clk = pxl_clk * 10 / 8;
50f005ef32Sjsg 	else if (timing->display_color_depth == COLOR_DEPTH_121212)
51f005ef32Sjsg 		pxl_clk = pxl_clk * 12 / 8;
52f005ef32Sjsg 
53f005ef32Sjsg 	return pxl_clk;
54f005ef32Sjsg }
55f005ef32Sjsg 
dp_active_dongle_validate_timing(const struct dc_crtc_timing * timing,const struct dpcd_caps * dpcd_caps)56f005ef32Sjsg static bool dp_active_dongle_validate_timing(
57f005ef32Sjsg 		const struct dc_crtc_timing *timing,
58f005ef32Sjsg 		const struct dpcd_caps *dpcd_caps)
59f005ef32Sjsg {
60f005ef32Sjsg 	const struct dc_dongle_caps *dongle_caps = &dpcd_caps->dongle_caps;
61f005ef32Sjsg 
62f005ef32Sjsg 	switch (dpcd_caps->dongle_type) {
63f005ef32Sjsg 	case DISPLAY_DONGLE_DP_VGA_CONVERTER:
64f005ef32Sjsg 	case DISPLAY_DONGLE_DP_DVI_CONVERTER:
65f005ef32Sjsg 	case DISPLAY_DONGLE_DP_DVI_DONGLE:
66f005ef32Sjsg 		if (timing->pixel_encoding == PIXEL_ENCODING_RGB)
67f005ef32Sjsg 			return true;
68f005ef32Sjsg 		else
69f005ef32Sjsg 			return false;
70f005ef32Sjsg 	default:
71f005ef32Sjsg 		break;
72f005ef32Sjsg 	}
73f005ef32Sjsg 
74f005ef32Sjsg 	if (dpcd_caps->dongle_type == DISPLAY_DONGLE_DP_HDMI_CONVERTER &&
75f005ef32Sjsg 			dongle_caps->extendedCapValid == true) {
76f005ef32Sjsg 		/* Check Pixel Encoding */
77f005ef32Sjsg 		switch (timing->pixel_encoding) {
78f005ef32Sjsg 		case PIXEL_ENCODING_RGB:
79f005ef32Sjsg 		case PIXEL_ENCODING_YCBCR444:
80f005ef32Sjsg 			break;
81f005ef32Sjsg 		case PIXEL_ENCODING_YCBCR422:
82f005ef32Sjsg 			if (!dongle_caps->is_dp_hdmi_ycbcr422_pass_through)
83f005ef32Sjsg 				return false;
84f005ef32Sjsg 			break;
85f005ef32Sjsg 		case PIXEL_ENCODING_YCBCR420:
86f005ef32Sjsg 			if (!dongle_caps->is_dp_hdmi_ycbcr420_pass_through)
87f005ef32Sjsg 				return false;
88f005ef32Sjsg 			break;
89f005ef32Sjsg 		default:
90f005ef32Sjsg 			/* Invalid Pixel Encoding*/
91f005ef32Sjsg 			return false;
92f005ef32Sjsg 		}
93f005ef32Sjsg 
94f005ef32Sjsg 		switch (timing->display_color_depth) {
95f005ef32Sjsg 		case COLOR_DEPTH_666:
96f005ef32Sjsg 		case COLOR_DEPTH_888:
97f005ef32Sjsg 			/*888 and 666 should always be supported*/
98f005ef32Sjsg 			break;
99f005ef32Sjsg 		case COLOR_DEPTH_101010:
100f005ef32Sjsg 			if (dongle_caps->dp_hdmi_max_bpc < 10)
101f005ef32Sjsg 				return false;
102f005ef32Sjsg 			break;
103f005ef32Sjsg 		case COLOR_DEPTH_121212:
104f005ef32Sjsg 			if (dongle_caps->dp_hdmi_max_bpc < 12)
105f005ef32Sjsg 				return false;
106f005ef32Sjsg 			break;
107f005ef32Sjsg 		case COLOR_DEPTH_141414:
108f005ef32Sjsg 		case COLOR_DEPTH_161616:
109f005ef32Sjsg 		default:
110f005ef32Sjsg 			/* These color depths are currently not supported */
111f005ef32Sjsg 			return false;
112f005ef32Sjsg 		}
113f005ef32Sjsg 
114f005ef32Sjsg 		/* Check 3D format */
115f005ef32Sjsg 		switch (timing->timing_3d_format) {
116f005ef32Sjsg 		case TIMING_3D_FORMAT_NONE:
117f005ef32Sjsg 		case TIMING_3D_FORMAT_FRAME_ALTERNATE:
118f005ef32Sjsg 			/*Only frame alternate 3D is supported on active dongle*/
119f005ef32Sjsg 			break;
120f005ef32Sjsg 		default:
121f005ef32Sjsg 			/*other 3D formats are not supported due to bad infoframe translation */
122f005ef32Sjsg 			return false;
123f005ef32Sjsg 		}
124f005ef32Sjsg 
125f005ef32Sjsg 		if (dongle_caps->dp_hdmi_frl_max_link_bw_in_kbps > 0) { // DP to HDMI FRL converter
126f005ef32Sjsg 			struct dc_crtc_timing outputTiming = *timing;
127f005ef32Sjsg 
128f005ef32Sjsg #if defined(CONFIG_DRM_AMD_DC_FP)
129f005ef32Sjsg 			if (timing->flags.DSC && !timing->dsc_cfg.is_frl)
130f005ef32Sjsg 				/* DP input has DSC, HDMI FRL output doesn't have DSC, remove DSC from output timing */
131f005ef32Sjsg 				outputTiming.flags.DSC = 0;
132f005ef32Sjsg #endif
133f005ef32Sjsg 			if (dc_bandwidth_in_kbps_from_timing(&outputTiming, DC_LINK_ENCODING_HDMI_FRL) >
134f005ef32Sjsg 					dongle_caps->dp_hdmi_frl_max_link_bw_in_kbps)
135f005ef32Sjsg 				return false;
136f005ef32Sjsg 		} else { // DP to HDMI TMDS converter
137f005ef32Sjsg 			if (get_tmds_output_pixel_clock_100hz(timing) > (dongle_caps->dp_hdmi_max_pixel_clk_in_khz * 10))
138f005ef32Sjsg 				return false;
139f005ef32Sjsg 		}
140f005ef32Sjsg 	}
141f005ef32Sjsg 
142f005ef32Sjsg 	if (dpcd_caps->channel_coding_cap.bits.DP_128b_132b_SUPPORTED == 0 &&
143f005ef32Sjsg 			dpcd_caps->dsc_caps.dsc_basic_caps.fields.dsc_support.DSC_PASSTHROUGH_SUPPORT == 0 &&
144f005ef32Sjsg 			dongle_caps->dfp_cap_ext.supported) {
145f005ef32Sjsg 
146f005ef32Sjsg 		if (dongle_caps->dfp_cap_ext.max_pixel_rate_in_mps < (timing->pix_clk_100hz / 10000))
147f005ef32Sjsg 			return false;
148f005ef32Sjsg 
149f005ef32Sjsg 		if (dongle_caps->dfp_cap_ext.max_video_h_active_width < timing->h_addressable)
150f005ef32Sjsg 			return false;
151f005ef32Sjsg 
152f005ef32Sjsg 		if (dongle_caps->dfp_cap_ext.max_video_v_active_height < timing->v_addressable)
153f005ef32Sjsg 			return false;
154f005ef32Sjsg 
155f005ef32Sjsg 		if (timing->pixel_encoding == PIXEL_ENCODING_RGB) {
156f005ef32Sjsg 			if (!dongle_caps->dfp_cap_ext.encoding_format_caps.support_rgb)
157f005ef32Sjsg 				return false;
158f005ef32Sjsg 			if (timing->display_color_depth == COLOR_DEPTH_666 &&
159f005ef32Sjsg 					!dongle_caps->dfp_cap_ext.rgb_color_depth_caps.support_6bpc)
160f005ef32Sjsg 				return false;
161f005ef32Sjsg 			else if (timing->display_color_depth == COLOR_DEPTH_888 &&
162f005ef32Sjsg 					!dongle_caps->dfp_cap_ext.rgb_color_depth_caps.support_8bpc)
163f005ef32Sjsg 				return false;
164f005ef32Sjsg 			else if (timing->display_color_depth == COLOR_DEPTH_101010 &&
165f005ef32Sjsg 					!dongle_caps->dfp_cap_ext.rgb_color_depth_caps.support_10bpc)
166f005ef32Sjsg 				return false;
167f005ef32Sjsg 			else if (timing->display_color_depth == COLOR_DEPTH_121212 &&
168f005ef32Sjsg 					!dongle_caps->dfp_cap_ext.rgb_color_depth_caps.support_12bpc)
169f005ef32Sjsg 				return false;
170f005ef32Sjsg 			else if (timing->display_color_depth == COLOR_DEPTH_161616 &&
171f005ef32Sjsg 					!dongle_caps->dfp_cap_ext.rgb_color_depth_caps.support_16bpc)
172f005ef32Sjsg 				return false;
173f005ef32Sjsg 		} else if (timing->pixel_encoding == PIXEL_ENCODING_YCBCR444) {
174f005ef32Sjsg 			if (!dongle_caps->dfp_cap_ext.encoding_format_caps.support_rgb)
175f005ef32Sjsg 				return false;
176f005ef32Sjsg 			if (timing->display_color_depth == COLOR_DEPTH_888 &&
177f005ef32Sjsg 					!dongle_caps->dfp_cap_ext.ycbcr444_color_depth_caps.support_8bpc)
178f005ef32Sjsg 				return false;
179f005ef32Sjsg 			else if (timing->display_color_depth == COLOR_DEPTH_101010 &&
180f005ef32Sjsg 					!dongle_caps->dfp_cap_ext.ycbcr444_color_depth_caps.support_10bpc)
181f005ef32Sjsg 				return false;
182f005ef32Sjsg 			else if (timing->display_color_depth == COLOR_DEPTH_121212 &&
183f005ef32Sjsg 					!dongle_caps->dfp_cap_ext.ycbcr444_color_depth_caps.support_12bpc)
184f005ef32Sjsg 				return false;
185f005ef32Sjsg 			else if (timing->display_color_depth == COLOR_DEPTH_161616 &&
186f005ef32Sjsg 					!dongle_caps->dfp_cap_ext.ycbcr444_color_depth_caps.support_16bpc)
187f005ef32Sjsg 				return false;
188f005ef32Sjsg 		} else if (timing->pixel_encoding == PIXEL_ENCODING_YCBCR422) {
189f005ef32Sjsg 			if (!dongle_caps->dfp_cap_ext.encoding_format_caps.support_rgb)
190f005ef32Sjsg 				return false;
191f005ef32Sjsg 			if (timing->display_color_depth == COLOR_DEPTH_888 &&
192f005ef32Sjsg 					!dongle_caps->dfp_cap_ext.ycbcr422_color_depth_caps.support_8bpc)
193f005ef32Sjsg 				return false;
194f005ef32Sjsg 			else if (timing->display_color_depth == COLOR_DEPTH_101010 &&
195f005ef32Sjsg 					!dongle_caps->dfp_cap_ext.ycbcr422_color_depth_caps.support_10bpc)
196f005ef32Sjsg 				return false;
197f005ef32Sjsg 			else if (timing->display_color_depth == COLOR_DEPTH_121212 &&
198f005ef32Sjsg 					!dongle_caps->dfp_cap_ext.ycbcr422_color_depth_caps.support_12bpc)
199f005ef32Sjsg 				return false;
200f005ef32Sjsg 			else if (timing->display_color_depth == COLOR_DEPTH_161616 &&
201f005ef32Sjsg 					!dongle_caps->dfp_cap_ext.ycbcr422_color_depth_caps.support_16bpc)
202f005ef32Sjsg 				return false;
203f005ef32Sjsg 		} else if (timing->pixel_encoding == PIXEL_ENCODING_YCBCR420) {
204f005ef32Sjsg 			if (!dongle_caps->dfp_cap_ext.encoding_format_caps.support_rgb)
205f005ef32Sjsg 				return false;
206f005ef32Sjsg 			if (timing->display_color_depth == COLOR_DEPTH_888 &&
207f005ef32Sjsg 					!dongle_caps->dfp_cap_ext.ycbcr420_color_depth_caps.support_8bpc)
208f005ef32Sjsg 				return false;
209f005ef32Sjsg 			else if (timing->display_color_depth == COLOR_DEPTH_101010 &&
210f005ef32Sjsg 					!dongle_caps->dfp_cap_ext.ycbcr420_color_depth_caps.support_10bpc)
211f005ef32Sjsg 				return false;
212f005ef32Sjsg 			else if (timing->display_color_depth == COLOR_DEPTH_121212 &&
213f005ef32Sjsg 					!dongle_caps->dfp_cap_ext.ycbcr420_color_depth_caps.support_12bpc)
214f005ef32Sjsg 				return false;
215f005ef32Sjsg 			else if (timing->display_color_depth == COLOR_DEPTH_161616 &&
216f005ef32Sjsg 					!dongle_caps->dfp_cap_ext.ycbcr420_color_depth_caps.support_16bpc)
217f005ef32Sjsg 				return false;
218f005ef32Sjsg 		}
219f005ef32Sjsg 	}
220f005ef32Sjsg 
221f005ef32Sjsg 	return true;
222f005ef32Sjsg }
223f005ef32Sjsg 
dp_link_bandwidth_kbps(const struct dc_link * link,const struct dc_link_settings * link_settings)224f005ef32Sjsg uint32_t dp_link_bandwidth_kbps(
225f005ef32Sjsg 	const struct dc_link *link,
226f005ef32Sjsg 	const struct dc_link_settings *link_settings)
227f005ef32Sjsg {
228f005ef32Sjsg 	uint32_t total_data_bw_efficiency_x10000 = 0;
229f005ef32Sjsg 	uint32_t link_rate_per_lane_kbps = 0;
230f005ef32Sjsg 
231f005ef32Sjsg 	switch (link_dp_get_encoding_format(link_settings)) {
232f005ef32Sjsg 	case DP_8b_10b_ENCODING:
233f005ef32Sjsg 		/* For 8b/10b encoding:
234f005ef32Sjsg 		 * link rate is defined in the unit of LINK_RATE_REF_FREQ_IN_KHZ per DP byte per lane.
235f005ef32Sjsg 		 * data bandwidth efficiency is 80% with additional 3% overhead if FEC is supported.
236f005ef32Sjsg 		 */
237f005ef32Sjsg 		link_rate_per_lane_kbps = link_settings->link_rate * LINK_RATE_REF_FREQ_IN_KHZ * BITS_PER_DP_BYTE;
238f005ef32Sjsg 		total_data_bw_efficiency_x10000 = DATA_EFFICIENCY_8b_10b_x10000;
239f005ef32Sjsg 		if (dp_should_enable_fec(link)) {
240f005ef32Sjsg 			total_data_bw_efficiency_x10000 /= 100;
241f005ef32Sjsg 			total_data_bw_efficiency_x10000 *= DATA_EFFICIENCY_8b_10b_FEC_EFFICIENCY_x100;
242f005ef32Sjsg 		}
243f005ef32Sjsg 		break;
244f005ef32Sjsg 	case DP_128b_132b_ENCODING:
245f005ef32Sjsg 		/* For 128b/132b encoding:
246f005ef32Sjsg 		 * link rate is defined in the unit of 10mbps per lane.
247f005ef32Sjsg 		 * total data bandwidth efficiency is always 96.71%.
248f005ef32Sjsg 		 */
249f005ef32Sjsg 		link_rate_per_lane_kbps = link_settings->link_rate * 10000;
250f005ef32Sjsg 		total_data_bw_efficiency_x10000 = DATA_EFFICIENCY_128b_132b_x10000;
251f005ef32Sjsg 		break;
252f005ef32Sjsg 	default:
253f005ef32Sjsg 		break;
254f005ef32Sjsg 	}
255f005ef32Sjsg 
256f005ef32Sjsg 	/* overall effective link bandwidth = link rate per lane * lane count * total data bandwidth efficiency */
257f005ef32Sjsg 	return link_rate_per_lane_kbps * link_settings->lane_count / 10000 * total_data_bw_efficiency_x10000;
258f005ef32Sjsg }
259f005ef32Sjsg 
dp_validate_mode_timing(struct dc_link * link,const struct dc_crtc_timing * timing)260f005ef32Sjsg static bool dp_validate_mode_timing(
261f005ef32Sjsg 	struct dc_link *link,
262f005ef32Sjsg 	const struct dc_crtc_timing *timing)
263f005ef32Sjsg {
264f005ef32Sjsg 	uint32_t req_bw;
265f005ef32Sjsg 	uint32_t max_bw;
266f005ef32Sjsg 
267f005ef32Sjsg 	const struct dc_link_settings *link_setting;
268f005ef32Sjsg 
269f005ef32Sjsg 	/* According to spec, VSC SDP should be used if pixel format is YCbCr420 */
270f005ef32Sjsg 	if (timing->pixel_encoding == PIXEL_ENCODING_YCBCR420 &&
271f005ef32Sjsg 			!link->dpcd_caps.dprx_feature.bits.VSC_SDP_COLORIMETRY_SUPPORTED &&
272f005ef32Sjsg 			dal_graphics_object_id_get_connector_id(link->link_id) != CONNECTOR_ID_VIRTUAL)
273f005ef32Sjsg 		return false;
274f005ef32Sjsg 
275f005ef32Sjsg 	/*always DP fail safe mode*/
276f005ef32Sjsg 	if ((timing->pix_clk_100hz / 10) == (uint32_t) 25175 &&
277f005ef32Sjsg 		timing->h_addressable == (uint32_t) 640 &&
278f005ef32Sjsg 		timing->v_addressable == (uint32_t) 480)
279f005ef32Sjsg 		return true;
280f005ef32Sjsg 
281f005ef32Sjsg 	link_setting = dp_get_verified_link_cap(link);
282f005ef32Sjsg 
283f005ef32Sjsg 	/* TODO: DYNAMIC_VALIDATION needs to be implemented */
284f005ef32Sjsg 	/*if (flags.DYNAMIC_VALIDATION == 1 &&
285f005ef32Sjsg 		link->verified_link_cap.lane_count != LANE_COUNT_UNKNOWN)
286f005ef32Sjsg 		link_setting = &link->verified_link_cap;
287f005ef32Sjsg 	*/
288f005ef32Sjsg 
289f005ef32Sjsg 	req_bw = dc_bandwidth_in_kbps_from_timing(timing, dc_link_get_highest_encoding_format(link));
290f005ef32Sjsg 	max_bw = dp_link_bandwidth_kbps(link, link_setting);
291f005ef32Sjsg 
292f005ef32Sjsg 	if (req_bw <= max_bw) {
293f005ef32Sjsg 		/* remember the biggest mode here, during
294f005ef32Sjsg 		 * initial link training (to get
295f005ef32Sjsg 		 * verified_link_cap), LS sends event about
296f005ef32Sjsg 		 * cannot train at reported cap to upper
297f005ef32Sjsg 		 * layer and upper layer will re-enumerate modes.
298f005ef32Sjsg 		 * this is not necessary if the lower
299f005ef32Sjsg 		 * verified_link_cap is enough to drive
300f005ef32Sjsg 		 * all the modes */
301f005ef32Sjsg 
302f005ef32Sjsg 		/* TODO: DYNAMIC_VALIDATION needs to be implemented */
303f005ef32Sjsg 		/* if (flags.DYNAMIC_VALIDATION == 1)
304f005ef32Sjsg 			dpsst->max_req_bw_for_verified_linkcap = dal_max(
305f005ef32Sjsg 				dpsst->max_req_bw_for_verified_linkcap, req_bw); */
306f005ef32Sjsg 		return true;
307f005ef32Sjsg 	} else
308f005ef32Sjsg 		return false;
309f005ef32Sjsg }
310f005ef32Sjsg 
link_validate_mode_timing(const struct dc_stream_state * stream,struct dc_link * link,const struct dc_crtc_timing * timing)311f005ef32Sjsg enum dc_status link_validate_mode_timing(
312f005ef32Sjsg 		const struct dc_stream_state *stream,
313f005ef32Sjsg 		struct dc_link *link,
314f005ef32Sjsg 		const struct dc_crtc_timing *timing)
315f005ef32Sjsg {
316f005ef32Sjsg 	uint32_t max_pix_clk = stream->link->dongle_max_pix_clk * 10;
317f005ef32Sjsg 	struct dpcd_caps *dpcd_caps = &link->dpcd_caps;
318f005ef32Sjsg 
319f005ef32Sjsg 	/* A hack to avoid failing any modes for EDID override feature on
320f005ef32Sjsg 	 * topology change such as lower quality cable for DP or different dongle
321f005ef32Sjsg 	 */
322f005ef32Sjsg 	if (link->remote_sinks[0] && link->remote_sinks[0]->sink_signal == SIGNAL_TYPE_VIRTUAL)
323f005ef32Sjsg 		return DC_OK;
324f005ef32Sjsg 
325f005ef32Sjsg 	/* Passive Dongle */
326f005ef32Sjsg 	if (max_pix_clk != 0 && get_tmds_output_pixel_clock_100hz(timing) > max_pix_clk)
327f005ef32Sjsg 		return DC_EXCEED_DONGLE_CAP;
328f005ef32Sjsg 
329f005ef32Sjsg 	/* Active Dongle*/
330f005ef32Sjsg 	if (!dp_active_dongle_validate_timing(timing, dpcd_caps))
331f005ef32Sjsg 		return DC_EXCEED_DONGLE_CAP;
332f005ef32Sjsg 
333f005ef32Sjsg 	switch (stream->signal) {
334f005ef32Sjsg 	case SIGNAL_TYPE_EDP:
335f005ef32Sjsg 	case SIGNAL_TYPE_DISPLAY_PORT:
336f005ef32Sjsg 		if (!dp_validate_mode_timing(
337f005ef32Sjsg 				link,
338f005ef32Sjsg 				timing))
339f005ef32Sjsg 			return DC_NO_DP_LINK_BANDWIDTH;
340f005ef32Sjsg 		break;
341f005ef32Sjsg 
342f005ef32Sjsg 	default:
343f005ef32Sjsg 		break;
344f005ef32Sjsg 	}
345f005ef32Sjsg 
346f005ef32Sjsg 	return DC_OK;
347f005ef32Sjsg }
348f005ef32Sjsg 
349f2c56d12Sjsg /*
350f2c56d12Sjsg  * This function calculates the bandwidth required for the stream timing
351f2c56d12Sjsg  * and aggregates the stream bandwidth for the respective dpia link
352f2c56d12Sjsg  *
353f2c56d12Sjsg  * @stream: pointer to the dc_stream_state struct instance
354f2c56d12Sjsg  * @num_streams: number of streams to be validated
355f2c56d12Sjsg  *
356f2c56d12Sjsg  * return: true if validation is succeeded
357f2c56d12Sjsg  */
link_validate_dpia_bandwidth(const struct dc_stream_state * stream,const unsigned int num_streams)358f005ef32Sjsg bool link_validate_dpia_bandwidth(const struct dc_stream_state *stream, const unsigned int num_streams)
359f005ef32Sjsg {
360f2c56d12Sjsg 	int bw_needed[MAX_DPIA_NUM] = {0};
361f2c56d12Sjsg 	struct dc_link *dpia_link[MAX_DPIA_NUM] = {0};
362f2c56d12Sjsg 	int num_dpias = 0;
363f005ef32Sjsg 
364*a8749599Sjsg 	for (unsigned int i = 0; i < num_streams; ++i) {
365f2c56d12Sjsg 		if (stream[i].signal == SIGNAL_TYPE_DISPLAY_PORT) {
366f2c56d12Sjsg 			/* new dpia sst stream, check whether it exceeds max dpia */
367f2c56d12Sjsg 			if (num_dpias >= MAX_DPIA_NUM)
368f2c56d12Sjsg 				return false;
369f005ef32Sjsg 
370f2c56d12Sjsg 			dpia_link[num_dpias] = stream[i].link;
371f2c56d12Sjsg 			bw_needed[num_dpias] = dc_bandwidth_in_kbps_from_timing(&stream[i].timing,
372f2c56d12Sjsg 					dc_link_get_highest_encoding_format(dpia_link[num_dpias]));
373f2c56d12Sjsg 			num_dpias++;
374f2c56d12Sjsg 		} else if (stream[i].signal == SIGNAL_TYPE_DISPLAY_PORT_MST) {
375f2c56d12Sjsg 			uint8_t j = 0;
376f2c56d12Sjsg 			/* check whether its a known dpia link */
377f2c56d12Sjsg 			for (; j < num_dpias; ++j) {
378f2c56d12Sjsg 				if (dpia_link[j] == stream[i].link)
379f2c56d12Sjsg 					break;
380f005ef32Sjsg 			}
381f005ef32Sjsg 
382f2c56d12Sjsg 			if (j == num_dpias) {
383f2c56d12Sjsg 				/* new dpia mst stream, check whether it exceeds max dpia */
384f2c56d12Sjsg 				if (num_dpias >= MAX_DPIA_NUM)
385f2c56d12Sjsg 					return false;
386f2c56d12Sjsg 				else {
387f2c56d12Sjsg 					dpia_link[j] = stream[i].link;
388f2c56d12Sjsg 					num_dpias++;
389f2c56d12Sjsg 				}
390f2c56d12Sjsg 			}
391f005ef32Sjsg 
392f2c56d12Sjsg 			bw_needed[j] += dc_bandwidth_in_kbps_from_timing(&stream[i].timing,
393f2c56d12Sjsg 				dc_link_get_highest_encoding_format(dpia_link[j]));
394f2c56d12Sjsg 		}
395f2c56d12Sjsg 	}
396f2c56d12Sjsg 
397f2c56d12Sjsg 	/* Include dp overheads */
398f2c56d12Sjsg 	for (uint8_t i = 0; i < num_dpias; ++i) {
399f2c56d12Sjsg 		int dp_overhead = 0;
400f2c56d12Sjsg 
401f2c56d12Sjsg 		dp_overhead = link_dp_dpia_get_dp_overhead_in_dp_tunneling(dpia_link[i]);
402f2c56d12Sjsg 		bw_needed[i] += dp_overhead;
403f2c56d12Sjsg 	}
404f2c56d12Sjsg 
405f2c56d12Sjsg 	return dpia_validate_usb4_bw(dpia_link, bw_needed, num_dpias);
406f005ef32Sjsg }
407