Commit a19de9db authored by Srinivasan Shanmugam's avatar Srinivasan Shanmugam Committed by Alex Deucher

drm/amd/display: Clean up style problems in amdgpu_dm_hdcp.c

Conform to Linux kernel coding style.

And promote sysfs entry for set/get srm to kdoc.
Suggested-by: default avatarRodrigo Siqueira <Rodrigo.Siqueira@amd.com>
Cc: Rodrigo Siqueira <Rodrigo.Siqueira@amd.com>
Cc: Aurabindo Pillai <aurabindo.pillai@amd.com>
Signed-off-by: default avatarSrinivasan Shanmugam <srinivasan.shanmugam@amd.com>
Reviewed-by: default avatarAurabindo Pillai <aurabindo.pillai@amd.com>
Signed-off-by: default avatarAlex Deucher <alexander.deucher@amd.com>
parent 5b29369b
...@@ -39,10 +39,10 @@ ...@@ -39,10 +39,10 @@
static bool static bool
lp_write_i2c(void *handle, uint32_t address, const uint8_t *data, uint32_t size) lp_write_i2c(void *handle, uint32_t address, const uint8_t *data, uint32_t size)
{ {
struct dc_link *link = handle; struct dc_link *link = handle;
struct i2c_payload i2c_payloads[] = {{true, address, size, (void *)data} }; struct i2c_payload i2c_payloads[] = {{true, address, size, (void *)data} };
struct i2c_command cmd = {i2c_payloads, 1, I2C_COMMAND_ENGINE_HW, link->dc->caps.i2c_speed_in_khz}; struct i2c_command cmd = {i2c_payloads, 1, I2C_COMMAND_ENGINE_HW,
link->dc->caps.i2c_speed_in_khz};
return dm_helpers_submit_i2c(link->ctx, link, &cmd); return dm_helpers_submit_i2c(link->ctx, link, &cmd);
} }
...@@ -52,8 +52,10 @@ lp_read_i2c(void *handle, uint32_t address, uint8_t offset, uint8_t *data, uint3 ...@@ -52,8 +52,10 @@ lp_read_i2c(void *handle, uint32_t address, uint8_t offset, uint8_t *data, uint3
{ {
struct dc_link *link = handle; struct dc_link *link = handle;
struct i2c_payload i2c_payloads[] = {{true, address, 1, &offset}, {false, address, size, data} }; struct i2c_payload i2c_payloads[] = {{true, address, 1, &offset},
struct i2c_command cmd = {i2c_payloads, 2, I2C_COMMAND_ENGINE_HW, link->dc->caps.i2c_speed_in_khz}; {false, address, size, data} };
struct i2c_command cmd = {i2c_payloads, 2, I2C_COMMAND_ENGINE_HW,
link->dc->caps.i2c_speed_in_khz};
return dm_helpers_submit_i2c(link->ctx, link, &cmd); return dm_helpers_submit_i2c(link->ctx, link, &cmd);
} }
...@@ -76,7 +78,6 @@ lp_read_dpcd(void *handle, uint32_t address, uint8_t *data, uint32_t size) ...@@ -76,7 +78,6 @@ lp_read_dpcd(void *handle, uint32_t address, uint8_t *data, uint32_t size)
static uint8_t *psp_get_srm(struct psp_context *psp, uint32_t *srm_version, uint32_t *srm_size) static uint8_t *psp_get_srm(struct psp_context *psp, uint32_t *srm_version, uint32_t *srm_size)
{ {
struct ta_hdcp_shared_memory *hdcp_cmd; struct ta_hdcp_shared_memory *hdcp_cmd;
if (!psp->hdcp_context.context.initialized) { if (!psp->hdcp_context.context.initialized) {
...@@ -96,13 +97,12 @@ static uint8_t *psp_get_srm(struct psp_context *psp, uint32_t *srm_version, uint ...@@ -96,13 +97,12 @@ static uint8_t *psp_get_srm(struct psp_context *psp, uint32_t *srm_version, uint
*srm_version = hdcp_cmd->out_msg.hdcp_get_srm.srm_version; *srm_version = hdcp_cmd->out_msg.hdcp_get_srm.srm_version;
*srm_size = hdcp_cmd->out_msg.hdcp_get_srm.srm_buf_size; *srm_size = hdcp_cmd->out_msg.hdcp_get_srm.srm_buf_size;
return hdcp_cmd->out_msg.hdcp_get_srm.srm_buf; return hdcp_cmd->out_msg.hdcp_get_srm.srm_buf;
} }
static int psp_set_srm(struct psp_context *psp, uint8_t *srm, uint32_t srm_size, uint32_t *srm_version) static int psp_set_srm(struct psp_context *psp,
u8 *srm, uint32_t srm_size, uint32_t *srm_version)
{ {
struct ta_hdcp_shared_memory *hdcp_cmd; struct ta_hdcp_shared_memory *hdcp_cmd;
if (!psp->hdcp_context.context.initialized) { if (!psp->hdcp_context.context.initialized) {
...@@ -119,7 +119,8 @@ static int psp_set_srm(struct psp_context *psp, uint8_t *srm, uint32_t srm_size, ...@@ -119,7 +119,8 @@ static int psp_set_srm(struct psp_context *psp, uint8_t *srm, uint32_t srm_size,
psp_hdcp_invoke(psp, hdcp_cmd->cmd_id); psp_hdcp_invoke(psp, hdcp_cmd->cmd_id);
if (hdcp_cmd->hdcp_status != TA_HDCP_STATUS__SUCCESS || hdcp_cmd->out_msg.hdcp_set_srm.valid_signature != 1 || if (hdcp_cmd->hdcp_status != TA_HDCP_STATUS__SUCCESS ||
hdcp_cmd->out_msg.hdcp_set_srm.valid_signature != 1 ||
hdcp_cmd->out_msg.hdcp_set_srm.srm_version == PSP_SRM_VERSION_MAX) hdcp_cmd->out_msg.hdcp_set_srm.srm_version == PSP_SRM_VERSION_MAX)
return -EINVAL; return -EINVAL;
...@@ -150,7 +151,6 @@ static void process_output(struct hdcp_workqueue *hdcp_work) ...@@ -150,7 +151,6 @@ static void process_output(struct hdcp_workqueue *hdcp_work)
static void link_lock(struct hdcp_workqueue *work, bool lock) static void link_lock(struct hdcp_workqueue *work, bool lock)
{ {
int i = 0; int i = 0;
for (i = 0; i < work->max_link; i++) { for (i = 0; i < work->max_link; i++) {
...@@ -160,10 +160,11 @@ static void link_lock(struct hdcp_workqueue *work, bool lock) ...@@ -160,10 +160,11 @@ static void link_lock(struct hdcp_workqueue *work, bool lock)
mutex_unlock(&work[i].mutex); mutex_unlock(&work[i].mutex);
} }
} }
void hdcp_update_display(struct hdcp_workqueue *hdcp_work, void hdcp_update_display(struct hdcp_workqueue *hdcp_work,
unsigned int link_index, unsigned int link_index,
struct amdgpu_dm_connector *aconnector, struct amdgpu_dm_connector *aconnector,
uint8_t content_type, u8 content_type,
bool enable_encryption) bool enable_encryption)
{ {
struct hdcp_workqueue *hdcp_w = &hdcp_work[link_index]; struct hdcp_workqueue *hdcp_w = &hdcp_work[link_index];
...@@ -178,18 +179,19 @@ void hdcp_update_display(struct hdcp_workqueue *hdcp_work, ...@@ -178,18 +179,19 @@ void hdcp_update_display(struct hdcp_workqueue *hdcp_work,
query.display = NULL; query.display = NULL;
mod_hdcp_query_display(&hdcp_w->hdcp, aconnector->base.index, &query); mod_hdcp_query_display(&hdcp_w->hdcp, aconnector->base.index, &query);
if (query.display != NULL) { if (query.display) {
memcpy(display, query.display, sizeof(struct mod_hdcp_display)); memcpy(display, query.display, sizeof(struct mod_hdcp_display));
mod_hdcp_remove_display(&hdcp_w->hdcp, aconnector->base.index, &hdcp_w->output); mod_hdcp_remove_display(&hdcp_w->hdcp, aconnector->base.index, &hdcp_w->output);
hdcp_w->link.adjust.hdcp2.force_type = MOD_HDCP_FORCE_TYPE_0; hdcp_w->link.adjust.hdcp2.force_type = MOD_HDCP_FORCE_TYPE_0;
if (enable_encryption) { if (enable_encryption) {
/* Explicitly set the saved SRM as sysfs call will be after we already enabled hdcp /* Explicitly set the saved SRM as sysfs call will be after
* (s3 resume case) * we already enabled hdcp (s3 resume case)
*/ */
if (hdcp_work->srm_size > 0) if (hdcp_work->srm_size > 0)
psp_set_srm(hdcp_work->hdcp.config.psp.handle, hdcp_work->srm, hdcp_work->srm_size, psp_set_srm(hdcp_work->hdcp.config.psp.handle, hdcp_work->srm,
hdcp_work->srm_size,
&hdcp_work->srm_version); &hdcp_work->srm_version);
display->adjust.disable = MOD_HDCP_DISPLAY_NOT_DISABLE; display->adjust.disable = MOD_HDCP_DISPLAY_NOT_DISABLE;
...@@ -219,7 +221,7 @@ void hdcp_update_display(struct hdcp_workqueue *hdcp_work, ...@@ -219,7 +221,7 @@ void hdcp_update_display(struct hdcp_workqueue *hdcp_work,
} }
static void hdcp_remove_display(struct hdcp_workqueue *hdcp_work, static void hdcp_remove_display(struct hdcp_workqueue *hdcp_work,
unsigned int link_index, unsigned int link_index,
struct amdgpu_dm_connector *aconnector) struct amdgpu_dm_connector *aconnector)
{ {
struct hdcp_workqueue *hdcp_w = &hdcp_work[link_index]; struct hdcp_workqueue *hdcp_w = &hdcp_work[link_index];
...@@ -238,7 +240,8 @@ static void hdcp_remove_display(struct hdcp_workqueue *hdcp_work, ...@@ -238,7 +240,8 @@ static void hdcp_remove_display(struct hdcp_workqueue *hdcp_work,
conn_state->content_protection = DRM_MODE_CONTENT_PROTECTION_DESIRED; conn_state->content_protection = DRM_MODE_CONTENT_PROTECTION_DESIRED;
DRM_DEBUG_DRIVER("[HDCP_DM] display %d, CP 2 -> 1, type %u, DPMS %u\n", DRM_DEBUG_DRIVER("[HDCP_DM] display %d, CP 2 -> 1, type %u, DPMS %u\n",
aconnector->base.index, conn_state->hdcp_content_type, aconnector->base.dpms); aconnector->base.index, conn_state->hdcp_content_type,
aconnector->base.dpms);
} }
mod_hdcp_remove_display(&hdcp_w->hdcp, aconnector->base.index, &hdcp_w->output); mod_hdcp_remove_display(&hdcp_w->hdcp, aconnector->base.index, &hdcp_w->output);
...@@ -246,6 +249,7 @@ static void hdcp_remove_display(struct hdcp_workqueue *hdcp_work, ...@@ -246,6 +249,7 @@ static void hdcp_remove_display(struct hdcp_workqueue *hdcp_work,
process_output(hdcp_w); process_output(hdcp_w);
mutex_unlock(&hdcp_w->mutex); mutex_unlock(&hdcp_w->mutex);
} }
void hdcp_reset_display(struct hdcp_workqueue *hdcp_work, unsigned int link_index) void hdcp_reset_display(struct hdcp_workqueue *hdcp_work, unsigned int link_index)
{ {
struct hdcp_workqueue *hdcp_w = &hdcp_work[link_index]; struct hdcp_workqueue *hdcp_w = &hdcp_work[link_index];
...@@ -274,15 +278,12 @@ void hdcp_handle_cpirq(struct hdcp_workqueue *hdcp_work, unsigned int link_index ...@@ -274,15 +278,12 @@ void hdcp_handle_cpirq(struct hdcp_workqueue *hdcp_work, unsigned int link_index
schedule_work(&hdcp_w->cpirq_work); schedule_work(&hdcp_w->cpirq_work);
} }
static void event_callback(struct work_struct *work) static void event_callback(struct work_struct *work)
{ {
struct hdcp_workqueue *hdcp_work; struct hdcp_workqueue *hdcp_work;
hdcp_work = container_of(to_delayed_work(work), struct hdcp_workqueue, hdcp_work = container_of(to_delayed_work(work), struct hdcp_workqueue,
callback_dwork); callback_dwork);
mutex_lock(&hdcp_work->mutex); mutex_lock(&hdcp_work->mutex);
...@@ -294,13 +295,12 @@ static void event_callback(struct work_struct *work) ...@@ -294,13 +295,12 @@ static void event_callback(struct work_struct *work)
process_output(hdcp_work); process_output(hdcp_work);
mutex_unlock(&hdcp_work->mutex); mutex_unlock(&hdcp_work->mutex);
} }
static void event_property_update(struct work_struct *work) static void event_property_update(struct work_struct *work)
{ {
struct hdcp_workqueue *hdcp_work = container_of(work, struct hdcp_workqueue, property_update_work); struct hdcp_workqueue *hdcp_work = container_of(work, struct hdcp_workqueue,
property_update_work);
struct amdgpu_dm_connector *aconnector = NULL; struct amdgpu_dm_connector *aconnector = NULL;
struct drm_device *dev; struct drm_device *dev;
long ret; long ret;
...@@ -334,11 +334,10 @@ static void event_property_update(struct work_struct *work) ...@@ -334,11 +334,10 @@ static void event_property_update(struct work_struct *work)
mutex_lock(&hdcp_work->mutex); mutex_lock(&hdcp_work->mutex);
if (conn_state->commit) { if (conn_state->commit) {
ret = wait_for_completion_interruptible_timeout( ret = wait_for_completion_interruptible_timeout(&conn_state->commit->hw_done,
&conn_state->commit->hw_done, 10 * HZ); 10 * HZ);
if (ret == 0) { if (ret == 0) {
DRM_ERROR( DRM_ERROR("HDCP state unknown! Setting it to DESIRED\n");
"HDCP state unknown! Setting it to DESIRED");
hdcp_work->encryption_status[conn_index] = hdcp_work->encryption_status[conn_index] =
MOD_HDCP_ENCRYPTION_STATUS_HDCP_OFF; MOD_HDCP_ENCRYPTION_STATUS_HDCP_OFF;
} }
...@@ -349,24 +348,20 @@ static void event_property_update(struct work_struct *work) ...@@ -349,24 +348,20 @@ static void event_property_update(struct work_struct *work)
DRM_MODE_HDCP_CONTENT_TYPE0 && DRM_MODE_HDCP_CONTENT_TYPE0 &&
hdcp_work->encryption_status[conn_index] <= hdcp_work->encryption_status[conn_index] <=
MOD_HDCP_ENCRYPTION_STATUS_HDCP2_TYPE0_ON) { MOD_HDCP_ENCRYPTION_STATUS_HDCP2_TYPE0_ON) {
DRM_DEBUG_DRIVER("[HDCP_DM] DRM_MODE_CONTENT_PROTECTION_ENABLED\n"); DRM_DEBUG_DRIVER("[HDCP_DM] DRM_MODE_CONTENT_PROTECTION_ENABLED\n");
drm_hdcp_update_content_protection( drm_hdcp_update_content_protection(connector,
connector, DRM_MODE_CONTENT_PROTECTION_ENABLED);
DRM_MODE_CONTENT_PROTECTION_ENABLED);
} else if (conn_state->hdcp_content_type == } else if (conn_state->hdcp_content_type ==
DRM_MODE_HDCP_CONTENT_TYPE1 && DRM_MODE_HDCP_CONTENT_TYPE1 &&
hdcp_work->encryption_status[conn_index] == hdcp_work->encryption_status[conn_index] ==
MOD_HDCP_ENCRYPTION_STATUS_HDCP2_TYPE1_ON) { MOD_HDCP_ENCRYPTION_STATUS_HDCP2_TYPE1_ON) {
drm_hdcp_update_content_protection( drm_hdcp_update_content_protection(connector,
connector, DRM_MODE_CONTENT_PROTECTION_ENABLED);
DRM_MODE_CONTENT_PROTECTION_ENABLED);
} }
} else { } else {
DRM_DEBUG_DRIVER("[HDCP_DM] DRM_MODE_CONTENT_PROTECTION_DESIRED\n"); DRM_DEBUG_DRIVER("[HDCP_DM] DRM_MODE_CONTENT_PROTECTION_DESIRED\n");
drm_hdcp_update_content_protection( drm_hdcp_update_content_protection(connector,
connector, DRM_MODE_CONTENT_PROTECTION_DESIRED); DRM_MODE_CONTENT_PROTECTION_DESIRED);
} }
mutex_unlock(&hdcp_work->mutex); mutex_unlock(&hdcp_work->mutex);
drm_modeset_unlock(&dev->mode_config.connection_mutex); drm_modeset_unlock(&dev->mode_config.connection_mutex);
...@@ -402,7 +397,7 @@ static void event_property_validate(struct work_struct *work) ...@@ -402,7 +397,7 @@ static void event_property_validate(struct work_struct *work)
&query); &query);
DRM_DEBUG_DRIVER("[HDCP_DM] disp %d, connector->CP %u, (query, work): (%d, %d)\n", DRM_DEBUG_DRIVER("[HDCP_DM] disp %d, connector->CP %u, (query, work): (%d, %d)\n",
aconnector->base.index, aconnector->base.index,
aconnector->base.state->content_protection, aconnector->base.state->content_protection,
query.encryption_status, query.encryption_status,
hdcp_work->encryption_status[conn_index]); hdcp_work->encryption_status[conn_index]);
...@@ -410,7 +405,8 @@ static void event_property_validate(struct work_struct *work) ...@@ -410,7 +405,8 @@ static void event_property_validate(struct work_struct *work)
if (query.encryption_status != if (query.encryption_status !=
hdcp_work->encryption_status[conn_index]) { hdcp_work->encryption_status[conn_index]) {
DRM_DEBUG_DRIVER("[HDCP_DM] encryption_status change from %x to %x\n", DRM_DEBUG_DRIVER("[HDCP_DM] encryption_status change from %x to %x\n",
hdcp_work->encryption_status[conn_index], query.encryption_status); hdcp_work->encryption_status[conn_index],
query.encryption_status);
hdcp_work->encryption_status[conn_index] = hdcp_work->encryption_status[conn_index] =
query.encryption_status; query.encryption_status;
...@@ -429,7 +425,7 @@ static void event_watchdog_timer(struct work_struct *work) ...@@ -429,7 +425,7 @@ static void event_watchdog_timer(struct work_struct *work)
struct hdcp_workqueue *hdcp_work; struct hdcp_workqueue *hdcp_work;
hdcp_work = container_of(to_delayed_work(work), hdcp_work = container_of(to_delayed_work(work),
struct hdcp_workqueue, struct hdcp_workqueue,
watchdog_timer_dwork); watchdog_timer_dwork);
mutex_lock(&hdcp_work->mutex); mutex_lock(&hdcp_work->mutex);
...@@ -443,7 +439,6 @@ static void event_watchdog_timer(struct work_struct *work) ...@@ -443,7 +439,6 @@ static void event_watchdog_timer(struct work_struct *work)
process_output(hdcp_work); process_output(hdcp_work);
mutex_unlock(&hdcp_work->mutex); mutex_unlock(&hdcp_work->mutex);
} }
static void event_cpirq(struct work_struct *work) static void event_cpirq(struct work_struct *work)
...@@ -459,10 +454,8 @@ static void event_cpirq(struct work_struct *work) ...@@ -459,10 +454,8 @@ static void event_cpirq(struct work_struct *work)
process_output(hdcp_work); process_output(hdcp_work);
mutex_unlock(&hdcp_work->mutex); mutex_unlock(&hdcp_work->mutex);
} }
void hdcp_destroy(struct kobject *kobj, struct hdcp_workqueue *hdcp_work) void hdcp_destroy(struct kobject *kobj, struct hdcp_workqueue *hdcp_work)
{ {
int i = 0; int i = 0;
...@@ -478,10 +471,8 @@ void hdcp_destroy(struct kobject *kobj, struct hdcp_workqueue *hdcp_work) ...@@ -478,10 +471,8 @@ void hdcp_destroy(struct kobject *kobj, struct hdcp_workqueue *hdcp_work)
kfree(hdcp_work); kfree(hdcp_work);
} }
static bool enable_assr(void *handle, struct dc_link *link) static bool enable_assr(void *handle, struct dc_link *link)
{ {
struct hdcp_workqueue *hdcp_work = handle; struct hdcp_workqueue *hdcp_work = handle;
struct mod_hdcp hdcp = hdcp_work->hdcp; struct mod_hdcp hdcp = hdcp_work->hdcp;
struct psp_context *psp = hdcp.config.psp.handle; struct psp_context *psp = hdcp.config.psp.handle;
...@@ -499,7 +490,8 @@ static bool enable_assr(void *handle, struct dc_link *link) ...@@ -499,7 +490,8 @@ static bool enable_assr(void *handle, struct dc_link *link)
memset(dtm_cmd, 0, sizeof(struct ta_dtm_shared_memory)); memset(dtm_cmd, 0, sizeof(struct ta_dtm_shared_memory));
dtm_cmd->cmd_id = TA_DTM_COMMAND__TOPOLOGY_ASSR_ENABLE; dtm_cmd->cmd_id = TA_DTM_COMMAND__TOPOLOGY_ASSR_ENABLE;
dtm_cmd->dtm_in_message.topology_assr_enable.display_topology_dig_be_index = link->link_enc_hw_inst; dtm_cmd->dtm_in_message.topology_assr_enable.display_topology_dig_be_index =
link->link_enc_hw_inst;
dtm_cmd->dtm_status = TA_DTM_STATUS__GENERIC_FAILURE; dtm_cmd->dtm_status = TA_DTM_STATUS__GENERIC_FAILURE;
psp_dtm_invoke(psp, dtm_cmd->cmd_id); psp_dtm_invoke(psp, dtm_cmd->cmd_id);
...@@ -541,7 +533,7 @@ static void update_config(void *handle, struct cp_psp_stream_config *config) ...@@ -541,7 +533,7 @@ static void update_config(void *handle, struct cp_psp_stream_config *config)
else if (aconnector->dc_em_sink) else if (aconnector->dc_em_sink)
sink = aconnector->dc_em_sink; sink = aconnector->dc_em_sink;
if (sink != NULL) if (sink)
link->mode = mod_hdcp_signal_type_to_operation_mode(sink->sink_signal); link->mode = mod_hdcp_signal_type_to_operation_mode(sink->sink_signal);
display->controller = CONTROLLER_ID_D0 + config->otg_inst; display->controller = CONTROLLER_ID_D0 + config->otg_inst;
...@@ -567,16 +559,20 @@ static void update_config(void *handle, struct cp_psp_stream_config *config) ...@@ -567,16 +559,20 @@ static void update_config(void *handle, struct cp_psp_stream_config *config)
conn_state = aconnector->base.state; conn_state = aconnector->base.state;
DRM_DEBUG_DRIVER("[HDCP_DM] display %d, CP %d, type %d\n", aconnector->base.index, DRM_DEBUG_DRIVER("[HDCP_DM] display %d, CP %d, type %d\n", aconnector->base.index,
(!!aconnector->base.state) ? aconnector->base.state->content_protection : -1, (!!aconnector->base.state) ?
(!!aconnector->base.state) ? aconnector->base.state->hdcp_content_type : -1); aconnector->base.state->content_protection : -1,
(!!aconnector->base.state) ?
aconnector->base.state->hdcp_content_type : -1);
if (conn_state) if (conn_state)
hdcp_update_display(hdcp_work, link_index, aconnector, hdcp_update_display(hdcp_work, link_index, aconnector,
conn_state->hdcp_content_type, false); conn_state->hdcp_content_type, false);
} }
/**
/* NOTE: From the usermodes prospective you only need to call write *ONCE*, the kernel * DOC: Add sysfs interface for set/get srm
*
* NOTE: From the usermodes prospective you only need to call write *ONCE*, the kernel
* will automatically call once or twice depending on the size * will automatically call once or twice depending on the size
* *
* call: "cat file > /sys/class/drm/card0/device/hdcp_srm" from usermode no matter what the size is * call: "cat file > /sys/class/drm/card0/device/hdcp_srm" from usermode no matter what the size is
...@@ -587,23 +583,23 @@ static void update_config(void *handle, struct cp_psp_stream_config *config) ...@@ -587,23 +583,23 @@ static void update_config(void *handle, struct cp_psp_stream_config *config)
* sysfs interface doesn't tell us the size we will get so we are sending partial SRMs to psp and on * sysfs interface doesn't tell us the size we will get so we are sending partial SRMs to psp and on
* the last call we will send the full SRM. PSP will fail on every call before the last. * the last call we will send the full SRM. PSP will fail on every call before the last.
* *
* This means we don't know if the SRM is good until the last call. And because of this limitation we * This means we don't know if the SRM is good until the last call. And because of this
* cannot throw errors early as it will stop the kernel from writing to sysfs * limitation we cannot throw errors early as it will stop the kernel from writing to sysfs
* *
* Example 1: * Example 1:
* Good SRM size = 5096 * Good SRM size = 5096
* first call to write 4096 -> PSP fails * first call to write 4096 -> PSP fails
* Second call to write 1000 -> PSP Pass -> SRM is set * Second call to write 1000 -> PSP Pass -> SRM is set
* *
* Example 2: * Example 2:
* Bad SRM size = 4096 * Bad SRM size = 4096
* first call to write 4096 -> PSP fails (This is the same as above, but we don't know if this * first call to write 4096 -> PSP fails (This is the same as above, but we don't know if this
* is the last call) * is the last call)
* *
* Solution?: * Solution?:
* 1: Parse the SRM? -> It is signed so we don't know the EOF * 1: Parse the SRM? -> It is signed so we don't know the EOF
* 2: We can have another sysfs that passes the size before calling set. -> simpler solution * 2: We can have another sysfs that passes the size before calling set. -> simpler solution
* below * below
* *
* Easy Solution: * Easy Solution:
* Always call get after Set to verify if set was successful. * Always call get after Set to verify if set was successful.
...@@ -612,20 +608,21 @@ static void update_config(void *handle, struct cp_psp_stream_config *config) ...@@ -612,20 +608,21 @@ static void update_config(void *handle, struct cp_psp_stream_config *config)
* +----------------------+ * +----------------------+
* PSP will only update its srm if its older than the one we are trying to load. * PSP will only update its srm if its older than the one we are trying to load.
* Always do set first than get. * Always do set first than get.
* -if we try to "1. SET" a older version PSP will reject it and we can "2. GET" the newer * -if we try to "1. SET" a older version PSP will reject it and we can "2. GET" the newer
* version and save it * version and save it
* *
* -if we try to "1. SET" a newer version PSP will accept it and we can "2. GET" the * -if we try to "1. SET" a newer version PSP will accept it and we can "2. GET" the
* same(newer) version back and save it * same(newer) version back and save it
* *
* -if we try to "1. SET" a newer version and PSP rejects it. That means the format is * -if we try to "1. SET" a newer version and PSP rejects it. That means the format is
* incorrect/corrupted and we should correct our SRM by getting it from PSP * incorrect/corrupted and we should correct our SRM by getting it from PSP
*/ */
static ssize_t srm_data_write(struct file *filp, struct kobject *kobj, struct bin_attribute *bin_attr, char *buffer, static ssize_t srm_data_write(struct file *filp, struct kobject *kobj,
struct bin_attribute *bin_attr, char *buffer,
loff_t pos, size_t count) loff_t pos, size_t count)
{ {
struct hdcp_workqueue *work; struct hdcp_workqueue *work;
uint32_t srm_version = 0; u32 srm_version = 0;
work = container_of(bin_attr, struct hdcp_workqueue, attr); work = container_of(bin_attr, struct hdcp_workqueue, attr);
link_lock(work, true); link_lock(work, true);
...@@ -639,19 +636,19 @@ static ssize_t srm_data_write(struct file *filp, struct kobject *kobj, struct bi ...@@ -639,19 +636,19 @@ static ssize_t srm_data_write(struct file *filp, struct kobject *kobj, struct bi
work->srm_version = srm_version; work->srm_version = srm_version;
} }
link_lock(work, false); link_lock(work, false);
return count; return count;
} }
static ssize_t srm_data_read(struct file *filp, struct kobject *kobj, struct bin_attribute *bin_attr, char *buffer, static ssize_t srm_data_read(struct file *filp, struct kobject *kobj,
struct bin_attribute *bin_attr, char *buffer,
loff_t pos, size_t count) loff_t pos, size_t count)
{ {
struct hdcp_workqueue *work; struct hdcp_workqueue *work;
uint8_t *srm = NULL; u8 *srm = NULL;
uint32_t srm_version; u32 srm_version;
uint32_t srm_size; u32 srm_size;
size_t ret = count; size_t ret = count;
work = container_of(bin_attr, struct hdcp_workqueue, attr); work = container_of(bin_attr, struct hdcp_workqueue, attr);
...@@ -684,12 +681,12 @@ static ssize_t srm_data_read(struct file *filp, struct kobject *kobj, struct bin ...@@ -684,12 +681,12 @@ static ssize_t srm_data_read(struct file *filp, struct kobject *kobj, struct bin
/* From the hdcp spec (5.Renewability) SRM needs to be stored in a non-volatile memory. /* From the hdcp spec (5.Renewability) SRM needs to be stored in a non-volatile memory.
* *
* For example, * For example,
* if Application "A" sets the SRM (ver 2) and we reboot/suspend and later when Application "B" * if Application "A" sets the SRM (ver 2) and we reboot/suspend and later when Application "B"
* needs to use HDCP, the version in PSP should be SRM(ver 2). So SRM should be persistent * needs to use HDCP, the version in PSP should be SRM(ver 2). So SRM should be persistent
* across boot/reboots/suspend/resume/shutdown * across boot/reboots/suspend/resume/shutdown
* *
* Currently when the system goes down (suspend/shutdown) the SRM is cleared from PSP. For HDCP we need * Currently when the system goes down (suspend/shutdown) the SRM is cleared from PSP. For HDCP
* to make the SRM persistent. * we need to make the SRM persistent.
* *
* -PSP owns the checking of SRM but doesn't have the ability to store it in a non-volatile memory. * -PSP owns the checking of SRM but doesn't have the ability to store it in a non-volatile memory.
* -The kernel cannot write to the file systems. * -The kernel cannot write to the file systems.
...@@ -699,8 +696,8 @@ static ssize_t srm_data_read(struct file *filp, struct kobject *kobj, struct bin ...@@ -699,8 +696,8 @@ static ssize_t srm_data_read(struct file *filp, struct kobject *kobj, struct bin
* *
* Usermode can read/write to/from PSP using the sysfs interface * Usermode can read/write to/from PSP using the sysfs interface
* For example: * For example:
* to save SRM from PSP to storage : cat /sys/class/drm/card0/device/hdcp_srm > srmfile * to save SRM from PSP to storage : cat /sys/class/drm/card0/device/hdcp_srm > srmfile
* to load from storage to PSP: cat srmfile > /sys/class/drm/card0/device/hdcp_srm * to load from storage to PSP: cat srmfile > /sys/class/drm/card0/device/hdcp_srm
*/ */
static const struct bin_attribute data_attr = { static const struct bin_attribute data_attr = {
.attr = {.name = "hdcp_srm", .mode = 0664}, .attr = {.name = "hdcp_srm", .mode = 0664},
...@@ -709,10 +706,9 @@ static const struct bin_attribute data_attr = { ...@@ -709,10 +706,9 @@ static const struct bin_attribute data_attr = {
.read = srm_data_read, .read = srm_data_read,
}; };
struct hdcp_workqueue *hdcp_create_workqueue(struct amdgpu_device *adev,
struct hdcp_workqueue *hdcp_create_workqueue(struct amdgpu_device *adev, struct cp_psp *cp_psp, struct dc *dc) struct cp_psp *cp_psp, struct dc *dc)
{ {
int max_caps = dc->caps.max_links; int max_caps = dc->caps.max_links;
struct hdcp_workqueue *hdcp_work; struct hdcp_workqueue *hdcp_work;
int i = 0; int i = 0;
...@@ -721,14 +717,16 @@ struct hdcp_workqueue *hdcp_create_workqueue(struct amdgpu_device *adev, struct ...@@ -721,14 +717,16 @@ struct hdcp_workqueue *hdcp_create_workqueue(struct amdgpu_device *adev, struct
if (ZERO_OR_NULL_PTR(hdcp_work)) if (ZERO_OR_NULL_PTR(hdcp_work))
return NULL; return NULL;
hdcp_work->srm = kcalloc(PSP_HDCP_SRM_FIRST_GEN_MAX_SIZE, sizeof(*hdcp_work->srm), GFP_KERNEL); hdcp_work->srm = kcalloc(PSP_HDCP_SRM_FIRST_GEN_MAX_SIZE,
sizeof(*hdcp_work->srm), GFP_KERNEL);
if (hdcp_work->srm == NULL) if (!hdcp_work->srm)
goto fail_alloc_context; goto fail_alloc_context;
hdcp_work->srm_temp = kcalloc(PSP_HDCP_SRM_FIRST_GEN_MAX_SIZE, sizeof(*hdcp_work->srm_temp), GFP_KERNEL); hdcp_work->srm_temp = kcalloc(PSP_HDCP_SRM_FIRST_GEN_MAX_SIZE,
sizeof(*hdcp_work->srm_temp), GFP_KERNEL);
if (hdcp_work->srm_temp == NULL) if (!hdcp_work->srm_temp)
goto fail_alloc_context; goto fail_alloc_context;
hdcp_work->max_link = max_caps; hdcp_work->max_link = max_caps;
...@@ -781,10 +779,5 @@ struct hdcp_workqueue *hdcp_create_workqueue(struct amdgpu_device *adev, struct ...@@ -781,10 +779,5 @@ struct hdcp_workqueue *hdcp_create_workqueue(struct amdgpu_device *adev, struct
kfree(hdcp_work); kfree(hdcp_work);
return NULL; return NULL;
} }
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