162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0
262306a36Sopenharmony_ci/*
362306a36Sopenharmony_ci * GPADC driver for sunxi platforms (D1, T113-S3 and R329)
462306a36Sopenharmony_ci * Copyright (c) 2023 Maksim Kiselev <bigunclemax@gmail.com>
562306a36Sopenharmony_ci */
662306a36Sopenharmony_ci
762306a36Sopenharmony_ci#include <linux/bitfield.h>
862306a36Sopenharmony_ci#include <linux/clk.h>
962306a36Sopenharmony_ci#include <linux/completion.h>
1062306a36Sopenharmony_ci#include <linux/interrupt.h>
1162306a36Sopenharmony_ci#include <linux/io.h>
1262306a36Sopenharmony_ci#include <linux/mod_devicetable.h>
1362306a36Sopenharmony_ci#include <linux/module.h>
1462306a36Sopenharmony_ci#include <linux/platform_device.h>
1562306a36Sopenharmony_ci#include <linux/property.h>
1662306a36Sopenharmony_ci#include <linux/reset.h>
1762306a36Sopenharmony_ci
1862306a36Sopenharmony_ci#include <linux/iio/iio.h>
1962306a36Sopenharmony_ci
2062306a36Sopenharmony_ci#define SUN20I_GPADC_DRIVER_NAME	"sun20i-gpadc"
2162306a36Sopenharmony_ci
2262306a36Sopenharmony_ci/* Register map definition */
2362306a36Sopenharmony_ci#define SUN20I_GPADC_SR			0x00
2462306a36Sopenharmony_ci#define SUN20I_GPADC_CTRL		0x04
2562306a36Sopenharmony_ci#define SUN20I_GPADC_CS_EN		0x08
2662306a36Sopenharmony_ci#define SUN20I_GPADC_FIFO_INTC		0x0c
2762306a36Sopenharmony_ci#define SUN20I_GPADC_FIFO_INTS		0x10
2862306a36Sopenharmony_ci#define SUN20I_GPADC_FIFO_DATA		0X14
2962306a36Sopenharmony_ci#define SUN20I_GPADC_CB_DATA		0X18
3062306a36Sopenharmony_ci#define SUN20I_GPADC_DATAL_INTC		0x20
3162306a36Sopenharmony_ci#define SUN20I_GPADC_DATAH_INTC		0x24
3262306a36Sopenharmony_ci#define SUN20I_GPADC_DATA_INTC		0x28
3362306a36Sopenharmony_ci#define SUN20I_GPADC_DATAL_INTS		0x30
3462306a36Sopenharmony_ci#define SUN20I_GPADC_DATAH_INTS		0x34
3562306a36Sopenharmony_ci#define SUN20I_GPADC_DATA_INTS		0x38
3662306a36Sopenharmony_ci#define SUN20I_GPADC_CH_CMP_DATA(x)	(0x40 + (x) * 4)
3762306a36Sopenharmony_ci#define SUN20I_GPADC_CH_DATA(x)		(0x80 + (x) * 4)
3862306a36Sopenharmony_ci
3962306a36Sopenharmony_ci#define SUN20I_GPADC_CTRL_ADC_AUTOCALI_EN_MASK		BIT(23)
4062306a36Sopenharmony_ci#define SUN20I_GPADC_CTRL_WORK_MODE_MASK		GENMASK(19, 18)
4162306a36Sopenharmony_ci#define SUN20I_GPADC_CTRL_ADC_EN_MASK			BIT(16)
4262306a36Sopenharmony_ci#define SUN20I_GPADC_CS_EN_ADC_CH(x)			BIT(x)
4362306a36Sopenharmony_ci#define SUN20I_GPADC_DATA_INTC_CH_DATA_IRQ_EN(x)	BIT(x)
4462306a36Sopenharmony_ci
4562306a36Sopenharmony_ci#define SUN20I_GPADC_WORK_MODE_SINGLE			0
4662306a36Sopenharmony_ci
4762306a36Sopenharmony_cistruct sun20i_gpadc_iio {
4862306a36Sopenharmony_ci	void __iomem		*regs;
4962306a36Sopenharmony_ci	struct completion	completion;
5062306a36Sopenharmony_ci	int			last_channel;
5162306a36Sopenharmony_ci	/*
5262306a36Sopenharmony_ci	 * Lock to protect the device state during a potential concurrent
5362306a36Sopenharmony_ci	 * read access from userspace. Reading a raw value requires a sequence
5462306a36Sopenharmony_ci	 * of register writes, then a wait for a completion callback,
5562306a36Sopenharmony_ci	 * and finally a register read, during which userspace could issue
5662306a36Sopenharmony_ci	 * another read request. This lock protects a read access from
5762306a36Sopenharmony_ci	 * ocurring before another one has finished.
5862306a36Sopenharmony_ci	 */
5962306a36Sopenharmony_ci	struct mutex		lock;
6062306a36Sopenharmony_ci};
6162306a36Sopenharmony_ci
6262306a36Sopenharmony_cistatic int sun20i_gpadc_adc_read(struct sun20i_gpadc_iio *info,
6362306a36Sopenharmony_ci				 struct iio_chan_spec const *chan, int *val)
6462306a36Sopenharmony_ci{
6562306a36Sopenharmony_ci	u32 ctrl;
6662306a36Sopenharmony_ci	int ret = IIO_VAL_INT;
6762306a36Sopenharmony_ci
6862306a36Sopenharmony_ci	mutex_lock(&info->lock);
6962306a36Sopenharmony_ci
7062306a36Sopenharmony_ci	reinit_completion(&info->completion);
7162306a36Sopenharmony_ci
7262306a36Sopenharmony_ci	if (info->last_channel != chan->channel) {
7362306a36Sopenharmony_ci		info->last_channel = chan->channel;
7462306a36Sopenharmony_ci
7562306a36Sopenharmony_ci		/* enable the analog input channel */
7662306a36Sopenharmony_ci		writel(SUN20I_GPADC_CS_EN_ADC_CH(chan->channel),
7762306a36Sopenharmony_ci		       info->regs + SUN20I_GPADC_CS_EN);
7862306a36Sopenharmony_ci
7962306a36Sopenharmony_ci		/* enable the data irq for input channel */
8062306a36Sopenharmony_ci		writel(SUN20I_GPADC_DATA_INTC_CH_DATA_IRQ_EN(chan->channel),
8162306a36Sopenharmony_ci		       info->regs + SUN20I_GPADC_DATA_INTC);
8262306a36Sopenharmony_ci	}
8362306a36Sopenharmony_ci
8462306a36Sopenharmony_ci	/* enable the ADC function */
8562306a36Sopenharmony_ci	ctrl = readl(info->regs + SUN20I_GPADC_CTRL);
8662306a36Sopenharmony_ci	ctrl |= FIELD_PREP(SUN20I_GPADC_CTRL_ADC_EN_MASK, 1);
8762306a36Sopenharmony_ci	writel(ctrl, info->regs + SUN20I_GPADC_CTRL);
8862306a36Sopenharmony_ci
8962306a36Sopenharmony_ci	/*
9062306a36Sopenharmony_ci	 * According to the datasheet maximum acquire time(TACQ) can be
9162306a36Sopenharmony_ci	 * (65535+1)/24Mhz and conversion time(CONV_TIME) is always constant
9262306a36Sopenharmony_ci	 * and equal to 14/24Mhz, so (TACQ+CONV_TIME) <= 2.73125ms.
9362306a36Sopenharmony_ci	 * A 10ms delay should be enough to make sure an interrupt occurs in
9462306a36Sopenharmony_ci	 * normal conditions. If it doesn't occur, then there is a timeout.
9562306a36Sopenharmony_ci	 */
9662306a36Sopenharmony_ci	if (!wait_for_completion_timeout(&info->completion, msecs_to_jiffies(10))) {
9762306a36Sopenharmony_ci		ret = -ETIMEDOUT;
9862306a36Sopenharmony_ci		goto err_unlock;
9962306a36Sopenharmony_ci	}
10062306a36Sopenharmony_ci
10162306a36Sopenharmony_ci	/* read the ADC data */
10262306a36Sopenharmony_ci	*val = readl(info->regs + SUN20I_GPADC_CH_DATA(chan->channel));
10362306a36Sopenharmony_ci
10462306a36Sopenharmony_cierr_unlock:
10562306a36Sopenharmony_ci	mutex_unlock(&info->lock);
10662306a36Sopenharmony_ci
10762306a36Sopenharmony_ci	return ret;
10862306a36Sopenharmony_ci}
10962306a36Sopenharmony_ci
11062306a36Sopenharmony_cistatic int sun20i_gpadc_read_raw(struct iio_dev *indio_dev,
11162306a36Sopenharmony_ci				 struct iio_chan_spec const *chan, int *val,
11262306a36Sopenharmony_ci				 int *val2, long mask)
11362306a36Sopenharmony_ci{
11462306a36Sopenharmony_ci	struct sun20i_gpadc_iio *info = iio_priv(indio_dev);
11562306a36Sopenharmony_ci
11662306a36Sopenharmony_ci	switch (mask) {
11762306a36Sopenharmony_ci	case IIO_CHAN_INFO_RAW:
11862306a36Sopenharmony_ci		return sun20i_gpadc_adc_read(info, chan, val);
11962306a36Sopenharmony_ci	case IIO_CHAN_INFO_SCALE:
12062306a36Sopenharmony_ci		/* value in mv = 1800mV / 4096 raw */
12162306a36Sopenharmony_ci		*val = 1800;
12262306a36Sopenharmony_ci		*val2 = 12;
12362306a36Sopenharmony_ci		return IIO_VAL_FRACTIONAL_LOG2;
12462306a36Sopenharmony_ci	default:
12562306a36Sopenharmony_ci		return -EINVAL;
12662306a36Sopenharmony_ci	}
12762306a36Sopenharmony_ci}
12862306a36Sopenharmony_ci
12962306a36Sopenharmony_cistatic irqreturn_t sun20i_gpadc_irq_handler(int irq, void *data)
13062306a36Sopenharmony_ci{
13162306a36Sopenharmony_ci	struct sun20i_gpadc_iio *info = data;
13262306a36Sopenharmony_ci
13362306a36Sopenharmony_ci	/* clear data interrupt status register */
13462306a36Sopenharmony_ci	writel(GENMASK(31, 0), info->regs + SUN20I_GPADC_DATA_INTS);
13562306a36Sopenharmony_ci
13662306a36Sopenharmony_ci	complete(&info->completion);
13762306a36Sopenharmony_ci
13862306a36Sopenharmony_ci	return IRQ_HANDLED;
13962306a36Sopenharmony_ci}
14062306a36Sopenharmony_ci
14162306a36Sopenharmony_cistatic const struct iio_info sun20i_gpadc_iio_info = {
14262306a36Sopenharmony_ci	.read_raw = sun20i_gpadc_read_raw,
14362306a36Sopenharmony_ci};
14462306a36Sopenharmony_ci
14562306a36Sopenharmony_cistatic void sun20i_gpadc_reset_assert(void *data)
14662306a36Sopenharmony_ci{
14762306a36Sopenharmony_ci	struct reset_control *rst = data;
14862306a36Sopenharmony_ci
14962306a36Sopenharmony_ci	reset_control_assert(rst);
15062306a36Sopenharmony_ci}
15162306a36Sopenharmony_ci
15262306a36Sopenharmony_cistatic int sun20i_gpadc_alloc_channels(struct iio_dev *indio_dev,
15362306a36Sopenharmony_ci				       struct device *dev)
15462306a36Sopenharmony_ci{
15562306a36Sopenharmony_ci	unsigned int channel;
15662306a36Sopenharmony_ci	int num_channels, i, ret;
15762306a36Sopenharmony_ci	struct iio_chan_spec *channels;
15862306a36Sopenharmony_ci	struct fwnode_handle *node;
15962306a36Sopenharmony_ci
16062306a36Sopenharmony_ci	num_channels = device_get_child_node_count(dev);
16162306a36Sopenharmony_ci	if (num_channels == 0)
16262306a36Sopenharmony_ci		return dev_err_probe(dev, -ENODEV, "no channel children\n");
16362306a36Sopenharmony_ci
16462306a36Sopenharmony_ci	channels = devm_kcalloc(dev, num_channels, sizeof(*channels),
16562306a36Sopenharmony_ci				GFP_KERNEL);
16662306a36Sopenharmony_ci	if (!channels)
16762306a36Sopenharmony_ci		return -ENOMEM;
16862306a36Sopenharmony_ci
16962306a36Sopenharmony_ci	i = 0;
17062306a36Sopenharmony_ci	device_for_each_child_node(dev, node) {
17162306a36Sopenharmony_ci		ret = fwnode_property_read_u32(node, "reg", &channel);
17262306a36Sopenharmony_ci		if (ret) {
17362306a36Sopenharmony_ci			fwnode_handle_put(node);
17462306a36Sopenharmony_ci			return dev_err_probe(dev, ret, "invalid channel number\n");
17562306a36Sopenharmony_ci		}
17662306a36Sopenharmony_ci
17762306a36Sopenharmony_ci		channels[i].type = IIO_VOLTAGE;
17862306a36Sopenharmony_ci		channels[i].indexed = 1;
17962306a36Sopenharmony_ci		channels[i].channel = channel;
18062306a36Sopenharmony_ci		channels[i].info_mask_separate = BIT(IIO_CHAN_INFO_RAW);
18162306a36Sopenharmony_ci		channels[i].info_mask_shared_by_type = BIT(IIO_CHAN_INFO_SCALE);
18262306a36Sopenharmony_ci
18362306a36Sopenharmony_ci		i++;
18462306a36Sopenharmony_ci	}
18562306a36Sopenharmony_ci
18662306a36Sopenharmony_ci	indio_dev->channels = channels;
18762306a36Sopenharmony_ci	indio_dev->num_channels = num_channels;
18862306a36Sopenharmony_ci
18962306a36Sopenharmony_ci	return 0;
19062306a36Sopenharmony_ci}
19162306a36Sopenharmony_ci
19262306a36Sopenharmony_cistatic int sun20i_gpadc_probe(struct platform_device *pdev)
19362306a36Sopenharmony_ci{
19462306a36Sopenharmony_ci	struct device *dev = &pdev->dev;
19562306a36Sopenharmony_ci	struct iio_dev *indio_dev;
19662306a36Sopenharmony_ci	struct sun20i_gpadc_iio *info;
19762306a36Sopenharmony_ci	struct reset_control *rst;
19862306a36Sopenharmony_ci	struct clk *clk;
19962306a36Sopenharmony_ci	int irq;
20062306a36Sopenharmony_ci	int ret;
20162306a36Sopenharmony_ci
20262306a36Sopenharmony_ci	indio_dev = devm_iio_device_alloc(dev, sizeof(*info));
20362306a36Sopenharmony_ci	if (!indio_dev)
20462306a36Sopenharmony_ci		return -ENOMEM;
20562306a36Sopenharmony_ci
20662306a36Sopenharmony_ci	info = iio_priv(indio_dev);
20762306a36Sopenharmony_ci	info->last_channel = -1;
20862306a36Sopenharmony_ci
20962306a36Sopenharmony_ci	mutex_init(&info->lock);
21062306a36Sopenharmony_ci	init_completion(&info->completion);
21162306a36Sopenharmony_ci
21262306a36Sopenharmony_ci	ret = sun20i_gpadc_alloc_channels(indio_dev, dev);
21362306a36Sopenharmony_ci	if (ret)
21462306a36Sopenharmony_ci		return ret;
21562306a36Sopenharmony_ci
21662306a36Sopenharmony_ci	indio_dev->info = &sun20i_gpadc_iio_info;
21762306a36Sopenharmony_ci	indio_dev->name = SUN20I_GPADC_DRIVER_NAME;
21862306a36Sopenharmony_ci
21962306a36Sopenharmony_ci	info->regs = devm_platform_ioremap_resource(pdev, 0);
22062306a36Sopenharmony_ci	if (IS_ERR(info->regs))
22162306a36Sopenharmony_ci		return PTR_ERR(info->regs);
22262306a36Sopenharmony_ci
22362306a36Sopenharmony_ci	clk = devm_clk_get_enabled(dev, NULL);
22462306a36Sopenharmony_ci	if (IS_ERR(clk))
22562306a36Sopenharmony_ci		return dev_err_probe(dev, PTR_ERR(clk), "failed to enable bus clock\n");
22662306a36Sopenharmony_ci
22762306a36Sopenharmony_ci	rst = devm_reset_control_get_exclusive(dev, NULL);
22862306a36Sopenharmony_ci	if (IS_ERR(rst))
22962306a36Sopenharmony_ci		return dev_err_probe(dev, PTR_ERR(rst), "failed to get reset control\n");
23062306a36Sopenharmony_ci
23162306a36Sopenharmony_ci	ret = reset_control_deassert(rst);
23262306a36Sopenharmony_ci	if (ret)
23362306a36Sopenharmony_ci		return dev_err_probe(dev, ret, "failed to deassert reset\n");
23462306a36Sopenharmony_ci
23562306a36Sopenharmony_ci	ret = devm_add_action_or_reset(dev, sun20i_gpadc_reset_assert, rst);
23662306a36Sopenharmony_ci	if (ret)
23762306a36Sopenharmony_ci		return ret;
23862306a36Sopenharmony_ci
23962306a36Sopenharmony_ci	irq = platform_get_irq(pdev, 0);
24062306a36Sopenharmony_ci	if (irq < 0)
24162306a36Sopenharmony_ci		return irq;
24262306a36Sopenharmony_ci
24362306a36Sopenharmony_ci	ret = devm_request_irq(dev, irq, sun20i_gpadc_irq_handler, 0,
24462306a36Sopenharmony_ci			       dev_name(dev), info);
24562306a36Sopenharmony_ci	if (ret)
24662306a36Sopenharmony_ci		return dev_err_probe(dev, ret, "failed requesting irq %d\n", irq);
24762306a36Sopenharmony_ci
24862306a36Sopenharmony_ci	writel(FIELD_PREP(SUN20I_GPADC_CTRL_ADC_AUTOCALI_EN_MASK, 1) |
24962306a36Sopenharmony_ci	       FIELD_PREP(SUN20I_GPADC_CTRL_WORK_MODE_MASK, SUN20I_GPADC_WORK_MODE_SINGLE),
25062306a36Sopenharmony_ci	       info->regs + SUN20I_GPADC_CTRL);
25162306a36Sopenharmony_ci
25262306a36Sopenharmony_ci	ret = devm_iio_device_register(dev, indio_dev);
25362306a36Sopenharmony_ci	if (ret)
25462306a36Sopenharmony_ci		return dev_err_probe(dev, ret, "could not register the device\n");
25562306a36Sopenharmony_ci
25662306a36Sopenharmony_ci	return 0;
25762306a36Sopenharmony_ci}
25862306a36Sopenharmony_ci
25962306a36Sopenharmony_cistatic const struct of_device_id sun20i_gpadc_of_id[] = {
26062306a36Sopenharmony_ci	{ .compatible = "allwinner,sun20i-d1-gpadc" },
26162306a36Sopenharmony_ci	{ /* sentinel */ }
26262306a36Sopenharmony_ci};
26362306a36Sopenharmony_ciMODULE_DEVICE_TABLE(of, sun20i_gpadc_of_id);
26462306a36Sopenharmony_ci
26562306a36Sopenharmony_cistatic struct platform_driver sun20i_gpadc_driver = {
26662306a36Sopenharmony_ci	.driver = {
26762306a36Sopenharmony_ci		.name = SUN20I_GPADC_DRIVER_NAME,
26862306a36Sopenharmony_ci		.of_match_table = sun20i_gpadc_of_id,
26962306a36Sopenharmony_ci	},
27062306a36Sopenharmony_ci	.probe = sun20i_gpadc_probe,
27162306a36Sopenharmony_ci};
27262306a36Sopenharmony_cimodule_platform_driver(sun20i_gpadc_driver);
27362306a36Sopenharmony_ci
27462306a36Sopenharmony_ciMODULE_DESCRIPTION("ADC driver for sunxi platforms");
27562306a36Sopenharmony_ciMODULE_AUTHOR("Maksim Kiselev <bigunclemax@gmail.com>");
27662306a36Sopenharmony_ciMODULE_LICENSE("GPL");
277