iio: adc: add helpers for parsing ADC nodes
authorMatti Vaittinen <mazziesaccount@gmail.com>
Mon, 24 Mar 2025 07:13:03 +0000 (09:13 +0200)
committerJonathan Cameron <Jonathan.Cameron@huawei.com>
Tue, 22 Apr 2025 18:09:52 +0000 (19:09 +0100)
There are ADC ICs which may have some of the AIN pins usable for other
functions. These ICs may have some of the AIN pins wired so that they
should not be used for ADC.

A common way of marking pins that can be used as ADC inputs is to add
corresponding channel@N nodes in the device tree as described in the ADC
      binding yaml.

Add couple of helper functions which can be used to retrieve the channel
information from the device node.

Signed-off-by: Matti Vaittinen <mazziesaccount@gmail.com>
Reviewed-by: Andy Shevchenko <andriy.shevchenko@linux.intel.com>
Reviewed-by: Marcelo Schmitt <marcelo.schmitt1@gmail.com>
Link: https://patch.msgid.link/f1d8b3e15237947738912c0d297b3e1e21d8b03e.1742560649.git.mazziesaccount@gmail.com
Signed-off-by: Jonathan Cameron <Jonathan.Cameron@huawei.com>
drivers/iio/adc/Kconfig
drivers/iio/adc/Makefile
drivers/iio/adc/industrialio-adc.c [new file with mode: 0644]
include/linux/iio/adc-helpers.h [new file with mode: 0644]

index 75ed633..b210d63 100644 (file)
@@ -6,6 +6,9 @@
 
 menu "Analog to digital converters"
 
+config IIO_ADC_HELPER
+       tristate
+
 config AB8500_GPADC
        bool "ST-Ericsson AB8500 GPADC driver"
        depends on AB8500_CORE && REGULATOR_AB8500
index 3e918c3..6cd2ec8 100644 (file)
@@ -3,6 +3,8 @@
 # Makefile for IIO ADC drivers
 #
 
+obj-$(CONFIG_IIO_ADC_HELPER) += industrialio-adc.o
+
 # When adding new entries keep the list in alphabetical order
 obj-$(CONFIG_AB8500_GPADC) += ab8500-gpadc.o
 obj-$(CONFIG_AD_SIGMA_DELTA) += ad_sigma_delta.o
diff --git a/drivers/iio/adc/industrialio-adc.c b/drivers/iio/adc/industrialio-adc.c
new file mode 100644 (file)
index 0000000..b405723
--- /dev/null
@@ -0,0 +1,82 @@
+// SPDX-License-Identifier: GPL-2.0-only
+/*
+ * Helpers for parsing common ADC information from a firmware node.
+ *
+ * Copyright (c) 2025 Matti Vaittinen <mazziesaccount@gmail.com>
+ */
+
+#include <linux/device.h>
+#include <linux/errno.h>
+#include <linux/export.h>
+#include <linux/module.h>
+#include <linux/property.h>
+#include <linux/types.h>
+
+#include <linux/iio/adc-helpers.h>
+#include <linux/iio/iio.h>
+
+/**
+ * devm_iio_adc_device_alloc_chaninfo_se - allocate and fill iio_chan_spec for ADC
+ *
+ * Scan the device node for single-ended ADC channel information. Channel ID is
+ * expected to be found from the "reg" property. Allocate and populate the
+ * iio_chan_spec structure corresponding to channels that are found. The memory
+ * for iio_chan_spec structure will be freed upon device detach.
+ *
+ * @dev:               Pointer to the ADC device.
+ * @template:          Template iio_chan_spec from which the fields of all
+ *                     found and allocated channels are initialized.
+ * @max_chan_id:       Maximum value of a channel ID. Use negative value if no
+ *                     checking is required.
+ * @cs:                        Location where pointer to allocated iio_chan_spec
+ *                     should be stored.
+ *
+ * Return:     Number of found channels on success. Negative value to indicate
+ *             failure. Specifically, -ENOENT if no channel nodes were found.
+ */
+int devm_iio_adc_device_alloc_chaninfo_se(struct device *dev,
+                                         const struct iio_chan_spec *template,
+                                         int max_chan_id,
+                                         struct iio_chan_spec **cs)
+{
+       struct iio_chan_spec *chan_array, *chan;
+       int num_chan, ret;
+
+       num_chan = iio_adc_device_num_channels(dev);
+       if (num_chan < 0)
+               return num_chan;
+
+       if (!num_chan)
+               return -ENOENT;
+
+       chan_array = devm_kcalloc(dev, num_chan, sizeof(*chan_array),
+                                 GFP_KERNEL);
+       if (!chan_array)
+               return -ENOMEM;
+
+       chan = &chan_array[0];
+
+       device_for_each_named_child_node_scoped(dev, child, "channel") {
+               u32 ch;
+
+               ret = fwnode_property_read_u32(child, "reg", &ch);
+               if (ret)
+                       return ret;
+
+               if (max_chan_id >= 0 && ch > max_chan_id)
+                       return -ERANGE;
+
+               *chan = *template;
+               chan->channel = ch;
+               chan++;
+       }
+
+       *cs = chan_array;
+
+       return num_chan;
+}
+EXPORT_SYMBOL_NS_GPL(devm_iio_adc_device_alloc_chaninfo_se, "IIO_DRIVER");
+
+MODULE_LICENSE("GPL");
+MODULE_AUTHOR("Matti Vaittinen <mazziesaccount@gmail.com>");
+MODULE_DESCRIPTION("IIO ADC fwnode parsing helpers");
diff --git a/include/linux/iio/adc-helpers.h b/include/linux/iio/adc-helpers.h
new file mode 100644 (file)
index 0000000..56b092a
--- /dev/null
@@ -0,0 +1,27 @@
+/* SPDX-License-Identifier: GPL-2.0-only */
+
+/*
+ * The industrial I/O ADC firmware property parsing helpers
+ *
+ * Copyright (c) 2025 Matti Vaittinen <mazziesaccount@gmail.com>
+ */
+
+#ifndef _INDUSTRIAL_IO_ADC_HELPERS_H_
+#define _INDUSTRIAL_IO_ADC_HELPERS_H_
+
+#include <linux/property.h>
+
+struct device;
+struct iio_chan_spec;
+
+static inline int iio_adc_device_num_channels(struct device *dev)
+{
+       return device_get_named_child_node_count(dev, "channel");
+}
+
+int devm_iio_adc_device_alloc_chaninfo_se(struct device *dev,
+                                         const struct iio_chan_spec *template,
+                                         int max_chan_id,
+                                         struct iio_chan_spec **cs);
+
+#endif /* _INDUSTRIAL_IO_ADC_HELPERS_H_ */