Re: [PATCH 5/5] usb: typec: ucsi: Clear UCSI_CCI_RESET_COMPLETE before reset
From: Heikki Krogerus
Date: Fri Mar 22 2024 - 06:07:02 EST
On Wed, Mar 20, 2024 at 08:39:26AM +0100, Christian A. Ehrhardt wrote:
> Check the UCSI_CCI_RESET_COMPLETE complete flag before starting
> another reset. Use a UCSI_SET_NOTIFICATION_ENABLE command to clear
> the flag if it is set.
>
> Signed-off-by: Christian A. Ehrhardt <lk@xxxxxxx>
Reviewed-by: Heikki Krogerus <heikki.krogerus@xxxxxxxxxxxxxxx>
> ---
> drivers/usb/typec/ucsi/ucsi.c | 36 ++++++++++++++++++++++++++++++++++-
> 1 file changed, 35 insertions(+), 1 deletion(-)
>
> diff --git a/drivers/usb/typec/ucsi/ucsi.c b/drivers/usb/typec/ucsi/ucsi.c
> index 63f340dbd867..85e507df7fa8 100644
> --- a/drivers/usb/typec/ucsi/ucsi.c
> +++ b/drivers/usb/typec/ucsi/ucsi.c
> @@ -1264,13 +1264,47 @@ static int ucsi_reset_connector(struct ucsi_connector *con, bool hard)
>
> static int ucsi_reset_ppm(struct ucsi *ucsi)
> {
> - u64 command = UCSI_PPM_RESET;
> + u64 command;
> unsigned long tmo;
> u32 cci;
> int ret;
>
> mutex_lock(&ucsi->ppm_lock);
>
> + ret = ucsi->ops->read(ucsi, UCSI_CCI, &cci, sizeof(cci));
> + if (ret < 0)
> + goto out;
> +
> + /*
> + * If UCSI_CCI_RESET_COMPLETE is already set we must clear
> + * the flag before we start another reset. Send a
> + * UCSI_SET_NOTIFICATION_ENABLE command to achieve this.
> + * Ignore a timeout and try the reset anyway if this fails.
> + */
> + if (cci & UCSI_CCI_RESET_COMPLETE) {
> + command = UCSI_SET_NOTIFICATION_ENABLE;
> + ret = ucsi->ops->async_write(ucsi, UCSI_CONTROL, &command,
> + sizeof(command));
> + if (ret < 0)
> + goto out;
> +
> + tmo = jiffies + msecs_to_jiffies(UCSI_TIMEOUT_MS);
> + do {
> + ret = ucsi->ops->read(ucsi, UCSI_CCI,
> + &cci, sizeof(cci));
> + if (ret < 0)
> + goto out;
> + if (cci & UCSI_CCI_COMMAND_COMPLETE)
> + break;
> + if (time_is_before_jiffies(tmo))
> + break;
> + msleep(20);
> + } while (1);
> +
> + WARN_ON(cci & UCSI_CCI_RESET_COMPLETE);
> + }
> +
> + command = UCSI_PPM_RESET;
> ret = ucsi->ops->async_write(ucsi, UCSI_CONTROL, &command,
> sizeof(command));
> if (ret < 0)
> --
> 2.40.1
--
heikki