blob: 178cbc55caea2223d6e89270d00dba628a9f0766 [file] [log] [blame]
Arun Murthydae2db32011-02-22 10:11:13 +01001/*
2 * Copyright (C) ST-Ericsson SA 2010
3 *
4 * License Terms: GNU General Public License v2
5 * Author: Arun R Murthy <arun.murthy@stericsson.com>
Daniel Willerud63219922011-03-05 11:46:13 +01006 * Author: Daniel Willerud <daniel.willerud@stericsson.com>
Arun Murthydae2db32011-02-22 10:11:13 +01007 */
8#include <linux/init.h>
9#include <linux/module.h>
10#include <linux/device.h>
11#include <linux/interrupt.h>
12#include <linux/spinlock.h>
13#include <linux/delay.h>
14#include <linux/platform_device.h>
15#include <linux/completion.h>
16#include <linux/regulator/consumer.h>
17#include <linux/err.h>
18#include <linux/slab.h>
Daniel Willerud63219922011-03-05 11:46:13 +010019#include <linux/list.h>
Arun Murthydae2db32011-02-22 10:11:13 +010020#include <linux/mfd/ab8500.h>
21#include <linux/mfd/abx500.h>
Daniel Willerudcf169432011-03-05 11:46:01 +010022#include <linux/mfd/ab8500/ab8500-gpadc.h>
Arun Murthydae2db32011-02-22 10:11:13 +010023
24/*
25 * GPADC register offsets
26 * Bank : 0x0A
27 */
28#define AB8500_GPADC_CTRL1_REG 0x00
29#define AB8500_GPADC_CTRL2_REG 0x01
30#define AB8500_GPADC_CTRL3_REG 0x02
31#define AB8500_GPADC_AUTO_TIMER_REG 0x03
32#define AB8500_GPADC_STAT_REG 0x04
33#define AB8500_GPADC_MANDATAL_REG 0x05
34#define AB8500_GPADC_MANDATAH_REG 0x06
35#define AB8500_GPADC_AUTODATAL_REG 0x07
36#define AB8500_GPADC_AUTODATAH_REG 0x08
37#define AB8500_GPADC_MUX_CTRL_REG 0x09
38
39/* gpadc constants */
40#define EN_VINTCORE12 0x04
41#define EN_VTVOUT 0x02
42#define EN_GPADC 0x01
43#define DIS_GPADC 0x00
44#define SW_AVG_16 0x60
45#define ADC_SW_CONV 0x04
46#define EN_BUF 0x40
47#define DIS_ZERO 0x00
48#define GPADC_BUSY 0x01
49
50/**
51 * struct ab8500_gpadc - ab8500 GPADC device information
52 * @dev: pointer to the struct device
Daniel Willerud63219922011-03-05 11:46:13 +010053 * @node: a list of AB8500 GPADCs, hence prepared for
54 reentrance
Arun Murthydae2db32011-02-22 10:11:13 +010055 * @ab8500_gpadc_complete: pointer to the struct completion, to indicate
56 * the completion of gpadc conversion
57 * @ab8500_gpadc_lock: structure of type mutex
58 * @regu: pointer to the struct regulator
59 * @irq: interrupt number that is used by gpadc
60 */
Daniel Willerud63219922011-03-05 11:46:13 +010061struct ab8500_gpadc {
Arun Murthydae2db32011-02-22 10:11:13 +010062 struct device *dev;
Daniel Willerud63219922011-03-05 11:46:13 +010063 struct list_head node;
Arun Murthydae2db32011-02-22 10:11:13 +010064 struct completion ab8500_gpadc_complete;
65 struct mutex ab8500_gpadc_lock;
66 struct regulator *regu;
67 int irq;
Daniel Willerud63219922011-03-05 11:46:13 +010068};
69
70static LIST_HEAD(ab8500_gpadc_list);
71
72/**
73 * ab8500_gpadc_get() - returns a reference to the primary AB8500 GPADC
74 * (i.e. the first GPADC in the instance list)
75 */
76struct ab8500_gpadc *ab8500_gpadc_get(char *name)
77{
78 struct ab8500_gpadc *gpadc;
79
80 list_for_each_entry(gpadc, &ab8500_gpadc_list, node) {
81 if (!strcmp(name, dev_name(gpadc->dev)))
82 return gpadc;
83 }
84
85 return ERR_PTR(-ENOENT);
86}
87EXPORT_SYMBOL(ab8500_gpadc_get);
Arun Murthydae2db32011-02-22 10:11:13 +010088
89/**
90 * ab8500_gpadc_convert() - gpadc conversion
91 * @input: analog input to be converted to digital data
92 *
93 * This function converts the selected analog i/p to digital
94 * data. Thereafter calibration has to be made to obtain the
95 * data in the required quantity measurement.
96 */
Daniel Willerud63219922011-03-05 11:46:13 +010097int ab8500_gpadc_convert(struct ab8500_gpadc *gpadc, u8 input)
Arun Murthydae2db32011-02-22 10:11:13 +010098{
99 int ret;
100 u16 data = 0;
101 int looplimit = 0;
102 u8 val, low_data, high_data;
103
Daniel Willerud63219922011-03-05 11:46:13 +0100104 if (!gpadc)
Arun Murthydae2db32011-02-22 10:11:13 +0100105 return -ENODEV;
106
Daniel Willerud63219922011-03-05 11:46:13 +0100107 mutex_lock(&gpadc->ab8500_gpadc_lock);
Arun Murthydae2db32011-02-22 10:11:13 +0100108 /* Enable VTVout LDO this is required for GPADC */
Daniel Willerud63219922011-03-05 11:46:13 +0100109 regulator_enable(gpadc->regu);
Arun Murthydae2db32011-02-22 10:11:13 +0100110
111 /* Check if ADC is not busy, lock and proceed */
112 do {
Daniel Willerud63219922011-03-05 11:46:13 +0100113 ret = abx500_get_register_interruptible(gpadc->dev,
114 AB8500_GPADC, AB8500_GPADC_STAT_REG, &val);
Arun Murthydae2db32011-02-22 10:11:13 +0100115 if (ret < 0)
116 goto out;
117 if (!(val & GPADC_BUSY))
118 break;
119 msleep(10);
120 } while (++looplimit < 10);
121 if (looplimit >= 10 && (val & GPADC_BUSY)) {
Daniel Willerud63219922011-03-05 11:46:13 +0100122 dev_err(gpadc->dev, "gpadc_conversion: GPADC busy");
Arun Murthydae2db32011-02-22 10:11:13 +0100123 ret = -EINVAL;
124 goto out;
125 }
126
127 /* Enable GPADC */
Daniel Willerud63219922011-03-05 11:46:13 +0100128 ret = abx500_mask_and_set_register_interruptible(gpadc->dev,
129 AB8500_GPADC, AB8500_GPADC_CTRL1_REG, EN_GPADC, EN_GPADC);
Arun Murthydae2db32011-02-22 10:11:13 +0100130 if (ret < 0) {
Daniel Willerud63219922011-03-05 11:46:13 +0100131 dev_err(gpadc->dev, "gpadc_conversion: enable gpadc failed\n");
Arun Murthydae2db32011-02-22 10:11:13 +0100132 goto out;
133 }
134 /* Select the input source and set average samples to 16 */
Daniel Willerud63219922011-03-05 11:46:13 +0100135 ret = abx500_set_register_interruptible(gpadc->dev, AB8500_GPADC,
Arun Murthydae2db32011-02-22 10:11:13 +0100136 AB8500_GPADC_CTRL2_REG, (input | SW_AVG_16));
137 if (ret < 0) {
Daniel Willerud63219922011-03-05 11:46:13 +0100138 dev_err(gpadc->dev,
Arun Murthydae2db32011-02-22 10:11:13 +0100139 "gpadc_conversion: set avg samples failed\n");
140 goto out;
141 }
142 /* Enable ADC, Buffering and select rising edge, start Conversion */
Daniel Willerud63219922011-03-05 11:46:13 +0100143 ret = abx500_mask_and_set_register_interruptible(gpadc->dev,
144 AB8500_GPADC, AB8500_GPADC_CTRL1_REG, EN_BUF, EN_BUF);
Arun Murthydae2db32011-02-22 10:11:13 +0100145 if (ret < 0) {
Daniel Willerud63219922011-03-05 11:46:13 +0100146 dev_err(gpadc->dev,
Arun Murthydae2db32011-02-22 10:11:13 +0100147 "gpadc_conversion: select falling edge failed\n");
148 goto out;
149 }
Daniel Willerud63219922011-03-05 11:46:13 +0100150 ret = abx500_mask_and_set_register_interruptible(gpadc->dev,
151 AB8500_GPADC, AB8500_GPADC_CTRL1_REG, ADC_SW_CONV, ADC_SW_CONV);
Arun Murthydae2db32011-02-22 10:11:13 +0100152 if (ret < 0) {
Daniel Willerud63219922011-03-05 11:46:13 +0100153 dev_err(gpadc->dev,
Arun Murthydae2db32011-02-22 10:11:13 +0100154 "gpadc_conversion: start s/w conversion failed\n");
155 goto out;
156 }
157 /* wait for completion of conversion */
Daniel Willerud63219922011-03-05 11:46:13 +0100158 if (!wait_for_completion_timeout(&gpadc->ab8500_gpadc_complete, 2*HZ)) {
159 dev_err(gpadc->dev,
Arun Murthydae2db32011-02-22 10:11:13 +0100160 "timeout: didnt recieve GPADC conversion interrupt\n");
161 ret = -EINVAL;
162 goto out;
163 }
164
165 /* Read the converted RAW data */
Daniel Willerud63219922011-03-05 11:46:13 +0100166 ret = abx500_get_register_interruptible(gpadc->dev, AB8500_GPADC,
Arun Murthydae2db32011-02-22 10:11:13 +0100167 AB8500_GPADC_MANDATAL_REG, &low_data);
168 if (ret < 0) {
Daniel Willerud63219922011-03-05 11:46:13 +0100169 dev_err(gpadc->dev, "gpadc_conversion: read low data failed\n");
Arun Murthydae2db32011-02-22 10:11:13 +0100170 goto out;
171 }
172
Daniel Willerud63219922011-03-05 11:46:13 +0100173 ret = abx500_get_register_interruptible(gpadc->dev, AB8500_GPADC,
Arun Murthydae2db32011-02-22 10:11:13 +0100174 AB8500_GPADC_MANDATAH_REG, &high_data);
175 if (ret < 0) {
Daniel Willerud63219922011-03-05 11:46:13 +0100176 dev_err(gpadc->dev,
177 "gpadc_conversion: read high data failed\n");
Arun Murthydae2db32011-02-22 10:11:13 +0100178 goto out;
179 }
180
181 data = (high_data << 8) | low_data;
182 /* Disable GPADC */
Daniel Willerud63219922011-03-05 11:46:13 +0100183 ret = abx500_set_register_interruptible(gpadc->dev, AB8500_GPADC,
Arun Murthydae2db32011-02-22 10:11:13 +0100184 AB8500_GPADC_CTRL1_REG, DIS_GPADC);
185 if (ret < 0) {
Daniel Willerud63219922011-03-05 11:46:13 +0100186 dev_err(gpadc->dev, "gpadc_conversion: disable gpadc failed\n");
Arun Murthydae2db32011-02-22 10:11:13 +0100187 goto out;
188 }
189 /* Disable VTVout LDO this is required for GPADC */
Daniel Willerud63219922011-03-05 11:46:13 +0100190 regulator_disable(gpadc->regu);
191 mutex_unlock(&gpadc->ab8500_gpadc_lock);
Arun Murthydae2db32011-02-22 10:11:13 +0100192 return data;
193
194out:
195 /*
196 * It has shown to be needed to turn off the GPADC if an error occurs,
197 * otherwise we might have problem when waiting for the busy bit in the
198 * GPADC status register to go low. In V1.1 there wait_for_completion
199 * seems to timeout when waiting for an interrupt.. Not seen in V2.0
200 */
Daniel Willerud63219922011-03-05 11:46:13 +0100201 (void) abx500_set_register_interruptible(gpadc->dev, AB8500_GPADC,
Arun Murthydae2db32011-02-22 10:11:13 +0100202 AB8500_GPADC_CTRL1_REG, DIS_GPADC);
Daniel Willerud63219922011-03-05 11:46:13 +0100203 regulator_disable(gpadc->regu);
204 mutex_unlock(&gpadc->ab8500_gpadc_lock);
205 dev_err(gpadc->dev,
206 "gpadc_conversion: Failed to AD convert channel %d\n", input);
Arun Murthydae2db32011-02-22 10:11:13 +0100207 return ret;
208}
209EXPORT_SYMBOL(ab8500_gpadc_convert);
210
211/**
212 * ab8500_bm_gpswadcconvend_handler() - isr for s/w gpadc conversion completion
213 * @irq: irq number
214 * @data: pointer to the data passed during request irq
215 *
216 * This is a interrupt service routine for s/w gpadc conversion completion.
217 * Notifies the gpadc completion is completed and the converted raw value
218 * can be read from the registers.
219 * Returns IRQ status(IRQ_HANDLED)
220 */
Daniel Willerud63219922011-03-05 11:46:13 +0100221static irqreturn_t ab8500_bm_gpswadcconvend_handler(int irq, void *_gpadc)
Arun Murthydae2db32011-02-22 10:11:13 +0100222{
Daniel Willerud63219922011-03-05 11:46:13 +0100223 struct ab8500_gpadc *gpadc = _gpadc;
Arun Murthydae2db32011-02-22 10:11:13 +0100224
225 complete(&gpadc->ab8500_gpadc_complete);
226
227 return IRQ_HANDLED;
228}
229
230static int __devinit ab8500_gpadc_probe(struct platform_device *pdev)
231{
232 int ret = 0;
233 struct ab8500_gpadc *gpadc;
234
235 gpadc = kzalloc(sizeof(struct ab8500_gpadc), GFP_KERNEL);
236 if (!gpadc) {
237 dev_err(&pdev->dev, "Error: No memory\n");
238 return -ENOMEM;
239 }
240
Arun Murthydae2db32011-02-22 10:11:13 +0100241 gpadc->irq = platform_get_irq_byname(pdev, "SW_CONV_END");
242 if (gpadc->irq < 0) {
Daniel Willerud63219922011-03-05 11:46:13 +0100243 dev_err(gpadc->dev, "failed to get platform irq-%d\n",
244 gpadc->irq);
Arun Murthydae2db32011-02-22 10:11:13 +0100245 ret = gpadc->irq;
246 goto fail;
247 }
248
249 gpadc->dev = &pdev->dev;
Daniel Willerud63219922011-03-05 11:46:13 +0100250 mutex_init(&gpadc->ab8500_gpadc_lock);
Arun Murthydae2db32011-02-22 10:11:13 +0100251
252 /* Initialize completion used to notify completion of conversion */
253 init_completion(&gpadc->ab8500_gpadc_complete);
254
255 /* Register interrupt - SwAdcComplete */
256 ret = request_threaded_irq(gpadc->irq, NULL,
257 ab8500_bm_gpswadcconvend_handler,
258 IRQF_NO_SUSPEND | IRQF_SHARED, "ab8500-gpadc", gpadc);
259 if (ret < 0) {
260 dev_err(gpadc->dev, "Failed to register interrupt, irq: %d\n",
261 gpadc->irq);
262 goto fail;
263 }
264
265 /* VTVout LDO used to power up ab8500-GPADC */
266 gpadc->regu = regulator_get(&pdev->dev, "vddadc");
267 if (IS_ERR(gpadc->regu)) {
268 ret = PTR_ERR(gpadc->regu);
269 dev_err(gpadc->dev, "failed to get vtvout LDO\n");
270 goto fail;
271 }
Daniel Willerud63219922011-03-05 11:46:13 +0100272 list_add_tail(&gpadc->node, &ab8500_gpadc_list);
Arun Murthydae2db32011-02-22 10:11:13 +0100273 dev_dbg(gpadc->dev, "probe success\n");
274 return 0;
275fail:
276 kfree(gpadc);
277 gpadc = NULL;
278 return ret;
279}
280
281static int __devexit ab8500_gpadc_remove(struct platform_device *pdev)
282{
283 struct ab8500_gpadc *gpadc = platform_get_drvdata(pdev);
284
Daniel Willerud63219922011-03-05 11:46:13 +0100285 /* remove this gpadc entry from the list */
286 list_del(&gpadc->node);
Arun Murthydae2db32011-02-22 10:11:13 +0100287 /* remove interrupt - completion of Sw ADC conversion */
Daniel Willerud63219922011-03-05 11:46:13 +0100288 free_irq(gpadc->irq, gpadc);
Arun Murthydae2db32011-02-22 10:11:13 +0100289 /* disable VTVout LDO that is being used by GPADC */
290 regulator_put(gpadc->regu);
291 kfree(gpadc);
292 gpadc = NULL;
293 return 0;
294}
295
296static struct platform_driver ab8500_gpadc_driver = {
297 .probe = ab8500_gpadc_probe,
298 .remove = __devexit_p(ab8500_gpadc_remove),
299 .driver = {
300 .name = "ab8500-gpadc",
301 .owner = THIS_MODULE,
302 },
303};
304
305static int __init ab8500_gpadc_init(void)
306{
307 return platform_driver_register(&ab8500_gpadc_driver);
308}
309
310static void __exit ab8500_gpadc_exit(void)
311{
312 platform_driver_unregister(&ab8500_gpadc_driver);
313}
314
315subsys_initcall_sync(ab8500_gpadc_init);
316module_exit(ab8500_gpadc_exit);
317
318MODULE_LICENSE("GPL v2");
Daniel Willerud63219922011-03-05 11:46:13 +0100319MODULE_AUTHOR("Arun R Murthy, Daniel Willerud");
Arun Murthydae2db32011-02-22 10:11:13 +0100320MODULE_ALIAS("platform:ab8500_gpadc");
321MODULE_DESCRIPTION("AB8500 GPADC driver");