2
0
mirror of https://github.com/edk2-porting/linux-next.git synced 2025-01-24 14:45:12 +08:00

iio: adc: sc27xx: Change to polling mode to read data

On Spreadtrum platform, the headphone will read one ADC channel multiple
times to identify the headphone type, and the headphone identification is
sensitive of the ADC reading time. And we found it will take longer time
to reading ADC data by using interrupt mode comparing with the polling
mode, thus we should change to polling mode to improve the efficiency
of reading data, which can identify the headphone type successfully.

Signed-off-by: Freeman Liu <freeman.liu@unisoc.com>
Signed-off-by: Baolin Wang <baolin.wang@linaro.org>
Signed-off-by: Jonathan Cameron <Jonathan.Cameron@huawei.com>
This commit is contained in:
Freeman Liu 2019-07-25 14:33:50 +08:00 committed by Jonathan Cameron
parent 473d12f763
commit 8de877d2bb

View File

@ -3,7 +3,6 @@
#include <linux/hwspinlock.h> #include <linux/hwspinlock.h>
#include <linux/iio/iio.h> #include <linux/iio/iio.h>
#include <linux/interrupt.h>
#include <linux/module.h> #include <linux/module.h>
#include <linux/nvmem-consumer.h> #include <linux/nvmem-consumer.h>
#include <linux/of.h> #include <linux/of.h>
@ -46,14 +45,18 @@
/* Bits definitions for SC27XX_ADC_INT_CLR registers */ /* Bits definitions for SC27XX_ADC_INT_CLR registers */
#define SC27XX_ADC_IRQ_CLR BIT(0) #define SC27XX_ADC_IRQ_CLR BIT(0)
/* Bits definitions for SC27XX_ADC_INT_RAW registers */
#define SC27XX_ADC_IRQ_RAW BIT(0)
/* Mask definition for SC27XX_ADC_DATA register */ /* Mask definition for SC27XX_ADC_DATA register */
#define SC27XX_ADC_DATA_MASK GENMASK(11, 0) #define SC27XX_ADC_DATA_MASK GENMASK(11, 0)
/* Timeout (ms) for the trylock of hardware spinlocks */ /* Timeout (ms) for the trylock of hardware spinlocks */
#define SC27XX_ADC_HWLOCK_TIMEOUT 5000 #define SC27XX_ADC_HWLOCK_TIMEOUT 5000
/* Timeout (ms) for ADC data conversion according to ADC datasheet */ /* Timeout (us) for ADC data conversion according to ADC datasheet */
#define SC27XX_ADC_RDY_TIMEOUT 100 #define SC27XX_ADC_RDY_TIMEOUT 1000000
#define SC27XX_ADC_POLL_RAW_STATUS 500
/* Maximum ADC channel number */ /* Maximum ADC channel number */
#define SC27XX_ADC_CHANNEL_MAX 32 #define SC27XX_ADC_CHANNEL_MAX 32
@ -72,10 +75,8 @@ struct sc27xx_adc_data {
* subsystems which will access the unique ADC controller. * subsystems which will access the unique ADC controller.
*/ */
struct hwspinlock *hwlock; struct hwspinlock *hwlock;
struct completion completion;
int channel_scale[SC27XX_ADC_CHANNEL_MAX]; int channel_scale[SC27XX_ADC_CHANNEL_MAX];
u32 base; u32 base;
int value;
int irq; int irq;
}; };
@ -188,9 +189,7 @@ static int sc27xx_adc_read(struct sc27xx_adc_data *data, int channel,
int scale, int *val) int scale, int *val)
{ {
int ret; int ret;
u32 tmp; u32 tmp, value, status;
reinit_completion(&data->completion);
ret = hwspin_lock_timeout_raw(data->hwlock, SC27XX_ADC_HWLOCK_TIMEOUT); ret = hwspin_lock_timeout_raw(data->hwlock, SC27XX_ADC_HWLOCK_TIMEOUT);
if (ret) { if (ret) {
@ -203,6 +202,11 @@ static int sc27xx_adc_read(struct sc27xx_adc_data *data, int channel,
if (ret) if (ret)
goto unlock_adc; goto unlock_adc;
ret = regmap_update_bits(data->regmap, data->base + SC27XX_ADC_INT_CLR,
SC27XX_ADC_IRQ_CLR, SC27XX_ADC_IRQ_CLR);
if (ret)
goto disable_adc;
/* Configure the channel id and scale */ /* Configure the channel id and scale */
tmp = (scale << SC27XX_ADC_SCALE_SHIFT) & SC27XX_ADC_SCALE_MASK; tmp = (scale << SC27XX_ADC_SCALE_SHIFT) & SC27XX_ADC_SCALE_MASK;
tmp |= channel & SC27XX_ADC_CHN_ID_MASK; tmp |= channel & SC27XX_ADC_CHN_ID_MASK;
@ -226,15 +230,22 @@ static int sc27xx_adc_read(struct sc27xx_adc_data *data, int channel,
if (ret) if (ret)
goto disable_adc; goto disable_adc;
ret = wait_for_completion_timeout(&data->completion, ret = regmap_read_poll_timeout(data->regmap,
msecs_to_jiffies(SC27XX_ADC_RDY_TIMEOUT)); data->base + SC27XX_ADC_INT_RAW,
if (!ret) { status, (status & SC27XX_ADC_IRQ_RAW),
dev_err(data->dev, "read ADC data timeout\n"); SC27XX_ADC_POLL_RAW_STATUS,
ret = -ETIMEDOUT; SC27XX_ADC_RDY_TIMEOUT);
} else { if (ret) {
ret = 0; dev_err(data->dev, "read adc timeout, status = 0x%x\n", status);
goto disable_adc;
} }
ret = regmap_read(data->regmap, data->base + SC27XX_ADC_DATA, &value);
if (ret)
goto disable_adc;
value &= SC27XX_ADC_DATA_MASK;
disable_adc: disable_adc:
regmap_update_bits(data->regmap, data->base + SC27XX_ADC_CTL, regmap_update_bits(data->regmap, data->base + SC27XX_ADC_CTL,
SC27XX_ADC_EN, 0); SC27XX_ADC_EN, 0);
@ -242,32 +253,11 @@ unlock_adc:
hwspin_unlock_raw(data->hwlock); hwspin_unlock_raw(data->hwlock);
if (!ret) if (!ret)
*val = data->value; *val = value;
return ret; return ret;
} }
static irqreturn_t sc27xx_adc_isr(int irq, void *dev_id)
{
struct sc27xx_adc_data *data = dev_id;
int ret;
ret = regmap_update_bits(data->regmap, data->base + SC27XX_ADC_INT_CLR,
SC27XX_ADC_IRQ_CLR, SC27XX_ADC_IRQ_CLR);
if (ret)
return IRQ_RETVAL(ret);
ret = regmap_read(data->regmap, data->base + SC27XX_ADC_DATA,
&data->value);
if (ret)
return IRQ_RETVAL(ret);
data->value &= SC27XX_ADC_DATA_MASK;
complete(&data->completion);
return IRQ_HANDLED;
}
static void sc27xx_adc_volt_ratio(struct sc27xx_adc_data *data, static void sc27xx_adc_volt_ratio(struct sc27xx_adc_data *data,
int channel, int scale, int channel, int scale,
u32 *div_numerator, u32 *div_denominator) u32 *div_numerator, u32 *div_denominator)
@ -454,11 +444,6 @@ static int sc27xx_adc_enable(struct sc27xx_adc_data *data)
if (ret) if (ret)
goto disable_adc; goto disable_adc;
ret = regmap_update_bits(data->regmap, data->base + SC27XX_ADC_INT_EN,
SC27XX_ADC_IRQ_EN, SC27XX_ADC_IRQ_EN);
if (ret)
goto disable_clk;
/* ADC channel scales' calibration from nvmem device */ /* ADC channel scales' calibration from nvmem device */
ret = sc27xx_adc_scale_calibration(data, true); ret = sc27xx_adc_scale_calibration(data, true);
if (ret) if (ret)
@ -484,9 +469,6 @@ static void sc27xx_adc_disable(void *_data)
{ {
struct sc27xx_adc_data *data = _data; struct sc27xx_adc_data *data = _data;
regmap_update_bits(data->regmap, data->base + SC27XX_ADC_INT_EN,
SC27XX_ADC_IRQ_EN, 0);
/* Disable ADC work clock and controller clock */ /* Disable ADC work clock and controller clock */
regmap_update_bits(data->regmap, SC27XX_ARM_CLK_EN, regmap_update_bits(data->regmap, SC27XX_ARM_CLK_EN,
SC27XX_CLK_ADC_EN | SC27XX_CLK_ADC_CLK_EN, 0); SC27XX_CLK_ADC_EN | SC27XX_CLK_ADC_CLK_EN, 0);
@ -551,7 +533,6 @@ static int sc27xx_adc_probe(struct platform_device *pdev)
return ret; return ret;
} }
init_completion(&sc27xx_data->completion);
sc27xx_data->dev = dev; sc27xx_data->dev = dev;
ret = sc27xx_adc_enable(sc27xx_data); ret = sc27xx_adc_enable(sc27xx_data);
@ -566,14 +547,6 @@ static int sc27xx_adc_probe(struct platform_device *pdev)
return ret; return ret;
} }
ret = devm_request_threaded_irq(dev, sc27xx_data->irq, NULL,
sc27xx_adc_isr, IRQF_ONESHOT,
pdev->name, sc27xx_data);
if (ret) {
dev_err(dev, "failed to request ADC irq\n");
return ret;
}
indio_dev->dev.parent = dev; indio_dev->dev.parent = dev;
indio_dev->name = dev_name(dev); indio_dev->name = dev_name(dev);
indio_dev->modes = INDIO_DIRECT_MODE; indio_dev->modes = INDIO_DIRECT_MODE;