blob: c50fba7be778488dd2964dc3953299402c1ab078 [file] [log] [blame]
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -05001/*
2 * MFD driver for TWL6040 audio device
3 *
4 * Authors: Misael Lopez Cruz <misael.lopez@ti.com>
5 * Jorge Eduardo Candelaria <jorge.candelaria@ti.com>
6 * Peter Ujfalusi <peter.ujfalusi@ti.com>
7 *
8 * Copyright: (C) 2011 Texas Instruments, Inc.
9 *
10 * This program is free software; you can redistribute it and/or modify
11 * it under the terms of the GNU General Public License version 2 as
12 * published by the Free Software Foundation.
13 *
14 * This program is distributed in the hope that it will be useful, but
15 * WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
17 * General Public License for more details.
18 *
19 * You should have received a copy of the GNU General Public License
20 * along with this program; if not, write to the Free Software
21 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
22 * 02110-1301 USA
23 *
24 */
25
26#include <linux/module.h>
27#include <linux/types.h>
28#include <linux/slab.h>
29#include <linux/kernel.h>
Peter Ujfalusi5af7df62012-05-02 16:54:42 +030030#include <linux/err.h>
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -050031#include <linux/platform_device.h>
32#include <linux/gpio.h>
33#include <linux/delay.h>
Peter Ujfalusi8eaeb932012-04-03 11:56:51 +030034#include <linux/i2c.h>
35#include <linux/regmap.h>
36#include <linux/err.h>
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -050037#include <linux/mfd/core.h>
38#include <linux/mfd/twl6040.h>
Peter Ujfalusi5af7df62012-05-02 16:54:42 +030039#include <linux/regulator/consumer.h>
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -050040
Peter Ujfalusi31b402e2011-10-12 11:57:54 +030041#define VIBRACTRL_MEMBER(reg) ((reg == TWL6040_REG_VIBCTLL) ? 0 : 1)
Peter Ujfalusi5af7df62012-05-02 16:54:42 +030042#define TWL6040_NUM_SUPPLIES (2)
Peter Ujfalusi31b402e2011-10-12 11:57:54 +030043
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -050044int twl6040_reg_read(struct twl6040 *twl6040, unsigned int reg)
45{
46 int ret;
Peter Ujfalusi8eaeb932012-04-03 11:56:51 +030047 unsigned int val;
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -050048
49 mutex_lock(&twl6040->io_mutex);
Peter Ujfalusi31b402e2011-10-12 11:57:54 +030050 /* Vibra control registers from cache */
51 if (unlikely(reg == TWL6040_REG_VIBCTLL ||
52 reg == TWL6040_REG_VIBCTLR)) {
53 val = twl6040->vibra_ctrl_cache[VIBRACTRL_MEMBER(reg)];
54 } else {
Peter Ujfalusi8eaeb932012-04-03 11:56:51 +030055 ret = regmap_read(twl6040->regmap, reg, &val);
Peter Ujfalusi31b402e2011-10-12 11:57:54 +030056 if (ret < 0) {
57 mutex_unlock(&twl6040->io_mutex);
58 return ret;
59 }
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -050060 }
61 mutex_unlock(&twl6040->io_mutex);
62
63 return val;
64}
65EXPORT_SYMBOL(twl6040_reg_read);
66
67int twl6040_reg_write(struct twl6040 *twl6040, unsigned int reg, u8 val)
68{
69 int ret;
70
71 mutex_lock(&twl6040->io_mutex);
Peter Ujfalusi8eaeb932012-04-03 11:56:51 +030072 ret = regmap_write(twl6040->regmap, reg, val);
Peter Ujfalusi31b402e2011-10-12 11:57:54 +030073 /* Cache the vibra control registers */
74 if (reg == TWL6040_REG_VIBCTLL || reg == TWL6040_REG_VIBCTLR)
75 twl6040->vibra_ctrl_cache[VIBRACTRL_MEMBER(reg)] = val;
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -050076 mutex_unlock(&twl6040->io_mutex);
77
78 return ret;
79}
80EXPORT_SYMBOL(twl6040_reg_write);
81
82int twl6040_set_bits(struct twl6040 *twl6040, unsigned int reg, u8 mask)
83{
84 int ret;
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -050085
86 mutex_lock(&twl6040->io_mutex);
Peter Ujfalusi8eaeb932012-04-03 11:56:51 +030087 ret = regmap_update_bits(twl6040->regmap, reg, mask, mask);
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -050088 mutex_unlock(&twl6040->io_mutex);
89 return ret;
90}
91EXPORT_SYMBOL(twl6040_set_bits);
92
93int twl6040_clear_bits(struct twl6040 *twl6040, unsigned int reg, u8 mask)
94{
95 int ret;
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -050096
97 mutex_lock(&twl6040->io_mutex);
Peter Ujfalusi8eaeb932012-04-03 11:56:51 +030098 ret = regmap_update_bits(twl6040->regmap, reg, mask, 0);
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -050099 mutex_unlock(&twl6040->io_mutex);
100 return ret;
101}
102EXPORT_SYMBOL(twl6040_clear_bits);
103
104/* twl6040 codec manual power-up sequence */
105static int twl6040_power_up(struct twl6040 *twl6040)
106{
107 u8 ldoctl, ncpctl, lppllctl;
108 int ret;
109
110 /* enable high-side LDO, reference system and internal oscillator */
111 ldoctl = TWL6040_HSLDOENA | TWL6040_REFENA | TWL6040_OSCENA;
112 ret = twl6040_reg_write(twl6040, TWL6040_REG_LDOCTL, ldoctl);
113 if (ret)
114 return ret;
115 usleep_range(10000, 10500);
116
117 /* enable negative charge pump */
118 ncpctl = TWL6040_NCPENA;
119 ret = twl6040_reg_write(twl6040, TWL6040_REG_NCPCTL, ncpctl);
120 if (ret)
121 goto ncp_err;
122 usleep_range(1000, 1500);
123
124 /* enable low-side LDO */
125 ldoctl |= TWL6040_LSLDOENA;
126 ret = twl6040_reg_write(twl6040, TWL6040_REG_LDOCTL, ldoctl);
127 if (ret)
128 goto lsldo_err;
129 usleep_range(1000, 1500);
130
131 /* enable low-power PLL */
132 lppllctl = TWL6040_LPLLENA;
133 ret = twl6040_reg_write(twl6040, TWL6040_REG_LPPLLCTL, lppllctl);
134 if (ret)
135 goto lppll_err;
136 usleep_range(5000, 5500);
137
138 /* disable internal oscillator */
139 ldoctl &= ~TWL6040_OSCENA;
140 ret = twl6040_reg_write(twl6040, TWL6040_REG_LDOCTL, ldoctl);
141 if (ret)
142 goto osc_err;
143
144 return 0;
145
146osc_err:
147 lppllctl &= ~TWL6040_LPLLENA;
148 twl6040_reg_write(twl6040, TWL6040_REG_LPPLLCTL, lppllctl);
149lppll_err:
150 ldoctl &= ~TWL6040_LSLDOENA;
151 twl6040_reg_write(twl6040, TWL6040_REG_LDOCTL, ldoctl);
152lsldo_err:
153 ncpctl &= ~TWL6040_NCPENA;
154 twl6040_reg_write(twl6040, TWL6040_REG_NCPCTL, ncpctl);
155ncp_err:
156 ldoctl &= ~(TWL6040_HSLDOENA | TWL6040_REFENA | TWL6040_OSCENA);
157 twl6040_reg_write(twl6040, TWL6040_REG_LDOCTL, ldoctl);
158
159 return ret;
160}
161
162/* twl6040 manual power-down sequence */
163static void twl6040_power_down(struct twl6040 *twl6040)
164{
165 u8 ncpctl, ldoctl, lppllctl;
166
167 ncpctl = twl6040_reg_read(twl6040, TWL6040_REG_NCPCTL);
168 ldoctl = twl6040_reg_read(twl6040, TWL6040_REG_LDOCTL);
169 lppllctl = twl6040_reg_read(twl6040, TWL6040_REG_LPPLLCTL);
170
171 /* enable internal oscillator */
172 ldoctl |= TWL6040_OSCENA;
173 twl6040_reg_write(twl6040, TWL6040_REG_LDOCTL, ldoctl);
174 usleep_range(1000, 1500);
175
176 /* disable low-power PLL */
177 lppllctl &= ~TWL6040_LPLLENA;
178 twl6040_reg_write(twl6040, TWL6040_REG_LPPLLCTL, lppllctl);
179
180 /* disable low-side LDO */
181 ldoctl &= ~TWL6040_LSLDOENA;
182 twl6040_reg_write(twl6040, TWL6040_REG_LDOCTL, ldoctl);
183
184 /* disable negative charge pump */
185 ncpctl &= ~TWL6040_NCPENA;
186 twl6040_reg_write(twl6040, TWL6040_REG_NCPCTL, ncpctl);
187
188 /* disable high-side LDO, reference system and internal oscillator */
189 ldoctl &= ~(TWL6040_HSLDOENA | TWL6040_REFENA | TWL6040_OSCENA);
190 twl6040_reg_write(twl6040, TWL6040_REG_LDOCTL, ldoctl);
191}
192
193static irqreturn_t twl6040_naudint_handler(int irq, void *data)
194{
195 struct twl6040 *twl6040 = data;
196 u8 intid, status;
197
198 intid = twl6040_reg_read(twl6040, TWL6040_REG_INTID);
199
200 if (intid & TWL6040_READYINT)
201 complete(&twl6040->ready);
202
203 if (intid & TWL6040_THINT) {
204 status = twl6040_reg_read(twl6040, TWL6040_REG_STATUS);
205 if (status & TWL6040_TSHUTDET) {
Peter Ujfalusi2d7c9572011-09-15 15:39:23 +0300206 dev_warn(twl6040->dev,
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -0500207 "Thermal shutdown, powering-off");
208 twl6040_power(twl6040, 0);
209 } else {
Peter Ujfalusi2d7c9572011-09-15 15:39:23 +0300210 dev_warn(twl6040->dev,
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -0500211 "Leaving thermal shutdown, powering-on");
212 twl6040_power(twl6040, 1);
213 }
214 }
215
216 return IRQ_HANDLED;
217}
218
219static int twl6040_power_up_completion(struct twl6040 *twl6040,
220 int naudint)
221{
222 int time_left;
223 u8 intid;
224
225 time_left = wait_for_completion_timeout(&twl6040->ready,
226 msecs_to_jiffies(144));
227 if (!time_left) {
228 intid = twl6040_reg_read(twl6040, TWL6040_REG_INTID);
229 if (!(intid & TWL6040_READYINT)) {
Peter Ujfalusi2d7c9572011-09-15 15:39:23 +0300230 dev_err(twl6040->dev,
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -0500231 "timeout waiting for READYINT\n");
232 return -ETIMEDOUT;
233 }
234 }
235
236 return 0;
237}
238
239int twl6040_power(struct twl6040 *twl6040, int on)
240{
241 int audpwron = twl6040->audpwron;
242 int naudint = twl6040->irq;
243 int ret = 0;
244
245 mutex_lock(&twl6040->mutex);
246
247 if (on) {
248 /* already powered-up */
249 if (twl6040->power_count++)
250 goto out;
251
252 if (gpio_is_valid(audpwron)) {
253 /* use AUDPWRON line */
254 gpio_set_value(audpwron, 1);
255 /* wait for power-up completion */
256 ret = twl6040_power_up_completion(twl6040, naudint);
257 if (ret) {
Peter Ujfalusi2d7c9572011-09-15 15:39:23 +0300258 dev_err(twl6040->dev,
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -0500259 "automatic power-down failed\n");
260 twl6040->power_count = 0;
261 goto out;
262 }
263 } else {
264 /* use manual power-up sequence */
265 ret = twl6040_power_up(twl6040);
266 if (ret) {
Peter Ujfalusi2d7c9572011-09-15 15:39:23 +0300267 dev_err(twl6040->dev,
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -0500268 "manual power-up failed\n");
269 twl6040->power_count = 0;
270 goto out;
271 }
272 }
Peter Ujfalusicfb7a332011-07-04 10:28:28 +0300273 /* Default PLL configuration after power up */
274 twl6040->pll = TWL6040_SYSCLK_SEL_LPPLL;
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -0500275 twl6040->sysclk = 19200000;
Peter Ujfalusif8447d62012-01-14 20:58:43 +0100276 twl6040->mclk = 32768;
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -0500277 } else {
278 /* already powered-down */
279 if (!twl6040->power_count) {
Peter Ujfalusi2d7c9572011-09-15 15:39:23 +0300280 dev_err(twl6040->dev,
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -0500281 "device is already powered-off\n");
282 ret = -EPERM;
283 goto out;
284 }
285
286 if (--twl6040->power_count)
287 goto out;
288
289 if (gpio_is_valid(audpwron)) {
290 /* use AUDPWRON line */
291 gpio_set_value(audpwron, 0);
292
293 /* power-down sequence latency */
294 usleep_range(500, 700);
295 } else {
296 /* use manual power-down sequence */
297 twl6040_power_down(twl6040);
298 }
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -0500299 twl6040->sysclk = 0;
Peter Ujfalusif8447d62012-01-14 20:58:43 +0100300 twl6040->mclk = 0;
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -0500301 }
302
303out:
304 mutex_unlock(&twl6040->mutex);
305 return ret;
306}
307EXPORT_SYMBOL(twl6040_power);
308
Peter Ujfalusicfb7a332011-07-04 10:28:28 +0300309int twl6040_set_pll(struct twl6040 *twl6040, int pll_id,
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -0500310 unsigned int freq_in, unsigned int freq_out)
311{
312 u8 hppllctl, lppllctl;
313 int ret = 0;
314
315 mutex_lock(&twl6040->mutex);
316
317 hppllctl = twl6040_reg_read(twl6040, TWL6040_REG_HPPLLCTL);
318 lppllctl = twl6040_reg_read(twl6040, TWL6040_REG_LPPLLCTL);
319
Peter Ujfalusi2bd05db2012-01-14 20:58:44 +0100320 /* Force full reconfiguration when switching between PLL */
321 if (pll_id != twl6040->pll) {
322 twl6040->sysclk = 0;
323 twl6040->mclk = 0;
324 }
325
Peter Ujfalusicfb7a332011-07-04 10:28:28 +0300326 switch (pll_id) {
327 case TWL6040_SYSCLK_SEL_LPPLL:
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -0500328 /* low-power PLL divider */
Peter Ujfalusi2bd05db2012-01-14 20:58:44 +0100329 /* Change the sysclk configuration only if it has been canged */
330 if (twl6040->sysclk != freq_out) {
331 switch (freq_out) {
332 case 17640000:
333 lppllctl |= TWL6040_LPLLFIN;
334 break;
335 case 19200000:
336 lppllctl &= ~TWL6040_LPLLFIN;
337 break;
338 default:
339 dev_err(twl6040->dev,
340 "freq_out %d not supported\n",
341 freq_out);
342 ret = -EINVAL;
343 goto pll_out;
344 }
345 twl6040_reg_write(twl6040, TWL6040_REG_LPPLLCTL,
346 lppllctl);
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -0500347 }
Peter Ujfalusi2bd05db2012-01-14 20:58:44 +0100348
349 /* The PLL in use has not been change, we can exit */
350 if (twl6040->pll == pll_id)
351 break;
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -0500352
353 switch (freq_in) {
354 case 32768:
355 lppllctl |= TWL6040_LPLLENA;
356 twl6040_reg_write(twl6040, TWL6040_REG_LPPLLCTL,
357 lppllctl);
358 mdelay(5);
359 lppllctl &= ~TWL6040_HPLLSEL;
360 twl6040_reg_write(twl6040, TWL6040_REG_LPPLLCTL,
361 lppllctl);
362 hppllctl &= ~TWL6040_HPLLENA;
363 twl6040_reg_write(twl6040, TWL6040_REG_HPPLLCTL,
364 hppllctl);
365 break;
366 default:
Peter Ujfalusi2d7c9572011-09-15 15:39:23 +0300367 dev_err(twl6040->dev,
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -0500368 "freq_in %d not supported\n", freq_in);
369 ret = -EINVAL;
370 goto pll_out;
371 }
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -0500372 break;
Peter Ujfalusicfb7a332011-07-04 10:28:28 +0300373 case TWL6040_SYSCLK_SEL_HPPLL:
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -0500374 /* high-performance PLL can provide only 19.2 MHz */
375 if (freq_out != 19200000) {
Peter Ujfalusi2d7c9572011-09-15 15:39:23 +0300376 dev_err(twl6040->dev,
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -0500377 "freq_out %d not supported\n", freq_out);
378 ret = -EINVAL;
379 goto pll_out;
380 }
381
Peter Ujfalusi2bd05db2012-01-14 20:58:44 +0100382 if (twl6040->mclk != freq_in) {
383 hppllctl &= ~TWL6040_MCLK_MSK;
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -0500384
Peter Ujfalusi2bd05db2012-01-14 20:58:44 +0100385 switch (freq_in) {
386 case 12000000:
387 /* PLL enabled, active mode */
388 hppllctl |= TWL6040_MCLK_12000KHZ |
389 TWL6040_HPLLENA;
390 break;
391 case 19200000:
392 /*
393 * PLL disabled
394 * (enable PLL if MCLK jitter quality
395 * doesn't meet specification)
396 */
397 hppllctl |= TWL6040_MCLK_19200KHZ;
398 break;
399 case 26000000:
400 /* PLL enabled, active mode */
401 hppllctl |= TWL6040_MCLK_26000KHZ |
402 TWL6040_HPLLENA;
403 break;
404 case 38400000:
405 /* PLL enabled, active mode */
406 hppllctl |= TWL6040_MCLK_38400KHZ |
407 TWL6040_HPLLENA;
408 break;
409 default:
410 dev_err(twl6040->dev,
411 "freq_in %d not supported\n", freq_in);
412 ret = -EINVAL;
413 goto pll_out;
414 }
415
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -0500416 /*
Peter Ujfalusi2bd05db2012-01-14 20:58:44 +0100417 * enable clock slicer to ensure input waveform is
418 * square
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -0500419 */
Peter Ujfalusi2bd05db2012-01-14 20:58:44 +0100420 hppllctl |= TWL6040_HPLLSQRENA;
421
422 twl6040_reg_write(twl6040, TWL6040_REG_HPPLLCTL,
423 hppllctl);
424 usleep_range(500, 700);
425 lppllctl |= TWL6040_HPLLSEL;
426 twl6040_reg_write(twl6040, TWL6040_REG_LPPLLCTL,
427 lppllctl);
428 lppllctl &= ~TWL6040_LPLLENA;
429 twl6040_reg_write(twl6040, TWL6040_REG_LPPLLCTL,
430 lppllctl);
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -0500431 }
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -0500432 break;
433 default:
Peter Ujfalusi2d7c9572011-09-15 15:39:23 +0300434 dev_err(twl6040->dev, "unknown pll id %d\n", pll_id);
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -0500435 ret = -EINVAL;
436 goto pll_out;
437 }
438
439 twl6040->sysclk = freq_out;
Peter Ujfalusif8447d62012-01-14 20:58:43 +0100440 twl6040->mclk = freq_in;
Peter Ujfalusicfb7a332011-07-04 10:28:28 +0300441 twl6040->pll = pll_id;
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -0500442
443pll_out:
444 mutex_unlock(&twl6040->mutex);
445 return ret;
446}
447EXPORT_SYMBOL(twl6040_set_pll);
448
Peter Ujfalusicfb7a332011-07-04 10:28:28 +0300449int twl6040_get_pll(struct twl6040 *twl6040)
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -0500450{
Peter Ujfalusicfb7a332011-07-04 10:28:28 +0300451 if (twl6040->power_count)
452 return twl6040->pll;
453 else
454 return -ENODEV;
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -0500455}
456EXPORT_SYMBOL(twl6040_get_pll);
457
458unsigned int twl6040_get_sysclk(struct twl6040 *twl6040)
459{
460 return twl6040->sysclk;
461}
462EXPORT_SYMBOL(twl6040_get_sysclk);
463
Peter Ujfalusi70601ec2011-10-12 11:57:55 +0300464/* Get the combined status of the vibra control register */
465int twl6040_get_vibralr_status(struct twl6040 *twl6040)
466{
467 u8 status;
468
469 status = twl6040->vibra_ctrl_cache[0] | twl6040->vibra_ctrl_cache[1];
470 status &= (TWL6040_VIBENA | TWL6040_VIBSEL);
471
472 return status;
473}
474EXPORT_SYMBOL(twl6040_get_vibralr_status);
475
Peter Ujfalusi0f962ae2011-07-05 11:40:33 +0300476static struct resource twl6040_vibra_rsrc[] = {
477 {
478 .flags = IORESOURCE_IRQ,
479 },
480};
481
482static struct resource twl6040_codec_rsrc[] = {
483 {
484 .flags = IORESOURCE_IRQ,
485 },
486};
487
Peter Ujfalusi8eaeb932012-04-03 11:56:51 +0300488static bool twl6040_readable_reg(struct device *dev, unsigned int reg)
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -0500489{
Peter Ujfalusi8eaeb932012-04-03 11:56:51 +0300490 /* Register 0 is not readable */
491 if (!reg)
492 return false;
493 return true;
494}
495
496static struct regmap_config twl6040_regmap_config = {
497 .reg_bits = 8,
498 .val_bits = 8,
499 .max_register = TWL6040_REG_STATUS, /* 0x2e */
500
501 .readable_reg = twl6040_readable_reg,
502};
503
504static int __devinit twl6040_probe(struct i2c_client *client,
505 const struct i2c_device_id *id)
506{
507 struct twl6040_platform_data *pdata = client->dev.platform_data;
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -0500508 struct twl6040 *twl6040;
509 struct mfd_cell *cell = NULL;
510 int ret, children = 0;
511
512 if (!pdata) {
Peter Ujfalusi8eaeb932012-04-03 11:56:51 +0300513 dev_err(&client->dev, "Platform data is missing\n");
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -0500514 return -EINVAL;
515 }
516
Peter Ujfalusid20e1d22011-07-04 20:15:19 +0300517 /* In order to operate correctly we need valid interrupt config */
Peter Ujfalusi67124192012-05-16 14:11:56 +0300518 if (!client->irq) {
Peter Ujfalusi8eaeb932012-04-03 11:56:51 +0300519 dev_err(&client->dev, "Invalid IRQ configuration\n");
Peter Ujfalusid20e1d22011-07-04 20:15:19 +0300520 return -EINVAL;
521 }
522
Peter Ujfalusi8eaeb932012-04-03 11:56:51 +0300523 twl6040 = devm_kzalloc(&client->dev, sizeof(struct twl6040),
524 GFP_KERNEL);
525 if (!twl6040) {
526 ret = -ENOMEM;
527 goto err;
528 }
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -0500529
Axel Linbbf6adc2012-04-25 10:09:46 +0800530 twl6040->regmap = devm_regmap_init_i2c(client, &twl6040_regmap_config);
Peter Ujfalusi8eaeb932012-04-03 11:56:51 +0300531 if (IS_ERR(twl6040->regmap)) {
532 ret = PTR_ERR(twl6040->regmap);
533 goto err;
534 }
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -0500535
Peter Ujfalusi8eaeb932012-04-03 11:56:51 +0300536 i2c_set_clientdata(client, twl6040);
537
Peter Ujfalusi5af7df62012-05-02 16:54:42 +0300538 twl6040->supplies[0].supply = "vio";
539 twl6040->supplies[1].supply = "v2v1";
540 ret = regulator_bulk_get(&client->dev, TWL6040_NUM_SUPPLIES,
541 twl6040->supplies);
542 if (ret != 0) {
543 dev_err(&client->dev, "Failed to get supplies: %d\n", ret);
544 goto regulator_get_err;
545 }
546
547 ret = regulator_bulk_enable(TWL6040_NUM_SUPPLIES, twl6040->supplies);
548 if (ret != 0) {
549 dev_err(&client->dev, "Failed to enable supplies: %d\n", ret);
550 goto power_err;
551 }
552
Peter Ujfalusi8eaeb932012-04-03 11:56:51 +0300553 twl6040->dev = &client->dev;
554 twl6040->irq = client->irq;
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -0500555
556 mutex_init(&twl6040->mutex);
557 mutex_init(&twl6040->io_mutex);
558 init_completion(&twl6040->ready);
559
560 twl6040->rev = twl6040_reg_read(twl6040, TWL6040_REG_ASICREV);
561
Peter Ujfalusi77f63e02011-09-15 15:39:26 +0300562 /* ERRATA: Automatic power-up is not possible in ES1.0 */
563 if (twl6040_get_revid(twl6040) > TWL6040_REV_ES1_0)
564 twl6040->audpwron = pdata->audpwron_gpio;
565 else
566 twl6040->audpwron = -EINVAL;
567
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -0500568 if (gpio_is_valid(twl6040->audpwron)) {
Axel Linb04edb92011-12-01 09:55:07 +0800569 ret = gpio_request_one(twl6040->audpwron, GPIOF_OUT_INIT_LOW,
570 "audpwron");
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -0500571 if (ret)
Peter Ujfalusi5af7df62012-05-02 16:54:42 +0300572 goto gpio_err;
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -0500573 }
574
Peter Ujfalusid20e1d22011-07-04 20:15:19 +0300575 /* codec interrupt */
576 ret = twl6040_irq_init(twl6040);
577 if (ret)
Peter Ujfalusi5af7df62012-05-02 16:54:42 +0300578 goto irq_init_err;
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -0500579
Peter Ujfalusi1b7c4722011-07-04 20:16:23 +0300580 ret = request_threaded_irq(twl6040->irq_base + TWL6040_IRQ_READY,
581 NULL, twl6040_naudint_handler, 0,
582 "twl6040_irq_ready", twl6040);
Peter Ujfalusid20e1d22011-07-04 20:15:19 +0300583 if (ret) {
584 dev_err(twl6040->dev, "READY IRQ request failed: %d\n",
585 ret);
586 goto irq_err;
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -0500587 }
588
589 /* dual-access registers controlled by I2C only */
590 twl6040_set_bits(twl6040, TWL6040_REG_ACCCTL, TWL6040_I2CSEL);
591
592 if (pdata->codec) {
Peter Ujfalusi0f962ae2011-07-05 11:40:33 +0300593 int irq = twl6040->irq_base + TWL6040_IRQ_PLUG;
594
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -0500595 cell = &twl6040->cells[children];
596 cell->name = "twl6040-codec";
Peter Ujfalusi0f962ae2011-07-05 11:40:33 +0300597 twl6040_codec_rsrc[0].start = irq;
598 twl6040_codec_rsrc[0].end = irq;
599 cell->resources = twl6040_codec_rsrc;
600 cell->num_resources = ARRAY_SIZE(twl6040_codec_rsrc);
Peter Ujfalusi6c448632011-06-01 13:05:10 +0300601 cell->platform_data = pdata->codec;
602 cell->pdata_size = sizeof(*pdata->codec);
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -0500603 children++;
604 }
605
606 if (pdata->vibra) {
Peter Ujfalusi0f962ae2011-07-05 11:40:33 +0300607 int irq = twl6040->irq_base + TWL6040_IRQ_VIB;
608
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -0500609 cell = &twl6040->cells[children];
610 cell->name = "twl6040-vibra";
Peter Ujfalusi0f962ae2011-07-05 11:40:33 +0300611 twl6040_vibra_rsrc[0].start = irq;
612 twl6040_vibra_rsrc[0].end = irq;
613 cell->resources = twl6040_vibra_rsrc;
614 cell->num_resources = ARRAY_SIZE(twl6040_vibra_rsrc);
615
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -0500616 cell->platform_data = pdata->vibra;
617 cell->pdata_size = sizeof(*pdata->vibra);
618 children++;
619 }
620
621 if (children) {
Peter Ujfalusi8eaeb932012-04-03 11:56:51 +0300622 ret = mfd_add_devices(&client->dev, -1, twl6040->cells,
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -0500623 children, NULL, 0);
624 if (ret)
625 goto mfd_err;
626 } else {
Peter Ujfalusi8eaeb932012-04-03 11:56:51 +0300627 dev_err(&client->dev, "No platform data found for children\n");
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -0500628 ret = -ENODEV;
629 goto mfd_err;
630 }
631
632 return 0;
633
634mfd_err:
Peter Ujfalusi1b7c4722011-07-04 20:16:23 +0300635 free_irq(twl6040->irq_base + TWL6040_IRQ_READY, twl6040);
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -0500636irq_err:
Peter Ujfalusid20e1d22011-07-04 20:15:19 +0300637 twl6040_irq_exit(twl6040);
Peter Ujfalusi5af7df62012-05-02 16:54:42 +0300638irq_init_err:
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -0500639 if (gpio_is_valid(twl6040->audpwron))
640 gpio_free(twl6040->audpwron);
Peter Ujfalusi5af7df62012-05-02 16:54:42 +0300641gpio_err:
642 regulator_bulk_disable(TWL6040_NUM_SUPPLIES, twl6040->supplies);
643power_err:
644 regulator_bulk_free(TWL6040_NUM_SUPPLIES, twl6040->supplies);
645regulator_get_err:
Peter Ujfalusi8eaeb932012-04-03 11:56:51 +0300646 i2c_set_clientdata(client, NULL);
Peter Ujfalusi8eaeb932012-04-03 11:56:51 +0300647err:
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -0500648 return ret;
649}
650
Peter Ujfalusi8eaeb932012-04-03 11:56:51 +0300651static int __devexit twl6040_remove(struct i2c_client *client)
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -0500652{
Peter Ujfalusi8eaeb932012-04-03 11:56:51 +0300653 struct twl6040 *twl6040 = i2c_get_clientdata(client);
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -0500654
655 if (twl6040->power_count)
656 twl6040_power(twl6040, 0);
657
658 if (gpio_is_valid(twl6040->audpwron))
659 gpio_free(twl6040->audpwron);
660
Peter Ujfalusi1b7c4722011-07-04 20:16:23 +0300661 free_irq(twl6040->irq_base + TWL6040_IRQ_READY, twl6040);
Peter Ujfalusid20e1d22011-07-04 20:15:19 +0300662 twl6040_irq_exit(twl6040);
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -0500663
Peter Ujfalusi8eaeb932012-04-03 11:56:51 +0300664 mfd_remove_devices(&client->dev);
665 i2c_set_clientdata(client, NULL);
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -0500666
Peter Ujfalusi5af7df62012-05-02 16:54:42 +0300667 regulator_bulk_disable(TWL6040_NUM_SUPPLIES, twl6040->supplies);
668 regulator_bulk_free(TWL6040_NUM_SUPPLIES, twl6040->supplies);
669
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -0500670 return 0;
671}
672
Peter Ujfalusi8eaeb932012-04-03 11:56:51 +0300673static const struct i2c_device_id twl6040_i2c_id[] = {
674 { "twl6040", 0, },
675 { },
676};
677MODULE_DEVICE_TABLE(i2c, twl6040_i2c_id);
678
679static struct i2c_driver twl6040_driver = {
680 .driver = {
681 .name = "twl6040",
682 .owner = THIS_MODULE,
683 },
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -0500684 .probe = twl6040_probe,
685 .remove = __devexit_p(twl6040_remove),
Peter Ujfalusi8eaeb932012-04-03 11:56:51 +0300686 .id_table = twl6040_i2c_id,
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -0500687};
688
Peter Ujfalusi8eaeb932012-04-03 11:56:51 +0300689module_i2c_driver(twl6040_driver);
Misael Lopez Cruzf19b2822011-04-27 02:14:07 -0500690
691MODULE_DESCRIPTION("TWL6040 MFD");
692MODULE_AUTHOR("Misael Lopez Cruz <misael.lopez@ti.com>");
693MODULE_AUTHOR("Jorge Eduardo Candelaria <jorge.candelaria@ti.com>");
694MODULE_LICENSE("GPL");
695MODULE_ALIAS("platform:twl6040");