Commit 5ca38a18 authored by Wenjing Liu's avatar Wenjing Liu Committed by Alex Deucher

drm/amd/display: move public dc link function implementation to dc_link_exports

[why]
Link is a subcomponent in dc. DM should be aware of dc link structure
as one of the abstracted objects maintained by dc. However it should
have no idea of the existence of a link component in dc dedicated to
maintain the states of dc link structure. As such we are moving link interfaces
out of dc_link.h and directly added to dc.h. We are grandually fading out
the explicit inclusion of dc_link header and eventually delete it.

On dc side, since link is a subcomponent behind dc interfaces, it is not
a good idea to implement dc interfaces in each individual subcomponent
of link which is already a subcomponent of dc. So we are decoupling it
by implementing a dc_link_exports in dc. This file will be a thin
translation layer that breaks the dependency so link is able to make
interface changes without breaking DM.
Reviewed-by: default avatarJun Lei <Jun.Lei@amd.com>
Acked-by: default avatarQingqing Zhuo <qingqing.zhuo@amd.com>
Signed-off-by: default avatarWenjing Liu <wenjing.liu@amd.com>
Tested-by: default avatarDaniel Wheeler <daniel.wheeler@amd.com>
Signed-off-by: default avatarAlex Deucher <alexander.deucher@amd.com>
parent 9b0f51e8
...@@ -65,7 +65,7 @@ AMD_DC = $(addsuffix /Makefile, $(addprefix $(FULL_AMD_DISPLAY_PATH)/dc/,$(DC_LI ...@@ -65,7 +65,7 @@ AMD_DC = $(addsuffix /Makefile, $(addprefix $(FULL_AMD_DISPLAY_PATH)/dc/,$(DC_LI
include $(AMD_DC) include $(AMD_DC)
DISPLAY_CORE = dc.o dc_stat.o dc_resource.o dc_hw_sequencer.o dc_sink.o \ DISPLAY_CORE = dc.o dc_stat.o dc_resource.o dc_hw_sequencer.o dc_sink.o \
dc_surface.o dc_debug.o dc_stream.o dc_link_enc_cfg.o dc_surface.o dc_debug.o dc_stream.o dc_link_enc_cfg.o dc_link_exports.o
DISPLAY_CORE += dc_vm_helper.o DISPLAY_CORE += dc_vm_helper.o
......
/*
* 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 provides single entrance to link functionality declared in dc
* public headers. The file is intended to be used as a thin translation layer
* that directly calls link internal functions without adding new functional
* behavior.
*
* When exporting a new link related dc function, add function declaration in
* dc.h with detail interface documentation, then add function implementation
* in this file which calls link functions.
*/
#include "link.h"
bool dc_link_detect(struct dc_link *link, enum dc_detect_reason reason)
{
return link_detect(link, reason);
}
bool dc_link_detect_connection_type(struct dc_link *link,
enum dc_connection_type *type)
{
return link_detect_connection_type(link, type);
}
const struct dc_link_status *dc_link_get_status(const struct dc_link *link)
{
return link_get_status(link);
}
#ifdef CONFIG_DRM_AMD_DC_HDCP
/* return true if the connected receiver supports the hdcp version */
bool dc_link_is_hdcp14(struct dc_link *link, enum signal_type signal)
{
return link_is_hdcp14(link, signal);
}
bool dc_link_is_hdcp22(struct dc_link *link, enum signal_type signal)
{
return link_is_hdcp22(link, signal);
}
#endif
void dc_link_clear_dprx_states(struct dc_link *link)
{
link_clear_dprx_states(link);
}
bool dc_link_reset_cur_dp_mst_topology(struct dc_link *link)
{
return link_reset_cur_dp_mst_topology(link);
}
uint32_t dc_link_bandwidth_kbps(
const struct dc_link *link,
const struct dc_link_settings *link_settings)
{
return dp_link_bandwidth_kbps(link, link_settings);
}
uint32_t dc_bandwidth_in_kbps_from_timing(
const struct dc_crtc_timing *timing)
{
return link_timing_bandwidth_kbps(timing);
}
void dc_get_cur_link_res_map(const struct dc *dc, uint32_t *map)
{
link_get_cur_res_map(dc, map);
}
void dc_restore_link_res_map(const struct dc *dc, uint32_t *map)
{
link_restore_res_map(dc, map);
}
bool dc_link_update_dsc_config(struct pipe_ctx *pipe_ctx)
{
return link_update_dsc_config(pipe_ctx);
}
...@@ -52,7 +52,6 @@ struct dmub_notification; ...@@ -52,7 +52,6 @@ struct dmub_notification;
#define MAX_SURFACES 3 #define MAX_SURFACES 3
#define MAX_PLANES 6 #define MAX_PLANES 6
#define MAX_STREAMS 6 #define MAX_STREAMS 6
#define MAX_SINKS_PER_LINK 4
#define MIN_VIEWPORT_SIZE 12 #define MIN_VIEWPORT_SIZE 12
#define MAX_NUM_EDP 2 #define MAX_NUM_EDP 2
...@@ -1372,109 +1371,128 @@ struct dc_state *dc_copy_state(struct dc_state *src_ctx); ...@@ -1372,109 +1371,128 @@ struct dc_state *dc_copy_state(struct dc_state *src_ctx);
void dc_retain_state(struct dc_state *context); void dc_retain_state(struct dc_state *context);
void dc_release_state(struct dc_state *context); void dc_release_state(struct dc_state *context);
struct dc_plane_state *dc_get_surface_for_mpcc(struct dc *dc,
struct dc_stream_state *stream,
int mpcc_inst);
uint32_t dc_get_opp_for_plane(struct dc *dc, struct dc_plane_state *plane);
/* Link Interfaces */ /* Link Interfaces */
/* TODO: remove this after resolving external dependencies */
#include "dc_link.h"
struct dpcd_caps { /* The function initiates detection handshake over the given link. It first
union dpcd_rev dpcd_rev; * determines if there are display connections over the link. If so it initiates
union max_lane_count max_ln_count; * detection protocols supported by the connected receiver device. The function
union max_down_spread max_down_spread; * contains protocol specific handshake sequences which are sometimes mandatory
union dprx_feature dprx_feature; * to establish a proper connection between TX and RX. So it is always
* recommended to call this function as the first link operation upon HPD event
/* valid only for eDP v1.4 or higher*/ * or power up event. Upon completion, the function will update link structure
uint8_t edp_supported_link_rates_count; * in place based on latest RX capabilities. The function may also cause dpms
enum dc_link_rate edp_supported_link_rates[8]; * to be reset to off for all currently enabled streams to the link. It is DM's
* responsibility to serialize detection and DPMS updates.
/* dongle type (DP converter, CV smart dongle) */ *
enum display_dongle_type dongle_type; * @reason - Indicate which event triggers this detection. dc may customize
bool is_dongle_type_one; * detection flow depending on the triggering events.
/* branch device or sink device */ * return false - if detection is not fully completed. This could happen when
bool is_branch_dev; * there is an unrecoverable error during detection or detection is partially
/* Dongle's downstream count. */ * completed (detection has been delegated to dm mst manager ie.
union sink_count sink_count; * link->connection_type == dc_connection_mst_branch when returning false).
bool is_mst_capable; * return true - detection is completed, link has been fully updated with latest
/* If dongle_type == DISPLAY_DONGLE_DP_HDMI_CONVERTER, * detection result.
indicates 'Frame Sequential-to-lllFrame Pack' conversion capability.*/ */
struct dc_dongle_caps dongle_caps; bool dc_link_detect(struct dc_link *link, enum dc_detect_reason reason);
uint32_t sink_dev_id;
int8_t sink_dev_id_str[6];
int8_t sink_hw_revision;
int8_t sink_fw_revision[2];
uint32_t branch_dev_id;
int8_t branch_dev_name[6];
int8_t branch_hw_revision;
int8_t branch_fw_revision[2];
bool allow_invalid_MSA_timing_param;
bool panel_mode_edp;
bool dpcd_display_control_capable;
bool ext_receiver_cap_field_present;
bool set_power_state_capable_edp;
bool dynamic_backlight_capable_edp;
union dpcd_fec_capability fec_cap;
struct dpcd_dsc_capabilities dsc_caps;
struct dc_lttpr_caps lttpr_caps;
struct adaptive_sync_caps adaptive_sync_caps;
struct dpcd_usb4_dp_tunneling_info usb4_dp_tun_info;
union dp_128b_132b_supported_link_rates dp_128b_132b_supported_link_rates;
union dp_main_line_channel_coding_cap channel_coding_cap;
union dp_sink_video_fallback_formats fallback_formats;
union dp_fec_capability1 fec_cap1;
union dp_cable_id cable_id;
uint8_t edp_rev;
union edp_alpm_caps alpm_caps;
struct edp_psr_info psr_info;
};
union dpcd_sink_ext_caps {
struct {
/* 0 - Sink supports backlight adjust via PWM during SDR/HDR mode
* 1 - Sink supports backlight adjust via AUX during SDR/HDR mode.
*/
uint8_t sdr_aux_backlight_control : 1;
uint8_t hdr_aux_backlight_control : 1;
uint8_t reserved_1 : 2;
uint8_t oled : 1;
uint8_t reserved : 3;
} bits;
uint8_t raw;
};
#if defined(CONFIG_DRM_AMD_DC_HDCP) /* determine if there is a sink connected to the link
union hdcp_rx_caps { *
struct { * @type - dc_connection_single if connected, dc_connection_none otherwise.
uint8_t version; * return - false if an unexpected error occurs, true otherwise.
uint8_t reserved; *
struct { * NOTE: This function doesn't detect downstream sink connections i.e
uint8_t repeater : 1; * dc_connection_mst_branch, dc_connection_sst_branch. In this case, it will
uint8_t hdcp_capable : 1; * return dc_connection_single if the branch device is connected despite of
uint8_t reserved : 6; * downstream sink's connection status.
} byte0; */
} fields; bool dc_link_detect_connection_type(struct dc_link *link,
uint8_t raw[3]; enum dc_connection_type *type);
};
union hdcp_bcaps { /* Getter for cached link status from given link */
struct { const struct dc_link_status *dc_link_get_status(const struct dc_link *link);
uint8_t HDCP_CAPABLE:1;
uint8_t REPEATER:1;
uint8_t RESERVED:6;
} bits;
uint8_t raw;
};
struct hdcp_caps { #ifdef CONFIG_DRM_AMD_DC_HDCP
union hdcp_rx_caps rx_caps; /* return true if the connected receiver supports the hdcp version */
union hdcp_bcaps bcaps; bool dc_link_is_hdcp14(struct dc_link *link, enum signal_type signal);
}; bool dc_link_is_hdcp22(struct dc_link *link, enum signal_type signal);
#endif #endif
#include "dc_link.h" /* The function clears recorded DP RX states in the link. DM should call this
* function when it is resuming from S3 power state to previously connected links.
*
* TODO - in the future we should consider to expand link resume interface to
* support clearing previous rx states. So we don't have to rely on dm to call
* this interface explicitly.
*/
void dc_link_clear_dprx_states(struct dc_link *link);
uint32_t dc_get_opp_for_plane(struct dc *dc, struct dc_plane_state *plane); /* Destruct the mst topology of the link and reset the allocated payload table
*
* NOTE: this should only be called if DM chooses not to call dc_link_detect but
* still wants to reset MST topology on an unplug event */
bool dc_link_reset_cur_dp_mst_topology(struct dc_link *link);
/* The function calculates effective DP link bandwidth when a given link is
* using the given link settings.
*
* return - total effective link bandwidth in kbps.
*/
uint32_t dc_link_bandwidth_kbps(
const struct dc_link *link,
const struct dc_link_settings *link_setting);
/* The function returns minimum bandwidth required to drive a given timing
* return - minimum required timing bandwidth in kbps.
*/
uint32_t dc_bandwidth_in_kbps_from_timing(
const struct dc_crtc_timing *timing);
/* The function takes a snapshot of current link resource allocation state
* @dc: pointer to dc of the dm calling this
* @map: a dc link resource snapshot defined internally to dc.
*
* DM needs to capture a snapshot of current link resource allocation mapping
* and store it in its persistent storage.
*
* Some of the link resource is using first come first serve policy.
* The allocation mapping depends on original hotplug order. This information
* is lost after driver is loaded next time. The snapshot is used in order to
* restore link resource to its previous state so user will get consistent
* link capability allocation across reboot.
*
*/
void dc_get_cur_link_res_map(const struct dc *dc, uint32_t *map);
/* This function restores link resource allocation state from a snapshot
* @dc: pointer to dc of the dm calling this
* @map: a dc link resource snapshot defined internally to dc.
*
* DM needs to call this function after initial link detection on boot and
* before first commit streams to restore link resource allocation state
* from previous boot session.
*
* Some of the link resource is using first come first serve policy.
* The allocation mapping depends on original hotplug order. This information
* is lost after driver is loaded next time. The snapshot is used in order to
* restore link resource to its previous state so user will get consistent
* link capability allocation across reboot.
*
*/
void dc_restore_link_res_map(const struct dc *dc, uint32_t *map);
/* TODO: this is not meant to be exposed to DM. Should switch to stream update
* interface i.e stream_update->dsc_config
*/
bool dc_link_update_dsc_config(struct pipe_ctx *pipe_ctx);
/* Sink Interfaces - A sink corresponds to a display output device */ /* Sink Interfaces - A sink corresponds to a display output device */
struct dc_container_id { struct dc_container_id {
......
...@@ -178,6 +178,9 @@ enum display_dongle_type { ...@@ -178,6 +178,9 @@ enum display_dongle_type {
DISPLAY_DONGLE_DP_HDMI_MISMATCHED_DONGLE, DISPLAY_DONGLE_DP_HDMI_MISMATCHED_DONGLE,
}; };
#define DC_MAX_EDID_BUFFER_SIZE 2048
#define DC_EDID_BLOCK_SIZE 128
struct ddc_service { struct ddc_service {
struct ddc *ddc_pin; struct ddc *ddc_pin;
struct ddc_flags flags; struct ddc_flags flags;
......
...@@ -27,6 +27,7 @@ ...@@ -27,6 +27,7 @@
#define DC_DP_TYPES_H #define DC_DP_TYPES_H
#include "os_types.h" #include "os_types.h"
#include "dc_ddc_types.h"
enum dc_lane_count { enum dc_lane_count {
LANE_COUNT_UNKNOWN = 0, LANE_COUNT_UNKNOWN = 0,
...@@ -1125,4 +1126,139 @@ struct edp_psr_info { ...@@ -1125,4 +1126,139 @@ struct edp_psr_info {
uint8_t force_psrsu_cap; uint8_t force_psrsu_cap;
}; };
struct dprx_states {
bool cable_id_written;
};
enum dpcd_downstream_port_max_bpc {
DOWN_STREAM_MAX_8BPC = 0,
DOWN_STREAM_MAX_10BPC,
DOWN_STREAM_MAX_12BPC,
DOWN_STREAM_MAX_16BPC
};
enum link_training_offset {
DPRX = 0,
LTTPR_PHY_REPEATER1 = 1,
LTTPR_PHY_REPEATER2 = 2,
LTTPR_PHY_REPEATER3 = 3,
LTTPR_PHY_REPEATER4 = 4,
LTTPR_PHY_REPEATER5 = 5,
LTTPR_PHY_REPEATER6 = 6,
LTTPR_PHY_REPEATER7 = 7,
LTTPR_PHY_REPEATER8 = 8
};
#define MAX_REPEATER_CNT 8
struct dc_lttpr_caps {
union dpcd_rev revision;
uint8_t mode;
uint8_t max_lane_count;
uint8_t max_link_rate;
uint8_t phy_repeater_cnt;
uint8_t max_ext_timeout;
union dp_main_link_channel_coding_lttpr_cap main_link_channel_coding;
union dp_128b_132b_supported_lttpr_link_rates supported_128b_132b_rates;
uint8_t aux_rd_interval[MAX_REPEATER_CNT - 1];
};
struct dc_dongle_dfp_cap_ext {
bool supported;
uint16_t max_pixel_rate_in_mps;
uint16_t max_video_h_active_width;
uint16_t max_video_v_active_height;
struct dp_encoding_format_caps encoding_format_caps;
struct dp_color_depth_caps rgb_color_depth_caps;
struct dp_color_depth_caps ycbcr444_color_depth_caps;
struct dp_color_depth_caps ycbcr422_color_depth_caps;
struct dp_color_depth_caps ycbcr420_color_depth_caps;
};
struct dc_dongle_caps {
/* dongle type (DP converter, CV smart dongle) */
enum display_dongle_type dongle_type;
bool extendedCapValid;
/* If dongle_type == DISPLAY_DONGLE_DP_HDMI_CONVERTER,
indicates 'Frame Sequential-to-lllFrame Pack' conversion capability.*/
bool is_dp_hdmi_s3d_converter;
bool is_dp_hdmi_ycbcr422_pass_through;
bool is_dp_hdmi_ycbcr420_pass_through;
bool is_dp_hdmi_ycbcr422_converter;
bool is_dp_hdmi_ycbcr420_converter;
uint32_t dp_hdmi_max_bpc;
uint32_t dp_hdmi_max_pixel_clk_in_khz;
uint32_t dp_hdmi_frl_max_link_bw_in_kbps;
struct dc_dongle_dfp_cap_ext dfp_cap_ext;
};
struct dpcd_caps {
union dpcd_rev dpcd_rev;
union max_lane_count max_ln_count;
union max_down_spread max_down_spread;
union dprx_feature dprx_feature;
/* valid only for eDP v1.4 or higher*/
uint8_t edp_supported_link_rates_count;
enum dc_link_rate edp_supported_link_rates[8];
/* dongle type (DP converter, CV smart dongle) */
enum display_dongle_type dongle_type;
bool is_dongle_type_one;
/* branch device or sink device */
bool is_branch_dev;
/* Dongle's downstream count. */
union sink_count sink_count;
bool is_mst_capable;
/* If dongle_type == DISPLAY_DONGLE_DP_HDMI_CONVERTER,
indicates 'Frame Sequential-to-lllFrame Pack' conversion capability.*/
struct dc_dongle_caps dongle_caps;
uint32_t sink_dev_id;
int8_t sink_dev_id_str[6];
int8_t sink_hw_revision;
int8_t sink_fw_revision[2];
uint32_t branch_dev_id;
int8_t branch_dev_name[6];
int8_t branch_hw_revision;
int8_t branch_fw_revision[2];
bool allow_invalid_MSA_timing_param;
bool panel_mode_edp;
bool dpcd_display_control_capable;
bool ext_receiver_cap_field_present;
bool set_power_state_capable_edp;
bool dynamic_backlight_capable_edp;
union dpcd_fec_capability fec_cap;
struct dpcd_dsc_capabilities dsc_caps;
struct dc_lttpr_caps lttpr_caps;
struct adaptive_sync_caps adaptive_sync_caps;
struct dpcd_usb4_dp_tunneling_info usb4_dp_tun_info;
union dp_128b_132b_supported_link_rates dp_128b_132b_supported_link_rates;
union dp_main_line_channel_coding_cap channel_coding_cap;
union dp_sink_video_fallback_formats fallback_formats;
union dp_fec_capability1 fec_cap1;
union dp_cable_id cable_id;
uint8_t edp_rev;
union edp_alpm_caps alpm_caps;
struct edp_psr_info psr_info;
};
union dpcd_sink_ext_caps {
struct {
/* 0 - Sink supports backlight adjust via PWM during SDR/HDR mode
* 1 - Sink supports backlight adjust via AUX during SDR/HDR mode.
*/
uint8_t sdr_aux_backlight_control : 1;
uint8_t hdr_aux_backlight_control : 1;
uint8_t reserved_1 : 2;
uint8_t oled : 1;
uint8_t reserved_2 : 1;
uint8_t miniled : 1;
uint8_t reserved : 1;
} bits;
uint8_t raw;
};
#endif /* DC_DP_TYPES_H */ #endif /* DC_DP_TYPES_H */
...@@ -39,15 +39,6 @@ enum dc_link_fec_state { ...@@ -39,15 +39,6 @@ enum dc_link_fec_state {
dc_link_fec_enabled dc_link_fec_enabled
}; };
struct dc_link_status {
bool link_active;
struct dpcd_caps *dpcd_caps;
};
struct dprx_states {
bool cable_id_written;
};
/* DP MST stream allocation (payload bandwidth number) */ /* DP MST stream allocation (payload bandwidth number) */
struct link_mst_stream_allocation { struct link_mst_stream_allocation {
/* DIG front */ /* DIG front */
...@@ -167,6 +158,8 @@ struct dc_dpia_bw_alloc { ...@@ -167,6 +158,8 @@ struct dc_dpia_bw_alloc {
bool response_ready; // Response ready from the CM side bool response_ready; // Response ready from the CM side
}; };
#define MAX_SINKS_PER_LINK 4
/* /*
* A link contains one or more sinks and their connected status. * A link contains one or more sinks and their connected status.
* The currently active signal type (HDMI, DP-SST, DP-MST) is also reported. * The currently active signal type (HDMI, DP-SST, DP-MST) is also reported.
...@@ -302,7 +295,6 @@ struct dc_link { ...@@ -302,7 +295,6 @@ struct dc_link {
struct phy_state phy_state; struct phy_state phy_state;
}; };
const struct dc_link_status *dc_link_get_status(const struct dc_link *dc_link);
/** /**
* dc_get_link_at_index() - Return an enumerated dc_link. * dc_get_link_at_index() - Return an enumerated dc_link.
...@@ -385,23 +377,6 @@ bool dc_link_setup_psr(struct dc_link *dc_link, ...@@ -385,23 +377,6 @@ bool dc_link_setup_psr(struct dc_link *dc_link,
const struct dc_stream_state *stream, struct psr_config *psr_config, const struct dc_stream_state *stream, struct psr_config *psr_config,
struct psr_context *psr_context); struct psr_context *psr_context);
/* Request DC to detect if there is a Panel connected.
* boot - If this call is during initial boot.
* Return false for any type of detection failure or MST detection
* true otherwise. True meaning further action is required (status update
* and OS notification).
*/
enum dc_detect_reason {
DETECT_REASON_BOOT,
DETECT_REASON_RESUMEFROMS3S4,
DETECT_REASON_HPD,
DETECT_REASON_HPDRX,
DETECT_REASON_FALLBACK,
DETECT_REASON_RETRAIN,
DETECT_REASON_TDR,
};
bool dc_link_detect(struct dc_link *dc_link, enum dc_detect_reason reason);
bool dc_link_get_hpd_state(struct dc_link *dc_link); bool dc_link_get_hpd_state(struct dc_link *dc_link);
/* Notify DC about DP RX Interrupt (aka Short Pulse Interrupt). /* Notify DC about DP RX Interrupt (aka Short Pulse Interrupt).
...@@ -471,16 +446,10 @@ bool dc_link_dp_get_max_link_enc_cap(const struct dc_link *link, struct dc_link_ ...@@ -471,16 +446,10 @@ bool dc_link_dp_get_max_link_enc_cap(const struct dc_link *link, struct dc_link_
void dc_link_enable_hpd_filter(struct dc_link *link, bool enable); void dc_link_enable_hpd_filter(struct dc_link *link, bool enable);
bool dc_link_is_dp_sink_present(struct dc_link *link); bool dc_link_is_dp_sink_present(struct dc_link *link);
bool dc_link_detect_connection_type(struct dc_link *link, enum dc_connection_type *type);
/* /*
* DPCD access interfaces * DPCD access interfaces
*/ */
#ifdef CONFIG_DRM_AMD_DC_HDCP
bool dc_link_is_hdcp14(struct dc_link *link, enum signal_type signal);
bool dc_link_is_hdcp22(struct dc_link *link, enum signal_type signal);
#endif
void dc_link_set_drive_settings(struct dc *dc, void dc_link_set_drive_settings(struct dc *dc,
struct link_training_settings *lt_settings, struct link_training_settings *lt_settings,
const struct dc_link *link); const struct dc_link *link);
...@@ -500,9 +469,6 @@ void dc_link_set_test_pattern(struct dc_link *link, ...@@ -500,9 +469,6 @@ void dc_link_set_test_pattern(struct dc_link *link,
const struct link_training_settings *p_link_settings, const struct link_training_settings *p_link_settings,
const unsigned char *p_custom_pattern, const unsigned char *p_custom_pattern,
unsigned int cust_pattern_size); unsigned int cust_pattern_size);
uint32_t dc_link_bandwidth_kbps(
const struct dc_link *link,
const struct dc_link_settings *link_setting);
const struct dc_link_settings *dc_link_get_link_cap( const struct dc_link_settings *dc_link_get_link_cap(
const struct dc_link *link); const struct dc_link *link);
...@@ -524,22 +490,16 @@ bool dc_submit_i2c_oem( ...@@ -524,22 +490,16 @@ bool dc_submit_i2c_oem(
struct dc *dc, struct dc *dc,
struct i2c_command *cmd); struct i2c_command *cmd);
uint32_t dc_bandwidth_in_kbps_from_timing(
const struct dc_crtc_timing *timing);
bool dc_link_is_fec_supported(const struct dc_link *link); bool dc_link_is_fec_supported(const struct dc_link *link);
bool dc_link_should_enable_fec(const struct dc_link *link); bool dc_link_should_enable_fec(const struct dc_link *link);
uint32_t dc_link_bw_kbps_from_raw_frl_link_rate_data(uint8_t bw); uint32_t dc_link_bw_kbps_from_raw_frl_link_rate_data(uint8_t bw);
enum dp_link_encoding dc_link_dp_mst_decide_link_encoding_format(const struct dc_link *link); enum dp_link_encoding dc_link_dp_mst_decide_link_encoding_format(const struct dc_link *link);
void dc_link_get_cur_link_res(const struct dc_link *link,
struct link_resource *link_res);
/* take a snapshot of current link resource allocation state */ /* take a snapshot of current link resource allocation state */
void dc_get_cur_link_res_map(const struct dc *dc, uint32_t *map); void dc_get_cur_link_res_map(const struct dc *dc, uint32_t *map);
/* restore link resource allocation state from a snapshot */ /* restore link resource allocation state from a snapshot */
void dc_restore_link_res_map(const struct dc *dc, uint32_t *map); void dc_restore_link_res_map(const struct dc *dc, uint32_t *map);
void dc_link_clear_dprx_states(struct dc_link *link);
void dp_trace_reset(struct dc_link *link); void dp_trace_reset(struct dc_link *link);
bool dc_dp_trace_is_initialized(struct dc_link *link); bool dc_dp_trace_is_initialized(struct dc_link *link);
unsigned long long dc_dp_trace_get_lt_end_timestamp(struct dc_link *link, unsigned long long dc_dp_trace_get_lt_end_timestamp(struct dc_link *link,
...@@ -553,9 +513,6 @@ struct dp_trace_lt_counts *dc_dp_trace_get_lt_counts(struct dc_link *link, ...@@ -553,9 +513,6 @@ struct dp_trace_lt_counts *dc_dp_trace_get_lt_counts(struct dc_link *link,
bool in_detection); bool in_detection);
unsigned int dc_dp_trace_get_link_loss_count(struct dc_link *link); unsigned int dc_dp_trace_get_link_loss_count(struct dc_link *link);
/* Destruct the mst topology of the link and reset the allocated payload table */
bool dc_link_reset_cur_dp_mst_topology(struct dc_link *link);
/* Attempt to transfer the given aux payload. This function does not perform /* Attempt to transfer the given aux payload. This function does not perform
* retries or handle error states. The reply is returned in the payload->reply * retries or handle error states. The reply is returned in the payload->reply
* and the result through operation_result. Returns the number of bytes * and the result through operation_result. Returns the number of bytes
......
...@@ -32,6 +32,7 @@ ...@@ -32,6 +32,7 @@
#include "os_types.h" #include "os_types.h"
#include "fixed31_32.h" #include "fixed31_32.h"
#include "irq_types.h" #include "irq_types.h"
#include "dc_ddc_types.h"
#include "dc_dp_types.h" #include "dc_dp_types.h"
#include "dc_hdmi_types.h" #include "dc_hdmi_types.h"
#include "dc_hw_types.h" #include "dc_hw_types.h"
...@@ -83,13 +84,8 @@ struct dc_perf_trace { ...@@ -83,13 +84,8 @@ struct dc_perf_trace {
unsigned long last_entry_write; unsigned long last_entry_write;
}; };
#define DC_MAX_EDID_BUFFER_SIZE 2048
#define DC_EDID_BLOCK_SIZE 128
#define MAX_SURFACE_NUM 4 #define MAX_SURFACE_NUM 4
#define NUM_PIXEL_FORMATS 10 #define NUM_PIXEL_FORMATS 10
#define MAX_REPEATER_CNT 8
#include "dc_ddc_types.h"
enum tiling_mode { enum tiling_mode {
TILING_MODE_INVALID, TILING_MODE_INVALID,
...@@ -375,66 +371,6 @@ struct dc_csc_adjustments { ...@@ -375,66 +371,6 @@ struct dc_csc_adjustments {
struct fixed31_32 hue; struct fixed31_32 hue;
}; };
enum dpcd_downstream_port_max_bpc {
DOWN_STREAM_MAX_8BPC = 0,
DOWN_STREAM_MAX_10BPC,
DOWN_STREAM_MAX_12BPC,
DOWN_STREAM_MAX_16BPC
};
enum link_training_offset {
DPRX = 0,
LTTPR_PHY_REPEATER1 = 1,
LTTPR_PHY_REPEATER2 = 2,
LTTPR_PHY_REPEATER3 = 3,
LTTPR_PHY_REPEATER4 = 4,
LTTPR_PHY_REPEATER5 = 5,
LTTPR_PHY_REPEATER6 = 6,
LTTPR_PHY_REPEATER7 = 7,
LTTPR_PHY_REPEATER8 = 8
};
struct dc_lttpr_caps {
union dpcd_rev revision;
uint8_t mode;
uint8_t max_lane_count;
uint8_t max_link_rate;
uint8_t phy_repeater_cnt;
uint8_t max_ext_timeout;
union dp_main_link_channel_coding_lttpr_cap main_link_channel_coding;
union dp_128b_132b_supported_lttpr_link_rates supported_128b_132b_rates;
uint8_t aux_rd_interval[MAX_REPEATER_CNT - 1];
};
struct dc_dongle_dfp_cap_ext {
bool supported;
uint16_t max_pixel_rate_in_mps;
uint16_t max_video_h_active_width;
uint16_t max_video_v_active_height;
struct dp_encoding_format_caps encoding_format_caps;
struct dp_color_depth_caps rgb_color_depth_caps;
struct dp_color_depth_caps ycbcr444_color_depth_caps;
struct dp_color_depth_caps ycbcr422_color_depth_caps;
struct dp_color_depth_caps ycbcr420_color_depth_caps;
};
struct dc_dongle_caps {
/* dongle type (DP converter, CV smart dongle) */
enum display_dongle_type dongle_type;
bool extendedCapValid;
/* If dongle_type == DISPLAY_DONGLE_DP_HDMI_CONVERTER,
indicates 'Frame Sequential-to-lllFrame Pack' conversion capability.*/
bool is_dp_hdmi_s3d_converter;
bool is_dp_hdmi_ycbcr422_pass_through;
bool is_dp_hdmi_ycbcr420_pass_through;
bool is_dp_hdmi_ycbcr422_converter;
bool is_dp_hdmi_ycbcr420_converter;
uint32_t dp_hdmi_max_bpc;
uint32_t dp_hdmi_max_pixel_clk_in_khz;
uint32_t dp_hdmi_frl_max_link_bw_in_kbps;
struct dc_dongle_dfp_cap_ext dfp_cap_ext;
};
/* Scaling format */ /* Scaling format */
enum scaling_transformation { enum scaling_transformation {
SCALING_TRANSFORMATION_UNINITIALIZED, SCALING_TRANSFORMATION_UNINITIALIZED,
...@@ -1003,4 +939,47 @@ struct otg_phy_mux { ...@@ -1003,4 +939,47 @@ struct otg_phy_mux {
}; };
#endif #endif
enum dc_detect_reason {
DETECT_REASON_BOOT,
DETECT_REASON_RESUMEFROMS3S4,
DETECT_REASON_HPD,
DETECT_REASON_HPDRX,
DETECT_REASON_FALLBACK,
DETECT_REASON_RETRAIN,
DETECT_REASON_TDR,
};
struct dc_link_status {
bool link_active;
struct dpcd_caps *dpcd_caps;
};
#if defined(CONFIG_DRM_AMD_DC_HDCP)
union hdcp_rx_caps {
struct {
uint8_t version;
uint8_t reserved;
struct {
uint8_t repeater : 1;
uint8_t hdcp_capable : 1;
uint8_t reserved : 6;
} byte0;
} fields;
uint8_t raw[3];
};
union hdcp_bcaps {
struct {
uint8_t HDCP_CAPABLE:1;
uint8_t REPEATER:1;
uint8_t RESERVED:6;
} bits;
uint8_t raw;
};
struct hdcp_caps {
union hdcp_rx_caps rx_caps;
union hdcp_bcaps bcaps;
};
#endif
#endif /* DC_TYPES_H_ */ #endif /* DC_TYPES_H_ */
...@@ -136,4 +136,22 @@ enum dc_status link_validate_mode_timing( ...@@ -136,4 +136,22 @@ enum dc_status link_validate_mode_timing(
const struct dc_stream_state *stream, const struct dc_stream_state *stream,
struct dc_link *link, struct dc_link *link,
const struct dc_crtc_timing *timing); const struct dc_crtc_timing *timing);
bool link_detect(struct dc_link *link, enum dc_detect_reason reason);
bool link_detect_connection_type(struct dc_link *link,
enum dc_connection_type *type);
const struct dc_link_status *link_get_status(const struct dc_link *link);
#ifdef CONFIG_DRM_AMD_DC_HDCP
/* return true if the connected receiver supports the hdcp version */
bool link_is_hdcp14(struct dc_link *link, enum signal_type signal);
bool link_is_hdcp22(struct dc_link *link, enum signal_type signal);
#endif
void link_clear_dprx_states(struct dc_link *link);
bool link_reset_cur_dp_mst_topology(struct dc_link *link);
uint32_t dp_link_bandwidth_kbps(
const struct dc_link *link,
const struct dc_link_settings *link_settings);
uint32_t link_timing_bandwidth_kbps(const struct dc_crtc_timing *timing);
void link_get_cur_res_map(const struct dc *dc, uint32_t *map);
void link_restore_res_map(const struct dc *dc, uint32_t *map);
#endif /* __DC_LINK_HPD_H__ */ #endif /* __DC_LINK_HPD_H__ */
...@@ -23,6 +23,7 @@ ...@@ -23,6 +23,7 @@
* *
*/ */
#include "link_dp_cts.h" #include "link_dp_cts.h"
#include "link/link_resource.h"
#include "link/protocols/link_dpcd.h" #include "link/protocols/link_dpcd.h"
#include "link/protocols/link_dp_training.h" #include "link/protocols/link_dp_training.h"
#include "link/protocols/link_dp_phy.h" #include "link/protocols/link_dp_phy.h"
...@@ -955,7 +956,7 @@ void dc_link_set_drive_settings(struct dc *dc, ...@@ -955,7 +956,7 @@ void dc_link_set_drive_settings(struct dc *dc,
if (i >= dc->link_count) if (i >= dc->link_count)
ASSERT_CRITICAL(false); ASSERT_CRITICAL(false);
dc_link_get_cur_link_res(link, &link_res); link_get_cur_link_res(link, &link_res);
dp_set_drive_settings(dc->links[i], &link_res, lt_settings); dp_set_drive_settings(dc->links[i], &link_res, lt_settings);
} }
......
...@@ -714,7 +714,7 @@ static bool discover_dp_mst_topology(struct dc_link *link, enum dc_detect_reason ...@@ -714,7 +714,7 @@ static bool discover_dp_mst_topology(struct dc_link *link, enum dc_detect_reason
return link->type == dc_connection_mst_branch; return link->type == dc_connection_mst_branch;
} }
static bool reset_cur_dp_mst_topology(struct dc_link *link) bool link_reset_cur_dp_mst_topology(struct dc_link *link)
{ {
DC_LOGGER_INIT(link->ctx->logger); DC_LOGGER_INIT(link->ctx->logger);
...@@ -725,10 +725,6 @@ static bool reset_cur_dp_mst_topology(struct dc_link *link) ...@@ -725,10 +725,6 @@ static bool reset_cur_dp_mst_topology(struct dc_link *link)
return dm_helpers_dp_mst_stop_top_mgr(link->ctx, link); return dm_helpers_dp_mst_stop_top_mgr(link->ctx, link);
} }
bool dc_link_reset_cur_dp_mst_topology(struct dc_link *link)
{
return reset_cur_dp_mst_topology(link);
}
static bool should_prepare_phy_clocks_for_link_verification(const struct dc *dc, static bool should_prepare_phy_clocks_for_link_verification(const struct dc *dc,
enum dc_detect_reason reason) enum dc_detect_reason reason)
{ {
...@@ -1199,7 +1195,7 @@ static bool detect_link_and_local_sink(struct dc_link *link, ...@@ -1199,7 +1195,7 @@ static bool detect_link_and_local_sink(struct dc_link *link,
* Does not detect downstream devices, such as MST sinks * Does not detect downstream devices, such as MST sinks
* or display connected through active dongles * or display connected through active dongles
*/ */
bool dc_link_detect_connection_type(struct dc_link *link, enum dc_connection_type *type) bool link_detect_connection_type(struct dc_link *link, enum dc_connection_type *type)
{ {
uint32_t is_hpd_high = 0; uint32_t is_hpd_high = 0;
...@@ -1242,7 +1238,7 @@ bool dc_link_detect_connection_type(struct dc_link *link, enum dc_connection_typ ...@@ -1242,7 +1238,7 @@ bool dc_link_detect_connection_type(struct dc_link *link, enum dc_connection_typ
return false; return false;
} }
bool dc_link_detect(struct dc_link *link, enum dc_detect_reason reason) bool link_detect(struct dc_link *link, enum dc_detect_reason reason)
{ {
bool is_local_sink_detect_success; bool is_local_sink_detect_success;
bool is_delegated_to_mst_top_mgr = false; bool is_delegated_to_mst_top_mgr = false;
...@@ -1261,18 +1257,18 @@ bool dc_link_detect(struct dc_link *link, enum dc_detect_reason reason) ...@@ -1261,18 +1257,18 @@ bool dc_link_detect(struct dc_link *link, enum dc_detect_reason reason)
if (is_local_sink_detect_success && if (is_local_sink_detect_success &&
pre_link_type == dc_connection_mst_branch && pre_link_type == dc_connection_mst_branch &&
link->type != dc_connection_mst_branch) link->type != dc_connection_mst_branch)
is_delegated_to_mst_top_mgr = reset_cur_dp_mst_topology(link); is_delegated_to_mst_top_mgr = link_reset_cur_dp_mst_topology(link);
return is_local_sink_detect_success && !is_delegated_to_mst_top_mgr; return is_local_sink_detect_success && !is_delegated_to_mst_top_mgr;
} }
void dc_link_clear_dprx_states(struct dc_link *link) void link_clear_dprx_states(struct dc_link *link)
{ {
memset(&link->dprx_states, 0, sizeof(link->dprx_states)); memset(&link->dprx_states, 0, sizeof(link->dprx_states));
} }
#if defined(CONFIG_DRM_AMD_DC_HDCP) #if defined(CONFIG_DRM_AMD_DC_HDCP)
bool dc_link_is_hdcp14(struct dc_link *link, enum signal_type signal) bool link_is_hdcp14(struct dc_link *link, enum signal_type signal)
{ {
bool ret = false; bool ret = false;
...@@ -1296,7 +1292,7 @@ bool dc_link_is_hdcp14(struct dc_link *link, enum signal_type signal) ...@@ -1296,7 +1292,7 @@ bool dc_link_is_hdcp14(struct dc_link *link, enum signal_type signal)
return ret; return ret;
} }
bool dc_link_is_hdcp22(struct dc_link *link, enum signal_type signal) bool link_is_hdcp22(struct dc_link *link, enum signal_type signal)
{ {
bool ret = false; bool ret = false;
...@@ -1320,7 +1316,7 @@ bool dc_link_is_hdcp22(struct dc_link *link, enum signal_type signal) ...@@ -1320,7 +1316,7 @@ bool dc_link_is_hdcp22(struct dc_link *link, enum signal_type signal)
} }
#endif // CONFIG_DRM_AMD_DC_HDCP #endif // CONFIG_DRM_AMD_DC_HDCP
const struct dc_link_status *dc_link_get_status(const struct dc_link *link) const struct dc_link_status *link_get_status(const struct dc_link *link)
{ {
return &link->link_status; return &link->link_status;
} }
......
...@@ -958,11 +958,6 @@ bool link_set_dsc_enable(struct pipe_ctx *pipe_ctx, bool enable) ...@@ -958,11 +958,6 @@ bool link_set_dsc_enable(struct pipe_ctx *pipe_ctx, bool enable)
return result; return result;
} }
bool dc_link_update_dsc_config(struct pipe_ctx *pipe_ctx)
{
return link_update_dsc_config(pipe_ctx);
}
bool link_update_dsc_config(struct pipe_ctx *pipe_ctx) bool link_update_dsc_config(struct pipe_ctx *pipe_ctx)
{ {
struct display_stream_compressor *dsc = pipe_ctx->stream_res.dsc; struct display_stream_compressor *dsc = pipe_ctx->stream_res.dsc;
......
...@@ -29,7 +29,7 @@ ...@@ -29,7 +29,7 @@
#include "link_resource.h" #include "link_resource.h"
#include "protocols/link_dp_capability.h" #include "protocols/link_dp_capability.h"
void dc_link_get_cur_link_res(const struct dc_link *link, void link_get_cur_link_res(const struct dc_link *link,
struct link_resource *link_res) struct link_resource *link_res)
{ {
int i; int i;
...@@ -49,24 +49,7 @@ void dc_link_get_cur_link_res(const struct dc_link *link, ...@@ -49,24 +49,7 @@ void dc_link_get_cur_link_res(const struct dc_link *link,
} }
/** void link_get_cur_res_map(const struct dc *dc, uint32_t *map)
* dc_get_cur_link_res_map() - take a snapshot of current link resource allocation state
* @dc: pointer to dc of the dm calling this
* @map: a dc link resource snapshot defined internally to dc.
*
* DM needs to capture a snapshot of current link resource allocation mapping
* and store it in its persistent storage.
*
* Some of the link resource is using first come first serve policy.
* The allocation mapping depends on original hotplug order. This information
* is lost after driver is loaded next time. The snapshot is used in order to
* restore link resource to its previous state so user will get consistent
* link capability allocation across reboot.
*
* Return: none (void function)
*
*/
void dc_get_cur_link_res_map(const struct dc *dc, uint32_t *map)
{ {
struct dc_link *link; struct dc_link *link;
uint32_t i; uint32_t i;
...@@ -89,25 +72,7 @@ void dc_get_cur_link_res_map(const struct dc *dc, uint32_t *map) ...@@ -89,25 +72,7 @@ void dc_get_cur_link_res_map(const struct dc *dc, uint32_t *map)
} }
} }
/** void link_restore_res_map(const struct dc *dc, uint32_t *map)
* dc_restore_link_res_map() - restore link resource allocation state from a snapshot
* @dc: pointer to dc of the dm calling this
* @map: a dc link resource snapshot defined internally to dc.
*
* DM needs to call this function after initial link detection on boot and
* before first commit streams to restore link resource allocation state
* from previous boot session.
*
* Some of the link resource is using first come first serve policy.
* The allocation mapping depends on original hotplug order. This information
* is lost after driver is loaded next time. The snapshot is used in order to
* restore link resource to its previous state so user will get consistent
* link capability allocation across reboot.
*
* Return: none (void function)
*
*/
void dc_restore_link_res_map(const struct dc *dc, uint32_t *map)
{ {
struct dc_link *link; struct dc_link *link;
uint32_t i; uint32_t i;
......
...@@ -25,5 +25,7 @@ ...@@ -25,5 +25,7 @@
#ifndef __LINK_RESOURCE_H__ #ifndef __LINK_RESOURCE_H__
#define __LINK_RESOURCE_H__ #define __LINK_RESOURCE_H__
#include "link.h" #include "link.h"
void link_get_cur_link_res(const struct dc_link *link,
struct link_resource *link_res);
#endif /* __LINK_RESOURCE_H__ */ #endif /* __LINK_RESOURCE_H__ */
...@@ -218,20 +218,20 @@ static bool dp_active_dongle_validate_timing( ...@@ -218,20 +218,20 @@ static bool dp_active_dongle_validate_timing(
return true; return true;
} }
uint32_t dc_link_bandwidth_kbps( uint32_t dp_link_bandwidth_kbps(
const struct dc_link *link, const struct dc_link *link,
const struct dc_link_settings *link_setting) const struct dc_link_settings *link_settings)
{ {
uint32_t total_data_bw_efficiency_x10000 = 0; uint32_t total_data_bw_efficiency_x10000 = 0;
uint32_t link_rate_per_lane_kbps = 0; uint32_t link_rate_per_lane_kbps = 0;
switch (link_dp_get_encoding_format(link_setting)) { switch (link_dp_get_encoding_format(link_settings)) {
case DP_8b_10b_ENCODING: case DP_8b_10b_ENCODING:
/* For 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. * 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. * data bandwidth efficiency is 80% with additional 3% overhead if FEC is supported.
*/ */
link_rate_per_lane_kbps = link_setting->link_rate * LINK_RATE_REF_FREQ_IN_KHZ * BITS_PER_DP_BYTE; 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; total_data_bw_efficiency_x10000 = DATA_EFFICIENCY_8b_10b_x10000;
if (dc_link_should_enable_fec(link)) { if (dc_link_should_enable_fec(link)) {
total_data_bw_efficiency_x10000 /= 100; total_data_bw_efficiency_x10000 /= 100;
...@@ -243,7 +243,7 @@ uint32_t dc_link_bandwidth_kbps( ...@@ -243,7 +243,7 @@ uint32_t dc_link_bandwidth_kbps(
* link rate is defined in the unit of 10mbps per lane. * link rate is defined in the unit of 10mbps per lane.
* total data bandwidth efficiency is always 96.71%. * total data bandwidth efficiency is always 96.71%.
*/ */
link_rate_per_lane_kbps = link_setting->link_rate * 10000; link_rate_per_lane_kbps = link_settings->link_rate * 10000;
total_data_bw_efficiency_x10000 = DATA_EFFICIENCY_128b_132b_x10000; total_data_bw_efficiency_x10000 = DATA_EFFICIENCY_128b_132b_x10000;
break; break;
default: default:
...@@ -251,10 +251,10 @@ uint32_t dc_link_bandwidth_kbps( ...@@ -251,10 +251,10 @@ uint32_t dc_link_bandwidth_kbps(
} }
/* overall effective link bandwidth = link rate per lane * lane count * total data bandwidth efficiency */ /* overall effective link bandwidth = link rate per lane * lane count * total data bandwidth efficiency */
return link_rate_per_lane_kbps * link_setting->lane_count / 10000 * total_data_bw_efficiency_x10000; return link_rate_per_lane_kbps * link_settings->lane_count / 10000 * total_data_bw_efficiency_x10000;
} }
uint32_t dc_bandwidth_in_kbps_from_timing( uint32_t link_timing_bandwidth_kbps(
const struct dc_crtc_timing *timing) const struct dc_crtc_timing *timing)
{ {
uint32_t bits_per_channel = 0; uint32_t bits_per_channel = 0;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment