Hi Olivier

On 12/15/22 13:51, Olivier Moysan wrote:
> Add support of offset and linear calibration for STM32MP15.
> The calibration is performed once at probe. The ADC is set in power on
> state for calibration. It remains in this state after calibration,
> to give to the kernel the opportunity to retrieve calibration data,
> directly from the ADC.
> 
> Signed-off-by: Olivier Moysan <olivier.moy...@foss.st.com>
> ---
> 
>  drivers/adc/stm32-adc.c | 134 ++++++++++++++++++++++++++++++++++------
>  1 file changed, 116 insertions(+), 18 deletions(-)
> 
> diff --git a/drivers/adc/stm32-adc.c b/drivers/adc/stm32-adc.c
> index 85efc119dbf1..1fba707c6f7d 100644
> --- a/drivers/adc/stm32-adc.c
> +++ b/drivers/adc/stm32-adc.c
> @@ -33,8 +33,11 @@
>  #define STM32H7_ADRDY                        BIT(0)
>  
>  /* STM32H7_ADC_CR - bit fields */
> +#define STM32H7_ADCAL                        BIT(31)
> +#define STM32H7_ADCALDIF             BIT(30)
>  #define STM32H7_DEEPPWD                      BIT(29)
>  #define STM32H7_ADVREGEN             BIT(28)
> +#define STM32H7_ADCALLIN             BIT(16)
>  #define STM32H7_BOOST                        BIT(8)
>  #define STM32H7_ADSTART                      BIT(2)
>  #define STM32H7_ADDIS                        BIT(1)
> @@ -65,47 +68,72 @@ struct stm32_adc {
>       const struct stm32_adc_cfg *cfg;
>  };
>  
> -static int stm32_adc_stop(struct udevice *dev)
> +static void stm32_adc_enter_pwr_down(struct udevice *dev)
>  {
>       struct stm32_adc *adc = dev_get_priv(dev);
>  
> -     setbits_le32(adc->regs + STM32H7_ADC_CR, STM32H7_ADDIS);
>       clrbits_le32(adc->regs + STM32H7_ADC_CR, STM32H7_BOOST);
>       /* Setting DEEPPWD disables ADC vreg and clears ADVREGEN */
>       setbits_le32(adc->regs + STM32H7_ADC_CR, STM32H7_DEEPPWD);
> -     adc->active_channel = -1;
> -
> -     return 0;
>  }
>  
> -static int stm32_adc_start_channel(struct udevice *dev, int channel)
> +static int stm32_adc_exit_pwr_down(struct udevice *dev)
>  {
> -     struct adc_uclass_plat *uc_pdata = dev_get_uclass_plat(dev);
>       struct stm32_adc_common *common = dev_get_priv(dev_get_parent(dev));
>       struct stm32_adc *adc = dev_get_priv(dev);
>       int ret;
>       u32 val;
>  
> +     /* return immediately if ADC is not in deep power down mode */
> +     if (!(readl(adc->regs + STM32H7_ADC_CR) & STM32H7_DEEPPWD))
> +             return 0;
> +
>       /* Exit deep power down, then enable ADC voltage regulator */
>       clrbits_le32(adc->regs + STM32H7_ADC_CR, STM32H7_DEEPPWD);
>       setbits_le32(adc->regs + STM32H7_ADC_CR, STM32H7_ADVREGEN);
> +
>       if (common->rate > STM32H7_BOOST_CLKRATE)
>               setbits_le32(adc->regs + STM32H7_ADC_CR, STM32H7_BOOST);
>  
>       /* Wait for startup time */
>       if (!adc->cfg->has_vregready) {
>               udelay(20);
> -     } else {
> -             ret = readl_poll_timeout(adc->regs + STM32H7_ADC_ISR, val,
> -                                      val & STM32MP1_VREGREADY,
> -                                      STM32_ADC_TIMEOUT_US);
> -             if (ret < 0) {
> -                     stm32_adc_stop(dev);
> -                     dev_err(dev, "Failed to enable vreg: %d\n", ret);
> -                     return ret;
> -             }
> +             return 0;
> +     }
> +
> +     ret = readl_poll_timeout(adc->regs + STM32H7_ADC_ISR, val,
> +                              val & STM32MP1_VREGREADY,
> +                              STM32_ADC_TIMEOUT_US);
> +     if (ret < 0) {
> +             stm32_adc_enter_pwr_down(dev);
> +             dev_err(dev, "Failed to enable vreg: %d\n", ret);
>       }
>  
> +     return ret;
> +}
> +
> +static int stm32_adc_stop(struct udevice *dev)
> +{
> +     struct stm32_adc *adc = dev_get_priv(dev);
> +
> +     setbits_le32(adc->regs + STM32H7_ADC_CR, STM32H7_ADDIS);
> +     stm32_adc_enter_pwr_down(dev);
> +     adc->active_channel = -1;
> +
> +     return 0;
> +}
> +
> +static int stm32_adc_start_channel(struct udevice *dev, int channel)
> +{
> +     struct adc_uclass_plat *uc_pdata = dev_get_uclass_plat(dev);
> +     struct stm32_adc *adc = dev_get_priv(dev);
> +     int ret;
> +     u32 val;
> +
> +     ret = stm32_adc_exit_pwr_down(dev);
> +     if (ret < 0)
> +             return ret;
> +
>       /* Only use single ended channels */
>       writel(0, adc->regs + STM32H7_ADC_DIFSEL);
>  
> @@ -162,6 +190,64 @@ static int stm32_adc_channel_data(struct udevice *dev, 
> int channel,
>       return 0;
>  }
>  
> +/**
> + * Fixed timeout value for ADC calibration.
> + * worst cases:
> + * - low clock frequency (0.12 MHz min)
> + * - maximum prescalers
> + * Calibration requires:
> + * - 16384 ADC clock cycle for the linear calibration
> + * - 20 ADC clock cycle for the offset calibration
> + *
> + * Set to 100ms for now
> + */
> +#define STM32H7_ADC_CALIB_TIMEOUT_US         100000
> +
> +static int stm32_adc_selfcalib(struct udevice *dev)
> +{
> +     struct stm32_adc *adc = dev_get_priv(dev);
> +     int ret;
> +     u32 val;
> +
> +     /*
> +      * Select calibration mode:
> +      * - Offset calibration for single ended inputs
> +      * - No linearity calibration. Done in next step.
> +      */
> +     clrbits_le32(adc->regs + STM32H7_ADC_CR, STM32H7_ADCALDIF | 
> STM32H7_ADCALLIN);
> +
> +     /* Start calibration, then wait for completion */
> +     setbits_le32(adc->regs + STM32H7_ADC_CR, STM32H7_ADCAL);
> +     ret = readl_poll_sleep_timeout(adc->regs + STM32H7_ADC_CR, val,
> +                                    !(val & STM32H7_ADCAL), 100,
> +                                    STM32H7_ADC_CALIB_TIMEOUT_US);
> +     if (ret) {
> +             dev_err(dev, "calibration failed\n");
> +             goto out;
> +     }
> +
> +     /*
> +      * Select calibration mode, then start calibration:
> +      * - Offset calibration for differential input
> +      * - Linearity calibration (needs to be done only once for single/diff)
> +      *   will run simultaneously with offset calibration.
> +      */
> +     setbits_le32(adc->regs + STM32H7_ADC_CR, STM32H7_ADCALDIF | 
> STM32H7_ADCALLIN);
> +
> +     /* Start calibration, then wait for completion */
> +     setbits_le32(adc->regs + STM32H7_ADC_CR, STM32H7_ADCAL);
> +     ret = readl_poll_sleep_timeout(adc->regs + STM32H7_ADC_CR, val,
> +                                    !(val & STM32H7_ADCAL), 100,
> +                                    STM32H7_ADC_CALIB_TIMEOUT_US);
> +     if (ret)
> +             dev_err(dev, "calibration failed\n");
> +
> +out:
> +     clrbits_le32(adc->regs + STM32H7_ADC_CR, STM32H7_ADCALDIF | 
> STM32H7_ADCALLIN);
> +
> +     return ret;
> +}
> +
>  static int stm32_adc_get_legacy_chan_count(struct udevice *dev)
>  {
>       int ret;
> @@ -272,7 +358,7 @@ static int stm32_adc_probe(struct udevice *dev)
>       struct adc_uclass_plat *uc_pdata = dev_get_uclass_plat(dev);
>       struct stm32_adc_common *common = dev_get_priv(dev_get_parent(dev));
>       struct stm32_adc *adc = dev_get_priv(dev);
> -     int offset;
> +     int offset, ret;
>  
>       offset = dev_read_u32_default(dev, "reg", -ENODATA);
>       if (offset < 0) {
> @@ -287,7 +373,19 @@ static int stm32_adc_probe(struct udevice *dev)
>       uc_pdata->vdd_microvolts = common->vref_uv;
>       uc_pdata->vss_microvolts = 0;
>  
> -     return stm32_adc_chan_of_init(dev);
> +     ret = stm32_adc_chan_of_init(dev);
> +     if (ret < 0)
> +             return ret;
> +
> +     ret = stm32_adc_exit_pwr_down(dev);
> +     if (ret < 0)
> +             return ret;
> +
> +     ret = stm32_adc_selfcalib(dev);
> +     if (ret)
> +             stm32_adc_enter_pwr_down(dev);
> +
> +     return ret;
>  }
>  
>  static const struct adc_ops stm32_adc_ops = {

Reviewed-by: Patrice Chotard <patrice.chot...@foss.st.com>

Thanks
Patrice

Reply via email to