#define PMIC_GLINK_MAX_PORTS 3
-#define UCSI_BUF_SIZE 48
+#define UCSI_BUF_V1_SIZE (UCSI_MESSAGE_OUT + (UCSI_MESSAGE_OUT - UCSI_MESSAGE_IN))
+#define UCSI_BUF_V2_SIZE (UCSIv2_MESSAGE_OUT + (UCSIv2_MESSAGE_OUT - UCSI_MESSAGE_IN))
#define MSG_TYPE_REQ_RESP 1
-#define UCSI_BUF_SIZE 48
#define UC_NOTIFY_RECEIVER_UCSI 0x0
#define UC_UCSI_READ_BUF_REQ 0x11
struct __packed ucsi_read_buf_resp_msg {
struct pmic_glink_hdr hdr;
- u8 buf[UCSI_BUF_SIZE];
+ union {
+ u8 v2_buf[UCSI_BUF_V2_SIZE];
+ u8 v1_buf[UCSI_BUF_V1_SIZE];
+ } buf;
u32 ret_code;
};
struct __packed ucsi_write_buf_req_msg {
struct pmic_glink_hdr hdr;
- u8 buf[UCSI_BUF_SIZE];
+ union {
+ u8 v2_buf[UCSI_BUF_V2_SIZE];
+ u8 v1_buf[UCSI_BUF_V1_SIZE];
+ } buf;
u32 reserved;
};
bool ucsi_registered;
bool pd_running;
- u8 read_buf[UCSI_BUF_SIZE];
+ u8 read_buf[UCSI_BUF_V2_SIZE];
};
static int pmic_glink_ucsi_read(struct ucsi *__ucsi, unsigned int offset,
const void *val, size_t val_len)
{
struct ucsi_write_buf_req_msg req = {};
+ size_t req_len, buf_len;
unsigned long left;
int ret;
+ u8 *buf;
req.hdr.owner = PMIC_GLINK_OWNER_USBC;
req.hdr.type = MSG_TYPE_REQ_RESP;
req.hdr.opcode = UC_UCSI_WRITE_BUF_REQ;
- memcpy(&req.buf[offset], val, val_len);
+
+ if (ucsi->ucsi->version >= UCSI_VERSION_2_0) {
+ buf_len = UCSI_BUF_V2_SIZE;
+ buf = req.buf.v2_buf;
+ } else if (ucsi->ucsi->version) {
+ buf_len = UCSI_BUF_V1_SIZE;
+ buf = req.buf.v1_buf;
+ } else {
+ dev_err(ucsi->dev, "UCSI version unknown\n");
+ return -EINVAL;
+ }
+ req_len = sizeof(struct pmic_glink_hdr) + buf_len + sizeof(u32);
+
+ if (offset + val_len > buf_len)
+ return -EINVAL;
+
+ memcpy(&buf[offset], val, val_len);
reinit_completion(&ucsi->write_ack);
- ret = pmic_glink_send(ucsi->client, &req, sizeof(req));
+ ret = pmic_glink_send(ucsi->client, &req, req_len);
if (ret < 0) {
dev_err(ucsi->dev, "failed to send UCSI write request: %d\n", ret);
return ret;
static void pmic_glink_ucsi_read_ack(struct pmic_glink_ucsi *ucsi, const void *data, int len)
{
- const struct ucsi_read_buf_resp_msg *resp = data;
+ u32 ret_code, resp_len, buf_len = 0;
+ u8 *buf;
+
+ if (ucsi->ucsi->version) {
+ if (ucsi->ucsi->version >= UCSI_VERSION_2_0) {
+ buf = ((struct ucsi_read_buf_resp_msg *)data)->buf.v2_buf;
+ buf_len = UCSI_BUF_V2_SIZE;
+ } else {
+ buf = ((struct ucsi_read_buf_resp_msg *)data)->buf.v1_buf;
+ buf_len = UCSI_BUF_V1_SIZE;
+ }
+ } else if (!ucsi->ucsi_registered) {
+ /*
+ * If UCSI version is not known yet because device is not registered, choose buffer
+ * size which best fits incoming data
+ */
+ if (len > sizeof(struct pmic_glink_hdr) + UCSI_BUF_V2_SIZE) {
+ buf = ((struct ucsi_read_buf_resp_msg *)data)->buf.v2_buf;
+ buf_len = UCSI_BUF_V2_SIZE;
+ } else {
+ buf = ((struct ucsi_read_buf_resp_msg *)data)->buf.v1_buf;
+ buf_len = UCSI_BUF_V1_SIZE;
+ }
+ } else {
+ dev_err(ucsi->dev, "Device has been registered but UCSI version is still unknown\n");
+ return;
+ }
- if (resp->ret_code)
+ resp_len = sizeof(struct pmic_glink_hdr) + buf_len + sizeof(u32);
+
+ if (len > resp_len)
+ return;
+
+ /* Ensure that buffer_len leaves space for ret_code to be read back from memory */
+ if (buf_len > len - sizeof(struct pmic_glink_hdr) - sizeof(u32))
+ buf_len = len - sizeof(struct pmic_glink_hdr) - sizeof(u32);
+
+ memcpy(&ret_code, buf + buf_len, sizeof(u32));
+ if (ret_code)
return;
- memcpy(ucsi->read_buf, resp->buf, UCSI_BUF_SIZE);
+ memcpy(ucsi->read_buf, buf, buf_len);
complete(&ucsi->read_ack);
}