mirror of https://gitee.com/openkylin/linux.git
iio: adc: rockchip_saradc: Add support iio buffers
Add the ability to also support access via (triggered) buffers next to the existing direct mode. Device in question is the Odroid Go Advance that connects a joystick to two of the saradc channels for X and Y axis and the new (and still pending) adc joystick driver of course wants to use triggered buffers from the iio subsystem. Signed-off-by: Simon Xue <xxm@rock-chips.com> [some simplifications and added commit description] Signed-off-by: Heiko Stuebner <heiko.stuebner@theobroma-systems.com> Signed-off-by: Jonathan Cameron <Jonathan.Cameron@huawei.com>
This commit is contained in:
parent
71eb7c855b
commit
4e130dc7b4
|
@ -15,7 +15,10 @@
|
||||||
#include <linux/delay.h>
|
#include <linux/delay.h>
|
||||||
#include <linux/reset.h>
|
#include <linux/reset.h>
|
||||||
#include <linux/regulator/consumer.h>
|
#include <linux/regulator/consumer.h>
|
||||||
|
#include <linux/iio/buffer.h>
|
||||||
#include <linux/iio/iio.h>
|
#include <linux/iio/iio.h>
|
||||||
|
#include <linux/iio/trigger_consumer.h>
|
||||||
|
#include <linux/iio/triggered_buffer.h>
|
||||||
|
|
||||||
#define SARADC_DATA 0x00
|
#define SARADC_DATA 0x00
|
||||||
|
|
||||||
|
@ -32,9 +35,9 @@
|
||||||
#define SARADC_DLY_PU_SOC_MASK 0x3f
|
#define SARADC_DLY_PU_SOC_MASK 0x3f
|
||||||
|
|
||||||
#define SARADC_TIMEOUT msecs_to_jiffies(100)
|
#define SARADC_TIMEOUT msecs_to_jiffies(100)
|
||||||
|
#define SARADC_MAX_CHANNELS 6
|
||||||
|
|
||||||
struct rockchip_saradc_data {
|
struct rockchip_saradc_data {
|
||||||
int num_bits;
|
|
||||||
const struct iio_chan_spec *channels;
|
const struct iio_chan_spec *channels;
|
||||||
int num_channels;
|
int num_channels;
|
||||||
unsigned long clk_rate;
|
unsigned long clk_rate;
|
||||||
|
@ -49,8 +52,37 @@ struct rockchip_saradc {
|
||||||
struct reset_control *reset;
|
struct reset_control *reset;
|
||||||
const struct rockchip_saradc_data *data;
|
const struct rockchip_saradc_data *data;
|
||||||
u16 last_val;
|
u16 last_val;
|
||||||
|
const struct iio_chan_spec *last_chan;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static void rockchip_saradc_power_down(struct rockchip_saradc *info)
|
||||||
|
{
|
||||||
|
/* Clear irq & power down adc */
|
||||||
|
writel_relaxed(0, info->regs + SARADC_CTRL);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int rockchip_saradc_conversion(struct rockchip_saradc *info,
|
||||||
|
struct iio_chan_spec const *chan)
|
||||||
|
{
|
||||||
|
reinit_completion(&info->completion);
|
||||||
|
|
||||||
|
/* 8 clock periods as delay between power up and start cmd */
|
||||||
|
writel_relaxed(8, info->regs + SARADC_DLY_PU_SOC);
|
||||||
|
|
||||||
|
info->last_chan = chan;
|
||||||
|
|
||||||
|
/* Select the channel to be used and trigger conversion */
|
||||||
|
writel(SARADC_CTRL_POWER_CTRL
|
||||||
|
| (chan->channel & SARADC_CTRL_CHN_MASK)
|
||||||
|
| SARADC_CTRL_IRQ_ENABLE,
|
||||||
|
info->regs + SARADC_CTRL);
|
||||||
|
|
||||||
|
if (!wait_for_completion_timeout(&info->completion, SARADC_TIMEOUT))
|
||||||
|
return -ETIMEDOUT;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
static int rockchip_saradc_read_raw(struct iio_dev *indio_dev,
|
static int rockchip_saradc_read_raw(struct iio_dev *indio_dev,
|
||||||
struct iio_chan_spec const *chan,
|
struct iio_chan_spec const *chan,
|
||||||
int *val, int *val2, long mask)
|
int *val, int *val2, long mask)
|
||||||
|
@ -62,22 +94,11 @@ static int rockchip_saradc_read_raw(struct iio_dev *indio_dev,
|
||||||
case IIO_CHAN_INFO_RAW:
|
case IIO_CHAN_INFO_RAW:
|
||||||
mutex_lock(&indio_dev->mlock);
|
mutex_lock(&indio_dev->mlock);
|
||||||
|
|
||||||
reinit_completion(&info->completion);
|
ret = rockchip_saradc_conversion(info, chan);
|
||||||
|
if (ret) {
|
||||||
/* 8 clock periods as delay between power up and start cmd */
|
rockchip_saradc_power_down(info);
|
||||||
writel_relaxed(8, info->regs + SARADC_DLY_PU_SOC);
|
|
||||||
|
|
||||||
/* Select the channel to be used and trigger conversion */
|
|
||||||
writel(SARADC_CTRL_POWER_CTRL
|
|
||||||
| (chan->channel & SARADC_CTRL_CHN_MASK)
|
|
||||||
| SARADC_CTRL_IRQ_ENABLE,
|
|
||||||
info->regs + SARADC_CTRL);
|
|
||||||
|
|
||||||
if (!wait_for_completion_timeout(&info->completion,
|
|
||||||
SARADC_TIMEOUT)) {
|
|
||||||
writel_relaxed(0, info->regs + SARADC_CTRL);
|
|
||||||
mutex_unlock(&indio_dev->mlock);
|
mutex_unlock(&indio_dev->mlock);
|
||||||
return -ETIMEDOUT;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
*val = info->last_val;
|
*val = info->last_val;
|
||||||
|
@ -91,7 +112,7 @@ static int rockchip_saradc_read_raw(struct iio_dev *indio_dev,
|
||||||
}
|
}
|
||||||
|
|
||||||
*val = ret / 1000;
|
*val = ret / 1000;
|
||||||
*val2 = info->data->num_bits;
|
*val2 = chan->scan_type.realbits;
|
||||||
return IIO_VAL_FRACTIONAL_LOG2;
|
return IIO_VAL_FRACTIONAL_LOG2;
|
||||||
default:
|
default:
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
@ -104,10 +125,9 @@ static irqreturn_t rockchip_saradc_isr(int irq, void *dev_id)
|
||||||
|
|
||||||
/* Read value */
|
/* Read value */
|
||||||
info->last_val = readl_relaxed(info->regs + SARADC_DATA);
|
info->last_val = readl_relaxed(info->regs + SARADC_DATA);
|
||||||
info->last_val &= GENMASK(info->data->num_bits - 1, 0);
|
info->last_val &= GENMASK(info->last_chan->scan_type.realbits - 1, 0);
|
||||||
|
|
||||||
/* Clear irq & power down adc */
|
rockchip_saradc_power_down(info);
|
||||||
writel_relaxed(0, info->regs + SARADC_CTRL);
|
|
||||||
|
|
||||||
complete(&info->completion);
|
complete(&info->completion);
|
||||||
|
|
||||||
|
@ -118,51 +138,55 @@ static const struct iio_info rockchip_saradc_iio_info = {
|
||||||
.read_raw = rockchip_saradc_read_raw,
|
.read_raw = rockchip_saradc_read_raw,
|
||||||
};
|
};
|
||||||
|
|
||||||
#define SARADC_CHANNEL(_index, _id) { \
|
#define SARADC_CHANNEL(_index, _id, _res) { \
|
||||||
.type = IIO_VOLTAGE, \
|
.type = IIO_VOLTAGE, \
|
||||||
.indexed = 1, \
|
.indexed = 1, \
|
||||||
.channel = _index, \
|
.channel = _index, \
|
||||||
.info_mask_separate = BIT(IIO_CHAN_INFO_RAW), \
|
.info_mask_separate = BIT(IIO_CHAN_INFO_RAW), \
|
||||||
.info_mask_shared_by_type = BIT(IIO_CHAN_INFO_SCALE), \
|
.info_mask_shared_by_type = BIT(IIO_CHAN_INFO_SCALE), \
|
||||||
.datasheet_name = _id, \
|
.datasheet_name = _id, \
|
||||||
|
.scan_index = _index, \
|
||||||
|
.scan_type = { \
|
||||||
|
.sign = 'u', \
|
||||||
|
.realbits = _res, \
|
||||||
|
.storagebits = 16, \
|
||||||
|
.endianness = IIO_CPU, \
|
||||||
|
}, \
|
||||||
}
|
}
|
||||||
|
|
||||||
static const struct iio_chan_spec rockchip_saradc_iio_channels[] = {
|
static const struct iio_chan_spec rockchip_saradc_iio_channels[] = {
|
||||||
SARADC_CHANNEL(0, "adc0"),
|
SARADC_CHANNEL(0, "adc0", 10),
|
||||||
SARADC_CHANNEL(1, "adc1"),
|
SARADC_CHANNEL(1, "adc1", 10),
|
||||||
SARADC_CHANNEL(2, "adc2"),
|
SARADC_CHANNEL(2, "adc2", 10),
|
||||||
};
|
};
|
||||||
|
|
||||||
static const struct rockchip_saradc_data saradc_data = {
|
static const struct rockchip_saradc_data saradc_data = {
|
||||||
.num_bits = 10,
|
|
||||||
.channels = rockchip_saradc_iio_channels,
|
.channels = rockchip_saradc_iio_channels,
|
||||||
.num_channels = ARRAY_SIZE(rockchip_saradc_iio_channels),
|
.num_channels = ARRAY_SIZE(rockchip_saradc_iio_channels),
|
||||||
.clk_rate = 1000000,
|
.clk_rate = 1000000,
|
||||||
};
|
};
|
||||||
|
|
||||||
static const struct iio_chan_spec rockchip_rk3066_tsadc_iio_channels[] = {
|
static const struct iio_chan_spec rockchip_rk3066_tsadc_iio_channels[] = {
|
||||||
SARADC_CHANNEL(0, "adc0"),
|
SARADC_CHANNEL(0, "adc0", 12),
|
||||||
SARADC_CHANNEL(1, "adc1"),
|
SARADC_CHANNEL(1, "adc1", 12),
|
||||||
};
|
};
|
||||||
|
|
||||||
static const struct rockchip_saradc_data rk3066_tsadc_data = {
|
static const struct rockchip_saradc_data rk3066_tsadc_data = {
|
||||||
.num_bits = 12,
|
|
||||||
.channels = rockchip_rk3066_tsadc_iio_channels,
|
.channels = rockchip_rk3066_tsadc_iio_channels,
|
||||||
.num_channels = ARRAY_SIZE(rockchip_rk3066_tsadc_iio_channels),
|
.num_channels = ARRAY_SIZE(rockchip_rk3066_tsadc_iio_channels),
|
||||||
.clk_rate = 50000,
|
.clk_rate = 50000,
|
||||||
};
|
};
|
||||||
|
|
||||||
static const struct iio_chan_spec rockchip_rk3399_saradc_iio_channels[] = {
|
static const struct iio_chan_spec rockchip_rk3399_saradc_iio_channels[] = {
|
||||||
SARADC_CHANNEL(0, "adc0"),
|
SARADC_CHANNEL(0, "adc0", 10),
|
||||||
SARADC_CHANNEL(1, "adc1"),
|
SARADC_CHANNEL(1, "adc1", 10),
|
||||||
SARADC_CHANNEL(2, "adc2"),
|
SARADC_CHANNEL(2, "adc2", 10),
|
||||||
SARADC_CHANNEL(3, "adc3"),
|
SARADC_CHANNEL(3, "adc3", 10),
|
||||||
SARADC_CHANNEL(4, "adc4"),
|
SARADC_CHANNEL(4, "adc4", 10),
|
||||||
SARADC_CHANNEL(5, "adc5"),
|
SARADC_CHANNEL(5, "adc5", 10),
|
||||||
};
|
};
|
||||||
|
|
||||||
static const struct rockchip_saradc_data rk3399_saradc_data = {
|
static const struct rockchip_saradc_data rk3399_saradc_data = {
|
||||||
.num_bits = 10,
|
|
||||||
.channels = rockchip_rk3399_saradc_iio_channels,
|
.channels = rockchip_rk3399_saradc_iio_channels,
|
||||||
.num_channels = ARRAY_SIZE(rockchip_rk3399_saradc_iio_channels),
|
.num_channels = ARRAY_SIZE(rockchip_rk3399_saradc_iio_channels),
|
||||||
.clk_rate = 1000000,
|
.clk_rate = 1000000,
|
||||||
|
@ -214,6 +238,46 @@ static void rockchip_saradc_regulator_disable(void *data)
|
||||||
regulator_disable(info->vref);
|
regulator_disable(info->vref);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static irqreturn_t rockchip_saradc_trigger_handler(int irq, void *p)
|
||||||
|
{
|
||||||
|
struct iio_poll_func *pf = p;
|
||||||
|
struct iio_dev *i_dev = pf->indio_dev;
|
||||||
|
struct rockchip_saradc *info = iio_priv(i_dev);
|
||||||
|
/*
|
||||||
|
* @values: each channel takes an u16 value
|
||||||
|
* @timestamp: will be 8-byte aligned automatically
|
||||||
|
*/
|
||||||
|
struct {
|
||||||
|
u16 values[SARADC_MAX_CHANNELS];
|
||||||
|
int64_t timestamp;
|
||||||
|
} data;
|
||||||
|
int ret;
|
||||||
|
int i, j = 0;
|
||||||
|
|
||||||
|
mutex_lock(&i_dev->mlock);
|
||||||
|
|
||||||
|
for_each_set_bit(i, i_dev->active_scan_mask, i_dev->masklength) {
|
||||||
|
const struct iio_chan_spec *chan = &i_dev->channels[i];
|
||||||
|
|
||||||
|
ret = rockchip_saradc_conversion(info, chan);
|
||||||
|
if (ret) {
|
||||||
|
rockchip_saradc_power_down(info);
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
|
|
||||||
|
data.values[j] = info->last_val;
|
||||||
|
j++;
|
||||||
|
}
|
||||||
|
|
||||||
|
iio_push_to_buffers_with_timestamp(i_dev, &data, iio_get_time_ns(i_dev));
|
||||||
|
out:
|
||||||
|
mutex_unlock(&i_dev->mlock);
|
||||||
|
|
||||||
|
iio_trigger_notify_done(i_dev->trig);
|
||||||
|
|
||||||
|
return IRQ_HANDLED;
|
||||||
|
}
|
||||||
|
|
||||||
static int rockchip_saradc_probe(struct platform_device *pdev)
|
static int rockchip_saradc_probe(struct platform_device *pdev)
|
||||||
{
|
{
|
||||||
struct rockchip_saradc *info = NULL;
|
struct rockchip_saradc *info = NULL;
|
||||||
|
@ -242,6 +306,12 @@ static int rockchip_saradc_probe(struct platform_device *pdev)
|
||||||
|
|
||||||
info->data = match->data;
|
info->data = match->data;
|
||||||
|
|
||||||
|
/* Sanity check for possible later IP variants with more channels */
|
||||||
|
if (info->data->num_channels > SARADC_MAX_CHANNELS) {
|
||||||
|
dev_err(&pdev->dev, "max channels exceeded");
|
||||||
|
return -EINVAL;
|
||||||
|
}
|
||||||
|
|
||||||
mem = platform_get_resource(pdev, IORESOURCE_MEM, 0);
|
mem = platform_get_resource(pdev, IORESOURCE_MEM, 0);
|
||||||
info->regs = devm_ioremap_resource(&pdev->dev, mem);
|
info->regs = devm_ioremap_resource(&pdev->dev, mem);
|
||||||
if (IS_ERR(info->regs))
|
if (IS_ERR(info->regs))
|
||||||
|
@ -354,6 +424,11 @@ static int rockchip_saradc_probe(struct platform_device *pdev)
|
||||||
|
|
||||||
indio_dev->channels = info->data->channels;
|
indio_dev->channels = info->data->channels;
|
||||||
indio_dev->num_channels = info->data->num_channels;
|
indio_dev->num_channels = info->data->num_channels;
|
||||||
|
ret = devm_iio_triggered_buffer_setup(&indio_dev->dev, indio_dev, NULL,
|
||||||
|
rockchip_saradc_trigger_handler,
|
||||||
|
NULL);
|
||||||
|
if (ret)
|
||||||
|
return ret;
|
||||||
|
|
||||||
return devm_iio_device_register(&pdev->dev, indio_dev);
|
return devm_iio_device_register(&pdev->dev, indio_dev);
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue