Message ID | 20201116201150.2919178-11-pmalani@chromium.org |
---|---|
State | New |
Headers | show |
Series | chrome/platform: cros_ec_typec: Register cables, partner altmodes and plug altmodes | expand |
On Mon, Nov 16, 2020 at 12:11:56PM -0800, Prashant Malani wrote: > In order to register cable alternate modes, we need to first register a > plug object. Use the Type C connector class framework to register a SOP' > plug for this purpose. > > Since a cable and plug go hand in hand, we can handle the registration > and removal together. > > Signed-off-by: Prashant Malani <pmalani@chromium.org> FWIW: Reviewed-by: Heikki Krogerus <heikki.krogerus@linux.intel.com> > --- > > Changes in v3: > - Re-arranged patch order and combined it with related series of > patches. > > No version v2. > > drivers/platform/chrome/cros_ec_typec.c | 35 ++++++++++++++++++------- > 1 file changed, 26 insertions(+), 9 deletions(-) > > diff --git a/drivers/platform/chrome/cros_ec_typec.c b/drivers/platform/chrome/cros_ec_typec.c > index ad5e37bfd45d..d2e154ae2362 100644 > --- a/drivers/platform/chrome/cros_ec_typec.c > +++ b/drivers/platform/chrome/cros_ec_typec.c > @@ -45,6 +45,8 @@ struct cros_typec_port { > struct typec_capability caps; > struct typec_partner *partner; > struct typec_cable *cable; > + /* SOP' plug. */ > + struct typec_plug *plug; > /* Port partner PD identity info. */ > struct usb_pd_identity p_identity; > /* Port cable PD identity info. */ > @@ -222,6 +224,8 @@ static void cros_typec_remove_cable(struct cros_typec_data *typec, > { > struct cros_typec_port *port = typec->ports[port_num]; > > + typec_unregister_plug(port->plug); > + port->plug = NULL; > typec_unregister_cable(port->cable); > port->cable = NULL; > memset(&port->c_identity, 0, sizeof(port->c_identity)); > @@ -712,7 +716,8 @@ static int cros_typec_handle_sop_prime_disc(struct cros_typec_data *typec, int p > { > struct cros_typec_port *port = typec->ports[port_num]; > struct ec_response_typec_discovery *disc = port->disc_data; > - struct typec_cable_desc desc = {}; > + struct typec_cable_desc c_desc = {}; > + struct typec_plug_desc p_desc; > struct ec_params_typec_discovery req = { > .port = port_num, > .partner_type = TYPEC_PARTNER_SOP_PRIME, > @@ -735,32 +740,44 @@ static int cros_typec_handle_sop_prime_disc(struct cros_typec_data *typec, int p > cable_plug_type = VDO_TYPEC_CABLE_TYPE(port->c_identity.vdo[0]); > switch (cable_plug_type) { > case CABLE_ATYPE: > - desc.type = USB_PLUG_TYPE_A; > + c_desc.type = USB_PLUG_TYPE_A; > break; > case CABLE_BTYPE: > - desc.type = USB_PLUG_TYPE_B; > + c_desc.type = USB_PLUG_TYPE_B; > break; > case CABLE_CTYPE: > - desc.type = USB_PLUG_TYPE_C; > + c_desc.type = USB_PLUG_TYPE_C; > break; > case CABLE_CAPTIVE: > - desc.type = USB_PLUG_CAPTIVE; > + c_desc.type = USB_PLUG_CAPTIVE; > break; > default: > - desc.type = USB_PLUG_NONE; > + c_desc.type = USB_PLUG_NONE; > } > - desc.active = PD_IDH_PTYPE(port->c_identity.id_header) == IDH_PTYPE_ACABLE; > + c_desc.active = PD_IDH_PTYPE(port->c_identity.id_header) == IDH_PTYPE_ACABLE; > } > > - desc.identity = &port->c_identity; > + c_desc.identity = &port->c_identity; > > - port->cable = typec_register_cable(port->port, &desc); > + port->cable = typec_register_cable(port->port, &c_desc); > if (IS_ERR(port->cable)) { > ret = PTR_ERR(port->cable); > port->cable = NULL; > + goto sop_prime_disc_exit; > + } > + > + p_desc.index = TYPEC_PLUG_SOP_P; > + port->plug = typec_register_plug(port->cable, &p_desc); > + if (IS_ERR(port->plug)) { > + ret = PTR_ERR(port->plug); > + port->plug = NULL; > + goto sop_prime_disc_exit; > } > > + return 0; > + > sop_prime_disc_exit: > + cros_typec_remove_cable(typec, port_num); > return ret; > } > > -- > 2.29.2.299.gdc1121823c-goog
diff --git a/drivers/platform/chrome/cros_ec_typec.c b/drivers/platform/chrome/cros_ec_typec.c index ad5e37bfd45d..d2e154ae2362 100644 --- a/drivers/platform/chrome/cros_ec_typec.c +++ b/drivers/platform/chrome/cros_ec_typec.c @@ -45,6 +45,8 @@ struct cros_typec_port { struct typec_capability caps; struct typec_partner *partner; struct typec_cable *cable; + /* SOP' plug. */ + struct typec_plug *plug; /* Port partner PD identity info. */ struct usb_pd_identity p_identity; /* Port cable PD identity info. */ @@ -222,6 +224,8 @@ static void cros_typec_remove_cable(struct cros_typec_data *typec, { struct cros_typec_port *port = typec->ports[port_num]; + typec_unregister_plug(port->plug); + port->plug = NULL; typec_unregister_cable(port->cable); port->cable = NULL; memset(&port->c_identity, 0, sizeof(port->c_identity)); @@ -712,7 +716,8 @@ static int cros_typec_handle_sop_prime_disc(struct cros_typec_data *typec, int p { struct cros_typec_port *port = typec->ports[port_num]; struct ec_response_typec_discovery *disc = port->disc_data; - struct typec_cable_desc desc = {}; + struct typec_cable_desc c_desc = {}; + struct typec_plug_desc p_desc; struct ec_params_typec_discovery req = { .port = port_num, .partner_type = TYPEC_PARTNER_SOP_PRIME, @@ -735,32 +740,44 @@ static int cros_typec_handle_sop_prime_disc(struct cros_typec_data *typec, int p cable_plug_type = VDO_TYPEC_CABLE_TYPE(port->c_identity.vdo[0]); switch (cable_plug_type) { case CABLE_ATYPE: - desc.type = USB_PLUG_TYPE_A; + c_desc.type = USB_PLUG_TYPE_A; break; case CABLE_BTYPE: - desc.type = USB_PLUG_TYPE_B; + c_desc.type = USB_PLUG_TYPE_B; break; case CABLE_CTYPE: - desc.type = USB_PLUG_TYPE_C; + c_desc.type = USB_PLUG_TYPE_C; break; case CABLE_CAPTIVE: - desc.type = USB_PLUG_CAPTIVE; + c_desc.type = USB_PLUG_CAPTIVE; break; default: - desc.type = USB_PLUG_NONE; + c_desc.type = USB_PLUG_NONE; } - desc.active = PD_IDH_PTYPE(port->c_identity.id_header) == IDH_PTYPE_ACABLE; + c_desc.active = PD_IDH_PTYPE(port->c_identity.id_header) == IDH_PTYPE_ACABLE; } - desc.identity = &port->c_identity; + c_desc.identity = &port->c_identity; - port->cable = typec_register_cable(port->port, &desc); + port->cable = typec_register_cable(port->port, &c_desc); if (IS_ERR(port->cable)) { ret = PTR_ERR(port->cable); port->cable = NULL; + goto sop_prime_disc_exit; + } + + p_desc.index = TYPEC_PLUG_SOP_P; + port->plug = typec_register_plug(port->cable, &p_desc); + if (IS_ERR(port->plug)) { + ret = PTR_ERR(port->plug); + port->plug = NULL; + goto sop_prime_disc_exit; } + return 0; + sop_prime_disc_exit: + cros_typec_remove_cable(typec, port_num); return ret; }
In order to register cable alternate modes, we need to first register a plug object. Use the Type C connector class framework to register a SOP' plug for this purpose. Since a cable and plug go hand in hand, we can handle the registration and removal together. Signed-off-by: Prashant Malani <pmalani@chromium.org> --- Changes in v3: - Re-arranged patch order and combined it with related series of patches. No version v2. drivers/platform/chrome/cros_ec_typec.c | 35 ++++++++++++++++++------- 1 file changed, 26 insertions(+), 9 deletions(-)