Message ID | 20200724174702.61754-3-hdegoede@redhat.com (mailing list archive) |
---|---|
State | Mainlined |
Commit | 8afe9a3548f9d1805dcea6d97978f2179c8403a3 |
Headers | show |
Series | [v2,1/6] usb: typec: tcpm: Move mod_delayed_work(&port->vdm_state_machine) call into tcpm_queue_vdm() | expand |
On Fri, Jul 24, 2020 at 07:46:59PM +0200, Hans de Goede wrote: > Refactor the tcpm_handle_vdm_request payload handling by doing the > endianness conversion only once directly inside tcpm_handle_vdm_request > itself instead of doing it multiple times inside various helper functions > called by tcpm_handle_vdm_request. > > This is a preparation patch for some further refactoring to fix an AB BA > lock inversion between the tcpm code and some altmode drivers. > > Reviewed-by: Guenter Roeck <linux@roeck-us.net> > Signed-off-by: Hans de Goede <hdegoede@redhat.com> One nit below, but nothing that would require a new version, so FWIW: Reviewed-by: Heikki Krogerus <heikki.krogerus@linux.intel.com> > --- > drivers/usb/typec/tcpm/tcpm.c | 49 ++++++++++++++++------------------- > 1 file changed, 22 insertions(+), 27 deletions(-) > > diff --git a/drivers/usb/typec/tcpm/tcpm.c b/drivers/usb/typec/tcpm/tcpm.c > index 862c474b3ebd..ee239b54bcd8 100644 > --- a/drivers/usb/typec/tcpm/tcpm.c > +++ b/drivers/usb/typec/tcpm/tcpm.c > @@ -981,16 +981,15 @@ static void tcpm_queue_vdm_unlocked(struct tcpm_port *port, const u32 header, > mutex_unlock(&port->lock); > } > > -static void svdm_consume_identity(struct tcpm_port *port, const __le32 *payload, > - int cnt) > +static void svdm_consume_identity(struct tcpm_port *port, const u32 *p, int cnt) > { > - u32 vdo = le32_to_cpu(payload[VDO_INDEX_IDH]); > - u32 product = le32_to_cpu(payload[VDO_INDEX_PRODUCT]); > + u32 vdo = p[VDO_INDEX_IDH]; > + u32 product = p[VDO_INDEX_PRODUCT]; > > memset(&port->mode_data, 0, sizeof(port->mode_data)); > > port->partner_ident.id_header = vdo; > - port->partner_ident.cert_stat = le32_to_cpu(payload[VDO_INDEX_CSTAT]); > + port->partner_ident.cert_stat = p[VDO_INDEX_CSTAT]; > port->partner_ident.product = product; > > typec_partner_set_identity(port->partner); > @@ -1000,17 +999,15 @@ static void svdm_consume_identity(struct tcpm_port *port, const __le32 *payload, > PD_PRODUCT_PID(product), product & 0xffff); > } > > -static bool svdm_consume_svids(struct tcpm_port *port, const __le32 *payload, > - int cnt) > +static bool svdm_consume_svids(struct tcpm_port *port, const u32 *p, int cnt) > { > struct pd_mode_data *pmdata = &port->mode_data; > int i; > > for (i = 1; i < cnt; i++) { > - u32 p = le32_to_cpu(payload[i]); > u16 svid; > > - svid = (p >> 16) & 0xffff; > + svid = (p[i] >> 16) & 0xffff; > if (!svid) > return false; > > @@ -1020,7 +1017,7 @@ static bool svdm_consume_svids(struct tcpm_port *port, const __le32 *payload, > pmdata->svids[pmdata->nsvids++] = svid; > tcpm_log(port, "SVID %d: 0x%x", pmdata->nsvids, svid); > > - svid = p & 0xffff; > + svid = p[i] & 0xffff; > if (!svid) > return false; > > @@ -1036,8 +1033,7 @@ static bool svdm_consume_svids(struct tcpm_port *port, const __le32 *payload, > return false; > } > > -static void svdm_consume_modes(struct tcpm_port *port, const __le32 *payload, > - int cnt) > +static void svdm_consume_modes(struct tcpm_port *port, const u32 *p, int cnt) > { > struct pd_mode_data *pmdata = &port->mode_data; > struct typec_altmode_desc *paltmode; > @@ -1054,7 +1050,7 @@ static void svdm_consume_modes(struct tcpm_port *port, const __le32 *payload, > > paltmode->svid = pmdata->svids[pmdata->svid_index]; > paltmode->mode = i; > - paltmode->vdo = le32_to_cpu(payload[i]); > + paltmode->vdo = p[i]; > > tcpm_log(port, " Alternate mode %d: SVID 0x%04x, VDO %d: 0x%08x", > pmdata->altmodes, paltmode->svid, > @@ -1082,21 +1078,17 @@ static void tcpm_register_partner_altmodes(struct tcpm_port *port) > > #define supports_modal(port) PD_IDH_MODAL_SUPP((port)->partner_ident.id_header) > > -static int tcpm_pd_svdm(struct tcpm_port *port, const __le32 *payload, int cnt, > +static int tcpm_pd_svdm(struct tcpm_port *port, const u32 *p, int cnt, > u32 *response) > { > struct typec_altmode *adev; > struct typec_altmode *pdev; > struct pd_mode_data *modep; > - u32 p[PD_MAX_PAYLOAD]; > int rlen = 0; > int cmd_type; > int cmd; > int i; > > - for (i = 0; i < cnt; i++) > - p[i] = le32_to_cpu(payload[i]); > - > cmd_type = PD_VDO_CMDT(p[0]); > cmd = PD_VDO_CMD(p[0]); > > @@ -1157,13 +1149,13 @@ static int tcpm_pd_svdm(struct tcpm_port *port, const __le32 *payload, int cnt, > switch (cmd) { > case CMD_DISCOVER_IDENT: > /* 6.4.4.3.1 */ > - svdm_consume_identity(port, payload, cnt); > + svdm_consume_identity(port, p, cnt); > response[0] = VDO(USB_SID_PD, 1, CMD_DISCOVER_SVID); > rlen = 1; > break; > case CMD_DISCOVER_SVID: > /* 6.4.4.3.2 */ > - if (svdm_consume_svids(port, payload, cnt)) { > + if (svdm_consume_svids(port, p, cnt)) { > response[0] = VDO(USB_SID_PD, 1, > CMD_DISCOVER_SVID); > rlen = 1; > @@ -1175,7 +1167,7 @@ static int tcpm_pd_svdm(struct tcpm_port *port, const __le32 *payload, int cnt, > break; > case CMD_DISCOVER_MODES: > /* 6.4.4.3.3 */ > - svdm_consume_modes(port, payload, cnt); > + svdm_consume_modes(port, p, cnt); > modep->svid_index++; > if (modep->svid_index < modep->nsvids) { > u16 svid = modep->svids[modep->svid_index]; > @@ -1238,15 +1230,18 @@ static int tcpm_pd_svdm(struct tcpm_port *port, const __le32 *payload, int cnt, > static void tcpm_handle_vdm_request(struct tcpm_port *port, > const __le32 *payload, int cnt) > { > - int rlen = 0; > + u32 p[PD_MAX_PAYLOAD]; > u32 response[8] = { }; > - u32 p0 = le32_to_cpu(payload[0]); > + int i, rlen = 0; > + > + for (i = 0; i < cnt; i++) > + p[i] = le32_to_cpu(payload[i]); > > if (port->vdm_state == VDM_STATE_BUSY) { > /* If UFP responded busy retry after timeout */ > - if (PD_VDO_CMDT(p0) == CMDT_RSP_BUSY) { > + if (PD_VDO_CMDT(p[0]) == CMDT_RSP_BUSY) { > port->vdm_state = VDM_STATE_WAIT_RSP_BUSY; > - port->vdo_retry = (p0 & ~VDO_CMDT_MASK) | > + port->vdo_retry = (p[0] & ~VDO_CMDT_MASK) | > CMDT_INIT; You could have changed that to: port->vdo_retry = (p[0] & ~VDO_CMDT_MASK) | CMDT_INIT; while at it. > mod_delayed_work(port->wq, &port->vdm_state_machine, > msecs_to_jiffies(PD_T_VDM_BUSY)); > @@ -1255,8 +1250,8 @@ static void tcpm_handle_vdm_request(struct tcpm_port *port, > port->vdm_state = VDM_STATE_DONE; > } > > - if (PD_VDO_SVDM(p0)) > - rlen = tcpm_pd_svdm(port, payload, cnt, response); > + if (PD_VDO_SVDM(p[0])) > + rlen = tcpm_pd_svdm(port, p, cnt, response); > > if (rlen > 0) > tcpm_queue_vdm(port, response[0], &response[1], rlen - 1); > -- > 2.26.2 thanks,
diff --git a/drivers/usb/typec/tcpm/tcpm.c b/drivers/usb/typec/tcpm/tcpm.c index 862c474b3ebd..ee239b54bcd8 100644 --- a/drivers/usb/typec/tcpm/tcpm.c +++ b/drivers/usb/typec/tcpm/tcpm.c @@ -981,16 +981,15 @@ static void tcpm_queue_vdm_unlocked(struct tcpm_port *port, const u32 header, mutex_unlock(&port->lock); } -static void svdm_consume_identity(struct tcpm_port *port, const __le32 *payload, - int cnt) +static void svdm_consume_identity(struct tcpm_port *port, const u32 *p, int cnt) { - u32 vdo = le32_to_cpu(payload[VDO_INDEX_IDH]); - u32 product = le32_to_cpu(payload[VDO_INDEX_PRODUCT]); + u32 vdo = p[VDO_INDEX_IDH]; + u32 product = p[VDO_INDEX_PRODUCT]; memset(&port->mode_data, 0, sizeof(port->mode_data)); port->partner_ident.id_header = vdo; - port->partner_ident.cert_stat = le32_to_cpu(payload[VDO_INDEX_CSTAT]); + port->partner_ident.cert_stat = p[VDO_INDEX_CSTAT]; port->partner_ident.product = product; typec_partner_set_identity(port->partner); @@ -1000,17 +999,15 @@ static void svdm_consume_identity(struct tcpm_port *port, const __le32 *payload, PD_PRODUCT_PID(product), product & 0xffff); } -static bool svdm_consume_svids(struct tcpm_port *port, const __le32 *payload, - int cnt) +static bool svdm_consume_svids(struct tcpm_port *port, const u32 *p, int cnt) { struct pd_mode_data *pmdata = &port->mode_data; int i; for (i = 1; i < cnt; i++) { - u32 p = le32_to_cpu(payload[i]); u16 svid; - svid = (p >> 16) & 0xffff; + svid = (p[i] >> 16) & 0xffff; if (!svid) return false; @@ -1020,7 +1017,7 @@ static bool svdm_consume_svids(struct tcpm_port *port, const __le32 *payload, pmdata->svids[pmdata->nsvids++] = svid; tcpm_log(port, "SVID %d: 0x%x", pmdata->nsvids, svid); - svid = p & 0xffff; + svid = p[i] & 0xffff; if (!svid) return false; @@ -1036,8 +1033,7 @@ static bool svdm_consume_svids(struct tcpm_port *port, const __le32 *payload, return false; } -static void svdm_consume_modes(struct tcpm_port *port, const __le32 *payload, - int cnt) +static void svdm_consume_modes(struct tcpm_port *port, const u32 *p, int cnt) { struct pd_mode_data *pmdata = &port->mode_data; struct typec_altmode_desc *paltmode; @@ -1054,7 +1050,7 @@ static void svdm_consume_modes(struct tcpm_port *port, const __le32 *payload, paltmode->svid = pmdata->svids[pmdata->svid_index]; paltmode->mode = i; - paltmode->vdo = le32_to_cpu(payload[i]); + paltmode->vdo = p[i]; tcpm_log(port, " Alternate mode %d: SVID 0x%04x, VDO %d: 0x%08x", pmdata->altmodes, paltmode->svid, @@ -1082,21 +1078,17 @@ static void tcpm_register_partner_altmodes(struct tcpm_port *port) #define supports_modal(port) PD_IDH_MODAL_SUPP((port)->partner_ident.id_header) -static int tcpm_pd_svdm(struct tcpm_port *port, const __le32 *payload, int cnt, +static int tcpm_pd_svdm(struct tcpm_port *port, const u32 *p, int cnt, u32 *response) { struct typec_altmode *adev; struct typec_altmode *pdev; struct pd_mode_data *modep; - u32 p[PD_MAX_PAYLOAD]; int rlen = 0; int cmd_type; int cmd; int i; - for (i = 0; i < cnt; i++) - p[i] = le32_to_cpu(payload[i]); - cmd_type = PD_VDO_CMDT(p[0]); cmd = PD_VDO_CMD(p[0]); @@ -1157,13 +1149,13 @@ static int tcpm_pd_svdm(struct tcpm_port *port, const __le32 *payload, int cnt, switch (cmd) { case CMD_DISCOVER_IDENT: /* 6.4.4.3.1 */ - svdm_consume_identity(port, payload, cnt); + svdm_consume_identity(port, p, cnt); response[0] = VDO(USB_SID_PD, 1, CMD_DISCOVER_SVID); rlen = 1; break; case CMD_DISCOVER_SVID: /* 6.4.4.3.2 */ - if (svdm_consume_svids(port, payload, cnt)) { + if (svdm_consume_svids(port, p, cnt)) { response[0] = VDO(USB_SID_PD, 1, CMD_DISCOVER_SVID); rlen = 1; @@ -1175,7 +1167,7 @@ static int tcpm_pd_svdm(struct tcpm_port *port, const __le32 *payload, int cnt, break; case CMD_DISCOVER_MODES: /* 6.4.4.3.3 */ - svdm_consume_modes(port, payload, cnt); + svdm_consume_modes(port, p, cnt); modep->svid_index++; if (modep->svid_index < modep->nsvids) { u16 svid = modep->svids[modep->svid_index]; @@ -1238,15 +1230,18 @@ static int tcpm_pd_svdm(struct tcpm_port *port, const __le32 *payload, int cnt, static void tcpm_handle_vdm_request(struct tcpm_port *port, const __le32 *payload, int cnt) { - int rlen = 0; + u32 p[PD_MAX_PAYLOAD]; u32 response[8] = { }; - u32 p0 = le32_to_cpu(payload[0]); + int i, rlen = 0; + + for (i = 0; i < cnt; i++) + p[i] = le32_to_cpu(payload[i]); if (port->vdm_state == VDM_STATE_BUSY) { /* If UFP responded busy retry after timeout */ - if (PD_VDO_CMDT(p0) == CMDT_RSP_BUSY) { + if (PD_VDO_CMDT(p[0]) == CMDT_RSP_BUSY) { port->vdm_state = VDM_STATE_WAIT_RSP_BUSY; - port->vdo_retry = (p0 & ~VDO_CMDT_MASK) | + port->vdo_retry = (p[0] & ~VDO_CMDT_MASK) | CMDT_INIT; mod_delayed_work(port->wq, &port->vdm_state_machine, msecs_to_jiffies(PD_T_VDM_BUSY)); @@ -1255,8 +1250,8 @@ static void tcpm_handle_vdm_request(struct tcpm_port *port, port->vdm_state = VDM_STATE_DONE; } - if (PD_VDO_SVDM(p0)) - rlen = tcpm_pd_svdm(port, payload, cnt, response); + if (PD_VDO_SVDM(p[0])) + rlen = tcpm_pd_svdm(port, p, cnt, response); if (rlen > 0) tcpm_queue_vdm(port, response[0], &response[1], rlen - 1);