Re: [PATCH v3 11/11] platform/chrome: cros_ec_typec: Register plug altmodes

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



On Mon, Nov 16, 2020 at 12:11:58PM -0800, Prashant Malani wrote:
> Modify the altmode registration (and unregistration) code so that it
> can be used by both partners and plugs.
> 
> Then, add code to register plug altmodes using the newly parameterized
> function. Also set the number of alternate modes for the plug using the
> associated Type C connector class function
> typec_plug_set_num_altmodes().
> 
> Signed-off-by: Prashant Malani <pmalani@xxxxxxxxxxxx>

Reviewed-by: Heikki Krogerus <heikki.krogerus@xxxxxxxxxxxxxxx>

> ---
> 
> 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 | 50 ++++++++++++++++++++-----
>  1 file changed, 40 insertions(+), 10 deletions(-)
> 
> diff --git a/drivers/platform/chrome/cros_ec_typec.c b/drivers/platform/chrome/cros_ec_typec.c
> index d2e154ae2362..65c5d0090ccd 100644
> --- a/drivers/platform/chrome/cros_ec_typec.c
> +++ b/drivers/platform/chrome/cros_ec_typec.c
> @@ -67,6 +67,7 @@ struct cros_typec_port {
>  	bool sop_prime_disc_done;
>  	struct ec_response_typec_discovery *disc_data;
>  	struct list_head partner_mode_list;
> +	struct list_head plug_mode_list;
>  };
>  
>  /* Platform-specific data for the Chrome OS EC Type C controller. */
> @@ -186,12 +187,15 @@ static int cros_typec_add_partner(struct cros_typec_data *typec, int port_num,
>  	return ret;
>  }
>  
> -static void cros_typec_unregister_altmodes(struct cros_typec_data *typec, int port_num)
> +static void cros_typec_unregister_altmodes(struct cros_typec_data *typec, int port_num,
> +					   bool is_partner)
>  {
>  	struct cros_typec_port *port = typec->ports[port_num];
>  	struct cros_typec_altmode_node *node, *tmp;
> +	struct list_head *head;
>  
> -	list_for_each_entry_safe(node, tmp, &port->partner_mode_list, list) {
> +	head = is_partner ? &port->partner_mode_list : &port->plug_mode_list;
> +	list_for_each_entry_safe(node, tmp, head, list) {
>  		list_del(&node->list);
>  		typec_unregister_altmode(node->amode);
>  		devm_kfree(typec->dev, node);
> @@ -203,7 +207,7 @@ static void cros_typec_remove_partner(struct cros_typec_data *typec,
>  {
>  	struct cros_typec_port *port = typec->ports[port_num];
>  
> -	cros_typec_unregister_altmodes(typec, port_num);
> +	cros_typec_unregister_altmodes(typec, port_num, true);
>  
>  	port->state.alt = NULL;
>  	port->state.mode = TYPEC_STATE_USB;
> @@ -224,6 +228,8 @@ static void cros_typec_remove_cable(struct cros_typec_data *typec,
>  {
>  	struct cros_typec_port *port = typec->ports[port_num];
>  
> +	cros_typec_unregister_altmodes(typec, port_num, false);
> +
>  	typec_unregister_plug(port->plug);
>  	port->plug = NULL;
>  	typec_unregister_cable(port->cable);
> @@ -352,6 +358,7 @@ static int cros_typec_init_ports(struct cros_typec_data *typec)
>  		}
>  
>  		INIT_LIST_HEAD(&cros_port->partner_mode_list);
> +		INIT_LIST_HEAD(&cros_port->plug_mode_list);
>  	}
>  
>  	return 0;
> @@ -639,7 +646,11 @@ static int cros_typec_get_mux_info(struct cros_typec_data *typec, int port_num,
>  				     sizeof(req), resp, sizeof(*resp));
>  }
>  
> -static int cros_typec_register_altmodes(struct cros_typec_data *typec, int port_num)
> +/*
> + * Helper function to register partner/plug altmodes.
> + */
> +static int cros_typec_register_altmodes(struct cros_typec_data *typec, int port_num,
> +					bool is_partner)
>  {
>  	struct cros_typec_port *port = typec->ports[port_num];
>  	struct ec_response_typec_discovery *sop_disc = port->disc_data;
> @@ -657,7 +668,11 @@ static int cros_typec_register_altmodes(struct cros_typec_data *typec, int port_
>  			desc.mode = j;
>  			desc.vdo = sop_disc->svids[i].mode_vdo[j];
>  
> -			amode = typec_partner_register_altmode(port->partner, &desc);
> +			if (is_partner)
> +				amode = typec_partner_register_altmode(port->partner, &desc);
> +			else
> +				amode = typec_plug_register_altmode(port->plug, &desc);
> +
>  			if (IS_ERR(amode)) {
>  				ret = PTR_ERR(amode);
>  				goto err_cleanup;
> @@ -672,21 +687,30 @@ static int cros_typec_register_altmodes(struct cros_typec_data *typec, int port_
>  			}
>  
>  			node->amode = amode;
> -			list_add_tail(&node->list, &port->partner_mode_list);
> +
> +			if (is_partner)
> +				list_add_tail(&node->list, &port->partner_mode_list);
> +			else
> +				list_add_tail(&node->list, &port->plug_mode_list);
>  			num_altmodes++;
>  		}
>  	}
>  
> -	ret = typec_partner_set_num_altmodes(port->partner, num_altmodes);
> +	if (is_partner)
> +		ret = typec_partner_set_num_altmodes(port->partner, num_altmodes);
> +	else
> +		ret = typec_plug_set_num_altmodes(port->plug, num_altmodes);
> +
>  	if (ret < 0) {
> -		dev_err(typec->dev, "Unable to set partner num_altmodes for port: %d\n", port_num);
> +		dev_err(typec->dev, "Unable to set %s num_altmodes for port: %d\n",
> +			is_partner ? "partner" : "plug", port_num);
>  		goto err_cleanup;
>  	}
>  
>  	return 0;
>  
>  err_cleanup:
> -	cros_typec_unregister_altmodes(typec, port_num);
> +	cros_typec_unregister_altmodes(typec, port_num, is_partner);
>  	return ret;
>  }
>  
> @@ -774,6 +798,12 @@ static int cros_typec_handle_sop_prime_disc(struct cros_typec_data *typec, int p
>  		goto sop_prime_disc_exit;
>  	}
>  
> +	ret = cros_typec_register_altmodes(typec, port_num, false);
> +	if (ret < 0) {
> +		dev_err(typec->dev, "Failed to register plug altmodes, port: %d\n", port_num);
> +		goto sop_prime_disc_exit;
> +	}
> +
>  	return 0;
>  
>  sop_prime_disc_exit:
> @@ -815,7 +845,7 @@ static int cros_typec_handle_sop_disc(struct cros_typec_data *typec, int port_nu
>  		goto disc_exit;
>  	}
>  
> -	ret = cros_typec_register_altmodes(typec, port_num);
> +	ret = cros_typec_register_altmodes(typec, port_num, true);
>  	if (ret < 0) {
>  		dev_err(typec->dev, "Failed to register partner altmodes, port: %d\n", port_num);
>  		goto disc_exit;
> -- 
> 2.29.2.299.gdc1121823c-goog

thanks,

-- 
heikki



[Index of Archives]     [Linux Media]     [Linux Input]     [Linux Audio Users]     [Yosemite News]     [Linux Kernel]     [Linux SCSI]     [Old Linux USB Devel Archive]

  Powered by Linux