Re: [PATCH v1 1/1] iio: adc: tlc4541: add support for TI tlc4541 adc

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 




On 11 January 2017 06:51:11 GMT+00:00, Phil Reid <preid@xxxxxxxxxxxxxxxxx> wrote:
>This adds TI's tlc4541 16-bit ADC driver. Which is a single channel
>ADC. Supports raw and trigger buffer access.
>Also supports the tlc3541 14-bit device, which has not been tested.
>Implementation of the tlc3541 is fairly straight forward thou.
>
>Signed-off-by: Phil Reid <preid@xxxxxxxxxxxxxxxxx>
One minor suggestion from me.

J
>---
>
>Notes:
>    Changes from v1:
>    - Add tlc3541 support and chan spec.
>    - remove fields that where already 0 from TLC4541_V_CHAN macro
>- Increase rx_buf size in tlc4541_state to avoid copy in
>tlc4541_trigger_handle
>    - Remove erroneous be16_to_cpu in tlc4541_trigger_handle
>    - Docs/binding: spi -> SPI & add ti,tlc3541
>    
>    I haven't add Rob's Ack due to adding a new compatible string.
>    
>I tried to ".index = 1" from the spec as suggested by Peter, but that
>didn't
>    seem to work. Perhaps remove of .channel was the intended target.
>    
>    Example output from iio_readdev
>    
>    with ".index = 1"
>    root@cyclone5:~# mkdir /sys/kernel/config/iio/triggers/hrtimer/hr1
>    root@cyclone5:~# iio_readdev -t hr1 -b 32 -s 10 tlc4541 | hexdump
>    WARNING: High-speed mode not enabled
>    0000000 af00 0000 0000 0000 b922 ca99 93da 1492
>    0000010 a800 00ff 0000 0000 b246 cb30 93da 1492
>    0000020 a900 0000 0000 0000 4f9c cbc9 93da 1492
>    0000030 aa00 00ff 0000 0000 bd2c cc61 93da 1492
>    0000040 aa00 00ff 0000 0000 544c ccfa 93da 1492
>    0000050 ab00 00ff 0000 0000 e806 cd92 93da 1492
>    0000060 a900 00ff 0000 0000 846c ce2b 93da 1492
>    0000070 ab00 0000 0000 0000 2efc cec8 93da 1492
>    0000080 a800 00ff 0000 0000 b090 cf5c 93da 1492
>    0000090 a900 00ff 0000 0000 476a cff5 93da 1492
>    
>    without .index
>    root@cyclone5:~# mkdir /sys/kernel/config/iio/triggers/hrtimer/hr1
>    root@cyclone5:~# iio_readdev -t hr1 -b 32 -s 10 tlc4541 | hexdump
>    WARNING: High-speed mode not enabled
>    0000000 6db0 eeb6 93e3 1492 35e0 ef4f 93e3 1492
>    0000010 4b34 efe5 93e3 1492 e9f2 f07d 93e3 1492
>    0000020 6182 f116 93e3 1492 090a f1af 93e3 1492
>    0000030 409c f249 93e3 1492 6c1a f2e0 93e3 1492
>    0000040 cd02 f378 93e3 1492 9582 f411 93e3 1492
>
> .../devicetree/bindings/iio/adc/ti-tlc4541.txt     |  17 ++
> drivers/iio/adc/Kconfig                            |  11 +
> drivers/iio/adc/Makefile                           |   1 +
>drivers/iio/adc/ti-tlc4541.c                       | 276
>+++++++++++++++++++++
> 4 files changed, 305 insertions(+)
>create mode 100644
>Documentation/devicetree/bindings/iio/adc/ti-tlc4541.txt
> create mode 100644 drivers/iio/adc/ti-tlc4541.c
>
>diff --git a/Documentation/devicetree/bindings/iio/adc/ti-tlc4541.txt
>b/Documentation/devicetree/bindings/iio/adc/ti-tlc4541.txt
>new file mode 100644
>index 0000000..e1de2bd
>--- /dev/null
>+++ b/Documentation/devicetree/bindings/iio/adc/ti-tlc4541.txt
>@@ -0,0 +1,17 @@
>+* Texas Instruments' TLC4541
>+
>+Required properties:
>+ - compatible: Should be one of
>+	* "ti,tlc4541"
>+	* "ti,tlc3541"
>+	- reg: SPI chip select number for the device
>+ - vref-supply: The regulator supply for ADC reference voltage
>+ - spi-max-frequency: Max SPI frequency to use (<= 200000)
>+
>+Example:
>+adc@0 {
>+	compatible = "ti,adc0832";
>+	reg = <0>;
>+	vref-supply = <&vdd_supply>;
>+	spi-max-frequency = <200000>;
>+};
>diff --git a/drivers/iio/adc/Kconfig b/drivers/iio/adc/Kconfig
>index 99c0514..4dda3f0 100644
>--- a/drivers/iio/adc/Kconfig
>+++ b/drivers/iio/adc/Kconfig
>@@ -525,6 +525,17 @@ config TI_AM335X_ADC
>	  To compile this driver as a module, choose M here: the module will
>be
> 	  called ti_am335x_adc.
> 
>+config TI_TLC4541
>+	tristate "Texas Instruments TLC4541 ADC driver"
>+	depends on SPI
>+	select IIO_BUFFER
>+	select IIO_TRIGGERED_BUFFER
>+	help
>+	  Say yes here to build support for Texas Instruments TLC4541 ADC
>chip.
>+
>+	  This driver can also be built as a module. If so, the module will
>be
>+	  called ti-tlc4541.
>+
> config TWL4030_MADC
> 	tristate "TWL4030 MADC (Monitoring A/D Converter)"
> 	depends on TWL4030_CORE
>diff --git a/drivers/iio/adc/Makefile b/drivers/iio/adc/Makefile
>index 7a40c04..9bf2377 100644
>--- a/drivers/iio/adc/Makefile
>+++ b/drivers/iio/adc/Makefile
>@@ -49,6 +49,7 @@ obj-$(CONFIG_TI_ADC161S626) += ti-adc161s626.o
> obj-$(CONFIG_TI_ADS1015) += ti-ads1015.o
> obj-$(CONFIG_TI_ADS8688) += ti-ads8688.o
> obj-$(CONFIG_TI_AM335X_ADC) += ti_am335x_adc.o
>+obj-$(CONFIG_TI_TLC4541) += ti-tlc4541.o
> obj-$(CONFIG_TWL4030_MADC) += twl4030-madc.o
> obj-$(CONFIG_TWL6030_GPADC) += twl6030-gpadc.o
> obj-$(CONFIG_VF610_ADC) += vf610_adc.o
>diff --git a/drivers/iio/adc/ti-tlc4541.c
>b/drivers/iio/adc/ti-tlc4541.c
>new file mode 100644
>index 0000000..a0cd5e1
>--- /dev/null
>+++ b/drivers/iio/adc/ti-tlc4541.c
>@@ -0,0 +1,276 @@
>+/*
>+ * TI tlc4541 ADC Driver
>+ *
>+ * Copyright (C) 2017 Phil Reid
>+ *
>+ * Datasheets can be found here:
>+ * http://www.ti.com/lit/gpn/tlc3541
>+ * http://www.ti.com/lit/gpn/tlc4541
>+ *
>+ * This program is free software; you can redistribute it and/or
>modify
>+ * it under the terms of the GNU General Public License version 2 as
>+ * published by the Free Software Foundation.
>+ *
>+ * The tlc4541 requires 24 clock cycles to start a transfer.
>+ * Conversion then takes 2.94us to complete before data is ready
>+ * Data is returned MSB first.
>+ */
>+
>+#include <linux/delay.h>
>+#include <linux/device.h>
>+#include <linux/err.h>
>+#include <linux/interrupt.h>
>+#include <linux/iio/iio.h>
>+#include <linux/iio/sysfs.h>
>+#include <linux/iio/buffer.h>
>+#include <linux/iio/trigger_consumer.h>
>+#include <linux/iio/triggered_buffer.h>
>+#include <linux/kernel.h>
>+#include <linux/module.h>
>+#include <linux/regulator/consumer.h>
>+#include <linux/slab.h>
>+#include <linux/spi/spi.h>
>+#include <linux/sysfs.h>
>+
>+struct tlc4541_state {
>+	struct spi_device               *spi;
>+	struct regulator                *reg;
>+	struct spi_transfer             scan_single_xfer[3];
>+	struct spi_message              scan_single_msg;
>+
>+	/*
>+	 * DMA (thus cache coherency maintenance) requires the
>+	 * transfer buffers to live in their own cache lines.
>+	 * 2 bytes data + 6 bytes padding + 8 bytes timestamp when
>+	 * call iio_push_to_buffers_with_timestamp.
>+	 */
>+	__be16                          rx_buf[8] ____cacheline_aligned;
>+};
>+
>+struct tlc4541_chip_info {
>+	const struct iio_chan_spec *channels;
>+	unsigned int num_channels;
>+};
>+
>+enum tlc4541_id {
>+	TLC3541,
>+	TLC4541,
>+};
>+
>+#define TLC4541_V_CHAN(bits, bitshift) {                             
>\
>+		.type = IIO_VOLTAGE,                                  \
>+		.indexed = 1,                                         \
>+		.info_mask_separate       = BIT(IIO_CHAN_INFO_RAW),   \
>+		.info_mask_shared_by_type = BIT(IIO_CHAN_INFO_SCALE), \
>+		.scan_type = {                                        \
>+			.sign = 'u',                                  \
>+			.realbits = (bits),                           \
>+			.storagebits = 16,                            \
>+			.shift = bitshift,                            \
>+			.endianness = IIO_BE,                         \
>+		},                                                    \
>+	}
>+
>+#define DECLARE_TLC4541_CHANNELS(name, bits, bitshift) \
>+const struct iio_chan_spec name ## _channels[] = { \
>+	TLC4541_V_CHAN(bits, bitshift), \
>+	IIO_CHAN_SOFT_TIMESTAMP(1), \
>+}
>+
>+static DECLARE_TLC4541_CHANNELS(tlc4541, 16, 0);
>+static DECLARE_TLC4541_CHANNELS(tlc3541, 14, 2);
>+
>+static const struct tlc4541_chip_info tlc4541_chip_info[] = {
>+	[TLC4541] = {
>+		.channels = tlc4541_channels,
>+		.num_channels = ARRAY_SIZE(tlc4541_channels),
>+	},
>+	[TLC3541] = {
>+		.channels = tlc3541_channels,
>+		.num_channels = ARRAY_SIZE(tlc3541_channels),
>+	},
>+};
>+
>+static irqreturn_t tlc4541_trigger_handler(int irq, void *p)
>+{
>+	struct iio_poll_func *pf = p;
>+	struct iio_dev *indio_dev = pf->indio_dev;
>+	struct tlc4541_state *st = iio_priv(indio_dev);
>+	int ret;
>+
>+	ret = spi_sync(st->spi, &st->scan_single_msg);
>+	if (ret < 0)
>+		goto done;
>+
>+	iio_push_to_buffers_with_timestamp(indio_dev, st->rx_buf,
>+					   iio_get_time_ns(indio_dev));
>+
>+done:
>+	iio_trigger_notify_done(indio_dev->trig);
>+	return IRQ_HANDLED;
>+}
>+
>+static int tlc4541_get_range(struct tlc4541_state *st)
>+{
>+	int vref;
>+
>+	vref = regulator_get_voltage(st->reg);
>+	if (vref < 0)
>+		return vref;
>+
>+	vref /= 1000;
>+
>+	return vref;
>+}
>+
>+static int tlc4541_read_raw(struct iio_dev *indio_dev,
>+			    struct iio_chan_spec const *chan,
>+			    int *val,
>+			    int *val2,
>+			    long m)
>+{
>+	int ret = 0;
>+	struct tlc4541_state *st = iio_priv(indio_dev);
>+
>+	switch (m) {
>+	case IIO_CHAN_INFO_RAW:
>+		ret = iio_device_claim_direct_mode(indio_dev);
>+		if (ret)
>+			return ret;
>+		ret = spi_sync(st->spi, &st->scan_single_msg);
>+		iio_device_release_direct_mode(indio_dev);
>+		if (ret < 0)
>+			return ret;
>+		*val = be16_to_cpu(st->rx_buf[0]);
>+		*val = *val >> chan->scan_type.shift;
>+		*val &= GENMASK(chan->scan_type.realbits - 1, 0);
>+		return IIO_VAL_INT;
>+	case IIO_CHAN_INFO_SCALE:
>+		ret = tlc4541_get_range(st);
>+		if (ret < 0)
>+			return ret;
>+		*val = ret;
>+		*val2 = chan->scan_type.realbits;
>+		return IIO_VAL_FRACTIONAL_LOG2;
>+	}
>+	return -EINVAL;
>+}
>+
>+static const struct iio_info tlc4541_info = {
>+	.read_raw = &tlc4541_read_raw,
>+	.driver_module = THIS_MODULE,
>+};
>+
>+static int tlc4541_probe(struct spi_device *spi)
>+{
>+	struct tlc4541_state *st;
>+	struct iio_dev *indio_dev;
>+	const struct tlc4541_chip_info *info;
>+	int ret;
>+	int8_t device_init = 0;
>+
>+	indio_dev = devm_iio_device_alloc(&spi->dev, sizeof(*st));
>+	if (indio_dev == NULL)
>+		return -ENOMEM;
>+
>+	st = iio_priv(indio_dev);
>+
>+	spi_set_drvdata(spi, indio_dev);
>+
>+	st->spi = spi;
>+
>+	info = &tlc4541_chip_info[spi_get_device_id(spi)->driver_data];
>+
>+	indio_dev->name = spi_get_device_id(spi)->name;
>+	indio_dev->dev.parent = &spi->dev;
>+	indio_dev->modes = INDIO_DIRECT_MODE;
>+	indio_dev->channels = info->channels;
>+	indio_dev->num_channels = info->num_channels;
>+	indio_dev->info = &tlc4541_info;
>+
>+	/* perform reset */
>+	spi_write(spi, &device_init, 1);
>+
>+	/* Setup default message */
>+	st->scan_single_xfer[0].rx_buf = &st->rx_buf[0];
>+	st->scan_single_xfer[0].len = 3;
>+	st->scan_single_xfer[1].delay_usecs = 3;
>+	st->scan_single_xfer[2].rx_buf = &st->rx_buf[0];
>+	st->scan_single_xfer[2].len = 2;
>+
>+	spi_message_init(&st->scan_single_msg);
>+	spi_message_add_tail(&st->scan_single_xfer[0], &st->scan_single_msg);
>+	spi_message_add_tail(&st->scan_single_xfer[1], &st->scan_single_msg);
>+	spi_message_add_tail(&st->scan_single_xfer[2], &st->scan_single_msg);
spi_init_with_transfers.
>+
>+	st->reg = devm_regulator_get(&spi->dev, "vref");
>+	if (IS_ERR(st->reg))
>+		return PTR_ERR(st->reg);
>+
>+	ret = regulator_enable(st->reg);
>+	if (ret)
>+		return ret;
>+
>+	ret = iio_triggered_buffer_setup(indio_dev, NULL,
>+			&tlc4541_trigger_handler, NULL);
>+	if (ret)
>+		goto error_disable_reg;
>+
>+	ret = iio_device_register(indio_dev);
>+	if (ret)
>+		goto error_cleanup_buffer;
>+
>+	return 0;
>+
>+error_cleanup_buffer:
>+	iio_triggered_buffer_cleanup(indio_dev);
>+error_disable_reg:
>+	regulator_disable(st->reg);
>+
>+	return ret;
>+}
>+
>+static int tlc4541_remove(struct spi_device *spi)
>+{
>+	struct iio_dev *indio_dev = spi_get_drvdata(spi);
>+	struct tlc4541_state *st = iio_priv(indio_dev);
>+
>+	iio_device_unregister(indio_dev);
>+	iio_triggered_buffer_cleanup(indio_dev);
>+	regulator_disable(st->reg);
>+
>+	return 0;
>+}
>+
>+#ifdef CONFIG_OF
>+
>+static const struct of_device_id tlc4541_dt_ids[] = {
>+	{ .compatible = "ti,tlc3541", },
>+	{ .compatible = "ti,tlc4541", },
>+	{}
>+};
>+MODULE_DEVICE_TABLE(of, tlc4541_dt_ids);
>+
>+#endif
>+
>+static const struct spi_device_id tlc4541_id[] = {
>+	{"tlc3541", TLC3541},
>+	{"tlc4541", TLC4541},
>+	{}
>+};
>+MODULE_DEVICE_TABLE(spi, tlc4541_id);
>+
>+static struct spi_driver tlc4541_driver = {
>+	.driver = {
>+		.name   = "tlc4541",
>+		.of_match_table = of_match_ptr(tlc4541_dt_ids),
>+	},
>+	.probe          = tlc4541_probe,
>+	.remove         = tlc4541_remove,
>+	.id_table       = tlc4541_id,
>+};
>+module_spi_driver(tlc4541_driver);
>+
>+MODULE_AUTHOR("Phil Reid <preid@xxxxxxxxxxxxxxxxx>");
>+MODULE_DESCRIPTION("Texas Instruments TLC4541 ADC");
>+MODULE_LICENSE("GPL v2");

-- 
Sent from my Android device with K-9 Mail. Please excuse my brevity.
--
To unsubscribe from this list: send the line "unsubscribe linux-iio" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at  http://vger.kernel.org/majordomo-info.html



[Index of Archives]     [Linux USB Devel]     [Video for Linux]     [Linux Audio Users]     [Yosemite News]     [Linux Input]     [Linux Kernel]     [Linux SCSI]     [X.org]

  Powered by Linux