1515 lines
38 KiB
C
1515 lines
38 KiB
C
/* Copyright (c) 2015, The Linux Foundation. All rights reserved.
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License version 2 and
|
|
* only version 2 as published by the Free Software Foundation.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*/
|
|
|
|
#define pr_fmt(fmt) "%s: " fmt, __func__
|
|
|
|
#include <linux/kernel.h>
|
|
#include <linux/slab.h>
|
|
#include <linux/module.h>
|
|
#include <linux/fs.h>
|
|
#include <linux/cdev.h>
|
|
#include <linux/sched.h>
|
|
#include <linux/list.h>
|
|
#include <linux/mutex.h>
|
|
#include <linux/io.h>
|
|
#include <linux/ion.h>
|
|
#include <linux/types.h>
|
|
#include <linux/device.h>
|
|
#include <linux/sched.h>
|
|
#include <linux/delay.h>
|
|
#include <linux/completion.h>
|
|
#include <linux/errno.h>
|
|
#include <linux/hdcp_qseecom.h>
|
|
#include <linux/kthread.h>
|
|
|
|
#include "qseecom_kernel.h"
|
|
|
|
#define TZAPP_NAME "hdcp2p2"
|
|
#define HDCP1_APP_NAME "hdcp1"
|
|
#define QSEECOM_SBUFF_SIZE 0x1000
|
|
|
|
#define MAX_TX_MESSAGE_SIZE 129
|
|
#define MAX_RX_MESSAGE_SIZE 534
|
|
#define MAX_TOPOLOGY_ELEMS 32
|
|
#define HDCP1_AKSV_SIZE 8
|
|
|
|
/* parameters related to LC_Init message */
|
|
#define MESSAGE_ID_SIZE 1
|
|
#define LC_INIT_MESSAGE_SIZE (MESSAGE_ID_SIZE+BITS_64_IN_BYTES)
|
|
|
|
/* parameters related to SKE_Send_EKS message */
|
|
#define SKE_SEND_EKS_MESSAGE_SIZE \
|
|
(MESSAGE_ID_SIZE+BITS_128_IN_BYTES+BITS_64_IN_BYTES)
|
|
|
|
/* all message IDs */
|
|
#define AKE_SEND_CERT_MESSAGE_ID 3
|
|
#define AKE_NO_STORED_KM_MESSAGE_ID 4
|
|
#define AKE_STORED_KM_MESSAGE_ID 5
|
|
#define AKE_SEND_H_PRIME_MESSAGE_ID 7
|
|
#define AKE_SEND_PAIRING_INFO_MESSAGE_ID 8
|
|
#define LC_INIT_MESSAGE_ID 9
|
|
#define LC_SEND_L_PRIME_MESSAGE_ID 10
|
|
#define SKE_SEND_EKS_MESSAGE_ID 11
|
|
#define REPEATER_AUTH_SEND_RECEIVERID_LIST_MESSAGE_ID 12
|
|
#define REPEATER_AUTH_SEND_ACK_MESSAGE_ID 15
|
|
#define REPEATER_AUTH_STREAM_MANAGE_MESSAGE_ID 16
|
|
#define REPEATER_AUTH_STREAM_READY_MESSAGE_ID 17
|
|
#define HDCP1_SET_KEY_MESSAGE_ID 202
|
|
#define HDCP1_SET_ENC_MESSAGE_ID 205
|
|
|
|
#define BITS_8_IN_BYTES 1
|
|
#define BITS_16_IN_BYTES 2
|
|
#define BITS_24_IN_BYTES 3
|
|
#define BITS_32_IN_BYTES 4
|
|
#define BITS_40_IN_BYTES 5
|
|
#define BITS_64_IN_BYTES 8
|
|
#define BITS_128_IN_BYTES 16
|
|
#define BITS_160_IN_BYTES 20
|
|
#define BITS_256_IN_BYTES 32
|
|
#define BITS_1024_IN_BYTES 128
|
|
#define BITS_3072_IN_BYTES 384
|
|
#define TXCAPS_SIZE 3
|
|
#define RXCAPS_SIZE 3
|
|
#define RXINFO_SIZE 2
|
|
#define SEQ_NUM_V_SIZE 3
|
|
|
|
#define RCVR_ID_SIZE BITS_40_IN_BYTES
|
|
#define MAX_RCVR_IDS_ALLOWED_IN_LIST 31
|
|
#define MAX_RCVR_ID_LIST_SIZE \
|
|
(RCVR_ID_SIZE*MAX_RCVR_IDS_ALLOWED_IN_LIST)
|
|
/*
|
|
* minimum wait as per standard is 200 ms. keep it 300 ms
|
|
* to be on safe side.
|
|
*/
|
|
#define SLEEP_SET_HW_KEY_MS 300
|
|
|
|
|
|
#define QSEECOM_ALIGN_SIZE 0x40
|
|
#define QSEECOM_ALIGN_MASK (QSEECOM_ALIGN_SIZE - 1)
|
|
#define QSEECOM_ALIGN(x)\
|
|
((x + QSEECOM_ALIGN_SIZE) & (~QSEECOM_ALIGN_MASK))
|
|
|
|
/* hdcp command status */
|
|
#define HDCP_SUCCESS 0
|
|
|
|
/* flags set by tz in response message */
|
|
#define HDCP_TXMTR_SUBSTATE_INIT 0
|
|
#define HDCP_TXMTR_SUBSTATE_WAITING_FOR_RECIEVERID_LIST 1
|
|
#define HDCP_TXMTR_SUBSTATE_PROCESSED_RECIEVERID_LIST 2
|
|
#define HDCP_TXMTR_SUBSTATE_WAITING_FOR_STREAM_READY_MESSAGE 3
|
|
#define HDCP_TXMTR_SUBSTATE_REPEATER_AUTH_COMPLETE 4
|
|
|
|
|
|
#define HDCP_TXMTR_SERVICE_ID 0x0001000
|
|
#define SERVICE_TXMTR_CREATE_CMD(x) (HDCP_TXMTR_SERVICE_ID | x)
|
|
|
|
#define HDCP_TXMTR_INIT SERVICE_TXMTR_CREATE_CMD(1)
|
|
#define HDCP_TXMTR_DEINIT SERVICE_TXMTR_CREATE_CMD(2)
|
|
#define HDCP_TXMTR_PROCESS_RECEIVED_MESSAGE SERVICE_TXMTR_CREATE_CMD(3)
|
|
#define HDCP_TXMTR_SEND_MESSAGE_TIMEOUT SERVICE_TXMTR_CREATE_CMD(4)
|
|
#define HDCP_TXMTR_SET_HW_KEY SERVICE_TXMTR_CREATE_CMD(5)
|
|
#define HDCP_TXMTR_QUERY_STREAM_TYPE SERVICE_TXMTR_CREATE_CMD(6)
|
|
#define HDCP_TXMTR_GET_KSXORLC128_AND_RIV SERVICE_TXMTR_CREATE_CMD(7)
|
|
#define HDCP_TXMTR_PROVISION_KEY SERVICE_TXMTR_CREATE_CMD(8)
|
|
#define HDCP_TXMTR_GET_TOPOLOGY_INFO SERVICE_TXMTR_CREATE_CMD(9)
|
|
#define HDCP_TXMTR_UPDATE_SRM SERVICE_TXMTR_CREATE_CMD(10)
|
|
/*This API calls the library init function */
|
|
#define HDCP_LIB_INIT SERVICE_TXMTR_CREATE_CMD(11)
|
|
/*This API calls the library deinit function */
|
|
#define HDCP_LIB_DEINIT SERVICE_TXMTR_CREATE_CMD(12)
|
|
|
|
#define HDCP_LIB_EXECUTE(x) {\
|
|
if (handle->tethered)\
|
|
hdcp_lib_##x(handle);\
|
|
else\
|
|
queue_kthread_work(&handle->worker, &handle->wk_##x);\
|
|
}
|
|
|
|
enum hdcp_state {
|
|
HDCP_STATE_INIT = 0x00,
|
|
HDCP_STATE_APP_LOADED = 0x01,
|
|
HDCP_STATE_TXMTR_INIT = 0x02,
|
|
HDCP_STATE_AUTHENTICATED = 0x04,
|
|
HDCP_STATE_ERROR = 0x08
|
|
};
|
|
|
|
enum hdcp_element {
|
|
HDCP_TYPE_UNKNOWN,
|
|
HDCP_TYPE_RECEIVER,
|
|
HDCP_TYPE_REPEATER,
|
|
};
|
|
|
|
enum hdcp_version {
|
|
HDCP_VERSION_UNKNOWN,
|
|
HDCP_VERSION_2_2,
|
|
HDCP_VERSION_1_4
|
|
};
|
|
|
|
struct receiver_info {
|
|
unsigned char rcvrInfo[RCVR_ID_SIZE];
|
|
enum hdcp_element elem_type;
|
|
enum hdcp_version hdcp_version;
|
|
};
|
|
|
|
struct topology_info {
|
|
unsigned int nNumRcvrs;
|
|
struct receiver_info rcvinfo[MAX_TOPOLOGY_ELEMS];
|
|
};
|
|
|
|
struct __attribute__ ((__packed__)) hdcp1_key_set_req {
|
|
uint32_t commandid;
|
|
};
|
|
|
|
struct __attribute__ ((__packed__)) hdcp1_key_set_rsp {
|
|
uint32_t commandid;
|
|
uint32_t ret;
|
|
uint8_t ksv[HDCP1_AKSV_SIZE];
|
|
};
|
|
|
|
struct __attribute__ ((__packed__)) hdcp_init_req {
|
|
uint32_t commandid;
|
|
};
|
|
|
|
struct __attribute__ ((__packed__)) hdcp_init_rsp {
|
|
uint32_t status;
|
|
uint32_t commandid;
|
|
uint32_t ctxhandle;
|
|
uint32_t timeout;
|
|
uint32_t msglen;
|
|
uint8_t message[MAX_TX_MESSAGE_SIZE];
|
|
};
|
|
|
|
|
|
struct __attribute__ ((__packed__)) hdcp_deinit_req {
|
|
uint32_t commandid;
|
|
uint32_t ctxhandle;
|
|
};
|
|
|
|
|
|
struct __attribute__ ((__packed__)) hdcp_deinit_rsp {
|
|
uint32_t status;
|
|
uint32_t commandid;
|
|
};
|
|
|
|
|
|
struct __attribute__ ((__packed__)) hdcp_rcvd_msg_req {
|
|
uint32_t commandid;
|
|
uint32_t ctxhandle;
|
|
uint32_t msglen;
|
|
uint8_t msg[MAX_RX_MESSAGE_SIZE];
|
|
};
|
|
|
|
|
|
struct __attribute__ ((__packed__)) hdcp_rcvd_msg_rsp {
|
|
uint32_t status;
|
|
uint32_t commandid;
|
|
uint32_t state;
|
|
uint32_t timeout;
|
|
uint32_t flag;
|
|
uint32_t msglen;
|
|
uint8_t msg[MAX_TX_MESSAGE_SIZE];
|
|
};
|
|
|
|
|
|
struct __attribute__ ((__packed__)) hdcp_set_hw_key_req {
|
|
uint32_t commandid;
|
|
uint32_t ctxhandle;
|
|
};
|
|
|
|
|
|
struct __attribute__ ((__packed__)) hdcp_set_hw_key_rsp {
|
|
uint32_t status;
|
|
uint32_t commandid;
|
|
};
|
|
|
|
|
|
struct __attribute__ ((__packed__)) hdcp_send_timeout_req {
|
|
uint32_t commandid;
|
|
uint32_t ctxhandle;
|
|
};
|
|
|
|
|
|
struct __attribute__ ((__packed__)) hdcp_send_timeout_rsp {
|
|
uint32_t status;
|
|
uint32_t commandid;
|
|
uint32_t timeout;
|
|
uint32_t msglen;
|
|
uint8_t message[MAX_TX_MESSAGE_SIZE];
|
|
};
|
|
|
|
struct __attribute__ ((__packed__)) hdcp_query_stream_type_req {
|
|
uint32_t commandid;
|
|
uint32_t ctxhandle;
|
|
};
|
|
|
|
struct __attribute__ ((__packed__)) hdcp_query_stream_type_rsp {
|
|
uint32_t status;
|
|
uint32_t commandid;
|
|
uint32_t timeout;
|
|
uint32_t msglen;
|
|
uint8_t msg[MAX_TX_MESSAGE_SIZE];
|
|
};
|
|
|
|
struct __attribute__ ((__packed__)) hdcp_set_stream_type_req {
|
|
uint32_t commandid;
|
|
uint32_t ctxhandle;
|
|
uint8_t streamtype;
|
|
};
|
|
|
|
struct __attribute__ ((__packed__)) hdcp_set_stream_type_rsp {
|
|
uint32_t status;
|
|
uint32_t commandid;
|
|
uint32_t timeout;
|
|
uint32_t msglen;
|
|
uint8_t message[MAX_TX_MESSAGE_SIZE];
|
|
};
|
|
|
|
|
|
struct __attribute__ ((__packed__)) hdcp_update_srm_req {
|
|
uint32_t commandid;
|
|
uint32_t ctxhandle;
|
|
uint32_t srmoffset;
|
|
uint32_t srmlength;
|
|
};
|
|
|
|
|
|
struct __attribute__ ((__packed__)) hdcp_update_srm_rsp {
|
|
uint32_t status;
|
|
uint32_t commandid;
|
|
};
|
|
|
|
|
|
struct __attribute__ ((__packed__)) hdcp_get_topology_req {
|
|
uint32_t commandid;
|
|
uint32_t ctxhandle;
|
|
};
|
|
|
|
|
|
struct __attribute__ ((__packed__)) hdcp_get_topology_rsp {
|
|
uint32_t status;
|
|
uint32_t commandid;
|
|
struct topology_info topologyinfo;
|
|
};
|
|
|
|
|
|
struct __attribute__ ((__packed__)) rxvr_info_struct {
|
|
uint8_t rcvrCert[522];
|
|
uint8_t rrx[BITS_64_IN_BYTES];
|
|
uint8_t rxcaps[RXCAPS_SIZE];
|
|
bool repeater;
|
|
};
|
|
|
|
|
|
struct __attribute__ ((__packed__)) repeater_info_struct {
|
|
uint8_t RxInfo[RXINFO_SIZE];
|
|
uint8_t seq_num_V[SEQ_NUM_V_SIZE];
|
|
bool seq_num_V_Rollover_flag;
|
|
uint8_t ReceiverIDList[MAX_RCVR_ID_LIST_SIZE];
|
|
uint32_t ReceiverIDListLen;
|
|
};
|
|
|
|
struct __attribute__ ((__packed__)) hdcp1_set_enc_req {
|
|
uint32_t commandid;
|
|
uint32_t enable;
|
|
};
|
|
|
|
struct __attribute__ ((__packed__)) hdcp1_set_enc_rsp {
|
|
uint32_t commandid;
|
|
uint32_t ret;
|
|
};
|
|
|
|
/*
|
|
* struct hdcp_lib_handle - handle for hdcp client
|
|
* @qseecom_handle - for sending commands to qseecom
|
|
* @listener_buf - buffer containing message shared with the client
|
|
* @msglen - size message in the buffer
|
|
* @tz_ctxhandle - context handle shared with tz
|
|
* @hdcp_timeout - timeout in msecs shared for hdcp messages
|
|
* @client_ctx - client context maintained by hdmi
|
|
* @client_ops - handle to call APIs exposed by hdcp client
|
|
* @timeout_lock - this lock protects hdcp_timeout field
|
|
* @msg_lock - this lock protects the message buffer
|
|
*/
|
|
struct hdcp_lib_handle {
|
|
unsigned char *listener_buf;
|
|
uint32_t msglen;
|
|
uint32_t tz_ctxhandle;
|
|
uint32_t hdcp_timeout;
|
|
uint32_t timeout_left;
|
|
bool no_stored_km_flag;
|
|
bool feature_supported;
|
|
void *client_ctx;
|
|
struct hdcp_client_ops *client_ops;
|
|
struct mutex msg_lock;
|
|
struct mutex wakeup_mutex;
|
|
enum hdcp_state hdcp_state;
|
|
enum hdcp_lib_wakeup_cmd wakeup_cmd;
|
|
bool repeater_flag;
|
|
bool tethered;
|
|
struct qseecom_handle *qseecom_handle;
|
|
int last_msg_sent;
|
|
char *last_msg_recvd_buf;
|
|
uint32_t last_msg_recvd_len;
|
|
atomic_t hdcp_off;
|
|
|
|
struct task_struct *thread;
|
|
struct completion topo_wait;
|
|
|
|
struct kthread_worker worker;
|
|
struct kthread_work wk_init;
|
|
struct kthread_work wk_msg_sent;
|
|
struct kthread_work wk_msg_recvd;
|
|
struct kthread_work wk_timeout;
|
|
struct kthread_work wk_clean;
|
|
struct kthread_work wk_topology;
|
|
struct kthread_work wk_stream;
|
|
};
|
|
|
|
struct hdcp_lib_message_map {
|
|
int msg_id;
|
|
const char *msg_name;
|
|
};
|
|
|
|
static void hdcp_lib_clean(struct hdcp_lib_handle *handle);
|
|
static void hdcp_lib_init(struct hdcp_lib_handle *handle);
|
|
static void hdcp_lib_msg_sent(struct hdcp_lib_handle *handle);
|
|
static void hdcp_lib_msg_recvd(struct hdcp_lib_handle *handle);
|
|
static void hdcp_lib_timeout(struct hdcp_lib_handle *handle);
|
|
static void hdcp_lib_stream(struct hdcp_lib_handle *handle);
|
|
|
|
static struct qseecom_handle *hdcp1_handle;
|
|
static bool hdcp1_supported = true;
|
|
|
|
static const char *hdcp_lib_message_name(int msg_id)
|
|
{
|
|
/*
|
|
* Message ID map. The first number indicates the message number
|
|
* assigned to the message by the HDCP 2.2 spec. This is also the first
|
|
* byte of every HDCP 2.2 authentication protocol message.
|
|
*/
|
|
static struct hdcp_lib_message_map hdcp_lib_msg_map[] = {
|
|
{2, "AKE_INIT"},
|
|
{3, "AKE_SEND_CERT"},
|
|
{4, "AKE_NO_STORED_KM"},
|
|
{5, "AKE_STORED_KM"},
|
|
{7, "AKE_SEND_H_PRIME"},
|
|
{8, "AKE_SEND_PAIRING_INFO"},
|
|
{9, "LC_INIT"},
|
|
{10, "LC_SEND_L_PRIME"},
|
|
{11, "SKE_SEND_EKS"},
|
|
{12, "REPEATER_AUTH_SEND_RECEIVERID_LIST"},
|
|
{15, "REPEATER_AUTH_SEND_ACK"},
|
|
{16, "REPEATER_AUTH_STREAM_MANAGE"},
|
|
{17, "REPEATER_AUTH_STREAM_READY"},
|
|
};
|
|
int i;
|
|
|
|
for (i = 0; i < ARRAY_SIZE(hdcp_lib_msg_map); i++) {
|
|
if (msg_id == hdcp_lib_msg_map[i].msg_id)
|
|
return hdcp_lib_msg_map[i].msg_name;
|
|
}
|
|
return "UNKNOWN";
|
|
}
|
|
|
|
static inline void hdcp_lib_wakeup_client(struct hdcp_lib_handle *handle,
|
|
struct hdmi_hdcp_wakeup_data *data)
|
|
{
|
|
int rc = 0;
|
|
|
|
if (handle && handle->client_ops && handle->client_ops->wakeup &&
|
|
data && (data->cmd != HDMI_HDCP_WKUP_CMD_INVALID)) {
|
|
rc = handle->client_ops->wakeup(data);
|
|
if (rc)
|
|
pr_err("error sending %s to client\n",
|
|
hdmi_hdcp_cmd_to_str(data->cmd));
|
|
}
|
|
}
|
|
|
|
static inline void hdcp_lib_send_message(struct hdcp_lib_handle *handle)
|
|
{
|
|
struct hdmi_hdcp_wakeup_data cdata = {HDMI_HDCP_WKUP_CMD_SEND_MESSAGE};
|
|
|
|
cdata.context = handle->client_ctx;
|
|
cdata.send_msg_buf = handle->listener_buf;
|
|
cdata.send_msg_len = handle->msglen;
|
|
cdata.timeout = handle->hdcp_timeout;
|
|
|
|
hdcp_lib_wakeup_client(handle, &cdata);
|
|
}
|
|
|
|
static int hdcp_lib_enable_encryption(struct hdcp_lib_handle *handle)
|
|
{
|
|
int rc = 0;
|
|
struct hdcp_set_hw_key_req *req_buf;
|
|
struct hdcp_set_hw_key_rsp *rsp_buf;
|
|
|
|
/*
|
|
* wait at least 200ms before enabling encryption
|
|
* as per hdcp2p2 sepcifications.
|
|
*/
|
|
msleep(SLEEP_SET_HW_KEY_MS);
|
|
|
|
req_buf = (struct hdcp_set_hw_key_req *)(
|
|
handle->qseecom_handle->sbuf);
|
|
req_buf->commandid = HDCP_TXMTR_SET_HW_KEY;
|
|
req_buf->ctxhandle = handle->tz_ctxhandle;
|
|
|
|
rsp_buf = (struct hdcp_set_hw_key_rsp *)(
|
|
handle->qseecom_handle->sbuf + QSEECOM_ALIGN(
|
|
sizeof(struct hdcp_set_hw_key_req)));
|
|
|
|
rc = qseecom_send_command(handle->qseecom_handle, req_buf,
|
|
QSEECOM_ALIGN(sizeof(struct hdcp_set_hw_key_req)),
|
|
rsp_buf, QSEECOM_ALIGN(sizeof(
|
|
struct hdcp_set_hw_key_rsp)));
|
|
|
|
if ((rc < 0) || (rsp_buf->status < 0)) {
|
|
pr_err("qseecom cmd failed with err = %d status = %d\n",
|
|
rc, rsp_buf->status);
|
|
rc = -EINVAL;
|
|
goto error;
|
|
}
|
|
|
|
/* reached an authenticated state */
|
|
handle->hdcp_state |= HDCP_STATE_AUTHENTICATED;
|
|
|
|
pr_debug("success\n");
|
|
return 0;
|
|
error:
|
|
if (!atomic_read(&handle->hdcp_off))
|
|
HDCP_LIB_EXECUTE(clean);
|
|
|
|
return rc;
|
|
}
|
|
|
|
static int hdcp_lib_library_load(struct hdcp_lib_handle *handle)
|
|
{
|
|
int rc = 0;
|
|
struct hdcp_init_req *req_buf;
|
|
struct hdcp_init_rsp *rsp_buf;
|
|
|
|
if (!handle) {
|
|
pr_err("invalid input\n");
|
|
goto exit;
|
|
}
|
|
|
|
if (handle->hdcp_state & HDCP_STATE_APP_LOADED) {
|
|
pr_err("library already loaded\n");
|
|
return rc;
|
|
}
|
|
|
|
/*
|
|
* allocating resource for qseecom handle
|
|
* the app is not loaded here
|
|
*/
|
|
rc = qseecom_start_app(&(handle->qseecom_handle),
|
|
TZAPP_NAME, QSEECOM_SBUFF_SIZE);
|
|
if (rc) {
|
|
pr_err("qseecom_start_app failed %d\n", rc);
|
|
goto exit;
|
|
}
|
|
|
|
pr_debug("qseecom_start_app success\n");
|
|
|
|
/* now load the app by sending hdcp_lib_init */
|
|
req_buf = (struct hdcp_init_req *)handle->qseecom_handle->sbuf;
|
|
req_buf->commandid = HDCP_LIB_INIT;
|
|
rsp_buf = (struct hdcp_init_rsp *)(handle->qseecom_handle->
|
|
sbuf + QSEECOM_ALIGN(sizeof(struct hdcp_init_req)));
|
|
|
|
rc = qseecom_send_command(handle->qseecom_handle,
|
|
req_buf, QSEECOM_ALIGN(sizeof(struct hdcp_init_req)),
|
|
rsp_buf, QSEECOM_ALIGN(sizeof(struct hdcp_init_rsp)));
|
|
|
|
if (rc < 0) {
|
|
pr_err("qseecom cmd failed err = %d\n", rc);
|
|
goto exit;
|
|
}
|
|
|
|
pr_debug("success\n");
|
|
|
|
handle->hdcp_state |= HDCP_STATE_APP_LOADED;
|
|
exit:
|
|
return rc;
|
|
}
|
|
|
|
static int hdcp_lib_library_unload(struct hdcp_lib_handle *handle)
|
|
{
|
|
int rc = 0;
|
|
struct hdcp_deinit_req *req_buf;
|
|
struct hdcp_deinit_rsp *rsp_buf;
|
|
|
|
if (!handle) {
|
|
pr_err("invalid input\n");
|
|
goto exit;
|
|
}
|
|
|
|
if (!(handle->hdcp_state & HDCP_STATE_APP_LOADED)) {
|
|
pr_err("library not loaded\n");
|
|
return rc;
|
|
}
|
|
|
|
/* unloading app by sending hdcp_lib_deinit cmd */
|
|
req_buf = (struct hdcp_deinit_req *)handle->
|
|
qseecom_handle->sbuf;
|
|
req_buf->commandid = HDCP_LIB_DEINIT;
|
|
rsp_buf = (struct hdcp_deinit_rsp *)(handle->qseecom_handle->
|
|
sbuf + QSEECOM_ALIGN(sizeof(struct hdcp_deinit_req)));
|
|
|
|
rc = qseecom_send_command(handle->qseecom_handle,
|
|
req_buf, QSEECOM_ALIGN(sizeof(struct hdcp_deinit_req)),
|
|
rsp_buf, QSEECOM_ALIGN(sizeof(struct hdcp_deinit_rsp)));
|
|
|
|
if (rc < 0) {
|
|
pr_err("qseecom cmd failed err = %d\n", rc);
|
|
goto exit;
|
|
}
|
|
|
|
/* deallocate the resources for qseecom handle */
|
|
rc = qseecom_shutdown_app(&handle->qseecom_handle);
|
|
if (rc) {
|
|
pr_err("qseecom_shutdown_app failed err: %d\n", rc);
|
|
goto exit;
|
|
}
|
|
|
|
handle->hdcp_state &= ~HDCP_STATE_APP_LOADED;
|
|
pr_debug("success\n");
|
|
exit:
|
|
return rc;
|
|
}
|
|
|
|
static int hdcp_lib_txmtr_init(struct hdcp_lib_handle *handle)
|
|
{
|
|
int rc = 0;
|
|
struct hdcp_init_req *req_buf;
|
|
struct hdcp_init_rsp *rsp_buf;
|
|
|
|
if (!handle) {
|
|
pr_err("invalid input\n");
|
|
return -EINVAL;
|
|
}
|
|
|
|
if (!(handle->hdcp_state & HDCP_STATE_APP_LOADED)) {
|
|
pr_err("app not loaded\n");
|
|
goto exit;
|
|
}
|
|
|
|
if (handle->hdcp_state & HDCP_STATE_TXMTR_INIT) {
|
|
pr_err("txmtr already initialized\n");
|
|
goto exit;
|
|
}
|
|
|
|
/* send HDCP_Txmtr_Init command to TZ */
|
|
req_buf = (struct hdcp_init_req *)handle->
|
|
qseecom_handle->sbuf;
|
|
req_buf->commandid = HDCP_TXMTR_INIT;
|
|
rsp_buf = (struct hdcp_init_rsp *)(handle->qseecom_handle->
|
|
sbuf + QSEECOM_ALIGN(sizeof(struct hdcp_init_req)));
|
|
|
|
rc = qseecom_send_command(handle->qseecom_handle, req_buf,
|
|
QSEECOM_ALIGN(sizeof(struct hdcp_init_req)), rsp_buf,
|
|
QSEECOM_ALIGN(sizeof(struct hdcp_init_rsp)));
|
|
|
|
if ((rc < 0) || (rsp_buf->status != HDCP_SUCCESS) ||
|
|
(rsp_buf->commandid != HDCP_TXMTR_INIT) ||
|
|
(rsp_buf->msglen <= 0) || (rsp_buf->message == NULL)) {
|
|
pr_err("qseecom cmd failed with err = %d, status = %d\n",
|
|
rc, rsp_buf->status);
|
|
rc = -EINVAL;
|
|
goto exit;
|
|
}
|
|
|
|
pr_debug("recvd %s from TZ at %dms\n",
|
|
hdcp_lib_message_name((int)rsp_buf->message[0]),
|
|
jiffies_to_msecs(jiffies));
|
|
|
|
/* send the response to HDMI driver */
|
|
memset(handle->listener_buf, 0, MAX_TX_MESSAGE_SIZE);
|
|
memcpy(handle->listener_buf, (unsigned char *)rsp_buf->message,
|
|
rsp_buf->msglen);
|
|
handle->msglen = rsp_buf->msglen;
|
|
handle->hdcp_timeout = rsp_buf->timeout;
|
|
|
|
handle->tz_ctxhandle = rsp_buf->ctxhandle;
|
|
handle->hdcp_state |= HDCP_STATE_TXMTR_INIT;
|
|
|
|
pr_debug("success\n");
|
|
exit:
|
|
return rc;
|
|
}
|
|
|
|
static int hdcp_lib_txmtr_deinit(struct hdcp_lib_handle *handle)
|
|
{
|
|
int rc = 0;
|
|
struct hdcp_deinit_req *req_buf;
|
|
struct hdcp_deinit_rsp *rsp_buf;
|
|
|
|
if (!handle) {
|
|
pr_err("invalid input\n");
|
|
rc = -EINVAL;
|
|
goto exit;
|
|
}
|
|
|
|
if (!(handle->hdcp_state & HDCP_STATE_APP_LOADED)) {
|
|
pr_err("app not loaded\n");
|
|
goto exit;
|
|
}
|
|
|
|
if (!(handle->hdcp_state & HDCP_STATE_TXMTR_INIT)) {
|
|
/* unload library here */
|
|
pr_err("txmtr not initialized\n");
|
|
goto exit;
|
|
}
|
|
|
|
/* send command to TZ */
|
|
req_buf = (struct hdcp_deinit_req *)handle->
|
|
qseecom_handle->sbuf;
|
|
req_buf->commandid = HDCP_TXMTR_DEINIT;
|
|
req_buf->ctxhandle = handle->tz_ctxhandle;
|
|
rsp_buf = (struct hdcp_deinit_rsp *)(handle->
|
|
qseecom_handle->sbuf + QSEECOM_ALIGN(sizeof(
|
|
struct hdcp_deinit_req)));
|
|
|
|
rc = qseecom_send_command(handle->qseecom_handle, req_buf,
|
|
QSEECOM_ALIGN(sizeof(struct hdcp_deinit_req)), rsp_buf,
|
|
QSEECOM_ALIGN(sizeof(struct hdcp_deinit_rsp)));
|
|
|
|
if ((rc < 0) || (rsp_buf->status < 0) ||
|
|
(rsp_buf->commandid != HDCP_TXMTR_DEINIT)) {
|
|
pr_err("qseecom cmd failed with err = %d status = %d\n",
|
|
rc, rsp_buf->status);
|
|
rc = -EINVAL;
|
|
goto exit;
|
|
}
|
|
|
|
handle->hdcp_state &= ~HDCP_STATE_TXMTR_INIT;
|
|
pr_debug("success\n");
|
|
exit:
|
|
return rc;
|
|
}
|
|
|
|
static void hdcp_lib_stream(struct hdcp_lib_handle *handle)
|
|
{
|
|
int rc = 0;
|
|
struct hdcp_query_stream_type_req *req_buf;
|
|
struct hdcp_query_stream_type_rsp *rsp_buf;
|
|
|
|
if (!handle) {
|
|
pr_err("invalid handle\n");
|
|
return;
|
|
}
|
|
|
|
/* send command to TZ */
|
|
req_buf = (struct hdcp_query_stream_type_req *)handle->
|
|
qseecom_handle->sbuf;
|
|
req_buf->commandid = HDCP_TXMTR_QUERY_STREAM_TYPE;
|
|
req_buf->ctxhandle = handle->tz_ctxhandle;
|
|
rsp_buf = (struct hdcp_query_stream_type_rsp *)(handle->
|
|
qseecom_handle->sbuf + QSEECOM_ALIGN(sizeof(
|
|
struct hdcp_query_stream_type_req)));
|
|
|
|
rc = qseecom_send_command(handle->qseecom_handle, req_buf,
|
|
QSEECOM_ALIGN(sizeof(struct hdcp_query_stream_type_req)), rsp_buf,
|
|
QSEECOM_ALIGN(sizeof(struct hdcp_query_stream_type_rsp)));
|
|
|
|
if ((rc < 0) || (rsp_buf->status < 0) || (rsp_buf->msglen <= 0) ||
|
|
(rsp_buf->commandid != HDCP_TXMTR_QUERY_STREAM_TYPE) ||
|
|
(rsp_buf->msg == NULL)) {
|
|
pr_err("qseecom cmd failed with err=%d status=%d\n",
|
|
rc, rsp_buf->status);
|
|
rc = -EINVAL;
|
|
goto exit;
|
|
}
|
|
|
|
pr_debug("message received from TZ: %s\n",
|
|
hdcp_lib_message_name((int)rsp_buf->msg[0]));
|
|
|
|
memset(handle->listener_buf, 0, MAX_TX_MESSAGE_SIZE);
|
|
memcpy(handle->listener_buf, (unsigned char *)rsp_buf->msg,
|
|
rsp_buf->msglen);
|
|
handle->hdcp_timeout = rsp_buf->timeout;
|
|
handle->msglen = rsp_buf->msglen;
|
|
exit:
|
|
if (!rc && !atomic_read(&handle->hdcp_off))
|
|
hdcp_lib_send_message(handle);
|
|
}
|
|
|
|
static void hdcp_lib_query_stream_work(struct kthread_work *work)
|
|
{
|
|
struct hdcp_lib_handle *handle = container_of(work,
|
|
struct hdcp_lib_handle, wk_stream);
|
|
|
|
hdcp_lib_stream(handle);
|
|
}
|
|
|
|
static bool hdcp_lib_client_feature_supported(void *phdcpcontext)
|
|
{
|
|
int rc = 0;
|
|
bool supported = false;
|
|
struct hdcp_lib_handle *handle = phdcpcontext;
|
|
|
|
if (!handle) {
|
|
pr_err("invalid input\n");
|
|
goto exit;
|
|
}
|
|
|
|
if (handle->feature_supported) {
|
|
supported = true;
|
|
goto exit;
|
|
}
|
|
|
|
rc = hdcp_lib_library_load(handle);
|
|
if (!rc) {
|
|
pr_debug("HDCP2p2 supported\n");
|
|
handle->feature_supported = true;
|
|
hdcp_lib_library_unload(handle);
|
|
supported = true;
|
|
}
|
|
exit:
|
|
return supported;
|
|
}
|
|
|
|
static void hdcp_lib_check_worker_status(struct hdcp_lib_handle *handle)
|
|
{
|
|
if (!list_empty(&handle->wk_init.node))
|
|
pr_debug("init work queued\n");
|
|
|
|
if (handle->worker.current_work == &handle->wk_init)
|
|
pr_debug("init work executing\n");
|
|
|
|
if (!list_empty(&handle->wk_msg_sent.node))
|
|
pr_debug("msg_sent work queued\n");
|
|
|
|
if (handle->worker.current_work == &handle->wk_msg_sent)
|
|
pr_debug("msg_sent work executing\n");
|
|
|
|
if (!list_empty(&handle->wk_msg_recvd.node))
|
|
pr_debug("msg_recvd work queued\n");
|
|
|
|
if (handle->worker.current_work == &handle->wk_msg_recvd)
|
|
pr_debug("msg_recvd work executing\n");
|
|
|
|
if (!list_empty(&handle->wk_timeout.node))
|
|
pr_debug("timeout work queued\n");
|
|
|
|
if (handle->worker.current_work == &handle->wk_timeout)
|
|
pr_debug("timeout work executing\n");
|
|
|
|
if (!list_empty(&handle->wk_clean.node))
|
|
pr_debug("clean work queued\n");
|
|
|
|
if (handle->worker.current_work == &handle->wk_clean)
|
|
pr_debug("clean work executing\n");
|
|
|
|
if (!list_empty(&handle->wk_topology.node))
|
|
pr_debug("topology work queued\n");
|
|
|
|
if (handle->worker.current_work == &handle->wk_topology)
|
|
pr_debug("topology work executing\n");
|
|
|
|
if (!list_empty(&handle->wk_stream.node))
|
|
pr_debug("stream work queued\n");
|
|
|
|
if (handle->worker.current_work == &handle->wk_stream)
|
|
pr_debug("stream work executing\n");
|
|
}
|
|
|
|
static int hdcp_lib_check_valid_state(struct hdcp_lib_handle *handle)
|
|
{
|
|
int rc = 0;
|
|
|
|
if (!list_empty(&handle->worker.work_list))
|
|
hdcp_lib_check_worker_status(handle);
|
|
|
|
if (handle->wakeup_cmd == HDCP_LIB_WKUP_CMD_START) {
|
|
if (!list_empty(&handle->worker.work_list)) {
|
|
rc = -EBUSY;
|
|
goto exit;
|
|
}
|
|
} else {
|
|
if (atomic_read(&handle->hdcp_off)) {
|
|
pr_warn("hdcp2.2 session tearing down\n");
|
|
goto exit;
|
|
}
|
|
|
|
if (!(handle->hdcp_state & HDCP_STATE_APP_LOADED)) {
|
|
pr_warn("hdcp 2.2 app not loaded\n");
|
|
goto exit;
|
|
}
|
|
}
|
|
exit:
|
|
return rc;
|
|
}
|
|
|
|
static void hdcp_lib_update_exec_type(void *ctx, bool tethered)
|
|
{
|
|
struct hdcp_lib_handle *handle = ctx;
|
|
|
|
if (!handle)
|
|
return;
|
|
|
|
mutex_lock(&handle->wakeup_mutex);
|
|
|
|
if (handle->tethered == tethered) {
|
|
pr_debug("exec mode same as %s\n",
|
|
tethered ? "tethered" : "threaded");
|
|
} else {
|
|
handle->tethered = tethered;
|
|
|
|
pr_debug("exec mode changed to %s\n",
|
|
tethered ? "tethered" : "threaded");
|
|
}
|
|
|
|
mutex_unlock(&handle->wakeup_mutex);
|
|
}
|
|
|
|
static int hdcp_lib_wakeup(struct hdcp_lib_wakeup_data *data)
|
|
{
|
|
struct hdcp_lib_handle *handle;
|
|
int rc = 0;
|
|
|
|
if (!data)
|
|
return -EINVAL;
|
|
|
|
handle = data->context;
|
|
if (!handle)
|
|
return -EINVAL;
|
|
|
|
mutex_lock(&handle->wakeup_mutex);
|
|
|
|
handle->wakeup_cmd = data->cmd;
|
|
handle->timeout_left = data->timeout;
|
|
|
|
pr_debug("%s, timeout left: %dms, tethered %d\n",
|
|
hdcp_lib_cmd_to_str(handle->wakeup_cmd),
|
|
handle->timeout_left, handle->tethered);
|
|
|
|
rc = hdcp_lib_check_valid_state(handle);
|
|
if (rc)
|
|
goto exit;
|
|
|
|
mutex_lock(&handle->msg_lock);
|
|
if (data->recvd_msg_len) {
|
|
kzfree(handle->last_msg_recvd_buf);
|
|
|
|
handle->last_msg_recvd_len = data->recvd_msg_len;
|
|
handle->last_msg_recvd_buf = kzalloc(data->recvd_msg_len,
|
|
GFP_KERNEL);
|
|
if (!handle->last_msg_recvd_buf) {
|
|
rc = -ENOMEM;
|
|
mutex_unlock(&handle->msg_lock);
|
|
goto exit;
|
|
}
|
|
|
|
memcpy(handle->last_msg_recvd_buf, data->recvd_msg_buf,
|
|
data->recvd_msg_len);
|
|
}
|
|
mutex_unlock(&handle->msg_lock);
|
|
|
|
if (!completion_done(&handle->topo_wait))
|
|
complete_all(&handle->topo_wait);
|
|
|
|
switch (handle->wakeup_cmd) {
|
|
case HDCP_LIB_WKUP_CMD_START:
|
|
handle->no_stored_km_flag = 0;
|
|
handle->repeater_flag = 0;
|
|
handle->last_msg_sent = 0;
|
|
handle->hdcp_timeout = 0;
|
|
handle->timeout_left = 0;
|
|
atomic_set(&handle->hdcp_off, 0);
|
|
handle->hdcp_state = HDCP_STATE_INIT;
|
|
|
|
HDCP_LIB_EXECUTE(init);
|
|
break;
|
|
case HDCP_LIB_WKUP_CMD_STOP:
|
|
atomic_set(&handle->hdcp_off, 1);
|
|
|
|
HDCP_LIB_EXECUTE(clean);
|
|
break;
|
|
case HDCP_LIB_WKUP_CMD_MSG_SEND_SUCCESS:
|
|
handle->last_msg_sent = handle->listener_buf[0];
|
|
|
|
HDCP_LIB_EXECUTE(msg_sent);
|
|
break;
|
|
case HDCP_LIB_WKUP_CMD_MSG_SEND_FAILED:
|
|
case HDCP_LIB_WKUP_CMD_MSG_RECV_FAILED:
|
|
HDCP_LIB_EXECUTE(clean);
|
|
break;
|
|
case HDCP_LIB_WKUP_CMD_MSG_RECV_SUCCESS:
|
|
HDCP_LIB_EXECUTE(msg_recvd);
|
|
break;
|
|
case HDCP_LIB_WKUP_CMD_MSG_RECV_TIMEOUT:
|
|
HDCP_LIB_EXECUTE(timeout);
|
|
break;
|
|
case HDCP_LIB_WKUP_CMD_QUERY_STREAM_TYPE:
|
|
HDCP_LIB_EXECUTE(stream);
|
|
break;
|
|
default:
|
|
pr_err("invalid wakeup command %d\n", handle->wakeup_cmd);
|
|
}
|
|
exit:
|
|
mutex_unlock(&handle->wakeup_mutex);
|
|
|
|
return rc;
|
|
}
|
|
|
|
static void hdcp_lib_msg_sent(struct hdcp_lib_handle *handle)
|
|
{
|
|
struct hdmi_hdcp_wakeup_data cdata = {HDMI_HDCP_WKUP_CMD_INVALID};
|
|
|
|
if (!handle) {
|
|
pr_err("invalid handle\n");
|
|
return;
|
|
}
|
|
|
|
cdata.context = handle->client_ctx;
|
|
|
|
switch (handle->last_msg_sent) {
|
|
case SKE_SEND_EKS_MESSAGE_ID:
|
|
if (handle->repeater_flag) {
|
|
if (!atomic_read(&handle->hdcp_off))
|
|
queue_kthread_work(&handle->worker,
|
|
&handle->wk_topology);
|
|
}
|
|
|
|
if (!hdcp_lib_enable_encryption(handle)) {
|
|
cdata.cmd = HDMI_HDCP_WKUP_CMD_STATUS_SUCCESS;
|
|
} else {
|
|
if (!atomic_read(&handle->hdcp_off))
|
|
HDCP_LIB_EXECUTE(clean);
|
|
}
|
|
break;
|
|
case REPEATER_AUTH_SEND_ACK_MESSAGE_ID:
|
|
pr_debug("Repeater authentication successful\n");
|
|
break;
|
|
default:
|
|
cdata.cmd = HDMI_HDCP_WKUP_CMD_RECV_MESSAGE;
|
|
cdata.timeout = handle->timeout_left;
|
|
}
|
|
|
|
hdcp_lib_wakeup_client(handle, &cdata);
|
|
}
|
|
|
|
static void hdcp_lib_msg_sent_work(struct kthread_work *work)
|
|
{
|
|
struct hdcp_lib_handle *handle = container_of(work,
|
|
struct hdcp_lib_handle, wk_msg_sent);
|
|
|
|
if (handle->wakeup_cmd != HDCP_LIB_WKUP_CMD_MSG_SEND_SUCCESS) {
|
|
pr_err("invalid wakeup command %d\n", handle->wakeup_cmd);
|
|
return;
|
|
}
|
|
|
|
hdcp_lib_msg_sent(handle);
|
|
}
|
|
|
|
static void hdcp_lib_init(struct hdcp_lib_handle *handle)
|
|
{
|
|
int rc = 0;
|
|
|
|
if (!handle) {
|
|
pr_err("invalid handle\n");
|
|
return;
|
|
}
|
|
|
|
if (handle->wakeup_cmd != HDCP_LIB_WKUP_CMD_START) {
|
|
pr_err("invalid wakeup command %d\n", handle->wakeup_cmd);
|
|
return;
|
|
}
|
|
|
|
rc = hdcp_lib_library_load(handle);
|
|
if (rc)
|
|
goto exit;
|
|
|
|
rc = hdcp_lib_txmtr_init(handle);
|
|
if (rc)
|
|
goto exit;
|
|
|
|
hdcp_lib_send_message(handle);
|
|
|
|
return;
|
|
exit:
|
|
HDCP_LIB_EXECUTE(clean);
|
|
}
|
|
|
|
static void hdcp_lib_init_work(struct kthread_work *work)
|
|
{
|
|
struct hdcp_lib_handle *handle = container_of(work,
|
|
struct hdcp_lib_handle, wk_init);
|
|
|
|
hdcp_lib_init(handle);
|
|
}
|
|
|
|
static void hdcp_lib_timeout(struct hdcp_lib_handle *handle)
|
|
{
|
|
int rc = 0;
|
|
struct hdcp_send_timeout_req *req_buf;
|
|
struct hdcp_send_timeout_rsp *rsp_buf;
|
|
|
|
if (!handle) {
|
|
pr_err("invalid handle\n");
|
|
return;
|
|
}
|
|
|
|
req_buf = (struct hdcp_send_timeout_req *)
|
|
(handle->qseecom_handle->sbuf);
|
|
req_buf->commandid = HDCP_TXMTR_SEND_MESSAGE_TIMEOUT;
|
|
req_buf->ctxhandle = handle->tz_ctxhandle;
|
|
|
|
rsp_buf = (struct hdcp_send_timeout_rsp *)(handle->
|
|
qseecom_handle->sbuf + QSEECOM_ALIGN(sizeof(
|
|
struct hdcp_send_timeout_req)));
|
|
|
|
rc = qseecom_send_command(handle->qseecom_handle, req_buf,
|
|
QSEECOM_ALIGN(sizeof(struct hdcp_send_timeout_req)), rsp_buf,
|
|
QSEECOM_ALIGN(sizeof(struct hdcp_send_timeout_rsp)));
|
|
|
|
if ((rc < 0) || (rsp_buf->status != HDCP_SUCCESS)) {
|
|
pr_err("qseecom cmd failed for with err = %d status = %d\n",
|
|
rc, rsp_buf->status);
|
|
rc = -EINVAL;
|
|
goto error;
|
|
}
|
|
|
|
if (rsp_buf->commandid == HDCP_TXMTR_SEND_MESSAGE_TIMEOUT) {
|
|
pr_err("HDCP_TXMTR_SEND_MESSAGE_TIMEOUT\n");
|
|
rc = -EINVAL;
|
|
goto error;
|
|
}
|
|
|
|
/*
|
|
* if the response contains LC_Init message
|
|
* send the message again to TZ
|
|
*/
|
|
if ((rsp_buf->commandid == HDCP_TXMTR_PROCESS_RECEIVED_MESSAGE) &&
|
|
((int)rsp_buf->message[0] == LC_INIT_MESSAGE_ID) &&
|
|
(rsp_buf->msglen == LC_INIT_MESSAGE_SIZE)) {
|
|
if (!atomic_read(&handle->hdcp_off)) {
|
|
/* keep local copy of TZ response */
|
|
memset(handle->listener_buf, 0, MAX_TX_MESSAGE_SIZE);
|
|
memcpy(handle->listener_buf,
|
|
(unsigned char *)rsp_buf->message,
|
|
rsp_buf->msglen);
|
|
handle->hdcp_timeout = rsp_buf->timeout;
|
|
handle->msglen = rsp_buf->msglen;
|
|
|
|
hdcp_lib_send_message(handle);
|
|
}
|
|
}
|
|
error:
|
|
if (!atomic_read(&handle->hdcp_off))
|
|
HDCP_LIB_EXECUTE(clean);
|
|
}
|
|
|
|
static void hdcp_lib_manage_timeout_work(struct kthread_work *work)
|
|
{
|
|
struct hdcp_lib_handle *handle = container_of(work,
|
|
struct hdcp_lib_handle, wk_timeout);
|
|
|
|
hdcp_lib_timeout(handle);
|
|
}
|
|
|
|
static void hdcp_lib_clean(struct hdcp_lib_handle *handle)
|
|
{
|
|
struct hdmi_hdcp_wakeup_data cdata = {HDMI_HDCP_WKUP_CMD_INVALID};
|
|
|
|
if (!handle) {
|
|
pr_err("invalid input\n");
|
|
return;
|
|
};
|
|
|
|
hdcp_lib_txmtr_deinit(handle);
|
|
hdcp_lib_library_unload(handle);
|
|
|
|
cdata.context = handle->client_ctx;
|
|
cdata.cmd = HDMI_HDCP_WKUP_CMD_STATUS_FAILED;
|
|
|
|
if (!atomic_read(&handle->hdcp_off))
|
|
hdcp_lib_wakeup_client(handle, &cdata);
|
|
|
|
atomic_set(&handle->hdcp_off, 1);
|
|
}
|
|
|
|
|
|
static void hdcp_lib_cleanup_work(struct kthread_work *work)
|
|
{
|
|
struct hdcp_lib_handle *handle = container_of(work,
|
|
struct hdcp_lib_handle, wk_clean);
|
|
|
|
hdcp_lib_clean(handle);
|
|
}
|
|
|
|
static void hdcp_lib_msg_recvd(struct hdcp_lib_handle *handle)
|
|
{
|
|
int rc = 0;
|
|
struct hdmi_hdcp_wakeup_data cdata = {HDMI_HDCP_WKUP_CMD_INVALID};
|
|
struct hdcp_rcvd_msg_req *req_buf;
|
|
struct hdcp_rcvd_msg_rsp *rsp_buf;
|
|
uint32_t msglen;
|
|
char *msg;
|
|
|
|
if (!handle) {
|
|
pr_err("invalid handle\n");
|
|
return;
|
|
}
|
|
|
|
cdata.context = handle->client_ctx;
|
|
|
|
mutex_lock(&handle->msg_lock);
|
|
msglen = handle->last_msg_recvd_len;
|
|
|
|
if (msglen <= 0) {
|
|
pr_err("invalid msg len\n");
|
|
mutex_unlock(&handle->msg_lock);
|
|
rc = -EINVAL;
|
|
goto exit;
|
|
}
|
|
|
|
msg = kzalloc(msglen, GFP_KERNEL);
|
|
if (!msg) {
|
|
mutex_unlock(&handle->msg_lock);
|
|
rc = -ENOMEM;
|
|
goto exit;
|
|
}
|
|
|
|
memcpy(msg, handle->last_msg_recvd_buf, msglen);
|
|
|
|
mutex_unlock(&handle->msg_lock);
|
|
|
|
pr_debug("msg received: %s from sink\n",
|
|
hdcp_lib_message_name((int)msg[0]));
|
|
|
|
/* send the message to QSEECOM */
|
|
req_buf = (struct hdcp_rcvd_msg_req *)(handle->
|
|
qseecom_handle->sbuf);
|
|
req_buf->commandid = HDCP_TXMTR_PROCESS_RECEIVED_MESSAGE;
|
|
memcpy(req_buf->msg, msg, msglen);
|
|
req_buf->msglen = msglen;
|
|
req_buf->ctxhandle = handle->tz_ctxhandle;
|
|
|
|
rsp_buf = (struct hdcp_rcvd_msg_rsp *)(handle->
|
|
qseecom_handle->sbuf + QSEECOM_ALIGN(sizeof(
|
|
struct hdcp_rcvd_msg_req)));
|
|
|
|
pr_debug("writing %s to TZ at %dms\n",
|
|
hdcp_lib_message_name((int)msg[0]),
|
|
jiffies_to_msecs(jiffies));
|
|
|
|
rc = qseecom_send_command(handle->qseecom_handle, req_buf,
|
|
QSEECOM_ALIGN(sizeof(struct hdcp_rcvd_msg_req)), rsp_buf,
|
|
QSEECOM_ALIGN(sizeof(struct hdcp_rcvd_msg_rsp)));
|
|
|
|
/* get next message from sink if we receive H PRIME on no store km */
|
|
if ((msg[0] == AKE_SEND_H_PRIME_MESSAGE_ID) &&
|
|
handle->no_stored_km_flag) {
|
|
handle->hdcp_timeout = rsp_buf->timeout;
|
|
|
|
cdata.cmd = HDMI_HDCP_WKUP_CMD_RECV_MESSAGE;
|
|
cdata.timeout = handle->hdcp_timeout;
|
|
|
|
goto exit;
|
|
}
|
|
|
|
if ((msg[0] == REPEATER_AUTH_STREAM_READY_MESSAGE_ID) &&
|
|
(rc == 0) && (rsp_buf->status == 0)) {
|
|
pr_debug("Got Auth_Stream_Ready, nothing sent to rx\n");
|
|
goto exit;
|
|
}
|
|
|
|
if ((rc < 0) || (rsp_buf->status < 0) || (rsp_buf->msglen <= 0) ||
|
|
(rsp_buf->commandid != HDCP_TXMTR_PROCESS_RECEIVED_MESSAGE) ||
|
|
(rsp_buf->msg == NULL)) {
|
|
pr_err("qseecom cmd failed with err=%d status=%d\n",
|
|
rc, rsp_buf->status);
|
|
rc = -EINVAL;
|
|
goto exit;
|
|
}
|
|
|
|
pr_debug("recvd %s from TZ at %dms\n",
|
|
hdcp_lib_message_name((int)rsp_buf->msg[0]),
|
|
jiffies_to_msecs(jiffies));
|
|
|
|
/* set the flag if response is AKE_No_Stored_km */
|
|
if (((int)rsp_buf->msg[0] == AKE_NO_STORED_KM_MESSAGE_ID)) {
|
|
pr_debug("Setting no_stored_km_flag\n");
|
|
handle->no_stored_km_flag = 1;
|
|
} else {
|
|
handle->no_stored_km_flag = 0;
|
|
}
|
|
|
|
/* check if it's a repeater */
|
|
if ((rsp_buf->msg[0] == SKE_SEND_EKS_MESSAGE_ID) &&
|
|
(rsp_buf->msglen == SKE_SEND_EKS_MESSAGE_SIZE)) {
|
|
if ((rsp_buf->flag ==
|
|
HDCP_TXMTR_SUBSTATE_WAITING_FOR_RECIEVERID_LIST) &&
|
|
(rsp_buf->timeout > 0))
|
|
handle->repeater_flag = 1;
|
|
}
|
|
|
|
memset(handle->listener_buf, 0, MAX_TX_MESSAGE_SIZE);
|
|
memcpy(handle->listener_buf, (unsigned char *)rsp_buf->msg,
|
|
rsp_buf->msglen);
|
|
handle->hdcp_timeout = rsp_buf->timeout;
|
|
handle->msglen = rsp_buf->msglen;
|
|
|
|
if (!atomic_read(&handle->hdcp_off)) {
|
|
cdata.cmd = HDMI_HDCP_WKUP_CMD_SEND_MESSAGE;
|
|
cdata.send_msg_buf = handle->listener_buf;
|
|
cdata.send_msg_len = handle->msglen;
|
|
cdata.timeout = handle->hdcp_timeout;
|
|
}
|
|
|
|
exit:
|
|
kzfree(msg);
|
|
|
|
hdcp_lib_wakeup_client(handle, &cdata);
|
|
|
|
if (rc && !atomic_read(&handle->hdcp_off))
|
|
HDCP_LIB_EXECUTE(clean);
|
|
}
|
|
|
|
static void hdcp_lib_msg_recvd_work(struct kthread_work *work)
|
|
{
|
|
struct hdcp_lib_handle *handle = container_of(work,
|
|
struct hdcp_lib_handle, wk_msg_recvd);
|
|
|
|
hdcp_lib_msg_recvd(handle);
|
|
}
|
|
|
|
static void hdcp_lib_topology_work(struct kthread_work *work)
|
|
{
|
|
u32 timeout;
|
|
struct hdcp_lib_handle *handle = container_of(work,
|
|
struct hdcp_lib_handle, wk_topology);
|
|
|
|
if (!handle) {
|
|
pr_err("invalid input\n");
|
|
return;
|
|
}
|
|
|
|
reinit_completion(&handle->topo_wait);
|
|
timeout = wait_for_completion_timeout(&handle->topo_wait, HZ * 3);
|
|
if (!timeout) {
|
|
pr_err("topology receiver id list timeout\n");
|
|
|
|
if (!atomic_read(&handle->hdcp_off))
|
|
HDCP_LIB_EXECUTE(clean);
|
|
}
|
|
}
|
|
|
|
bool hdcp1_check_if_supported_load_app(void)
|
|
{
|
|
int rc = 0;
|
|
|
|
/* start hdcp1 app */
|
|
if (hdcp1_supported && !hdcp1_handle) {
|
|
rc = qseecom_start_app(&hdcp1_handle, HDCP1_APP_NAME,
|
|
QSEECOM_SBUFF_SIZE);
|
|
if (rc) {
|
|
pr_err("qseecom_start_app failed %d\n", rc);
|
|
hdcp1_supported = false;
|
|
}
|
|
}
|
|
|
|
pr_debug("hdcp1 app %s loaded\n",
|
|
hdcp1_supported ? "successfully" : "not");
|
|
|
|
return hdcp1_supported;
|
|
}
|
|
|
|
/* APIs exposed to all clients */
|
|
int hdcp1_set_keys(uint32_t *aksv_msb, uint32_t *aksv_lsb)
|
|
{
|
|
int rc = 0;
|
|
struct hdcp1_key_set_req *key_set_req;
|
|
struct hdcp1_key_set_rsp *key_set_rsp;
|
|
|
|
if (aksv_msb == NULL || aksv_lsb == NULL)
|
|
return -EINVAL;
|
|
|
|
if (!hdcp1_supported || !hdcp1_handle)
|
|
return -EINVAL;
|
|
|
|
/* set keys and request aksv */
|
|
key_set_req = (struct hdcp1_key_set_req *)hdcp1_handle->sbuf;
|
|
key_set_req->commandid = HDCP1_SET_KEY_MESSAGE_ID;
|
|
key_set_rsp = (struct hdcp1_key_set_rsp *)(hdcp1_handle->sbuf +
|
|
QSEECOM_ALIGN(sizeof(struct hdcp1_key_set_req)));
|
|
rc = qseecom_send_command(hdcp1_handle,
|
|
key_set_req, QSEECOM_ALIGN(sizeof(struct hdcp1_key_set_req)),
|
|
key_set_rsp, QSEECOM_ALIGN(sizeof(struct hdcp1_key_set_rsp)));
|
|
|
|
if (rc < 0) {
|
|
pr_err("qseecom cmd failed err=%d\n", rc);
|
|
return -ENOKEY;
|
|
}
|
|
|
|
rc = key_set_rsp->ret;
|
|
if (rc) {
|
|
pr_err("set key cmd failed, rsp=%d\n",
|
|
key_set_rsp->ret);
|
|
return -ENOKEY;
|
|
}
|
|
|
|
/* copy bytes into msb and lsb */
|
|
*aksv_msb = key_set_rsp->ksv[0] << 24;
|
|
*aksv_msb |= key_set_rsp->ksv[1] << 16;
|
|
*aksv_msb |= key_set_rsp->ksv[2] << 8;
|
|
*aksv_msb |= key_set_rsp->ksv[3];
|
|
*aksv_lsb = key_set_rsp->ksv[4] << 24;
|
|
*aksv_lsb |= key_set_rsp->ksv[5] << 16;
|
|
*aksv_lsb |= key_set_rsp->ksv[6] << 8;
|
|
*aksv_lsb |= key_set_rsp->ksv[7];
|
|
|
|
return 0;
|
|
}
|
|
|
|
int hdcp1_set_enc(bool enable)
|
|
{
|
|
int rc = 0;
|
|
struct hdcp1_set_enc_req *set_enc_req;
|
|
struct hdcp1_set_enc_rsp *set_enc_rsp;
|
|
|
|
if (!hdcp1_supported || !hdcp1_handle)
|
|
return -EINVAL;
|
|
|
|
/* set keys and request aksv */
|
|
set_enc_req = (struct hdcp1_set_enc_req *)hdcp1_handle->sbuf;
|
|
set_enc_req->commandid = HDCP1_SET_ENC_MESSAGE_ID;
|
|
set_enc_req->enable = enable;
|
|
set_enc_rsp = (struct hdcp1_set_enc_rsp *)(hdcp1_handle->sbuf +
|
|
QSEECOM_ALIGN(sizeof(struct hdcp1_set_enc_req)));
|
|
rc = qseecom_send_command(hdcp1_handle,
|
|
set_enc_req, QSEECOM_ALIGN(sizeof(struct hdcp1_set_enc_req)),
|
|
set_enc_rsp, QSEECOM_ALIGN(sizeof(struct hdcp1_set_enc_rsp)));
|
|
|
|
if (rc < 0) {
|
|
pr_err("qseecom cmd failed err=%d\n", rc);
|
|
return -EINVAL;
|
|
}
|
|
|
|
rc = set_enc_rsp->ret;
|
|
if (rc) {
|
|
pr_err("enc cmd failed, rsp=%d\n",
|
|
set_enc_rsp->ret);
|
|
return -EINVAL;
|
|
}
|
|
|
|
pr_debug("success\n");
|
|
return 0;
|
|
}
|
|
|
|
int hdcp_library_register(struct hdcp_register_data *data)
|
|
{
|
|
int rc = 0;
|
|
struct hdcp_lib_handle *handle = NULL;
|
|
|
|
if (!data) {
|
|
pr_err("invalid input\n");
|
|
return -EINVAL;
|
|
}
|
|
|
|
if (!data->txmtr_ops) {
|
|
pr_err("invalid input: txmtr context\n");
|
|
return -EINVAL;
|
|
}
|
|
|
|
if (!data->client_ops) {
|
|
pr_err("invalid input: client_ops\n");
|
|
return -EINVAL;
|
|
}
|
|
|
|
if (!data->hdcp_ctx) {
|
|
pr_err("invalid input: hdcp_ctx\n");
|
|
return -EINVAL;
|
|
}
|
|
|
|
/* populate ops to be called by client */
|
|
data->txmtr_ops->feature_supported = hdcp_lib_client_feature_supported;
|
|
data->txmtr_ops->wakeup = hdcp_lib_wakeup;
|
|
data->txmtr_ops->update_exec_type = hdcp_lib_update_exec_type;
|
|
|
|
handle = kzalloc(sizeof(*handle), GFP_KERNEL);
|
|
if (!handle) {
|
|
rc = -ENOMEM;
|
|
goto unlock;
|
|
}
|
|
|
|
handle->client_ctx = data->client_ctx;
|
|
handle->client_ops = data->client_ops;
|
|
handle->tethered = data->tethered;
|
|
|
|
pr_debug("tethered %d\n", handle->tethered);
|
|
|
|
atomic_set(&handle->hdcp_off, 0);
|
|
|
|
mutex_init(&handle->msg_lock);
|
|
mutex_init(&handle->wakeup_mutex);
|
|
|
|
init_kthread_worker(&handle->worker);
|
|
|
|
init_kthread_work(&handle->wk_init, hdcp_lib_init_work);
|
|
init_kthread_work(&handle->wk_msg_sent, hdcp_lib_msg_sent_work);
|
|
init_kthread_work(&handle->wk_msg_recvd, hdcp_lib_msg_recvd_work);
|
|
init_kthread_work(&handle->wk_timeout, hdcp_lib_manage_timeout_work);
|
|
init_kthread_work(&handle->wk_clean, hdcp_lib_cleanup_work);
|
|
init_kthread_work(&handle->wk_topology, hdcp_lib_topology_work);
|
|
init_kthread_work(&handle->wk_stream, hdcp_lib_query_stream_work);
|
|
|
|
init_completion(&handle->topo_wait);
|
|
|
|
handle->listener_buf = kzalloc(MAX_TX_MESSAGE_SIZE, GFP_KERNEL);
|
|
if (!(handle->listener_buf)) {
|
|
rc = -ENOMEM;
|
|
goto error;
|
|
}
|
|
|
|
*data->hdcp_ctx = handle;
|
|
|
|
handle->thread = kthread_run(kthread_worker_fn,
|
|
&handle->worker, "hdcp_tz_lib");
|
|
|
|
if (IS_ERR(handle->thread)) {
|
|
pr_err("unable to start lib thread\n");
|
|
rc = PTR_ERR(handle->thread);
|
|
handle->thread = NULL;
|
|
goto error;
|
|
}
|
|
|
|
return 0;
|
|
error:
|
|
kzfree(handle->listener_buf);
|
|
handle->listener_buf = NULL;
|
|
kzfree(handle);
|
|
handle = NULL;
|
|
unlock:
|
|
return rc;
|
|
}
|
|
|
|
void hdcp_library_deregister(void *phdcpcontext)
|
|
{
|
|
struct hdcp_lib_handle *handle = phdcpcontext;
|
|
|
|
if (!handle)
|
|
return;
|
|
|
|
kthread_stop(handle->thread);
|
|
|
|
kzfree(handle->qseecom_handle);
|
|
kzfree(handle->last_msg_recvd_buf);
|
|
|
|
mutex_destroy(&handle->wakeup_mutex);
|
|
|
|
kzfree(handle->listener_buf);
|
|
kzfree(handle);
|
|
}
|