591 lines
20 KiB
C
591 lines
20 KiB
C
/*
|
|
* Copyright 2023 Advanced Micro Devices, Inc.
|
|
*
|
|
* Permission is hereby granted, free of charge, to any person obtaining a
|
|
* copy of this software and associated documentation files (the "Software"),
|
|
* to deal in the Software without restriction, including without limitation
|
|
* the rights to use, copy, modify, merge, publish, distribute, sublicense,
|
|
* and/or sell copies of the Software, and to permit persons to whom the
|
|
* Software is furnished to do so, subject to the following conditions:
|
|
*
|
|
* The above copyright notice and this permission notice shall be included in
|
|
* all copies or substantial portions of the Software.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
|
|
* THE COPYRIGHT HOLDER(S) OR AUTHOR(S) BE LIABLE FOR ANY CLAIM, DAMAGES OR
|
|
* OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
|
|
* ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
|
|
* OTHER DEALINGS IN THE SOFTWARE.
|
|
*
|
|
* Authors: AMD
|
|
*
|
|
*/
|
|
|
|
/* FILE POLICY AND INTENDED USAGE:
|
|
* This file owns timing validation against various link limitations. (ex.
|
|
* link bandwidth, receiver capability or our hardware capability) It also
|
|
* provides helper functions exposing bandwidth formulas used in validation.
|
|
*/
|
|
#include "link_validation.h"
|
|
#include "protocols/link_dp_capability.h"
|
|
#include "protocols/link_dp_dpia_bw.h"
|
|
#include "resource.h"
|
|
|
|
#define DC_LOGGER_INIT(logger)
|
|
|
|
static uint32_t get_tmds_output_pixel_clock_100hz(const struct dc_crtc_timing *timing)
|
|
{
|
|
|
|
uint32_t pxl_clk = timing->pix_clk_100hz;
|
|
|
|
if (timing->pixel_encoding == PIXEL_ENCODING_YCBCR420)
|
|
pxl_clk /= 2;
|
|
else if (timing->pixel_encoding == PIXEL_ENCODING_YCBCR422)
|
|
pxl_clk = pxl_clk * 2 / 3;
|
|
|
|
if (timing->display_color_depth == COLOR_DEPTH_101010)
|
|
pxl_clk = pxl_clk * 10 / 8;
|
|
else if (timing->display_color_depth == COLOR_DEPTH_121212)
|
|
pxl_clk = pxl_clk * 12 / 8;
|
|
|
|
return pxl_clk;
|
|
}
|
|
|
|
static bool dp_active_dongle_validate_timing(
|
|
const struct dc_crtc_timing *timing,
|
|
const struct dpcd_caps *dpcd_caps)
|
|
{
|
|
const struct dc_dongle_caps *dongle_caps = &dpcd_caps->dongle_caps;
|
|
|
|
switch (dpcd_caps->dongle_type) {
|
|
case DISPLAY_DONGLE_DP_VGA_CONVERTER:
|
|
case DISPLAY_DONGLE_DP_DVI_CONVERTER:
|
|
case DISPLAY_DONGLE_DP_DVI_DONGLE:
|
|
if (timing->pixel_encoding == PIXEL_ENCODING_RGB)
|
|
return true;
|
|
else
|
|
return false;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
if (dpcd_caps->dongle_type == DISPLAY_DONGLE_DP_HDMI_CONVERTER &&
|
|
dongle_caps->extendedCapValid == true) {
|
|
/* Check Pixel Encoding */
|
|
switch (timing->pixel_encoding) {
|
|
case PIXEL_ENCODING_RGB:
|
|
case PIXEL_ENCODING_YCBCR444:
|
|
break;
|
|
case PIXEL_ENCODING_YCBCR422:
|
|
if (!dongle_caps->is_dp_hdmi_ycbcr422_pass_through)
|
|
return false;
|
|
break;
|
|
case PIXEL_ENCODING_YCBCR420:
|
|
if (!dongle_caps->is_dp_hdmi_ycbcr420_pass_through)
|
|
return false;
|
|
break;
|
|
default:
|
|
/* Invalid Pixel Encoding*/
|
|
return false;
|
|
}
|
|
|
|
switch (timing->display_color_depth) {
|
|
case COLOR_DEPTH_666:
|
|
case COLOR_DEPTH_888:
|
|
/*888 and 666 should always be supported*/
|
|
break;
|
|
case COLOR_DEPTH_101010:
|
|
if (dongle_caps->dp_hdmi_max_bpc < 10)
|
|
return false;
|
|
break;
|
|
case COLOR_DEPTH_121212:
|
|
if (dongle_caps->dp_hdmi_max_bpc < 12)
|
|
return false;
|
|
break;
|
|
case COLOR_DEPTH_141414:
|
|
case COLOR_DEPTH_161616:
|
|
default:
|
|
/* These color depths are currently not supported */
|
|
return false;
|
|
}
|
|
|
|
/* Check 3D format */
|
|
switch (timing->timing_3d_format) {
|
|
case TIMING_3D_FORMAT_NONE:
|
|
case TIMING_3D_FORMAT_FRAME_ALTERNATE:
|
|
/*Only frame alternate 3D is supported on active dongle*/
|
|
break;
|
|
default:
|
|
/*other 3D formats are not supported due to bad infoframe translation */
|
|
return false;
|
|
}
|
|
|
|
if (dongle_caps->dp_hdmi_frl_max_link_bw_in_kbps > 0) { // DP to HDMI FRL converter
|
|
struct dc_crtc_timing outputTiming = *timing;
|
|
|
|
if (timing->flags.DSC && !timing->dsc_cfg.is_frl)
|
|
/* DP input has DSC, HDMI FRL output doesn't have DSC, remove DSC from output timing */
|
|
outputTiming.flags.DSC = 0;
|
|
if (dc_bandwidth_in_kbps_from_timing(&outputTiming, DC_LINK_ENCODING_HDMI_FRL) >
|
|
dongle_caps->dp_hdmi_frl_max_link_bw_in_kbps)
|
|
return false;
|
|
} else { // DP to HDMI TMDS converter
|
|
if (get_tmds_output_pixel_clock_100hz(timing) > (dongle_caps->dp_hdmi_max_pixel_clk_in_khz * 10))
|
|
return false;
|
|
}
|
|
}
|
|
|
|
if (dpcd_caps->channel_coding_cap.bits.DP_128b_132b_SUPPORTED == 0 &&
|
|
dpcd_caps->dsc_caps.dsc_basic_caps.fields.dsc_support.DSC_PASSTHROUGH_SUPPORT == 0 &&
|
|
dongle_caps->dfp_cap_ext.supported) {
|
|
|
|
if (dongle_caps->dfp_cap_ext.max_pixel_rate_in_mps < (timing->pix_clk_100hz / 10000))
|
|
return false;
|
|
|
|
if (dongle_caps->dfp_cap_ext.max_video_h_active_width < timing->h_addressable)
|
|
return false;
|
|
|
|
if (dongle_caps->dfp_cap_ext.max_video_v_active_height < timing->v_addressable)
|
|
return false;
|
|
|
|
if (timing->pixel_encoding == PIXEL_ENCODING_RGB) {
|
|
if (!dongle_caps->dfp_cap_ext.encoding_format_caps.support_rgb)
|
|
return false;
|
|
if (timing->display_color_depth == COLOR_DEPTH_666 &&
|
|
!dongle_caps->dfp_cap_ext.rgb_color_depth_caps.support_6bpc)
|
|
return false;
|
|
else if (timing->display_color_depth == COLOR_DEPTH_888 &&
|
|
!dongle_caps->dfp_cap_ext.rgb_color_depth_caps.support_8bpc)
|
|
return false;
|
|
else if (timing->display_color_depth == COLOR_DEPTH_101010 &&
|
|
!dongle_caps->dfp_cap_ext.rgb_color_depth_caps.support_10bpc)
|
|
return false;
|
|
else if (timing->display_color_depth == COLOR_DEPTH_121212 &&
|
|
!dongle_caps->dfp_cap_ext.rgb_color_depth_caps.support_12bpc)
|
|
return false;
|
|
else if (timing->display_color_depth == COLOR_DEPTH_161616 &&
|
|
!dongle_caps->dfp_cap_ext.rgb_color_depth_caps.support_16bpc)
|
|
return false;
|
|
} else if (timing->pixel_encoding == PIXEL_ENCODING_YCBCR444) {
|
|
if (!dongle_caps->dfp_cap_ext.encoding_format_caps.support_rgb)
|
|
return false;
|
|
if (timing->display_color_depth == COLOR_DEPTH_888 &&
|
|
!dongle_caps->dfp_cap_ext.ycbcr444_color_depth_caps.support_8bpc)
|
|
return false;
|
|
else if (timing->display_color_depth == COLOR_DEPTH_101010 &&
|
|
!dongle_caps->dfp_cap_ext.ycbcr444_color_depth_caps.support_10bpc)
|
|
return false;
|
|
else if (timing->display_color_depth == COLOR_DEPTH_121212 &&
|
|
!dongle_caps->dfp_cap_ext.ycbcr444_color_depth_caps.support_12bpc)
|
|
return false;
|
|
else if (timing->display_color_depth == COLOR_DEPTH_161616 &&
|
|
!dongle_caps->dfp_cap_ext.ycbcr444_color_depth_caps.support_16bpc)
|
|
return false;
|
|
} else if (timing->pixel_encoding == PIXEL_ENCODING_YCBCR422) {
|
|
if (!dongle_caps->dfp_cap_ext.encoding_format_caps.support_rgb)
|
|
return false;
|
|
if (timing->display_color_depth == COLOR_DEPTH_888 &&
|
|
!dongle_caps->dfp_cap_ext.ycbcr422_color_depth_caps.support_8bpc)
|
|
return false;
|
|
else if (timing->display_color_depth == COLOR_DEPTH_101010 &&
|
|
!dongle_caps->dfp_cap_ext.ycbcr422_color_depth_caps.support_10bpc)
|
|
return false;
|
|
else if (timing->display_color_depth == COLOR_DEPTH_121212 &&
|
|
!dongle_caps->dfp_cap_ext.ycbcr422_color_depth_caps.support_12bpc)
|
|
return false;
|
|
else if (timing->display_color_depth == COLOR_DEPTH_161616 &&
|
|
!dongle_caps->dfp_cap_ext.ycbcr422_color_depth_caps.support_16bpc)
|
|
return false;
|
|
} else if (timing->pixel_encoding == PIXEL_ENCODING_YCBCR420) {
|
|
if (!dongle_caps->dfp_cap_ext.encoding_format_caps.support_rgb)
|
|
return false;
|
|
if (timing->display_color_depth == COLOR_DEPTH_888 &&
|
|
!dongle_caps->dfp_cap_ext.ycbcr420_color_depth_caps.support_8bpc)
|
|
return false;
|
|
else if (timing->display_color_depth == COLOR_DEPTH_101010 &&
|
|
!dongle_caps->dfp_cap_ext.ycbcr420_color_depth_caps.support_10bpc)
|
|
return false;
|
|
else if (timing->display_color_depth == COLOR_DEPTH_121212 &&
|
|
!dongle_caps->dfp_cap_ext.ycbcr420_color_depth_caps.support_12bpc)
|
|
return false;
|
|
else if (timing->display_color_depth == COLOR_DEPTH_161616 &&
|
|
!dongle_caps->dfp_cap_ext.ycbcr420_color_depth_caps.support_16bpc)
|
|
return false;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
uint32_t dp_link_bandwidth_kbps(
|
|
const struct dc_link *link,
|
|
const struct dc_link_settings *link_settings)
|
|
{
|
|
uint32_t total_data_bw_efficiency_x10000 = 0;
|
|
uint32_t link_rate_per_lane_kbps = 0;
|
|
|
|
switch (link_dp_get_encoding_format(link_settings)) {
|
|
case DP_8b_10b_ENCODING:
|
|
/* For 8b/10b encoding:
|
|
* link rate is defined in the unit of LINK_RATE_REF_FREQ_IN_KHZ per DP byte per lane.
|
|
* data bandwidth efficiency is 80% with additional 3% overhead if FEC is supported.
|
|
*/
|
|
link_rate_per_lane_kbps = link_settings->link_rate * LINK_RATE_REF_FREQ_IN_KHZ * BITS_PER_DP_BYTE;
|
|
total_data_bw_efficiency_x10000 = DATA_EFFICIENCY_8b_10b_x10000;
|
|
if (dp_should_enable_fec(link)) {
|
|
total_data_bw_efficiency_x10000 /= 100;
|
|
total_data_bw_efficiency_x10000 *= DATA_EFFICIENCY_8b_10b_FEC_EFFICIENCY_x100;
|
|
}
|
|
break;
|
|
case DP_128b_132b_ENCODING:
|
|
/* For 128b/132b encoding:
|
|
* link rate is defined in the unit of 10mbps per lane.
|
|
* total data bandwidth efficiency is always 96.71%.
|
|
*/
|
|
link_rate_per_lane_kbps = link_settings->link_rate * 10000;
|
|
total_data_bw_efficiency_x10000 = DATA_EFFICIENCY_128b_132b_x10000;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
/* overall effective link bandwidth = link rate per lane * lane count * total data bandwidth efficiency */
|
|
return link_rate_per_lane_kbps * link_settings->lane_count / 10000 * total_data_bw_efficiency_x10000;
|
|
}
|
|
|
|
static bool dp_validate_mode_timing(
|
|
struct dc_link *link,
|
|
const struct dc_crtc_timing *timing)
|
|
{
|
|
uint32_t req_bw;
|
|
uint32_t max_bw;
|
|
|
|
const struct dc_link_settings *link_setting;
|
|
|
|
/* According to spec, VSC SDP should be used if pixel format is YCbCr420 */
|
|
if (timing->pixel_encoding == PIXEL_ENCODING_YCBCR420 &&
|
|
!link->dpcd_caps.dprx_feature.bits.VSC_SDP_COLORIMETRY_SUPPORTED &&
|
|
dal_graphics_object_id_get_connector_id(link->link_id) != CONNECTOR_ID_VIRTUAL)
|
|
return false;
|
|
|
|
/*always DP fail safe mode*/
|
|
if ((timing->pix_clk_100hz / 10) == (uint32_t) 25175 &&
|
|
timing->h_addressable == (uint32_t) 640 &&
|
|
timing->v_addressable == (uint32_t) 480)
|
|
return true;
|
|
|
|
link_setting = dp_get_verified_link_cap(link);
|
|
|
|
/* TODO: DYNAMIC_VALIDATION needs to be implemented */
|
|
/*if (flags.DYNAMIC_VALIDATION == 1 &&
|
|
link->verified_link_cap.lane_count != LANE_COUNT_UNKNOWN)
|
|
link_setting = &link->verified_link_cap;
|
|
*/
|
|
|
|
req_bw = dc_bandwidth_in_kbps_from_timing(timing, dc_link_get_highest_encoding_format(link));
|
|
max_bw = dp_link_bandwidth_kbps(link, link_setting);
|
|
|
|
bool is_max_uncompressed_pixel_rate_exceeded = link->dpcd_caps.max_uncompressed_pixel_rate_cap.bits.valid &&
|
|
timing->pix_clk_100hz > link->dpcd_caps.max_uncompressed_pixel_rate_cap.bits.max_uncompressed_pixel_rate_cap * 10000;
|
|
|
|
if (is_max_uncompressed_pixel_rate_exceeded && !timing->flags.DSC) {
|
|
return false;
|
|
}
|
|
|
|
if (req_bw <= max_bw) {
|
|
/* remember the biggest mode here, during
|
|
* initial link training (to get
|
|
* verified_link_cap), LS sends event about
|
|
* cannot train at reported cap to upper
|
|
* layer and upper layer will re-enumerate modes.
|
|
* this is not necessary if the lower
|
|
* verified_link_cap is enough to drive
|
|
* all the modes */
|
|
|
|
/* TODO: DYNAMIC_VALIDATION needs to be implemented */
|
|
/* if (flags.DYNAMIC_VALIDATION == 1)
|
|
dpsst->max_req_bw_for_verified_linkcap = dal_max(
|
|
dpsst->max_req_bw_for_verified_linkcap, req_bw); */
|
|
return true;
|
|
} else
|
|
return false;
|
|
}
|
|
|
|
enum dc_status link_validate_mode_timing(
|
|
const struct dc_stream_state *stream,
|
|
struct dc_link *link,
|
|
const struct dc_crtc_timing *timing)
|
|
{
|
|
uint32_t max_pix_clk = stream->link->dongle_max_pix_clk * 10;
|
|
struct dpcd_caps *dpcd_caps = &link->dpcd_caps;
|
|
|
|
/* A hack to avoid failing any modes for EDID override feature on
|
|
* topology change such as lower quality cable for DP or different dongle
|
|
*/
|
|
if (link->remote_sinks[0] && link->remote_sinks[0]->sink_signal == SIGNAL_TYPE_VIRTUAL)
|
|
return DC_OK;
|
|
|
|
/* Passive Dongle */
|
|
if (max_pix_clk != 0 && get_tmds_output_pixel_clock_100hz(timing) > max_pix_clk)
|
|
return DC_EXCEED_DONGLE_CAP;
|
|
|
|
/* Active Dongle*/
|
|
if (!dp_active_dongle_validate_timing(timing, dpcd_caps))
|
|
return DC_EXCEED_DONGLE_CAP;
|
|
|
|
switch (stream->signal) {
|
|
case SIGNAL_TYPE_EDP:
|
|
case SIGNAL_TYPE_DISPLAY_PORT:
|
|
if (!dp_validate_mode_timing(
|
|
link,
|
|
timing))
|
|
return DC_NO_DP_LINK_BANDWIDTH;
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return DC_OK;
|
|
}
|
|
|
|
/*
|
|
* This function calculates the bandwidth required for the stream timing
|
|
* and aggregates the stream bandwidth for the respective dpia link
|
|
*
|
|
* @stream: pointer to the dc_stream_state struct instance
|
|
* @num_streams: number of streams to be validated
|
|
*
|
|
* return: true if validation is succeeded
|
|
*/
|
|
bool link_validate_dpia_bandwidth(const struct dc_stream_state *stream, const unsigned int num_streams)
|
|
{
|
|
int bw_needed[MAX_DPIA_NUM] = {0};
|
|
struct dc_link *dpia_link[MAX_DPIA_NUM] = {0};
|
|
int num_dpias = 0;
|
|
|
|
for (unsigned int i = 0; i < num_streams; ++i) {
|
|
if (stream[i].signal == SIGNAL_TYPE_DISPLAY_PORT) {
|
|
/* new dpia sst stream, check whether it exceeds max dpia */
|
|
if (num_dpias >= MAX_DPIA_NUM)
|
|
return false;
|
|
|
|
dpia_link[num_dpias] = stream[i].link;
|
|
bw_needed[num_dpias] = dc_bandwidth_in_kbps_from_timing(&stream[i].timing,
|
|
dc_link_get_highest_encoding_format(dpia_link[num_dpias]));
|
|
num_dpias++;
|
|
} else if (stream[i].signal == SIGNAL_TYPE_DISPLAY_PORT_MST) {
|
|
uint8_t j = 0;
|
|
/* check whether its a known dpia link */
|
|
for (; j < num_dpias; ++j) {
|
|
if (dpia_link[j] == stream[i].link)
|
|
break;
|
|
}
|
|
|
|
if (j == num_dpias) {
|
|
/* new dpia mst stream, check whether it exceeds max dpia */
|
|
if (num_dpias >= MAX_DPIA_NUM)
|
|
return false;
|
|
else {
|
|
dpia_link[j] = stream[i].link;
|
|
num_dpias++;
|
|
}
|
|
}
|
|
|
|
bw_needed[j] += dc_bandwidth_in_kbps_from_timing(&stream[i].timing,
|
|
dc_link_get_highest_encoding_format(dpia_link[j]));
|
|
}
|
|
}
|
|
|
|
/* Include dp overheads */
|
|
for (uint8_t i = 0; i < num_dpias; ++i) {
|
|
int dp_overhead = 0;
|
|
|
|
dp_overhead = link_dp_dpia_get_dp_overhead_in_dp_tunneling(dpia_link[i]);
|
|
bw_needed[i] += dp_overhead;
|
|
}
|
|
|
|
return dpia_validate_usb4_bw(dpia_link, bw_needed, num_dpias);
|
|
}
|
|
|
|
struct dp_audio_layout_config {
|
|
uint8_t layouts_per_sample_denom;
|
|
uint8_t symbols_per_layout;
|
|
uint8_t max_layouts_per_audio_sdp;
|
|
};
|
|
|
|
static void get_audio_layout_config(
|
|
uint32_t channel_count,
|
|
enum dp_link_encoding encoding,
|
|
struct dp_audio_layout_config *output)
|
|
{
|
|
memset(output, 0, sizeof(struct dp_audio_layout_config));
|
|
|
|
/* Assuming L-PCM audio. Current implementation uses max 1 layout per SDP,
|
|
* with each layout being the same size (8ch layout).
|
|
*/
|
|
if (encoding == DP_8b_10b_ENCODING) {
|
|
if (channel_count == 2) {
|
|
output->layouts_per_sample_denom = 4;
|
|
output->symbols_per_layout = 40;
|
|
output->max_layouts_per_audio_sdp = 1;
|
|
} else if (channel_count == 8 || channel_count == 6) {
|
|
output->layouts_per_sample_denom = 1;
|
|
output->symbols_per_layout = 40;
|
|
output->max_layouts_per_audio_sdp = 1;
|
|
}
|
|
} else if (encoding == DP_128b_132b_ENCODING) {
|
|
if (channel_count == 2) {
|
|
output->layouts_per_sample_denom = 4;
|
|
output->symbols_per_layout = 10;
|
|
output->max_layouts_per_audio_sdp = 1;
|
|
} else if (channel_count == 8 || channel_count == 6) {
|
|
output->layouts_per_sample_denom = 1;
|
|
output->symbols_per_layout = 10;
|
|
output->max_layouts_per_audio_sdp = 1;
|
|
}
|
|
}
|
|
}
|
|
|
|
static uint32_t get_av_stream_map_lane_count(
|
|
enum dp_link_encoding encoding,
|
|
enum dc_lane_count lane_count,
|
|
bool is_mst)
|
|
{
|
|
uint32_t av_stream_map_lane_count = 0;
|
|
|
|
if (encoding == DP_8b_10b_ENCODING) {
|
|
if (!is_mst)
|
|
av_stream_map_lane_count = lane_count;
|
|
else
|
|
av_stream_map_lane_count = 4;
|
|
} else if (encoding == DP_128b_132b_ENCODING) {
|
|
av_stream_map_lane_count = 4;
|
|
}
|
|
|
|
ASSERT(av_stream_map_lane_count != 0);
|
|
|
|
return av_stream_map_lane_count;
|
|
}
|
|
|
|
static uint32_t get_audio_sdp_overhead(
|
|
enum dp_link_encoding encoding,
|
|
enum dc_lane_count lane_count,
|
|
bool is_mst)
|
|
{
|
|
uint32_t audio_sdp_overhead = 0;
|
|
|
|
if (encoding == DP_8b_10b_ENCODING) {
|
|
if (is_mst)
|
|
audio_sdp_overhead = 16; /* 4 * 2 + 8 */
|
|
else
|
|
audio_sdp_overhead = lane_count * 2 + 8;
|
|
} else if (encoding == DP_128b_132b_ENCODING) {
|
|
audio_sdp_overhead = 10; /* 4 x 2.5 */
|
|
}
|
|
|
|
ASSERT(audio_sdp_overhead != 0);
|
|
|
|
return audio_sdp_overhead;
|
|
}
|
|
|
|
/* Current calculation only applicable for 8b/10b MST and 128b/132b SST/MST.
|
|
*/
|
|
static uint32_t calculate_overhead_hblank_bw_in_symbols(
|
|
uint32_t max_slice_h)
|
|
{
|
|
uint32_t overhead_hblank_bw = 0; /* in stream symbols */
|
|
|
|
overhead_hblank_bw += max_slice_h * 4; /* EOC overhead */
|
|
overhead_hblank_bw += 12; /* Main link overhead (VBID, BS/BE) */
|
|
|
|
return overhead_hblank_bw;
|
|
}
|
|
|
|
uint32_t dp_required_hblank_size_bytes(
|
|
const struct dc_link *link,
|
|
struct dp_audio_bandwidth_params *audio_params)
|
|
{
|
|
/* Main logic from dce_audio is duplicated here, with the main
|
|
* difference being:
|
|
* - Pre-determined lane count of 4
|
|
* - Assumed 16 dsc slices for worst case
|
|
* - Assumed SDP split disabled for worst case
|
|
* TODO: Unify logic from dce_audio to prevent duplicated logic.
|
|
*/
|
|
|
|
const struct dc_crtc_timing *timing = audio_params->crtc_timing;
|
|
const uint32_t channel_count = audio_params->channel_count;
|
|
const uint32_t sample_rate_hz = audio_params->sample_rate_hz;
|
|
const enum dp_link_encoding link_encoding = audio_params->link_encoding;
|
|
|
|
// 8b/10b MST and 128b/132b are always 4 logical lanes.
|
|
const uint32_t lane_count = 4;
|
|
const bool is_mst = (link->connector_signal == SIGNAL_TYPE_DISPLAY_PORT);
|
|
// Maximum slice count is with ODM 4:1, 4 slices per DSC
|
|
const uint32_t max_slices_h = 16;
|
|
|
|
const uint32_t av_stream_map_lane_count = get_av_stream_map_lane_count(
|
|
link_encoding, lane_count, is_mst);
|
|
const uint32_t audio_sdp_overhead = get_audio_sdp_overhead(
|
|
link_encoding, lane_count, is_mst);
|
|
struct dp_audio_layout_config layout_config;
|
|
|
|
if (link_encoding == DP_8b_10b_ENCODING && link->connector_signal == SIGNAL_TYPE_DISPLAY_PORT)
|
|
return 0;
|
|
|
|
get_audio_layout_config(
|
|
channel_count, link_encoding, &layout_config);
|
|
|
|
/* DP spec recommends between 1.05 to 1.1 safety margin to prevent sample under-run */
|
|
struct fixed31_32 audio_sdp_margin = dc_fixpt_from_fraction(110, 100);
|
|
struct fixed31_32 horizontal_line_freq_khz = dc_fixpt_from_fraction(
|
|
timing->pix_clk_100hz, (long long)timing->h_total * 10);
|
|
struct fixed31_32 samples_per_line;
|
|
struct fixed31_32 layouts_per_line;
|
|
struct fixed31_32 symbols_per_sdp_max_layout;
|
|
struct fixed31_32 remainder;
|
|
uint32_t num_sdp_with_max_layouts;
|
|
uint32_t required_symbols_per_hblank;
|
|
uint32_t required_bytes_per_hblank = 0;
|
|
|
|
samples_per_line = dc_fixpt_from_fraction(sample_rate_hz, 1000);
|
|
samples_per_line = dc_fixpt_div(samples_per_line, horizontal_line_freq_khz);
|
|
layouts_per_line = dc_fixpt_div_int(samples_per_line, layout_config.layouts_per_sample_denom);
|
|
// HBlank expansion usage assumes SDP split disabled to allow for worst case.
|
|
layouts_per_line = dc_fixpt_from_int(dc_fixpt_ceil(layouts_per_line));
|
|
|
|
num_sdp_with_max_layouts = dc_fixpt_floor(
|
|
dc_fixpt_div_int(layouts_per_line, layout_config.max_layouts_per_audio_sdp));
|
|
symbols_per_sdp_max_layout = dc_fixpt_from_int(
|
|
layout_config.max_layouts_per_audio_sdp * layout_config.symbols_per_layout);
|
|
symbols_per_sdp_max_layout = dc_fixpt_add_int(symbols_per_sdp_max_layout, audio_sdp_overhead);
|
|
symbols_per_sdp_max_layout = dc_fixpt_mul(symbols_per_sdp_max_layout, audio_sdp_margin);
|
|
required_symbols_per_hblank = num_sdp_with_max_layouts;
|
|
required_symbols_per_hblank *= ((dc_fixpt_ceil(symbols_per_sdp_max_layout) + av_stream_map_lane_count) /
|
|
av_stream_map_lane_count) * av_stream_map_lane_count;
|
|
|
|
if (num_sdp_with_max_layouts != dc_fixpt_ceil(
|
|
dc_fixpt_div_int(layouts_per_line, layout_config.max_layouts_per_audio_sdp))) {
|
|
remainder = dc_fixpt_sub_int(layouts_per_line,
|
|
num_sdp_with_max_layouts * layout_config.max_layouts_per_audio_sdp);
|
|
remainder = dc_fixpt_mul_int(remainder, layout_config.symbols_per_layout);
|
|
remainder = dc_fixpt_add_int(remainder, audio_sdp_overhead);
|
|
remainder = dc_fixpt_mul(remainder, audio_sdp_margin);
|
|
required_symbols_per_hblank += ((dc_fixpt_ceil(remainder) + av_stream_map_lane_count) /
|
|
av_stream_map_lane_count) * av_stream_map_lane_count;
|
|
}
|
|
|
|
required_symbols_per_hblank += calculate_overhead_hblank_bw_in_symbols(max_slices_h);
|
|
|
|
if (link_encoding == DP_8b_10b_ENCODING)
|
|
required_bytes_per_hblank = required_symbols_per_hblank; // 8 bits per 8b/10b symbol
|
|
else if (link_encoding == DP_128b_132b_ENCODING)
|
|
required_bytes_per_hblank = required_symbols_per_hblank * 4; // 32 bits per 128b/132b symbol
|
|
|
|
return required_bytes_per_hblank;
|
|
}
|
|
|