[ Shuffling the recipients list ]

On 04/06/2019 09:20, Bjorn Andersson wrote:

> Acquire the device-reset GPIO and toggle this to reset the UFS device
> during initialization and host reset.
> 
> Signed-off-by: Bjorn Andersson <bjorn.anders...@linaro.org>
> ---
>  drivers/scsi/ufs/ufshcd.c | 44 +++++++++++++++++++++++++++++++++++++++
>  drivers/scsi/ufs/ufshcd.h |  4 ++++
>  2 files changed, 48 insertions(+)
> 
> diff --git a/drivers/scsi/ufs/ufshcd.c b/drivers/scsi/ufs/ufshcd.c
> index 8c1c551f2b42..951a0efee536 100644
> --- a/drivers/scsi/ufs/ufshcd.c
> +++ b/drivers/scsi/ufs/ufshcd.c
> @@ -42,6 +42,7 @@
>  #include <linux/nls.h>
>  #include <linux/of.h>
>  #include <linux/bitfield.h>
> +#include <linux/gpio/consumer.h>
>  #include "ufshcd.h"
>  #include "ufs_quirks.h"
>  #include "unipro.h"
> @@ -6104,6 +6105,25 @@ static int ufshcd_abort(struct scsi_cmnd *cmd)
>       return err;
>  }
>  
> +/**
> + ufshcd_device_reset() - toggle the (optional) device reset line
> + * @hba: per-adapter instance
> + *
> + * Toggles the (optional) reset line to reset the attached device.
> + */
> +static void ufshcd_device_reset(struct ufs_hba *hba)
> +{
> +     /*
> +      * The USB device shall detect reset pulses of 1us, sleep for 10us to
> +      * be on the safe side.
> +      */
> +     gpiod_set_value_cansleep(hba->device_reset, 1);
> +     usleep_range(10, 15);
> +
> +     gpiod_set_value_cansleep(hba->device_reset, 0);
> +     usleep_range(10, 15);
> +}
> +
>  /**
>   * ufshcd_host_reset_and_restore - reset and restore host controller
>   * @hba: per-adapter instance
> @@ -6159,6 +6179,9 @@ static int ufshcd_reset_and_restore(struct ufs_hba *hba)
>       int retries = MAX_HOST_RESET_RETRIES;
>  
>       do {
> +             /* Reset the attached device */
> +             ufshcd_device_reset(hba);
> +
>               err = ufshcd_host_reset_and_restore(hba);
>       } while (err && --retries);
>  
> @@ -7355,6 +7378,18 @@ static void ufshcd_variant_hba_exit(struct ufs_hba 
> *hba)
>       ufshcd_vops_exit(hba);
>  }
>  
> +static int ufshcd_init_device_reset(struct ufs_hba *hba)
> +{
> +     hba->device_reset = devm_gpiod_get_optional(hba->dev, "device-reset",
> +                                                 GPIOD_OUT_HIGH);
> +     if (IS_ERR(hba->device_reset)) {
> +             dev_err(hba->dev, "failed to acquire reset gpio: %ld\n",
> +                     PTR_ERR(hba->device_reset));
> +     }
> +
> +     return PTR_ERR_OR_ZERO(hba->device_reset);
> +}
> +
>  static int ufshcd_hba_init(struct ufs_hba *hba)
>  {
>       int err;
> @@ -7394,9 +7429,15 @@ static int ufshcd_hba_init(struct ufs_hba *hba)
>       if (err)
>               goto out_disable_vreg;
>  
> +     err = ufshcd_init_device_reset(hba);
> +     if (err)
> +             goto out_disable_variant;
> +
>       hba->is_powered = true;
>       goto out;
>  
> +out_disable_variant:
> +     ufshcd_vops_setup_regulators(hba, false);
>  out_disable_vreg:
>       ufshcd_setup_vreg(hba, false);
>  out_disable_clks:
> @@ -8290,6 +8331,9 @@ int ufshcd_init(struct ufs_hba *hba, void __iomem 
> *mmio_base, unsigned int irq)
>               goto exit_gating;
>       }
>  
> +     /* Reset the attached device */
> +     ufshcd_device_reset(hba);
> +
>       /* Host controller enable */
>       err = ufshcd_hba_enable(hba);
>       if (err) {
> diff --git a/drivers/scsi/ufs/ufshcd.h b/drivers/scsi/ufs/ufshcd.h
> index ecfa898b9ccc..d8be67742168 100644
> --- a/drivers/scsi/ufs/ufshcd.h
> +++ b/drivers/scsi/ufs/ufshcd.h
> @@ -72,6 +72,8 @@
>  #define UFSHCD "ufshcd"
>  #define UFSHCD_DRIVER_VERSION "0.2"
>  
> +struct gpio_desc;
> +
>  struct ufs_hba;
>  
>  enum dev_cmd_type {
> @@ -706,6 +708,8 @@ struct ufs_hba {
>  
>       struct device           bsg_dev;
>       struct request_queue    *bsg_queue;
> +
> +     struct gpio_desc *device_reset;
>  };
>  
>  /* Returns true if clocks can be gated. Otherwise false */
> 

Why is this needed on 845 and not on 8998?

On 8998 we already have:

                        resets = <&gcc GCC_UFS_BCR>;
                        reset-names = "rst";

The above reset line gets wiggled/frobbed when appropriate.

(What's the difference between gpio and pinctrl? vs a reset "clock" as above)

ufshcd_device_reset_ctrl() vs ufshcd_init_device_reset()

Sounds like the nomenclature could be unified or clarified.

Regards.

Reply via email to