提交 a98cdd8c 编写于 作者: W Wenjing Liu 提交者: Alex Deucher

drm/amd/display: refactor ddc logic from dc_link_ddc to link_ddc

[why]
1. Move dd_link_ddc functions to link_ddc.
2. Move link ddc functions declaration exposed in dc to link.h
3. Move link ddc functions declaration exposed in dm to dc_link.h
4. Remove i2caux_interface.h file
Tested-by: NDaniel Wheeler <Daniel.Wheeler@amd.com>
Reviewed-by: NJun Lei <Jun.Lei@amd.com>
Acked-by: NRodrigo Siqueira <Rodrigo.Siqueira@amd.com>
Signed-off-by: NWenjing Liu <wenjing.liu@amd.com>
Signed-off-by: NAlex Deucher <alexander.deucher@amd.com>
上级 4370f72e
...@@ -66,7 +66,6 @@ ...@@ -66,7 +66,6 @@
#include "ivsrcid/ivsrcid_vislands30.h" #include "ivsrcid/ivsrcid_vislands30.h"
#include "i2caux_interface.h"
#include <linux/module.h> #include <linux/module.h>
#include <linux/moduleparam.h> #include <linux/moduleparam.h>
#include <linux/types.h> #include <linux/types.h>
......
...@@ -39,12 +39,10 @@ ...@@ -39,12 +39,10 @@
#include "dc.h" #include "dc.h"
#include "dm_helpers.h" #include "dm_helpers.h"
#include "dc_link_ddc.h"
#include "dc_link_dp.h" #include "dc_link_dp.h"
#include "ddc_service_types.h" #include "ddc_service_types.h"
#include "dpcd_defs.h" #include "dpcd_defs.h"
#include "i2caux_interface.h"
#include "dmub_cmd.h" #include "dmub_cmd.h"
#if defined(CONFIG_DEBUG_FS) #if defined(CONFIG_DEBUG_FS)
#include "amdgpu_dm_debugfs.h" #include "amdgpu_dm_debugfs.h"
......
...@@ -64,8 +64,8 @@ AMD_DC = $(addsuffix /Makefile, $(addprefix $(FULL_AMD_DISPLAY_PATH)/dc/,$(DC_LI ...@@ -64,8 +64,8 @@ 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_link.o dc_resource.o dc_hw_sequencer.o dc_sink.o \ DISPLAY_CORE = dc.o dc_stat.o dc_link.o dc_resource.o dc_hw_sequencer.o dc_sink.o \
dc_surface.o dc_link_dp.o dc_link_ddc.o dc_debug.o dc_stream.o \ dc_surface.o dc_link_dp.o dc_debug.o dc_stream.o \
dc_link_enc_cfg.o dc_link_dpia.o dc_link_dpcd.o dc_link_enc_cfg.o dc_link_dpia.o dc_link_dpcd.o
DISPLAY_CORE += dc_vm_helper.o DISPLAY_CORE += dc_vm_helper.o
......
...@@ -33,7 +33,6 @@ ...@@ -33,7 +33,6 @@
#include "include/gpio_service_interface.h" #include "include/gpio_service_interface.h"
#include "include/grph_object_ctrl_defs.h" #include "include/grph_object_ctrl_defs.h"
#include "include/bios_parser_interface.h" #include "include/bios_parser_interface.h"
#include "include/i2caux_interface.h"
#include "include/logger_interface.h" #include "include/logger_interface.h"
#include "command_table.h" #include "command_table.h"
......
...@@ -32,7 +32,6 @@ ...@@ -32,7 +32,6 @@
#include "dc_bios_types.h" #include "dc_bios_types.h"
#include "include/grph_object_ctrl_defs.h" #include "include/grph_object_ctrl_defs.h"
#include "include/bios_parser_interface.h" #include "include/bios_parser_interface.h"
#include "include/i2caux_interface.h"
#include "include/logger_interface.h" #include "include/logger_interface.h"
#include "command_table2.h" #include "command_table2.h"
......
...@@ -33,6 +33,7 @@ ...@@ -33,6 +33,7 @@
#include "resource.h" #include "resource.h"
#include "gpio_service_interface.h"
#include "clk_mgr.h" #include "clk_mgr.h"
#include "clock_source.h" #include "clock_source.h"
#include "dc_bios_types.h" #include "dc_bios_types.h"
...@@ -53,7 +54,7 @@ ...@@ -53,7 +54,7 @@
#include "link_enc_cfg.h" #include "link_enc_cfg.h"
#include "dc_link.h" #include "dc_link.h"
#include "dc_link_ddc.h" #include "link.h"
#include "dm_helpers.h" #include "dm_helpers.h"
#include "mem_input.h" #include "mem_input.h"
...@@ -68,8 +69,6 @@ ...@@ -68,8 +69,6 @@
#include "dmub/dmub_srv.h" #include "dmub/dmub_srv.h"
#include "i2caux_interface.h"
#include "dce/dmub_psr.h" #include "dce/dmub_psr.h"
#include "dce/dmub_hw_lock_mgr.h" #include "dce/dmub_hw_lock_mgr.h"
......
...@@ -34,7 +34,7 @@ ...@@ -34,7 +34,7 @@
#include "core_status.h" #include "core_status.h"
#include "dc_link_dp.h" #include "dc_link_dp.h"
#include "dc_link_dpia.h" #include "dc_link_dpia.h"
#include "dc_link_ddc.h" #include "link/link_ddc.h"
#include "link_hwss.h" #include "link_hwss.h"
#include "link.h" #include "link.h"
#include "opp.h" #include "opp.h"
...@@ -80,7 +80,7 @@ static void dc_link_destruct(struct dc_link *link) ...@@ -80,7 +80,7 @@ static void dc_link_destruct(struct dc_link *link)
} }
if (link->ddc) if (link->ddc)
dal_ddc_service_destroy(&link->ddc); link_destroy_ddc_service(&link->ddc);
if (link->panel_cntl) if (link->panel_cntl)
link->panel_cntl->funcs->destroy(&link->panel_cntl); link->panel_cntl->funcs->destroy(&link->panel_cntl);
...@@ -277,7 +277,7 @@ bool dc_link_is_dp_sink_present(struct dc_link *link) ...@@ -277,7 +277,7 @@ bool dc_link_is_dp_sink_present(struct dc_link *link)
(connector_id == CONNECTOR_ID_EDP) || (connector_id == CONNECTOR_ID_EDP) ||
(connector_id == CONNECTOR_ID_USBC)); (connector_id == CONNECTOR_ID_USBC));
ddc = dal_ddc_service_get_ddc_pin(link->ddc); ddc = get_ddc_pin(link->ddc);
if (!ddc) { if (!ddc) {
BREAK_TO_DEBUGGER(); BREAK_TO_DEBUGGER();
...@@ -422,11 +422,179 @@ static enum signal_type decide_signal_from_strap_and_dongle_type(enum display_do ...@@ -422,11 +422,179 @@ static enum signal_type decide_signal_from_strap_and_dongle_type(enum display_do
return signal; return signal;
} }
static bool i2c_read(
struct ddc_service *ddc,
uint32_t address,
uint8_t *buffer,
uint32_t len)
{
uint8_t offs_data = 0;
struct i2c_payload payloads[2] = {
{
.write = true,
.address = address,
.length = 1,
.data = &offs_data },
{
.write = false,
.address = address,
.length = len,
.data = buffer } };
struct i2c_command command = {
.payloads = payloads,
.number_of_payloads = 2,
.engine = DDC_I2C_COMMAND_ENGINE,
.speed = ddc->ctx->dc->caps.i2c_speed_in_khz };
return dm_helpers_submit_i2c(
ddc->ctx,
ddc->link,
&command);
}
enum {
DP_SINK_CAP_SIZE =
DP_EDP_CONFIGURATION_CAP - DP_DPCD_REV + 1
};
static void query_dp_dual_mode_adaptor(
struct ddc_service *ddc,
struct display_sink_capability *sink_cap)
{
uint8_t i;
bool is_valid_hdmi_signature;
enum display_dongle_type *dongle = &sink_cap->dongle_type;
uint8_t type2_dongle_buf[DP_ADAPTOR_TYPE2_SIZE];
bool is_type2_dongle = false;
int retry_count = 2;
struct dp_hdmi_dongle_signature_data *dongle_signature;
/* Assume we have no valid DP passive dongle connected */
*dongle = DISPLAY_DONGLE_NONE;
sink_cap->max_hdmi_pixel_clock = DP_ADAPTOR_HDMI_SAFE_MAX_TMDS_CLK;
/* Read DP-HDMI dongle I2c (no response interpreted as DP-DVI dongle)*/
if (!i2c_read(
ddc,
DP_HDMI_DONGLE_ADDRESS,
type2_dongle_buf,
sizeof(type2_dongle_buf))) {
/* Passive HDMI dongles can sometimes fail here without retrying*/
while (retry_count > 0) {
if (i2c_read(ddc,
DP_HDMI_DONGLE_ADDRESS,
type2_dongle_buf,
sizeof(type2_dongle_buf)))
break;
retry_count--;
}
if (retry_count == 0) {
*dongle = DISPLAY_DONGLE_DP_DVI_DONGLE;
sink_cap->max_hdmi_pixel_clock = DP_ADAPTOR_DVI_MAX_TMDS_CLK;
CONN_DATA_DETECT(ddc->link, type2_dongle_buf, sizeof(type2_dongle_buf),
"DP-DVI passive dongle %dMhz: ",
DP_ADAPTOR_DVI_MAX_TMDS_CLK / 1000);
return;
}
}
/* Check if Type 2 dongle.*/
if (type2_dongle_buf[DP_ADAPTOR_TYPE2_REG_ID] == DP_ADAPTOR_TYPE2_ID)
is_type2_dongle = true;
dongle_signature =
(struct dp_hdmi_dongle_signature_data *)type2_dongle_buf;
is_valid_hdmi_signature = true;
/* Check EOT */
if (dongle_signature->eot != DP_HDMI_DONGLE_SIGNATURE_EOT) {
is_valid_hdmi_signature = false;
}
/* Check signature */
for (i = 0; i < sizeof(dongle_signature->id); ++i) {
/* If its not the right signature,
* skip mismatch in subversion byte.*/
if (dongle_signature->id[i] !=
dp_hdmi_dongle_signature_str[i] && i != 3) {
if (is_type2_dongle) {
is_valid_hdmi_signature = false;
break;
}
}
}
if (is_type2_dongle) {
uint32_t max_tmds_clk =
type2_dongle_buf[DP_ADAPTOR_TYPE2_REG_MAX_TMDS_CLK];
max_tmds_clk = max_tmds_clk * 2 + max_tmds_clk / 2;
if (0 == max_tmds_clk ||
max_tmds_clk < DP_ADAPTOR_TYPE2_MIN_TMDS_CLK ||
max_tmds_clk > DP_ADAPTOR_TYPE2_MAX_TMDS_CLK) {
*dongle = DISPLAY_DONGLE_DP_DVI_DONGLE;
CONN_DATA_DETECT(ddc->link, type2_dongle_buf,
sizeof(type2_dongle_buf),
"DP-DVI passive dongle %dMhz: ",
DP_ADAPTOR_DVI_MAX_TMDS_CLK / 1000);
} else {
if (is_valid_hdmi_signature == true) {
*dongle = DISPLAY_DONGLE_DP_HDMI_DONGLE;
CONN_DATA_DETECT(ddc->link, type2_dongle_buf,
sizeof(type2_dongle_buf),
"Type 2 DP-HDMI passive dongle %dMhz: ",
max_tmds_clk);
} else {
*dongle = DISPLAY_DONGLE_DP_HDMI_MISMATCHED_DONGLE;
CONN_DATA_DETECT(ddc->link, type2_dongle_buf,
sizeof(type2_dongle_buf),
"Type 2 DP-HDMI passive dongle (no signature) %dMhz: ",
max_tmds_clk);
}
/* Multiply by 1000 to convert to kHz. */
sink_cap->max_hdmi_pixel_clock =
max_tmds_clk * 1000;
}
sink_cap->is_dongle_type_one = false;
} else {
if (is_valid_hdmi_signature == true) {
*dongle = DISPLAY_DONGLE_DP_HDMI_DONGLE;
CONN_DATA_DETECT(ddc->link, type2_dongle_buf,
sizeof(type2_dongle_buf),
"Type 1 DP-HDMI passive dongle %dMhz: ",
sink_cap->max_hdmi_pixel_clock / 1000);
} else {
*dongle = DISPLAY_DONGLE_DP_HDMI_MISMATCHED_DONGLE;
CONN_DATA_DETECT(ddc->link, type2_dongle_buf,
sizeof(type2_dongle_buf),
"Type 1 DP-HDMI passive dongle (no signature) %dMhz: ",
sink_cap->max_hdmi_pixel_clock / 1000);
}
sink_cap->is_dongle_type_one = true;
}
return;
}
static enum signal_type dp_passive_dongle_detection(struct ddc_service *ddc, static enum signal_type dp_passive_dongle_detection(struct ddc_service *ddc,
struct display_sink_capability *sink_cap, struct display_sink_capability *sink_cap,
struct audio_support *audio_support) struct audio_support *audio_support)
{ {
dal_ddc_service_i2c_query_dp_dual_mode_adaptor(ddc, sink_cap); query_dp_dual_mode_adaptor(ddc, sink_cap);
return decide_signal_from_strap_and_dongle_type(sink_cap->dongle_type, return decide_signal_from_strap_and_dongle_type(sink_cap->dongle_type,
audio_support); audio_support);
...@@ -1046,11 +1214,11 @@ static bool detect_link_and_local_sink(struct dc_link *link, ...@@ -1046,11 +1214,11 @@ static bool detect_link_and_local_sink(struct dc_link *link,
else else
link->dpcd_sink_count = 1; link->dpcd_sink_count = 1;
dal_ddc_service_set_transaction_type(link->ddc, set_ddc_transaction_type(link->ddc,
sink_caps.transaction_type); sink_caps.transaction_type);
link->aux_mode = link->aux_mode =
dal_ddc_service_is_in_aux_transaction_mode(link->ddc); link_is_in_aux_transaction_mode(link->ddc);
sink_init_data.link = link; sink_init_data.link = link;
sink_init_data.sink_signal = sink_caps.signal; sink_init_data.sink_signal = sink_caps.signal;
...@@ -1265,7 +1433,7 @@ static enum channel_id get_ddc_line(struct dc_link *link) ...@@ -1265,7 +1433,7 @@ static enum channel_id get_ddc_line(struct dc_link *link)
channel = CHANNEL_ID_UNKNOWN; channel = CHANNEL_ID_UNKNOWN;
ddc = dal_ddc_service_get_ddc_pin(link->ddc); ddc = get_ddc_pin(link->ddc);
if (ddc) { if (ddc) {
switch (dal_ddc_get_line(ddc)) { switch (dal_ddc_get_line(ddc)) {
...@@ -1502,7 +1670,7 @@ static bool dc_link_construct_legacy(struct dc_link *link, ...@@ -1502,7 +1670,7 @@ static bool dc_link_construct_legacy(struct dc_link *link,
ddc_service_init_data.ctx = link->ctx; ddc_service_init_data.ctx = link->ctx;
ddc_service_init_data.id = link->link_id; ddc_service_init_data.id = link->link_id;
ddc_service_init_data.link = link; ddc_service_init_data.link = link;
link->ddc = dal_ddc_service_create(&ddc_service_init_data); link->ddc = link_create_ddc_service(&ddc_service_init_data);
if (!link->ddc) { if (!link->ddc) {
DC_ERROR("Failed to create ddc_service!\n"); DC_ERROR("Failed to create ddc_service!\n");
...@@ -1515,7 +1683,7 @@ static bool dc_link_construct_legacy(struct dc_link *link, ...@@ -1515,7 +1683,7 @@ static bool dc_link_construct_legacy(struct dc_link *link,
} }
link->ddc_hw_inst = link->ddc_hw_inst =
dal_ddc_get_line(dal_ddc_service_get_ddc_pin(link->ddc)); dal_ddc_get_line(get_ddc_pin(link->ddc));
if (link->dc->res_pool->funcs->panel_cntl_create && if (link->dc->res_pool->funcs->panel_cntl_create &&
...@@ -1652,7 +1820,7 @@ static bool dc_link_construct_legacy(struct dc_link *link, ...@@ -1652,7 +1820,7 @@ static bool dc_link_construct_legacy(struct dc_link *link,
if (link->panel_cntl != NULL) if (link->panel_cntl != NULL)
link->panel_cntl->funcs->destroy(&link->panel_cntl); link->panel_cntl->funcs->destroy(&link->panel_cntl);
panel_cntl_create_fail: panel_cntl_create_fail:
dal_ddc_service_destroy(&link->ddc); link_destroy_ddc_service(&link->ddc);
ddc_create_fail: ddc_create_fail:
create_fail: create_fail:
...@@ -1710,7 +1878,7 @@ static bool dc_link_construct_dpia(struct dc_link *link, ...@@ -1710,7 +1878,7 @@ static bool dc_link_construct_dpia(struct dc_link *link,
/* Set indicator for dpia link so that ddc won't be created */ /* Set indicator for dpia link so that ddc won't be created */
ddc_service_init_data.is_dpia_link = true; ddc_service_init_data.is_dpia_link = true;
link->ddc = dal_ddc_service_create(&ddc_service_init_data); link->ddc = link_create_ddc_service(&ddc_service_init_data);
if (!link->ddc) { if (!link->ddc) {
DC_ERROR("Failed to create ddc_service!\n"); DC_ERROR("Failed to create ddc_service!\n");
goto ddc_create_fail; goto ddc_create_fail;
...@@ -2178,7 +2346,7 @@ static void write_i2c_retimer_setting( ...@@ -2178,7 +2346,7 @@ static void write_i2c_retimer_setting(
value = settings->reg_settings[i].i2c_reg_val; value = settings->reg_settings[i].i2c_reg_val;
else { else {
i2c_success = i2c_success =
dal_ddc_service_query_ddc_data( link_query_ddc_data(
pipe_ctx->stream->link->ddc, pipe_ctx->stream->link->ddc,
slave_address, &offset, 1, &value, 1); slave_address, &offset, 1, &value, 1);
if (!i2c_success) if (!i2c_success)
...@@ -2228,7 +2396,7 @@ static void write_i2c_retimer_setting( ...@@ -2228,7 +2396,7 @@ static void write_i2c_retimer_setting(
value = settings->reg_settings_6g[i].i2c_reg_val; value = settings->reg_settings_6g[i].i2c_reg_val;
else { else {
i2c_success = i2c_success =
dal_ddc_service_query_ddc_data( link_query_ddc_data(
pipe_ctx->stream->link->ddc, pipe_ctx->stream->link->ddc,
slave_address, &offset, 1, &value, 1); slave_address, &offset, 1, &value, 1);
if (!i2c_success) if (!i2c_success)
...@@ -2526,7 +2694,7 @@ static void enable_link_hdmi(struct pipe_ctx *pipe_ctx) ...@@ -2526,7 +2694,7 @@ static void enable_link_hdmi(struct pipe_ctx *pipe_ctx)
} }
if (dc_is_hdmi_signal(pipe_ctx->stream->signal)) if (dc_is_hdmi_signal(pipe_ctx->stream->signal))
dal_ddc_service_write_scdc_data( write_scdc_data(
stream->link->ddc, stream->link->ddc,
stream->phy_pix_clk, stream->phy_pix_clk,
stream->timing.flags.LTE_340MCSC_SCRAMBLE); stream->timing.flags.LTE_340MCSC_SCRAMBLE);
...@@ -2547,7 +2715,7 @@ static void enable_link_hdmi(struct pipe_ctx *pipe_ctx) ...@@ -2547,7 +2715,7 @@ static void enable_link_hdmi(struct pipe_ctx *pipe_ctx)
stream->phy_pix_clk); stream->phy_pix_clk);
if (dc_is_hdmi_signal(pipe_ctx->stream->signal)) if (dc_is_hdmi_signal(pipe_ctx->stream->signal))
dal_ddc_service_read_scdc_data(link->ddc); read_scdc_data(link->ddc);
} }
static void enable_link_lvds(struct pipe_ctx *pipe_ctx) static void enable_link_lvds(struct pipe_ctx *pipe_ctx)
...@@ -4312,7 +4480,7 @@ void core_link_disable_stream(struct pipe_ctx *pipe_ctx) ...@@ -4312,7 +4480,7 @@ void core_link_disable_stream(struct pipe_ctx *pipe_ctx)
unsigned short masked_chip_caps = link->chip_caps & unsigned short masked_chip_caps = link->chip_caps &
EXT_DISPLAY_PATH_CAPS__EXT_CHIP_MASK; EXT_DISPLAY_PATH_CAPS__EXT_CHIP_MASK;
//Need to inform that sink is going to use legacy HDMI mode. //Need to inform that sink is going to use legacy HDMI mode.
dal_ddc_service_write_scdc_data( write_scdc_data(
link->ddc, link->ddc,
165000,//vbios only handles 165Mhz. 165000,//vbios only handles 165Mhz.
false); false);
......
...@@ -32,7 +32,7 @@ ...@@ -32,7 +32,7 @@
#include "inc/core_types.h" #include "inc/core_types.h"
#include "link_hwss.h" #include "link_hwss.h"
#include "dc_link_ddc.h" #include "link/link_ddc.h"
#include "core_status.h" #include "core_status.h"
#include "dpcd_defs.h" #include "dpcd_defs.h"
#include "dc_dmub_srv.h" #include "dc_dmub_srv.h"
...@@ -4866,7 +4866,7 @@ static void get_active_converter_info( ...@@ -4866,7 +4866,7 @@ static void get_active_converter_info(
/* decode converter info*/ /* decode converter info*/
if (!ds_port.fields.PORT_PRESENT) { if (!ds_port.fields.PORT_PRESENT) {
link->dpcd_caps.dongle_type = DISPLAY_DONGLE_NONE; link->dpcd_caps.dongle_type = DISPLAY_DONGLE_NONE;
ddc_service_set_dongle_type(link->ddc, set_dongle_type(link->ddc,
link->dpcd_caps.dongle_type); link->dpcd_caps.dongle_type);
link->dpcd_caps.is_branch_dev = false; link->dpcd_caps.is_branch_dev = false;
return; return;
...@@ -4974,7 +4974,7 @@ static void get_active_converter_info( ...@@ -4974,7 +4974,7 @@ static void get_active_converter_info(
} }
} }
ddc_service_set_dongle_type(link->ddc, link->dpcd_caps.dongle_type); set_dongle_type(link->ddc, link->dpcd_caps.dongle_type);
{ {
struct dp_sink_hw_fw_revision dp_hw_fw_revision; struct dp_sink_hw_fw_revision dp_hw_fw_revision;
...@@ -5352,7 +5352,7 @@ static bool retrieve_link_cap(struct dc_link *link) ...@@ -5352,7 +5352,7 @@ static bool retrieve_link_cap(struct dc_link *link)
* default to LTTPR timeout (3.2ms) first as a W/A for DP link layer * default to LTTPR timeout (3.2ms) first as a W/A for DP link layer
* CTS 4.2.1.1 regression introduced by CTS specs requirement update. * CTS 4.2.1.1 regression introduced by CTS specs requirement update.
*/ */
dc_link_aux_try_to_configure_timeout(link->ddc, try_to_configure_aux_timeout(link->ddc,
LINK_AUX_DEFAULT_LTTPR_TIMEOUT_PERIOD); LINK_AUX_DEFAULT_LTTPR_TIMEOUT_PERIOD);
status = dp_retrieve_lttpr_cap(link); status = dp_retrieve_lttpr_cap(link);
...@@ -5393,7 +5393,7 @@ static bool retrieve_link_cap(struct dc_link *link) ...@@ -5393,7 +5393,7 @@ static bool retrieve_link_cap(struct dc_link *link)
} }
if (!dp_is_lttpr_present(link)) if (!dp_is_lttpr_present(link))
dc_link_aux_try_to_configure_timeout(link->ddc, LINK_AUX_DEFAULT_TIMEOUT_PERIOD); try_to_configure_aux_timeout(link->ddc, LINK_AUX_DEFAULT_TIMEOUT_PERIOD);
{ {
union training_aux_rd_interval aux_rd_interval; union training_aux_rd_interval aux_rd_interval;
......
...@@ -77,6 +77,32 @@ struct aux_reply_transaction_data { ...@@ -77,6 +77,32 @@ struct aux_reply_transaction_data {
uint8_t *data; uint8_t *data;
}; };
struct aux_payload {
/* set following flag to read/write I2C data,
* reset it to read/write DPCD data */
bool i2c_over_aux;
/* set following flag to write data,
* reset it to read data */
bool write;
bool mot;
bool write_status_update;
uint32_t address;
uint32_t length;
uint8_t *data;
/*
* used to return the reply type of the transaction
* ignored if NULL
*/
uint8_t *reply;
/* expressed in milliseconds
* zero means "use default value"
*/
uint32_t defer_delay;
};
#define DEFAULT_AUX_MAX_DATA_SIZE 16
struct i2c_payload { struct i2c_payload {
bool write; bool write;
uint8_t address; uint8_t address;
...@@ -90,6 +116,8 @@ enum i2c_command_engine { ...@@ -90,6 +116,8 @@ enum i2c_command_engine {
I2C_COMMAND_ENGINE_HW I2C_COMMAND_ENGINE_HW
}; };
#define DDC_I2C_COMMAND_ENGINE I2C_COMMAND_ENGINE_SW
struct i2c_command { struct i2c_command {
struct i2c_payload *payloads; struct i2c_payload *payloads;
uint8_t number_of_payloads; uint8_t number_of_payloads;
......
/* /*
* Copyright 2012-15 Advanced Micro Devices, Inc. * Copyright 2022 Advanced Micro Devices, Inc.
* *
* Permission is hereby granted, free of charge, to any person obtaining a * Permission is hereby granted, free of charge, to any person obtaining a
* copy of this software and associated documentation files (the "Software"), * copy of this software and associated documentation files (the "Software"),
...@@ -23,13 +23,10 @@ ...@@ -23,13 +23,10 @@
* *
*/ */
#ifndef __DAL_DDC_SERVICE_H__ #ifndef DC_HDMI_TYPES_H
#define __DAL_DDC_SERVICE_H__ #define DC_HDMI_TYPES_H
#include "include/ddc_service_types.h" #include "os_types.h"
#include "include/i2caux_interface.h"
#define EDID_SEGMENT_SIZE 256
/* Address range from 0x00 to 0x1F.*/ /* Address range from 0x00 to 0x1F.*/
#define DP_ADAPTOR_TYPE2_SIZE 0x20 #define DP_ADAPTOR_TYPE2_SIZE 0x20
...@@ -46,88 +43,72 @@ ...@@ -46,88 +43,72 @@
/* kHZ*/ /* kHZ*/
#define DP_ADAPTOR_HDMI_SAFE_MAX_TMDS_CLK 165000 #define DP_ADAPTOR_HDMI_SAFE_MAX_TMDS_CLK 165000
#define DDC_I2C_COMMAND_ENGINE I2C_COMMAND_ENGINE_SW struct dp_hdmi_dongle_signature_data {
int8_t id[15];/* "DP-HDMI ADAPTOR"*/
struct ddc_service; uint8_t eot;/* end of transmition '\x4' */
struct graphics_object_id;
enum ddc_result;
struct av_sync_data;
struct dp_receiver_id_info;
struct i2c_payloads;
struct aux_payloads;
enum aux_return_code_type;
void dal_ddc_i2c_payloads_add(
struct i2c_payloads *payloads,
uint32_t address,
uint32_t len,
uint8_t *data,
bool write);
struct ddc_service_init_data {
struct graphics_object_id id;
struct dc_context *ctx;
struct dc_link *link;
bool is_dpia_link;
}; };
struct ddc_service *dal_ddc_service_create( /* DP-HDMI dongle slave address for retrieving dongle signature*/
struct ddc_service_init_data *ddc_init_data); #define DP_HDMI_DONGLE_ADDRESS 0x40
static const uint8_t dp_hdmi_dongle_signature_str[] = "DP-HDMI ADAPTOR";
void dal_ddc_service_destroy(struct ddc_service **ddc); #define DP_HDMI_DONGLE_SIGNATURE_EOT 0x04
enum ddc_service_type dal_ddc_service_get_type(struct ddc_service *ddc);
/* SCDC Address defines (HDMI 2.0)*/
void dal_ddc_service_set_transaction_type( #define HDMI_SCDC_WRITE_UPDATE_0_ARRAY 3
struct ddc_service *ddc, #define HDMI_SCDC_ADDRESS 0x54
enum ddc_transaction_type type); #define HDMI_SCDC_SINK_VERSION 0x01
#define HDMI_SCDC_SOURCE_VERSION 0x02
bool dal_ddc_service_is_in_aux_transaction_mode(struct ddc_service *ddc); #define HDMI_SCDC_UPDATE_0 0x10
#define HDMI_SCDC_TMDS_CONFIG 0x20
void dal_ddc_service_i2c_query_dp_dual_mode_adaptor( #define HDMI_SCDC_SCRAMBLER_STATUS 0x21
struct ddc_service *ddc, #define HDMI_SCDC_CONFIG_0 0x30
struct display_sink_capability *sink_cap); #define HDMI_SCDC_CONFIG_1 0x31
#define HDMI_SCDC_SOURCE_TEST_REQ 0x35
bool dal_ddc_service_query_ddc_data( #define HDMI_SCDC_STATUS_FLAGS 0x40
struct ddc_service *ddc, #define HDMI_SCDC_ERR_DETECT 0x50
uint32_t address, #define HDMI_SCDC_TEST_CONFIG 0xC0
uint8_t *write_buf,
uint32_t write_size, union hdmi_scdc_update_read_data {
uint8_t *read_buf, uint8_t byte[2];
uint32_t read_size); struct {
uint8_t STATUS_UPDATE:1;
bool dal_ddc_submit_aux_command(struct ddc_service *ddc, uint8_t CED_UPDATE:1;
struct aux_payload *payload); uint8_t RR_TEST:1;
uint8_t RESERVED:5;
int dc_link_aux_transfer_raw(struct ddc_service *ddc, uint8_t RESERVED2:8;
struct aux_payload *payload, } fields;
enum aux_return_code_type *operation_result); };
bool dc_link_aux_transfer_with_retries(struct ddc_service *ddc,
struct aux_payload *payload);
bool dc_link_aux_try_to_configure_timeout(struct ddc_service *ddc,
uint32_t timeout);
void dal_ddc_service_write_scdc_data(
struct ddc_service *ddc_service,
uint32_t pix_clk,
bool lte_340_scramble);
void dal_ddc_service_read_scdc_data(
struct ddc_service *ddc_service);
void ddc_service_set_dongle_type(struct ddc_service *ddc,
enum display_dongle_type dongle_type);
void dal_ddc_service_set_ddc_pin(
struct ddc_service *ddc_service,
struct ddc *ddc);
struct ddc *dal_ddc_service_get_ddc_pin(struct ddc_service *ddc_service);
uint32_t get_defer_delay(struct ddc_service *ddc); union hdmi_scdc_status_flags_data {
uint8_t byte;
struct {
uint8_t CLOCK_DETECTED:1;
uint8_t CH0_LOCKED:1;
uint8_t CH1_LOCKED:1;
uint8_t CH2_LOCKED:1;
uint8_t RESERVED:4;
} fields;
};
#endif /* __DAL_DDC_SERVICE_H__ */ union hdmi_scdc_ced_data {
uint8_t byte[11];
struct {
uint8_t CH0_8LOW:8;
uint8_t CH0_7HIGH:7;
uint8_t CH0_VALID:1;
uint8_t CH1_8LOW:8;
uint8_t CH1_7HIGH:7;
uint8_t CH1_VALID:1;
uint8_t CH2_8LOW:8;
uint8_t CH2_7HIGH:7;
uint8_t CH2_VALID:1;
uint8_t CHECKSUM:8;
uint8_t RESERVED:8;
uint8_t RESERVED2:8;
uint8_t RESERVED3:8;
uint8_t RESERVED4:4;
} fields;
};
#endif /* DC_HDMI_TYPES_H */
...@@ -31,6 +31,7 @@ ...@@ -31,6 +31,7 @@
#include "grph_object_defs.h" #include "grph_object_defs.h"
struct link_resource; struct link_resource;
enum aux_return_code_type;
enum dc_link_fec_state { enum dc_link_fec_state {
dc_link_fec_not_ready, dc_link_fec_not_ready,
...@@ -591,4 +592,13 @@ unsigned int dc_dp_trace_get_link_loss_count(struct dc_link *link); ...@@ -591,4 +592,13 @@ 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 */ /* Destruct the mst topology of the link and reset the allocated payload table */
bool reset_cur_dp_mst_topology(struct dc_link *link); bool reset_cur_dp_mst_topology(struct dc_link *link);
/* 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
* and the result through operation_result. Returns the number of bytes
* transferred,or -1 on a failure.
*/
int dc_link_aux_transfer_raw(struct ddc_service *ddc,
struct aux_payload *payload,
enum aux_return_code_type *operation_result);
#endif /* DC_LINK_H_ */ #endif /* DC_LINK_H_ */
...@@ -33,6 +33,7 @@ ...@@ -33,6 +33,7 @@
#include "fixed31_32.h" #include "fixed31_32.h"
#include "irq_types.h" #include "irq_types.h"
#include "dc_dp_types.h" #include "dc_dp_types.h"
#include "dc_hdmi_types.h"
#include "dc_hw_types.h" #include "dc_hw_types.h"
#include "dal_types.h" #include "dal_types.h"
#include "grph_object_defs.h" #include "grph_object_defs.h"
......
...@@ -26,7 +26,7 @@ ...@@ -26,7 +26,7 @@
#ifndef __DAL_AUX_ENGINE_DCE110_H__ #ifndef __DAL_AUX_ENGINE_DCE110_H__
#define __DAL_AUX_ENGINE_DCE110_H__ #define __DAL_AUX_ENGINE_DCE110_H__
#include "i2caux_interface.h" #include "gpio_service_interface.h"
#include "inc/hw/aux_engine.h" #include "inc/hw/aux_engine.h"
enum aux_return_code_type; enum aux_return_code_type;
......
...@@ -29,7 +29,6 @@ ...@@ -29,7 +29,6 @@
#include "link_encoder.h" #include "link_encoder.h"
#include "dce_link_encoder.h" #include "dce_link_encoder.h"
#include "stream_encoder.h" #include "stream_encoder.h"
#include "i2caux_interface.h"
#include "dc_bios_types.h" #include "dc_bios_types.h"
#include "gpio_service_interface.h" #include "gpio_service_interface.h"
......
...@@ -29,7 +29,6 @@ ...@@ -29,7 +29,6 @@
#include "link_encoder.h" #include "link_encoder.h"
#include "dcn10_link_encoder.h" #include "dcn10_link_encoder.h"
#include "stream_encoder.h" #include "stream_encoder.h"
#include "i2caux_interface.h"
#include "dc_bios_types.h" #include "dc_bios_types.h"
#include "gpio_service_interface.h" #include "gpio_service_interface.h"
......
...@@ -29,7 +29,6 @@ ...@@ -29,7 +29,6 @@
#include "link_encoder.h" #include "link_encoder.h"
#include "dcn20_link_encoder.h" #include "dcn20_link_encoder.h"
#include "stream_encoder.h" #include "stream_encoder.h"
#include "i2caux_interface.h"
#include "dc_bios_types.h" #include "dc_bios_types.h"
#include "gpio_service_interface.h" #include "gpio_service_interface.h"
......
...@@ -62,7 +62,6 @@ ...@@ -62,7 +62,6 @@
#include "dml/display_mode_vba.h" #include "dml/display_mode_vba.h"
#include "dcn20_dccg.h" #include "dcn20_dccg.h"
#include "dcn20_vmid.h" #include "dcn20_vmid.h"
#include "dc_link_ddc.h"
#include "dce/dce_panel_cntl.h" #include "dce/dce_panel_cntl.h"
#include "navi10_ip_offset.h" #include "navi10_ip_offset.h"
...@@ -90,6 +89,7 @@ ...@@ -90,6 +89,7 @@
#include "amdgpu_socbb.h" #include "amdgpu_socbb.h"
#include "link.h"
#define DC_LOGGER_INIT(logger) #define DC_LOGGER_INIT(logger)
#ifndef mmDP0_DP_DPHY_INTERNAL_CTRL #ifndef mmDP0_DP_DPHY_INTERNAL_CTRL
...@@ -1214,7 +1214,7 @@ static void dcn20_resource_destruct(struct dcn20_resource_pool *pool) ...@@ -1214,7 +1214,7 @@ static void dcn20_resource_destruct(struct dcn20_resource_pool *pool)
dcn20_pp_smu_destroy(&pool->base.pp_smu); dcn20_pp_smu_destroy(&pool->base.pp_smu);
if (pool->base.oem_device != NULL) if (pool->base.oem_device != NULL)
dal_ddc_service_destroy(&pool->base.oem_device); link_destroy_ddc_service(&pool->base.oem_device);
} }
struct hubp *dcn20_hubp_create( struct hubp *dcn20_hubp_create(
...@@ -2769,7 +2769,7 @@ static bool dcn20_resource_construct( ...@@ -2769,7 +2769,7 @@ static bool dcn20_resource_construct(
ddc_init_data.id.id = dc->ctx->dc_bios->fw_info.oem_i2c_obj_id; ddc_init_data.id.id = dc->ctx->dc_bios->fw_info.oem_i2c_obj_id;
ddc_init_data.id.enum_id = 0; ddc_init_data.id.enum_id = 0;
ddc_init_data.id.type = OBJECT_TYPE_GENERIC; ddc_init_data.id.type = OBJECT_TYPE_GENERIC;
pool->base.oem_device = dal_ddc_service_create(&ddc_init_data); pool->base.oem_device = link_create_ddc_service(&ddc_init_data);
} else { } else {
pool->base.oem_device = NULL; pool->base.oem_device = NULL;
} }
......
...@@ -29,7 +29,6 @@ ...@@ -29,7 +29,6 @@
#include "link_encoder.h" #include "link_encoder.h"
#include "dcn201_link_encoder.h" #include "dcn201_link_encoder.h"
#include "stream_encoder.h" #include "stream_encoder.h"
#include "i2caux_interface.h"
#include "dc_bios_types.h" #include "dc_bios_types.h"
#include "gpio_service_interface.h" #include "gpio_service_interface.h"
......
...@@ -31,7 +31,6 @@ ...@@ -31,7 +31,6 @@
#include "dcn21_link_encoder.h" #include "dcn21_link_encoder.h"
#include "stream_encoder.h" #include "stream_encoder.h"
#include "i2caux_interface.h"
#include "dc_bios_types.h" #include "dc_bios_types.h"
#include "gpio_service_interface.h" #include "gpio_service_interface.h"
......
...@@ -29,7 +29,6 @@ ...@@ -29,7 +29,6 @@
#include "link_encoder.h" #include "link_encoder.h"
#include "dcn30_dio_link_encoder.h" #include "dcn30_dio_link_encoder.h"
#include "stream_encoder.h" #include "stream_encoder.h"
#include "i2caux_interface.h"
#include "dc_bios_types.h" #include "dc_bios_types.h"
/* #include "dcn3ag/dcn3ag_phy_fw.h" */ /* #include "dcn3ag/dcn3ag_phy_fw.h" */
......
...@@ -60,7 +60,7 @@ ...@@ -60,7 +60,7 @@
#include "dml/display_mode_vba.h" #include "dml/display_mode_vba.h"
#include "dcn30/dcn30_dccg.h" #include "dcn30/dcn30_dccg.h"
#include "dcn10/dcn10_resource.h" #include "dcn10/dcn10_resource.h"
#include "dc_link_ddc.h" #include "link.h"
#include "dce/dce_panel_cntl.h" #include "dce/dce_panel_cntl.h"
#include "dcn30/dcn30_dwb.h" #include "dcn30/dcn30_dwb.h"
...@@ -1208,7 +1208,7 @@ static void dcn30_resource_destruct(struct dcn30_resource_pool *pool) ...@@ -1208,7 +1208,7 @@ static void dcn30_resource_destruct(struct dcn30_resource_pool *pool)
dcn_dccg_destroy(&pool->base.dccg); dcn_dccg_destroy(&pool->base.dccg);
if (pool->base.oem_device != NULL) if (pool->base.oem_device != NULL)
dal_ddc_service_destroy(&pool->base.oem_device); link_destroy_ddc_service(&pool->base.oem_device);
} }
static struct hubp *dcn30_hubp_create( static struct hubp *dcn30_hubp_create(
...@@ -2590,7 +2590,7 @@ static bool dcn30_resource_construct( ...@@ -2590,7 +2590,7 @@ static bool dcn30_resource_construct(
ddc_init_data.id.id = dc->ctx->dc_bios->fw_info.oem_i2c_obj_id; ddc_init_data.id.id = dc->ctx->dc_bios->fw_info.oem_i2c_obj_id;
ddc_init_data.id.enum_id = 0; ddc_init_data.id.enum_id = 0;
ddc_init_data.id.type = OBJECT_TYPE_GENERIC; ddc_init_data.id.type = OBJECT_TYPE_GENERIC;
pool->base.oem_device = dal_ddc_service_create(&ddc_init_data); pool->base.oem_device = link_create_ddc_service(&ddc_init_data);
} else { } else {
pool->base.oem_device = NULL; pool->base.oem_device = NULL;
} }
......
...@@ -29,7 +29,6 @@ ...@@ -29,7 +29,6 @@
#include "link_encoder.h" #include "link_encoder.h"
#include "dcn301_dio_link_encoder.h" #include "dcn301_dio_link_encoder.h"
#include "stream_encoder.h" #include "stream_encoder.h"
#include "i2caux_interface.h"
#include "dc_bios_types.h" #include "dc_bios_types.h"
#include "gpio_service_interface.h" #include "gpio_service_interface.h"
......
...@@ -47,6 +47,7 @@ ...@@ -47,6 +47,7 @@
#include "dcn10/dcn10_resource.h" #include "dcn10/dcn10_resource.h"
#include "link.h"
#include "dce/dce_abm.h" #include "dce/dce_abm.h"
#include "dce/dce_audio.h" #include "dce/dce_audio.h"
#include "dce/dce_aux.h" #include "dce/dce_aux.h"
...@@ -1125,6 +1126,9 @@ static void dcn302_resource_destruct(struct resource_pool *pool) ...@@ -1125,6 +1126,9 @@ static void dcn302_resource_destruct(struct resource_pool *pool)
if (pool->dccg != NULL) if (pool->dccg != NULL)
dcn_dccg_destroy(&pool->dccg); dcn_dccg_destroy(&pool->dccg);
if (pool->oem_device != NULL)
link_destroy_ddc_service(&pool->oem_device);
} }
static void dcn302_destroy_resource_pool(struct resource_pool **pool) static void dcn302_destroy_resource_pool(struct resource_pool **pool)
...@@ -1216,6 +1220,7 @@ static bool dcn302_resource_construct( ...@@ -1216,6 +1220,7 @@ static bool dcn302_resource_construct(
int i; int i;
struct dc_context *ctx = dc->ctx; struct dc_context *ctx = dc->ctx;
struct irq_service_init_data init_data; struct irq_service_init_data init_data;
struct ddc_service_init_data ddc_init_data = {0};
ctx->dc_bios->regs = &bios_regs; ctx->dc_bios->regs = &bios_regs;
...@@ -1497,6 +1502,17 @@ static bool dcn302_resource_construct( ...@@ -1497,6 +1502,17 @@ static bool dcn302_resource_construct(
dc->cap_funcs = cap_funcs; dc->cap_funcs = cap_funcs;
if (dc->ctx->dc_bios->fw_info.oem_i2c_present) {
ddc_init_data.ctx = dc->ctx;
ddc_init_data.link = NULL;
ddc_init_data.id.id = dc->ctx->dc_bios->fw_info.oem_i2c_obj_id;
ddc_init_data.id.enum_id = 0;
ddc_init_data.id.type = OBJECT_TYPE_GENERIC;
pool->oem_device = link_create_ddc_service(&ddc_init_data);
} else {
pool->oem_device = NULL;
}
return true; return true;
create_fail: create_fail:
......
...@@ -29,7 +29,7 @@ ...@@ -29,7 +29,7 @@
#include "dcn10/dcn10_resource.h" #include "dcn10/dcn10_resource.h"
#include "dc_link_ddc.h" #include "link.h"
#include "dce/dce_abm.h" #include "dce/dce_abm.h"
#include "dce/dce_audio.h" #include "dce/dce_audio.h"
...@@ -1054,7 +1054,7 @@ static void dcn303_resource_destruct(struct resource_pool *pool) ...@@ -1054,7 +1054,7 @@ static void dcn303_resource_destruct(struct resource_pool *pool)
dcn_dccg_destroy(&pool->dccg); dcn_dccg_destroy(&pool->dccg);
if (pool->oem_device != NULL) if (pool->oem_device != NULL)
dal_ddc_service_destroy(&pool->oem_device); link_destroy_ddc_service(&pool->oem_device);
} }
static void dcn303_destroy_resource_pool(struct resource_pool **pool) static void dcn303_destroy_resource_pool(struct resource_pool **pool)
...@@ -1421,7 +1421,7 @@ static bool dcn303_resource_construct( ...@@ -1421,7 +1421,7 @@ static bool dcn303_resource_construct(
ddc_init_data.id.id = dc->ctx->dc_bios->fw_info.oem_i2c_obj_id; ddc_init_data.id.id = dc->ctx->dc_bios->fw_info.oem_i2c_obj_id;
ddc_init_data.id.enum_id = 0; ddc_init_data.id.enum_id = 0;
ddc_init_data.id.type = OBJECT_TYPE_GENERIC; ddc_init_data.id.type = OBJECT_TYPE_GENERIC;
pool->oem_device = dal_ddc_service_create(&ddc_init_data); pool->oem_device = link_create_ddc_service(&ddc_init_data);
} else { } else {
pool->oem_device = NULL; pool->oem_device = NULL;
} }
......
...@@ -30,7 +30,6 @@ ...@@ -30,7 +30,6 @@
#include "link_encoder.h" #include "link_encoder.h"
#include "dcn31_dio_link_encoder.h" #include "dcn31_dio_link_encoder.h"
#include "stream_encoder.h" #include "stream_encoder.h"
#include "i2caux_interface.h"
#include "dc_bios_types.h" #include "dc_bios_types.h"
#include "gpio_service_interface.h" #include "gpio_service_interface.h"
......
...@@ -31,7 +31,6 @@ ...@@ -31,7 +31,6 @@
#include "dcn31/dcn31_dio_link_encoder.h" #include "dcn31/dcn31_dio_link_encoder.h"
#include "dcn32_dio_link_encoder.h" #include "dcn32_dio_link_encoder.h"
#include "stream_encoder.h" #include "stream_encoder.h"
#include "i2caux_interface.h"
#include "dc_bios_types.h" #include "dc_bios_types.h"
#include "link_enc_cfg.h" #include "link_enc_cfg.h"
......
...@@ -69,7 +69,7 @@ ...@@ -69,7 +69,7 @@
#include "dml/display_mode_vba.h" #include "dml/display_mode_vba.h"
#include "dcn32/dcn32_dccg.h" #include "dcn32/dcn32_dccg.h"
#include "dcn10/dcn10_resource.h" #include "dcn10/dcn10_resource.h"
#include "dc_link_ddc.h" #include "link.h"
#include "dcn31/dcn31_panel_cntl.h" #include "dcn31/dcn31_panel_cntl.h"
#include "dcn30/dcn30_dwb.h" #include "dcn30/dcn30_dwb.h"
...@@ -1508,7 +1508,7 @@ static void dcn32_resource_destruct(struct dcn32_resource_pool *pool) ...@@ -1508,7 +1508,7 @@ static void dcn32_resource_destruct(struct dcn32_resource_pool *pool)
dcn_dccg_destroy(&pool->base.dccg); dcn_dccg_destroy(&pool->base.dccg);
if (pool->base.oem_device != NULL) if (pool->base.oem_device != NULL)
dal_ddc_service_destroy(&pool->base.oem_device); link_destroy_ddc_service(&pool->base.oem_device);
} }
...@@ -2450,7 +2450,7 @@ static bool dcn32_resource_construct( ...@@ -2450,7 +2450,7 @@ static bool dcn32_resource_construct(
ddc_init_data.id.id = dc->ctx->dc_bios->fw_info.oem_i2c_obj_id; ddc_init_data.id.id = dc->ctx->dc_bios->fw_info.oem_i2c_obj_id;
ddc_init_data.id.enum_id = 0; ddc_init_data.id.enum_id = 0;
ddc_init_data.id.type = OBJECT_TYPE_GENERIC; ddc_init_data.id.type = OBJECT_TYPE_GENERIC;
pool->base.oem_device = dal_ddc_service_create(&ddc_init_data); pool->base.oem_device = link_create_ddc_service(&ddc_init_data);
} else { } else {
pool->base.oem_device = NULL; pool->base.oem_device = NULL;
} }
......
...@@ -31,7 +31,6 @@ ...@@ -31,7 +31,6 @@
#include "dcn321_dio_link_encoder.h" #include "dcn321_dio_link_encoder.h"
#include "dcn31/dcn31_dio_link_encoder.h" #include "dcn31/dcn31_dio_link_encoder.h"
#include "stream_encoder.h" #include "stream_encoder.h"
#include "i2caux_interface.h"
#include "dc_bios_types.h" #include "dc_bios_types.h"
#include "gpio_service_interface.h" #include "gpio_service_interface.h"
......
...@@ -73,7 +73,7 @@ ...@@ -73,7 +73,7 @@
#include "dml/display_mode_vba.h" #include "dml/display_mode_vba.h"
#include "dcn32/dcn32_dccg.h" #include "dcn32/dcn32_dccg.h"
#include "dcn10/dcn10_resource.h" #include "dcn10/dcn10_resource.h"
#include "dc_link_ddc.h" #include "link.h"
#include "dcn31/dcn31_panel_cntl.h" #include "dcn31/dcn31_panel_cntl.h"
#include "dcn30/dcn30_dwb.h" #include "dcn30/dcn30_dwb.h"
...@@ -1493,7 +1493,7 @@ static void dcn321_resource_destruct(struct dcn321_resource_pool *pool) ...@@ -1493,7 +1493,7 @@ static void dcn321_resource_destruct(struct dcn321_resource_pool *pool)
dcn_dccg_destroy(&pool->base.dccg); dcn_dccg_destroy(&pool->base.dccg);
if (pool->base.oem_device != NULL) if (pool->base.oem_device != NULL)
dal_ddc_service_destroy(&pool->base.oem_device); link_destroy_ddc_service(&pool->base.oem_device);
} }
...@@ -1991,7 +1991,7 @@ static bool dcn321_resource_construct( ...@@ -1991,7 +1991,7 @@ static bool dcn321_resource_construct(
ddc_init_data.id.id = dc->ctx->dc_bios->fw_info.oem_i2c_obj_id; ddc_init_data.id.id = dc->ctx->dc_bios->fw_info.oem_i2c_obj_id;
ddc_init_data.id.enum_id = 0; ddc_init_data.id.enum_id = 0;
ddc_init_data.id.type = OBJECT_TYPE_GENERIC; ddc_init_data.id.type = OBJECT_TYPE_GENERIC;
pool->base.oem_device = dal_ddc_service_create(&ddc_init_data); pool->base.oem_device = link_create_ddc_service(&ddc_init_data);
} else { } else {
pool->base.oem_device = NULL; pool->base.oem_device = NULL;
} }
......
...@@ -28,10 +28,9 @@ ...@@ -28,10 +28,9 @@
#include "dm_services.h" #include "dm_services.h"
#include "dm_helpers.h" #include "dm_helpers.h"
#include "include/hdcp_types.h" #include "include/hdcp_types.h"
#include "include/i2caux_interface.h"
#include "include/signal_types.h" #include "include/signal_types.h"
#include "core_types.h" #include "core_types.h"
#include "dc_link_ddc.h" #include "link.h"
#include "link_hwss.h" #include "link_hwss.h"
#include "inc/link_dpcd.h" #include "inc/link_dpcd.h"
......
...@@ -27,7 +27,6 @@ ...@@ -27,7 +27,6 @@
#define __DAL_AUX_ENGINE_H__ #define __DAL_AUX_ENGINE_H__
#include "dc_ddc_types.h" #include "dc_ddc_types.h"
#include "include/i2caux_interface.h"
enum aux_return_code_type; enum aux_return_code_type;
...@@ -81,7 +80,12 @@ enum i2c_default_speed { ...@@ -81,7 +80,12 @@ enum i2c_default_speed {
I2CAUX_DEFAULT_I2C_SW_SPEED = 50 I2CAUX_DEFAULT_I2C_SW_SPEED = 50
}; };
union aux_config; union aux_config {
struct {
uint32_t ALLOW_AUX_WHEN_HPD_LOW:1;
} bits;
uint32_t raw;
};
struct aux_engine { struct aux_engine {
uint32_t inst; uint32_t inst;
......
...@@ -44,4 +44,41 @@ struct gpio *link_get_hpd_gpio(struct dc_bios *dcb, ...@@ -44,4 +44,41 @@ struct gpio *link_get_hpd_gpio(struct dc_bios *dcb,
struct graphics_object_id link_id, struct graphics_object_id link_id,
struct gpio_service *gpio_service); struct gpio_service *gpio_service);
struct ddc_service_init_data {
struct graphics_object_id id;
struct dc_context *ctx;
struct dc_link *link;
bool is_dpia_link;
};
struct ddc_service *link_create_ddc_service(
struct ddc_service_init_data *ddc_init_data);
void link_destroy_ddc_service(struct ddc_service **ddc);
bool link_is_in_aux_transaction_mode(struct ddc_service *ddc);
bool link_query_ddc_data(
struct ddc_service *ddc,
uint32_t address,
uint8_t *write_buf,
uint32_t write_size,
uint8_t *read_buf,
uint32_t read_size);
/* Attempt to submit an aux payload, retrying on timeouts, defers, and busy
* states as outlined in the DP spec. Returns true if the request was
* successful.
*
* NOTE: The function requires explicit mutex on DM side in order to prevent
* potential race condition. DC components should call the dpcd read/write
* function in dm_helpers in order to access dpcd safely
*/
bool link_aux_transfer_with_retries_no_mutex(struct ddc_service *ddc,
struct aux_payload *payload);
uint32_t link_get_aux_defer_delay(struct ddc_service *ddc);
#endif /* __DC_LINK_HPD_H__ */ #endif /* __DC_LINK_HPD_H__ */
...@@ -24,7 +24,7 @@ ...@@ -24,7 +24,7 @@
# PHY, HPD, DDC and etc). # PHY, HPD, DDC and etc).
LINK = link_hwss_dio.o link_hwss_dpia.o link_hwss_hpo_dp.o link_dp_trace.o \ LINK = link_hwss_dio.o link_hwss_dpia.o link_hwss_hpo_dp.o link_dp_trace.o \
link_hpd.o link_hpd.o link_ddc.o
AMD_DAL_LINK = $(addprefix $(AMDDALPATH)/dc/link/,$(LINK)) AMD_DAL_LINK = $(addprefix $(AMDDALPATH)/dc/link/,$(LINK))
......
...@@ -23,20 +23,20 @@ ...@@ -23,20 +23,20 @@
* *
*/ */
#include "dm_services.h" /* FILE POLICY AND INTENDED USAGE:
#include "dm_helpers.h" *
#include "gpio_service_interface.h" * This file implements generic display communication protocols such as i2c, aux
#include "include/ddc_service_types.h" * and scdc. The file should not contain any specific applications of these
#include "include/grph_object_id.h" * protocols such as display capability query, detection, or handshaking such as
#include "include/dpcd_defs.h" * link training.
#include "include/logger_interface.h" */
#include "include/vector.h" #include "link_ddc.h"
#include "core_types.h" #include "vector.h"
#include "dc_link_ddc.h"
#include "dce/dce_aux.h" #include "dce/dce_aux.h"
#include "dmub/inc/dmub_cmd.h" #include "dal_asic_id.h"
#include "link_dpcd.h" #include "link_dpcd.h"
#include "include/dal_asic_id.h" #include "dm_helpers.h"
#include "atomfirmware.h"
#define DC_LOGGER_INIT(logger) #define DC_LOGGER_INIT(logger)
...@@ -45,86 +45,6 @@ static const uint8_t DP_VGA_DONGLE_BRANCH_DEV_NAME[] = "DpVga"; ...@@ -45,86 +45,6 @@ static const uint8_t DP_VGA_DONGLE_BRANCH_DEV_NAME[] = "DpVga";
static const uint8_t DP_DVI_CONVERTER_ID_4[] = "m2DVIa"; static const uint8_t DP_DVI_CONVERTER_ID_4[] = "m2DVIa";
static const uint8_t DP_DVI_CONVERTER_ID_5[] = "3393N2"; static const uint8_t DP_DVI_CONVERTER_ID_5[] = "3393N2";
#define AUX_POWER_UP_WA_DELAY 500
#define I2C_OVER_AUX_DEFER_WA_DELAY 70
#define DPVGA_DONGLE_AUX_DEFER_WA_DELAY 40
#define I2C_OVER_AUX_DEFER_WA_DELAY_1MS 1
/* CV smart dongle slave address for retrieving supported HDTV modes*/
#define CV_SMART_DONGLE_ADDRESS 0x20
/* DVI-HDMI dongle slave address for retrieving dongle signature*/
#define DVI_HDMI_DONGLE_ADDRESS 0x68
struct dvi_hdmi_dongle_signature_data {
int8_t vendor[3];/* "AMD" */
uint8_t version[2];
uint8_t size;
int8_t id[11];/* "6140063500G"*/
};
/* DP-HDMI dongle slave address for retrieving dongle signature*/
#define DP_HDMI_DONGLE_ADDRESS 0x40
static const uint8_t dp_hdmi_dongle_signature_str[] = "DP-HDMI ADAPTOR";
#define DP_HDMI_DONGLE_SIGNATURE_EOT 0x04
struct dp_hdmi_dongle_signature_data {
int8_t id[15];/* "DP-HDMI ADAPTOR"*/
uint8_t eot;/* end of transmition '\x4' */
};
/* SCDC Address defines (HDMI 2.0)*/
#define HDMI_SCDC_WRITE_UPDATE_0_ARRAY 3
#define HDMI_SCDC_ADDRESS 0x54
#define HDMI_SCDC_SINK_VERSION 0x01
#define HDMI_SCDC_SOURCE_VERSION 0x02
#define HDMI_SCDC_UPDATE_0 0x10
#define HDMI_SCDC_TMDS_CONFIG 0x20
#define HDMI_SCDC_SCRAMBLER_STATUS 0x21
#define HDMI_SCDC_CONFIG_0 0x30
#define HDMI_SCDC_STATUS_FLAGS 0x40
#define HDMI_SCDC_ERR_DETECT 0x50
#define HDMI_SCDC_TEST_CONFIG 0xC0
union hdmi_scdc_update_read_data {
uint8_t byte[2];
struct {
uint8_t STATUS_UPDATE:1;
uint8_t CED_UPDATE:1;
uint8_t RR_TEST:1;
uint8_t RESERVED:5;
uint8_t RESERVED2:8;
} fields;
};
union hdmi_scdc_status_flags_data {
uint8_t byte;
struct {
uint8_t CLOCK_DETECTED:1;
uint8_t CH0_LOCKED:1;
uint8_t CH1_LOCKED:1;
uint8_t CH2_LOCKED:1;
uint8_t RESERVED:4;
} fields;
};
union hdmi_scdc_ced_data {
uint8_t byte[7];
struct {
uint8_t CH0_8LOW:8;
uint8_t CH0_7HIGH:7;
uint8_t CH0_VALID:1;
uint8_t CH1_8LOW:8;
uint8_t CH1_7HIGH:7;
uint8_t CH1_VALID:1;
uint8_t CH2_8LOW:8;
uint8_t CH2_7HIGH:7;
uint8_t CH2_VALID:1;
uint8_t CHECKSUM:8;
uint8_t RESERVED:8;
uint8_t RESERVED2:8;
uint8_t RESERVED3:8;
uint8_t RESERVED4:4;
} fields;
};
struct i2c_payloads { struct i2c_payloads {
struct vector payloads; struct vector payloads;
}; };
...@@ -157,7 +77,7 @@ static uint32_t dal_ddc_i2c_payloads_get_count(struct i2c_payloads *p) ...@@ -157,7 +77,7 @@ static uint32_t dal_ddc_i2c_payloads_get_count(struct i2c_payloads *p)
#define DDC_MIN(a, b) (((a) < (b)) ? (a) : (b)) #define DDC_MIN(a, b) (((a) < (b)) ? (a) : (b))
void dal_ddc_i2c_payloads_add( static void i2c_payloads_add(
struct i2c_payloads *payloads, struct i2c_payloads *payloads,
uint32_t address, uint32_t address,
uint32_t len, uint32_t len,
...@@ -225,7 +145,7 @@ static void ddc_service_construct( ...@@ -225,7 +145,7 @@ static void ddc_service_construct(
ddc_service->wa.raw = 0; ddc_service->wa.raw = 0;
} }
struct ddc_service *dal_ddc_service_create( struct ddc_service *link_create_ddc_service(
struct ddc_service_init_data *init_data) struct ddc_service_init_data *init_data)
{ {
struct ddc_service *ddc_service; struct ddc_service *ddc_service;
...@@ -245,7 +165,7 @@ static void ddc_service_destruct(struct ddc_service *ddc) ...@@ -245,7 +165,7 @@ static void ddc_service_destruct(struct ddc_service *ddc)
dal_gpio_destroy_ddc(&ddc->ddc_pin); dal_gpio_destroy_ddc(&ddc->ddc_pin);
} }
void dal_ddc_service_destroy(struct ddc_service **ddc) void link_destroy_ddc_service(struct ddc_service **ddc)
{ {
if (!ddc || !*ddc) { if (!ddc || !*ddc) {
BREAK_TO_DEBUGGER(); BREAK_TO_DEBUGGER();
...@@ -256,19 +176,14 @@ void dal_ddc_service_destroy(struct ddc_service **ddc) ...@@ -256,19 +176,14 @@ void dal_ddc_service_destroy(struct ddc_service **ddc)
*ddc = NULL; *ddc = NULL;
} }
enum ddc_service_type dal_ddc_service_get_type(struct ddc_service *ddc) void set_ddc_transaction_type(
{
return DDC_SERVICE_TYPE_CONNECTOR;
}
void dal_ddc_service_set_transaction_type(
struct ddc_service *ddc, struct ddc_service *ddc,
enum ddc_transaction_type type) enum ddc_transaction_type type)
{ {
ddc->transaction_type = type; ddc->transaction_type = type;
} }
bool dal_ddc_service_is_in_aux_transaction_mode(struct ddc_service *ddc) bool link_is_in_aux_transaction_mode(struct ddc_service *ddc)
{ {
switch (ddc->transaction_type) { switch (ddc->transaction_type) {
case DDC_TRANSACTION_TYPE_I2C_OVER_AUX: case DDC_TRANSACTION_TYPE_I2C_OVER_AUX:
...@@ -281,7 +196,7 @@ bool dal_ddc_service_is_in_aux_transaction_mode(struct ddc_service *ddc) ...@@ -281,7 +196,7 @@ bool dal_ddc_service_is_in_aux_transaction_mode(struct ddc_service *ddc)
return false; return false;
} }
void ddc_service_set_dongle_type(struct ddc_service *ddc, void set_dongle_type(struct ddc_service *ddc,
enum display_dongle_type dongle_type) enum display_dongle_type dongle_type)
{ {
ddc->dongle_type = dongle_type; ddc->dongle_type = dongle_type;
...@@ -323,7 +238,7 @@ static uint32_t defer_delay_converter_wa( ...@@ -323,7 +238,7 @@ static uint32_t defer_delay_converter_wa(
#define DP_TRANSLATOR_DELAY 5 #define DP_TRANSLATOR_DELAY 5
uint32_t get_defer_delay(struct ddc_service *ddc) uint32_t link_get_aux_defer_delay(struct ddc_service *ddc)
{ {
uint32_t defer_delay = 0; uint32_t defer_delay = 0;
...@@ -351,175 +266,45 @@ uint32_t get_defer_delay(struct ddc_service *ddc) ...@@ -351,175 +266,45 @@ uint32_t get_defer_delay(struct ddc_service *ddc)
return defer_delay; return defer_delay;
} }
static bool i2c_read( static bool submit_aux_command(struct ddc_service *ddc,
struct ddc_service *ddc, struct aux_payload *payload)
uint32_t address,
uint8_t *buffer,
uint32_t len)
{
uint8_t offs_data = 0;
struct i2c_payload payloads[2] = {
{
.write = true,
.address = address,
.length = 1,
.data = &offs_data },
{
.write = false,
.address = address,
.length = len,
.data = buffer } };
struct i2c_command command = {
.payloads = payloads,
.number_of_payloads = 2,
.engine = DDC_I2C_COMMAND_ENGINE,
.speed = ddc->ctx->dc->caps.i2c_speed_in_khz };
return dm_helpers_submit_i2c(
ddc->ctx,
ddc->link,
&command);
}
void dal_ddc_service_i2c_query_dp_dual_mode_adaptor(
struct ddc_service *ddc,
struct display_sink_capability *sink_cap)
{ {
uint8_t i; uint32_t retrieved = 0;
bool is_valid_hdmi_signature; bool ret = false;
enum display_dongle_type *dongle = &sink_cap->dongle_type;
uint8_t type2_dongle_buf[DP_ADAPTOR_TYPE2_SIZE];
bool is_type2_dongle = false;
int retry_count = 2;
struct dp_hdmi_dongle_signature_data *dongle_signature;
/* Assume we have no valid DP passive dongle connected */
*dongle = DISPLAY_DONGLE_NONE;
sink_cap->max_hdmi_pixel_clock = DP_ADAPTOR_HDMI_SAFE_MAX_TMDS_CLK;
/* Read DP-HDMI dongle I2c (no response interpreted as DP-DVI dongle)*/
if (!i2c_read(
ddc,
DP_HDMI_DONGLE_ADDRESS,
type2_dongle_buf,
sizeof(type2_dongle_buf))) {
/* Passive HDMI dongles can sometimes fail here without retrying*/
while (retry_count > 0) {
if (i2c_read(ddc,
DP_HDMI_DONGLE_ADDRESS,
type2_dongle_buf,
sizeof(type2_dongle_buf)))
break;
retry_count--;
}
if (retry_count == 0) {
*dongle = DISPLAY_DONGLE_DP_DVI_DONGLE;
sink_cap->max_hdmi_pixel_clock = DP_ADAPTOR_DVI_MAX_TMDS_CLK;
CONN_DATA_DETECT(ddc->link, type2_dongle_buf, sizeof(type2_dongle_buf),
"DP-DVI passive dongle %dMhz: ",
DP_ADAPTOR_DVI_MAX_TMDS_CLK / 1000);
return;
}
}
/* Check if Type 2 dongle.*/
if (type2_dongle_buf[DP_ADAPTOR_TYPE2_REG_ID] == DP_ADAPTOR_TYPE2_ID)
is_type2_dongle = true;
dongle_signature =
(struct dp_hdmi_dongle_signature_data *)type2_dongle_buf;
is_valid_hdmi_signature = true; if (!ddc)
return false;
/* Check EOT */ if (!payload)
if (dongle_signature->eot != DP_HDMI_DONGLE_SIGNATURE_EOT) { return false;
is_valid_hdmi_signature = false;
}
/* Check signature */ do {
for (i = 0; i < sizeof(dongle_signature->id); ++i) { struct aux_payload current_payload;
/* If its not the right signature, bool is_end_of_payload = (retrieved + DEFAULT_AUX_MAX_DATA_SIZE) >=
* skip mismatch in subversion byte.*/ payload->length;
if (dongle_signature->id[i] != uint32_t payload_length = is_end_of_payload ?
dp_hdmi_dongle_signature_str[i] && i != 3) { payload->length - retrieved : DEFAULT_AUX_MAX_DATA_SIZE;
if (is_type2_dongle) { current_payload.address = payload->address;
is_valid_hdmi_signature = false; current_payload.data = &payload->data[retrieved];
break; current_payload.defer_delay = payload->defer_delay;
} current_payload.i2c_over_aux = payload->i2c_over_aux;
current_payload.length = payload_length;
/* set mot (middle of transaction) to false if it is the last payload */
current_payload.mot = is_end_of_payload ? payload->mot:true;
current_payload.write_status_update = false;
current_payload.reply = payload->reply;
current_payload.write = payload->write;
} ret = link_aux_transfer_with_retries_no_mutex(ddc, &current_payload);
}
if (is_type2_dongle) { retrieved += payload_length;
uint32_t max_tmds_clk = } while (retrieved < payload->length && ret == true);
type2_dongle_buf[DP_ADAPTOR_TYPE2_REG_MAX_TMDS_CLK];
max_tmds_clk = max_tmds_clk * 2 + max_tmds_clk / 2;
if (0 == max_tmds_clk ||
max_tmds_clk < DP_ADAPTOR_TYPE2_MIN_TMDS_CLK ||
max_tmds_clk > DP_ADAPTOR_TYPE2_MAX_TMDS_CLK) {
*dongle = DISPLAY_DONGLE_DP_DVI_DONGLE;
CONN_DATA_DETECT(ddc->link, type2_dongle_buf,
sizeof(type2_dongle_buf),
"DP-DVI passive dongle %dMhz: ",
DP_ADAPTOR_DVI_MAX_TMDS_CLK / 1000);
} else {
if (is_valid_hdmi_signature == true) {
*dongle = DISPLAY_DONGLE_DP_HDMI_DONGLE;
CONN_DATA_DETECT(ddc->link, type2_dongle_buf,
sizeof(type2_dongle_buf),
"Type 2 DP-HDMI passive dongle %dMhz: ",
max_tmds_clk);
} else {
*dongle = DISPLAY_DONGLE_DP_HDMI_MISMATCHED_DONGLE;
CONN_DATA_DETECT(ddc->link, type2_dongle_buf,
sizeof(type2_dongle_buf),
"Type 2 DP-HDMI passive dongle (no signature) %dMhz: ",
max_tmds_clk);
}
/* Multiply by 1000 to convert to kHz. */
sink_cap->max_hdmi_pixel_clock =
max_tmds_clk * 1000;
}
sink_cap->is_dongle_type_one = false;
} else {
if (is_valid_hdmi_signature == true) {
*dongle = DISPLAY_DONGLE_DP_HDMI_DONGLE;
CONN_DATA_DETECT(ddc->link, type2_dongle_buf,
sizeof(type2_dongle_buf),
"Type 1 DP-HDMI passive dongle %dMhz: ",
sink_cap->max_hdmi_pixel_clock / 1000);
} else {
*dongle = DISPLAY_DONGLE_DP_HDMI_MISMATCHED_DONGLE;
CONN_DATA_DETECT(ddc->link, type2_dongle_buf,
sizeof(type2_dongle_buf),
"Type 1 DP-HDMI passive dongle (no signature) %dMhz: ",
sink_cap->max_hdmi_pixel_clock / 1000);
}
sink_cap->is_dongle_type_one = true;
}
return; return ret;
} }
enum { bool link_query_ddc_data(
DP_SINK_CAP_SIZE =
DP_EDP_CONFIGURATION_CAP - DP_DPCD_REV + 1
};
bool dal_ddc_service_query_ddc_data(
struct ddc_service *ddc, struct ddc_service *ddc,
uint32_t address, uint32_t address,
uint8_t *write_buf, uint8_t *write_buf,
...@@ -529,7 +314,7 @@ bool dal_ddc_service_query_ddc_data( ...@@ -529,7 +314,7 @@ bool dal_ddc_service_query_ddc_data(
{ {
bool success = true; bool success = true;
uint32_t payload_size = uint32_t payload_size =
dal_ddc_service_is_in_aux_transaction_mode(ddc) ? link_is_in_aux_transaction_mode(ddc) ?
DEFAULT_AUX_MAX_DATA_SIZE : EDID_SEGMENT_SIZE; DEFAULT_AUX_MAX_DATA_SIZE : EDID_SEGMENT_SIZE;
uint32_t write_payloads = uint32_t write_payloads =
...@@ -543,13 +328,13 @@ bool dal_ddc_service_query_ddc_data( ...@@ -543,13 +328,13 @@ bool dal_ddc_service_query_ddc_data(
if (!payloads_num) if (!payloads_num)
return false; return false;
if (dal_ddc_service_is_in_aux_transaction_mode(ddc)) { if (link_is_in_aux_transaction_mode(ddc)) {
struct aux_payload payload; struct aux_payload payload;
payload.i2c_over_aux = true; payload.i2c_over_aux = true;
payload.address = address; payload.address = address;
payload.reply = NULL; payload.reply = NULL;
payload.defer_delay = get_defer_delay(ddc); payload.defer_delay = link_get_aux_defer_delay(ddc);
payload.write_status_update = false; payload.write_status_update = false;
if (write_size != 0) { if (write_size != 0) {
...@@ -561,7 +346,7 @@ bool dal_ddc_service_query_ddc_data( ...@@ -561,7 +346,7 @@ bool dal_ddc_service_query_ddc_data(
payload.length = write_size; payload.length = write_size;
payload.data = write_buf; payload.data = write_buf;
success = dal_ddc_submit_aux_command(ddc, &payload); success = submit_aux_command(ddc, &payload);
} }
if (read_size != 0 && success) { if (read_size != 0 && success) {
...@@ -573,7 +358,7 @@ bool dal_ddc_service_query_ddc_data( ...@@ -573,7 +358,7 @@ bool dal_ddc_service_query_ddc_data(
payload.length = read_size; payload.length = read_size;
payload.data = read_buf; payload.data = read_buf;
success = dal_ddc_submit_aux_command(ddc, &payload); success = submit_aux_command(ddc, &payload);
} }
} else { } else {
struct i2c_command command = {0}; struct i2c_command command = {0};
...@@ -587,10 +372,10 @@ bool dal_ddc_service_query_ddc_data( ...@@ -587,10 +372,10 @@ bool dal_ddc_service_query_ddc_data(
command.engine = DDC_I2C_COMMAND_ENGINE; command.engine = DDC_I2C_COMMAND_ENGINE;
command.speed = ddc->ctx->dc->caps.i2c_speed_in_khz; command.speed = ddc->ctx->dc->caps.i2c_speed_in_khz;
dal_ddc_i2c_payloads_add( i2c_payloads_add(
&payloads, address, write_size, write_buf, true); &payloads, address, write_size, write_buf, true);
dal_ddc_i2c_payloads_add( i2c_payloads_add(
&payloads, address, read_size, read_buf, false); &payloads, address, read_size, read_buf, false);
command.number_of_payloads = command.number_of_payloads =
...@@ -607,51 +392,6 @@ bool dal_ddc_service_query_ddc_data( ...@@ -607,51 +392,6 @@ bool dal_ddc_service_query_ddc_data(
return success; return success;
} }
bool dal_ddc_submit_aux_command(struct ddc_service *ddc,
struct aux_payload *payload)
{
uint32_t retrieved = 0;
bool ret = false;
if (!ddc)
return false;
if (!payload)
return false;
do {
struct aux_payload current_payload;
bool is_end_of_payload = (retrieved + DEFAULT_AUX_MAX_DATA_SIZE) >=
payload->length;
uint32_t payload_length = is_end_of_payload ?
payload->length - retrieved : DEFAULT_AUX_MAX_DATA_SIZE;
current_payload.address = payload->address;
current_payload.data = &payload->data[retrieved];
current_payload.defer_delay = payload->defer_delay;
current_payload.i2c_over_aux = payload->i2c_over_aux;
current_payload.length = payload_length;
/* set mot (middle of transaction) to false if it is the last payload */
current_payload.mot = is_end_of_payload ? payload->mot:true;
current_payload.write_status_update = false;
current_payload.reply = payload->reply;
current_payload.write = payload->write;
ret = dc_link_aux_transfer_with_retries(ddc, &current_payload);
retrieved += payload_length;
} while (retrieved < payload->length && ret == true);
return ret;
}
/* dc_link_aux_transfer_raw() - 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 and the result through
* *operation_result. Returns the number of bytes transferred,
* or -1 on a failure.
*/
int dc_link_aux_transfer_raw(struct ddc_service *ddc, int dc_link_aux_transfer_raw(struct ddc_service *ddc,
struct aux_payload *payload, struct aux_payload *payload,
enum aux_return_code_type *operation_result) enum aux_return_code_type *operation_result)
...@@ -664,22 +404,14 @@ int dc_link_aux_transfer_raw(struct ddc_service *ddc, ...@@ -664,22 +404,14 @@ int dc_link_aux_transfer_raw(struct ddc_service *ddc,
} }
} }
/* dc_link_aux_transfer_with_retries() - Attempt to submit an bool link_aux_transfer_with_retries_no_mutex(struct ddc_service *ddc,
* aux payload, retrying on timeouts, defers, and busy states
* as outlined in the DP spec. Returns true if the request
* was successful.
*
* Unless you want to implement your own retry semantics, this
* is probably the one you want.
*/
bool dc_link_aux_transfer_with_retries(struct ddc_service *ddc,
struct aux_payload *payload) struct aux_payload *payload)
{ {
return dce_aux_transfer_with_retries(ddc, payload); return dce_aux_transfer_with_retries(ddc, payload);
} }
bool dc_link_aux_try_to_configure_timeout(struct ddc_service *ddc, bool try_to_configure_aux_timeout(struct ddc_service *ddc,
uint32_t timeout) uint32_t timeout)
{ {
bool result = false; bool result = false;
...@@ -712,20 +444,12 @@ bool dc_link_aux_try_to_configure_timeout(struct ddc_service *ddc, ...@@ -712,20 +444,12 @@ bool dc_link_aux_try_to_configure_timeout(struct ddc_service *ddc,
return result; return result;
} }
/*test only function*/ struct ddc *get_ddc_pin(struct ddc_service *ddc_service)
void dal_ddc_service_set_ddc_pin(
struct ddc_service *ddc_service,
struct ddc *ddc)
{
ddc_service->ddc_pin = ddc;
}
struct ddc *dal_ddc_service_get_ddc_pin(struct ddc_service *ddc_service)
{ {
return ddc_service->ddc_pin; return ddc_service->ddc_pin;
} }
void dal_ddc_service_write_scdc_data(struct ddc_service *ddc_service, void write_scdc_data(struct ddc_service *ddc_service,
uint32_t pix_clk, uint32_t pix_clk,
bool lte_340_scramble) bool lte_340_scramble)
{ {
...@@ -740,13 +464,13 @@ void dal_ddc_service_write_scdc_data(struct ddc_service *ddc_service, ...@@ -740,13 +464,13 @@ void dal_ddc_service_write_scdc_data(struct ddc_service *ddc_service,
ddc_service->link->local_sink->edid_caps.panel_patch.skip_scdc_overwrite) ddc_service->link->local_sink->edid_caps.panel_patch.skip_scdc_overwrite)
return; return;
dal_ddc_service_query_ddc_data(ddc_service, slave_address, &offset, link_query_ddc_data(ddc_service, slave_address, &offset,
sizeof(offset), &sink_version, sizeof(sink_version)); sizeof(offset), &sink_version, sizeof(sink_version));
if (sink_version == 1) { if (sink_version == 1) {
/*Source Version = 1*/ /*Source Version = 1*/
write_buffer[0] = HDMI_SCDC_SOURCE_VERSION; write_buffer[0] = HDMI_SCDC_SOURCE_VERSION;
write_buffer[1] = 1; write_buffer[1] = 1;
dal_ddc_service_query_ddc_data(ddc_service, slave_address, link_query_ddc_data(ddc_service, slave_address,
write_buffer, sizeof(write_buffer), NULL, 0); write_buffer, sizeof(write_buffer), NULL, 0);
/*Read Request from SCDC caps*/ /*Read Request from SCDC caps*/
} }
...@@ -759,11 +483,11 @@ void dal_ddc_service_write_scdc_data(struct ddc_service *ddc_service, ...@@ -759,11 +483,11 @@ void dal_ddc_service_write_scdc_data(struct ddc_service *ddc_service,
} else { } else {
write_buffer[1] = 0; write_buffer[1] = 0;
} }
dal_ddc_service_query_ddc_data(ddc_service, slave_address, write_buffer, link_query_ddc_data(ddc_service, slave_address, write_buffer,
sizeof(write_buffer), NULL, 0); sizeof(write_buffer), NULL, 0);
} }
void dal_ddc_service_read_scdc_data(struct ddc_service *ddc_service) void read_scdc_data(struct ddc_service *ddc_service)
{ {
uint8_t slave_address = HDMI_SCDC_ADDRESS; uint8_t slave_address = HDMI_SCDC_ADDRESS;
uint8_t offset = HDMI_SCDC_TMDS_CONFIG; uint8_t offset = HDMI_SCDC_TMDS_CONFIG;
...@@ -773,20 +497,19 @@ void dal_ddc_service_read_scdc_data(struct ddc_service *ddc_service) ...@@ -773,20 +497,19 @@ void dal_ddc_service_read_scdc_data(struct ddc_service *ddc_service)
ddc_service->link->local_sink->edid_caps.panel_patch.skip_scdc_overwrite) ddc_service->link->local_sink->edid_caps.panel_patch.skip_scdc_overwrite)
return; return;
dal_ddc_service_query_ddc_data(ddc_service, slave_address, &offset, link_query_ddc_data(ddc_service, slave_address, &offset,
sizeof(offset), &tmds_config, sizeof(tmds_config)); sizeof(offset), &tmds_config, sizeof(tmds_config));
if (tmds_config & 0x1) { if (tmds_config & 0x1) {
union hdmi_scdc_status_flags_data status_data = {0}; union hdmi_scdc_status_flags_data status_data = {0};
uint8_t scramble_status = 0; uint8_t scramble_status = 0;
offset = HDMI_SCDC_SCRAMBLER_STATUS; offset = HDMI_SCDC_SCRAMBLER_STATUS;
dal_ddc_service_query_ddc_data(ddc_service, slave_address, link_query_ddc_data(ddc_service, slave_address,
&offset, sizeof(offset), &scramble_status, &offset, sizeof(offset), &scramble_status,
sizeof(scramble_status)); sizeof(scramble_status));
offset = HDMI_SCDC_STATUS_FLAGS; offset = HDMI_SCDC_STATUS_FLAGS;
dal_ddc_service_query_ddc_data(ddc_service, slave_address, link_query_ddc_data(ddc_service, slave_address,
&offset, sizeof(offset), &status_data.byte, &offset, sizeof(offset), &status_data.byte,
sizeof(status_data.byte)); sizeof(status_data.byte));
} }
} }
...@@ -23,60 +23,37 @@ ...@@ -23,60 +23,37 @@
* *
*/ */
#ifndef __DAL_I2CAUX_INTERFACE_H__ #ifndef __DAL_DDC_SERVICE_H__
#define __DAL_I2CAUX_INTERFACE_H__ #define __DAL_DDC_SERVICE_H__
#include "dc_types.h" #include "link.h"
#include "gpio_service_interface.h"
#define AUX_POWER_UP_WA_DELAY 500
#define I2C_OVER_AUX_DEFER_WA_DELAY 70
#define DPVGA_DONGLE_AUX_DEFER_WA_DELAY 40
#define I2C_OVER_AUX_DEFER_WA_DELAY_1MS 1
#define DEFAULT_AUX_MAX_DATA_SIZE 16 #define EDID_SEGMENT_SIZE 256
#define AUX_MAX_DEFER_WRITE_RETRY 20
struct aux_payload { void set_ddc_transaction_type(
/* set following flag to read/write I2C data, struct ddc_service *ddc,
* reset it to read/write DPCD data */ enum ddc_transaction_type type);
bool i2c_over_aux;
/* set following flag to write data,
* reset it to read data */
bool write;
bool mot;
bool write_status_update;
uint32_t address; bool try_to_configure_aux_timeout(struct ddc_service *ddc,
uint32_t length; uint32_t timeout);
uint8_t *data;
/*
* used to return the reply type of the transaction
* ignored if NULL
*/
uint8_t *reply;
/* expressed in milliseconds
* zero means "use default value"
*/
uint32_t defer_delay;
}; void write_scdc_data(
struct ddc_service *ddc_service,
uint32_t pix_clk,
bool lte_340_scramble);
struct aux_command { void read_scdc_data(
struct aux_payload *payloads; struct ddc_service *ddc_service);
uint8_t number_of_payloads;
/* expressed in milliseconds void set_dongle_type(struct ddc_service *ddc,
* zero means "use default value" */ enum display_dongle_type dongle_type);
uint32_t defer_delay;
/* zero means "use default value" */ struct ddc *get_ddc_pin(struct ddc_service *ddc_service);
uint32_t max_defer_write_retry;
enum i2c_mot_mode mot; #endif /* __DAL_DDC_SERVICE_H__ */
};
union aux_config {
struct {
uint32_t ALLOW_AUX_WHEN_HPD_LOW:1;
} bits;
uint32_t raw;
};
#endif
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册