blob: 3d87f0b31889f47a8e9f2fa49abb71ea42dc40a5 [file] [log] [blame]
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001/* Copyright (c) 2010-2011, Code Aurora Forum. All rights reserved.
2 *
3 * This program is free software; you can redistribute it and/or modify
4 * it under the terms of the GNU General Public License version 2 and
5 * only version 2 as published by the Free Software Foundation.
6 *
7 * This program is distributed in the hope that it will be useful,
8 * but WITHOUT ANY WARRANTY; without even the implied warranty of
9 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
10 * GNU General Public License for more details.
11 */
12
13#include <linux/interrupt.h>
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070014#include <linux/slab.h>
15#include <linux/ratelimit.h>
Anirudh Ghayalc8051a82011-11-17 09:28:24 +053016#include <linux/gpio.h>
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070017#include <linux/mfd/core.h>
Anirudh Ghayalc8051a82011-11-17 09:28:24 +053018#include <linux/msm_ssbi.h>
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070019#include <linux/mfd/pmic8901.h>
20#include <linux/platform_device.h>
21#include <linux/debugfs.h>
David Collins88ebc9c2011-11-15 08:46:08 -080022#include <linux/delay.h>
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070023
24/* PMIC8901 Revision */
25#define SSBI_REG_REV 0x002 /* PMIC4 revision */
26
27/* PMIC8901 IRQ */
28#define SSBI_REG_ADDR_IRQ_BASE 0xD5
29
30#define SSBI_REG_ADDR_IRQ_ROOT (SSBI_REG_ADDR_IRQ_BASE + 0)
31#define SSBI_REG_ADDR_IRQ_M_STATUS1 (SSBI_REG_ADDR_IRQ_BASE + 1)
32#define SSBI_REG_ADDR_IRQ_M_STATUS2 (SSBI_REG_ADDR_IRQ_BASE + 2)
33#define SSBI_REG_ADDR_IRQ_M_STATUS3 (SSBI_REG_ADDR_IRQ_BASE + 3)
34#define SSBI_REG_ADDR_IRQ_M_STATUS4 (SSBI_REG_ADDR_IRQ_BASE + 4)
35#define SSBI_REG_ADDR_IRQ_BLK_SEL (SSBI_REG_ADDR_IRQ_BASE + 5)
36#define SSBI_REG_ADDR_IRQ_IT_STATUS (SSBI_REG_ADDR_IRQ_BASE + 6)
37#define SSBI_REG_ADDR_IRQ_CONFIG (SSBI_REG_ADDR_IRQ_BASE + 7)
38#define SSBI_REG_ADDR_IRQ_RT_STATUS (SSBI_REG_ADDR_IRQ_BASE + 8)
39
40#define PM8901_IRQF_LVL_SEL 0x01 /* level select */
41#define PM8901_IRQF_MASK_FE 0x02 /* mask falling edge */
42#define PM8901_IRQF_MASK_RE 0x04 /* mask rising edge */
43#define PM8901_IRQF_CLR 0x08 /* clear interrupt */
44#define PM8901_IRQF_BITS_MASK 0x70
45#define PM8901_IRQF_BITS_SHIFT 4
46#define PM8901_IRQF_WRITE 0x80
47
48#define PM8901_IRQF_MASK_ALL (PM8901_IRQF_MASK_FE | \
49 PM8901_IRQF_MASK_RE)
50#define PM8901_IRQF_W_C_M (PM8901_IRQF_WRITE | \
51 PM8901_IRQF_CLR | \
52 PM8901_IRQF_MASK_ALL)
53
54#define MAX_PM_IRQ 72
55#define MAX_PM_BLOCKS (MAX_PM_IRQ / 8 + 1)
56#define MAX_PM_MASTERS (MAX_PM_BLOCKS / 8 + 1)
57
58#define MPP_IRQ_BLOCK 1
59
60/* FTS regulator PMR registers */
61#define SSBI_REG_ADDR_S1_PMR (0xA7)
62#define SSBI_REG_ADDR_S2_PMR (0xA8)
63#define SSBI_REG_ADDR_S3_PMR (0xA9)
64#define SSBI_REG_ADDR_S4_PMR (0xAA)
65
66#define REGULATOR_PMR_STATE_MASK 0x60
67#define REGULATOR_PMR_STATE_OFF 0x20
68
David Collins88ebc9c2011-11-15 08:46:08 -080069/* Shutdown/restart delays to allow for LDO 7/dVdd regulator load settling. */
70#define DELAY_AFTER_REG_DISABLE_MS 4
71#define DELAY_BEFORE_SHUTDOWN_MS 8
72
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070073struct pm8901_chip {
74 struct pm8901_platform_data pdata;
Anirudh Ghayalc8051a82011-11-17 09:28:24 +053075 struct device *dev;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070076
77 u8 irqs_allowed[MAX_PM_BLOCKS];
78 u8 blocks_allowed[MAX_PM_MASTERS];
79 u8 masters_allowed;
80 int pm_max_irq;
81 int pm_max_blocks;
82 int pm_max_masters;
83
84 u8 config[MAX_PM_IRQ];
85 u8 wake_enable[MAX_PM_IRQ];
86 u16 count_wakeable;
87
88 u8 revision;
89
90 spinlock_t pm_lock;
91};
92
93#if defined(CONFIG_DEBUG_FS)
94struct pm8901_dbg_device {
95 struct mutex dbg_mutex;
96 struct pm8901_chip *pm_chip;
97 struct dentry *dent;
98 int addr;
99};
100
101static struct pm8901_dbg_device *pmic_dbg_device;
102#endif
103
104static struct pm8901_chip *pmic_chip;
105
106/* Helper Functions */
107DEFINE_RATELIMIT_STATE(pm8901_msg_ratelimit, 60 * HZ, 10);
108
109static inline int pm8901_can_print(void)
110{
111 return __ratelimit(&pm8901_msg_ratelimit);
112}
113
114static inline int
Anirudh Ghayalc8051a82011-11-17 09:28:24 +0530115ssbi_read(struct device *dev, u16 addr, u8 *buf, size_t len)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700116{
Anirudh Ghayalc8051a82011-11-17 09:28:24 +0530117 return msm_ssbi_read(dev->parent, addr, buf, len);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700118}
119
120static inline int
Anirudh Ghayalc8051a82011-11-17 09:28:24 +0530121ssbi_write(struct device *dev, u16 addr, u8 *buf, size_t len)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700122{
Anirudh Ghayalc8051a82011-11-17 09:28:24 +0530123 return msm_ssbi_write(dev->parent, addr, buf, len);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700124}
125
126/* External APIs */
127int pm8901_rev(struct pm8901_chip *chip)
128{
129 if (chip == NULL) {
130 if (pmic_chip != NULL)
131 return pmic_chip->revision;
132 else
133 return -EINVAL;
134 }
135
136 return chip->revision;
137}
138EXPORT_SYMBOL(pm8901_rev);
139
140int pm8901_read(struct pm8901_chip *chip, u16 addr, u8 *values,
141 unsigned int len)
142{
143 if (chip == NULL)
144 return -EINVAL;
145
146 return ssbi_read(chip->dev, addr, values, len);
147}
148EXPORT_SYMBOL(pm8901_read);
149
150int pm8901_write(struct pm8901_chip *chip, u16 addr, u8 *values,
151 unsigned int len)
152{
153 if (chip == NULL)
154 return -EINVAL;
155
156 return ssbi_write(chip->dev, addr, values, len);
157}
158EXPORT_SYMBOL(pm8901_write);
159
160int pm8901_irq_get_rt_status(struct pm8901_chip *chip, int irq)
161{
162 int rc;
163 u8 block, bits, bit;
164 unsigned long irqsave;
165
166 if (chip == NULL || irq < chip->pdata.irq_base ||
167 irq >= chip->pdata.irq_base + MAX_PM_IRQ)
168 return -EINVAL;
169
170 irq -= chip->pdata.irq_base;
171
172 block = irq / 8;
173 bit = irq % 8;
174
175 spin_lock_irqsave(&chip->pm_lock, irqsave);
176
177 rc = ssbi_write(chip->dev, SSBI_REG_ADDR_IRQ_BLK_SEL, &block, 1);
178 if (rc) {
179 pr_err("%s: FAIL ssbi_write(): rc=%d (Select Block)\n",
180 __func__, rc);
181 goto bail_out;
182 }
183
184 rc = ssbi_read(chip->dev, SSBI_REG_ADDR_IRQ_RT_STATUS, &bits, 1);
185 if (rc) {
186 pr_err("%s: FAIL ssbi_read(): rc=%d (Read RT Status)\n",
187 __func__, rc);
188 goto bail_out;
189 }
190
191 rc = (bits & (1 << bit)) ? 1 : 0;
192
193bail_out:
194 spin_unlock_irqrestore(&chip->pm_lock, irqsave);
195
196 return rc;
197}
198EXPORT_SYMBOL(pm8901_irq_get_rt_status);
199
200int pm8901_reset_pwr_off(int reset)
201{
202 int rc = 0, i;
203 u8 pmr;
204 u8 pmr_addr[4] = {
205 SSBI_REG_ADDR_S2_PMR,
206 SSBI_REG_ADDR_S3_PMR,
207 SSBI_REG_ADDR_S4_PMR,
208 SSBI_REG_ADDR_S1_PMR,
209 };
210
211 if (pmic_chip == NULL)
212 return -ENODEV;
213
214 /* Turn off regulators S1, S2, S3, S4 when shutting down. */
215 if (!reset) {
216 for (i = 0; i < 4; i++) {
217 rc = ssbi_read(pmic_chip->dev, pmr_addr[i], &pmr, 1);
218 if (rc) {
219 pr_err("%s: FAIL ssbi_read(0x%x): rc=%d\n",
220 __func__, pmr_addr[i], rc);
221 goto get_out;
222 }
223
224 pmr &= ~REGULATOR_PMR_STATE_MASK;
225 pmr |= REGULATOR_PMR_STATE_OFF;
226
227 rc = ssbi_write(pmic_chip->dev, pmr_addr[i], &pmr, 1);
228 if (rc) {
229 pr_err("%s: FAIL ssbi_write(0x%x)=0x%x: rc=%d"
230 "\n", __func__, pmr_addr[i], pmr, rc);
231 goto get_out;
232 }
David Collins88ebc9c2011-11-15 08:46:08 -0800233 mdelay(DELAY_AFTER_REG_DISABLE_MS);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700234 }
235 }
236
237get_out:
David Collins88ebc9c2011-11-15 08:46:08 -0800238 mdelay(DELAY_BEFORE_SHUTDOWN_MS);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700239 return rc;
240}
241EXPORT_SYMBOL(pm8901_reset_pwr_off);
242
243/* Internal functions */
244static inline int
245pm8901_config_irq(struct pm8901_chip *chip, u8 *bp, u8 *cp)
246{
247 int rc;
248
249 rc = ssbi_write(chip->dev, SSBI_REG_ADDR_IRQ_BLK_SEL, bp, 1);
250 if (rc) {
251 pr_err("%s: ssbi_write: rc=%d (Select block)\n",
252 __func__, rc);
253 goto bail_out;
254 }
255
256 rc = ssbi_write(chip->dev, SSBI_REG_ADDR_IRQ_CONFIG, cp, 1);
257 if (rc)
258 pr_err("%s: ssbi_write: rc=%d (Configure IRQ)\n",
259 __func__, rc);
260
261bail_out:
262 return rc;
263}
264
265static void pm8901_irq_mask(struct irq_data *d)
266{
267 int master, irq_bit;
268 struct pm8901_chip *chip = irq_data_get_irq_handler_data(d);
269 u8 block, config;
270 unsigned int irq = d->irq;
271
272 irq -= chip->pdata.irq_base;
273 block = irq / 8;
274 master = block / 8;
275 irq_bit = irq % 8;
276
277 chip->irqs_allowed[block] &= ~(1 << irq_bit);
278 if (!chip->irqs_allowed[block]) {
279 chip->blocks_allowed[master] &= ~(1 << (block % 8));
280
281 if (!chip->blocks_allowed[master])
282 chip->masters_allowed &= ~(1 << master);
283 }
284
285 config = PM8901_IRQF_WRITE | chip->config[irq] |
286 PM8901_IRQF_MASK_FE | PM8901_IRQF_MASK_RE;
287 pm8901_config_irq(chip, &block, &config);
288}
289
290static void pm8901_irq_unmask(struct irq_data *d)
291{
292 int master, irq_bit;
293 struct pm8901_chip *chip = irq_data_get_irq_handler_data(d);
294 u8 block, config, old_irqs_allowed, old_blocks_allowed;
295 unsigned int irq = d->irq;
296
297 irq -= chip->pdata.irq_base;
298 block = irq / 8;
299 master = block / 8;
300 irq_bit = irq % 8;
301
302 old_irqs_allowed = chip->irqs_allowed[block];
303 chip->irqs_allowed[block] |= 1 << irq_bit;
304 if (!old_irqs_allowed) {
305 master = block / 8;
306
307 old_blocks_allowed = chip->blocks_allowed[master];
308 chip->blocks_allowed[master] |= 1 << (block % 8);
309
310 if (!old_blocks_allowed)
311 chip->masters_allowed |= 1 << master;
312 }
313
314 config = PM8901_IRQF_WRITE | chip->config[irq];
315 pm8901_config_irq(chip, &block, &config);
316}
317
318static void pm8901_irq_ack(struct irq_data *d)
319{
320 struct pm8901_chip *chip = irq_data_get_irq_handler_data(d);
321 u8 block, config;
322 unsigned int irq = d->irq;
323
324 irq -= chip->pdata.irq_base;
325 block = irq / 8;
326
327 config = PM8901_IRQF_WRITE | chip->config[irq] | PM8901_IRQF_CLR;
328 pm8901_config_irq(chip, &block, &config);
329}
330
331static int pm8901_irq_set_type(struct irq_data *d, unsigned int flow_type)
332{
333 int master, irq_bit;
334 struct pm8901_chip *chip = irq_data_get_irq_handler_data(d);
335 u8 block, config;
336 unsigned int irq = d->irq;
337
338 irq -= chip->pdata.irq_base;
339 if (irq > chip->pm_max_irq) {
340 chip->pm_max_irq = irq;
341 chip->pm_max_blocks =
342 chip->pm_max_irq / 8 + 1;
343 chip->pm_max_masters =
344 chip->pm_max_blocks / 8 + 1;
345 }
346 block = irq / 8;
347 master = block / 8;
348 irq_bit = irq % 8;
349
350 chip->config[irq] = (irq_bit << PM8901_IRQF_BITS_SHIFT) |
351 PM8901_IRQF_MASK_RE | PM8901_IRQF_MASK_FE;
352 if (flow_type & (IRQF_TRIGGER_RISING | IRQF_TRIGGER_FALLING)) {
353 if (flow_type & IRQF_TRIGGER_RISING)
354 chip->config[irq] &= ~PM8901_IRQF_MASK_RE;
355 if (flow_type & IRQF_TRIGGER_FALLING)
356 chip->config[irq] &= ~PM8901_IRQF_MASK_FE;
357 } else {
358 chip->config[irq] |= PM8901_IRQF_LVL_SEL;
359
360 if (flow_type & IRQF_TRIGGER_HIGH)
361 chip->config[irq] &= ~PM8901_IRQF_MASK_RE;
362 else
363 chip->config[irq] &= ~PM8901_IRQF_MASK_FE;
364 }
365
366 config = PM8901_IRQF_WRITE | chip->config[irq] | PM8901_IRQF_CLR;
367 return pm8901_config_irq(chip, &block, &config);
368}
369
370static int pm8901_irq_set_wake(struct irq_data *d, unsigned int on)
371{
372 struct pm8901_chip *chip = irq_data_get_irq_handler_data(d);
373 unsigned int irq = d->irq;
374
375 irq -= chip->pdata.irq_base;
376 if (on) {
377 if (!chip->wake_enable[irq]) {
378 chip->wake_enable[irq] = 1;
379 chip->count_wakeable++;
380 }
381 } else {
382 if (chip->wake_enable[irq]) {
383 chip->wake_enable[irq] = 0;
384 chip->count_wakeable--;
385 }
386 }
387
388 return 0;
389}
390
391static inline int
392pm8901_read_root(struct pm8901_chip *chip, u8 *rp)
393{
394 int rc;
395
396 rc = ssbi_read(chip->dev, SSBI_REG_ADDR_IRQ_ROOT, rp, 1);
397 if (rc) {
398 pr_err("%s: FAIL ssbi_read(): rc=%d (Read Root)\n",
399 __func__, rc);
400 *rp = 0;
401 }
402
403 return rc;
404}
405
406static inline int
407pm8901_read_master(struct pm8901_chip *chip, u8 m, u8 *bp)
408{
409 int rc;
410
411 rc = ssbi_read(chip->dev, SSBI_REG_ADDR_IRQ_M_STATUS1 + m, bp, 1);
412 if (rc) {
413 pr_err("%s: FAIL ssbi_read(): rc=%d (Read Master)\n",
414 __func__, rc);
415 *bp = 0;
416 }
417
418 return rc;
419}
420
421static inline int
422pm8901_read_block(struct pm8901_chip *chip, u8 *bp, u8 *ip)
423{
424 int rc;
425
426 rc = ssbi_write(chip->dev, SSBI_REG_ADDR_IRQ_BLK_SEL, bp, 1);
427 if (rc) {
428 pr_err("%s: FAIL ssbi_write(): rc=%d (Select Block)\n",
429 __func__, rc);
430 *bp = 0;
431 goto bail_out;
432 }
433
434 rc = ssbi_read(chip->dev, SSBI_REG_ADDR_IRQ_IT_STATUS, ip, 1);
435 if (rc)
436 pr_err("%s: FAIL ssbi_read(): rc=%d (Read Status)\n",
437 __func__, rc);
438
439bail_out:
440 return rc;
441}
442
443static irqreturn_t pm8901_isr_thread(int irq_requested, void *data)
444{
445 struct pm8901_chip *chip = data;
446 int i, j, k;
447 u8 root, block, config, bits;
448 u8 blocks[MAX_PM_MASTERS];
449 int masters = 0, irq, handled = 0, spurious = 0;
450 u16 irqs_to_handle[MAX_PM_IRQ];
451 unsigned long irqsave;
452
453 spin_lock_irqsave(&chip->pm_lock, irqsave);
454
455 /* Read root for masters */
456 if (pm8901_read_root(chip, &root))
457 goto bail_out;
458
459 masters = root >> 1;
460
461 if (!(masters & chip->masters_allowed) ||
462 (masters & ~chip->masters_allowed)) {
463 spurious = 1000000;
464 }
465
466 /* Read allowed masters for blocks. */
467 for (i = 0; i < chip->pm_max_masters; i++) {
468 if (masters & (1 << i)) {
469 if (pm8901_read_master(chip, i, &blocks[i]))
470 goto bail_out;
471
472 if (!blocks[i]) {
473 if (pm8901_can_print())
474 pr_err("%s: Spurious master: %d "
475 "(blocks=0)", __func__, i);
476 spurious += 10000;
477 }
478 } else
479 blocks[i] = 0;
480 }
481
482 /* Select block, read status and call isr */
483 for (i = 0; i < chip->pm_max_masters; i++) {
484 if (!blocks[i])
485 continue;
486
487 for (j = 0; j < 8; j++) {
488 if (!(blocks[i] & (1 << j)))
489 continue;
490
491 block = i * 8 + j; /* block # */
492 if (pm8901_read_block(chip, &block, &bits))
493 goto bail_out;
494
495 if (!bits) {
496 if (pm8901_can_print())
497 pr_err("%s: Spurious block: "
498 "[master, block]=[%d, %d] "
499 "(bits=0)\n", __func__, i, j);
500 spurious += 100;
501 continue;
502 }
503
504 /* Check IRQ bits */
505 for (k = 0; k < 8; k++) {
506 if (!(bits & (1 << k)))
507 continue;
508
509 /* Check spurious interrupts */
510 if (((1 << i) & chip->masters_allowed) &&
511 (blocks[i] & chip->blocks_allowed[i]) &&
512 (bits & chip->irqs_allowed[block])) {
513
514 /* Found one */
515 irq = block * 8 + k;
516 irqs_to_handle[handled] = irq +
517 chip->pdata.irq_base;
518 handled++;
519 } else {
520 /* Clear and mask wrong one */
521 config = PM8901_IRQF_W_C_M |
522 (k < PM8901_IRQF_BITS_SHIFT);
523
524 pm8901_config_irq(chip,
525 &block, &config);
526
527 if (pm8901_can_print())
528 pr_err("%s: Spurious IRQ: "
529 "[master, block, bit]="
530 "[%d, %d (%d), %d]\n",
531 __func__,
532 i, j, block, k);
533 spurious++;
534 }
535 }
536 }
537
538 }
539
540bail_out:
541
542 spin_unlock_irqrestore(&chip->pm_lock, irqsave);
543
544 for (i = 0; i < handled; i++)
545 generic_handle_irq(irqs_to_handle[i]);
546
547 if (spurious) {
548 if (!pm8901_can_print())
549 return IRQ_HANDLED;
550
551 pr_err("%s: spurious = %d (handled = %d)\n",
552 __func__, spurious, handled);
553 pr_err(" root = 0x%x (masters_allowed<<1 = 0x%x)\n",
554 root, chip->masters_allowed << 1);
555 for (i = 0; i < chip->pm_max_masters; i++) {
556 if (masters & (1 << i))
557 pr_err(" blocks[%d]=0x%x, "
558 "allowed[%d]=0x%x\n",
559 i, blocks[i],
560 i, chip->blocks_allowed[i]);
561 }
562 }
563
564 return IRQ_HANDLED;
565}
566
567#if defined(CONFIG_DEBUG_FS)
568
569static int check_addr(int addr, const char *func_name)
570{
571 if (addr < 0 || addr > 0x3FF) {
572 pr_err("%s: PMIC 8901 register address is invalid: %d\n",
573 func_name, addr);
574 return -EINVAL;
575 }
576 return 0;
577}
578
579static int data_set(void *data, u64 val)
580{
581 struct pm8901_dbg_device *dbgdev = data;
582 u8 reg = val;
583 int rc;
584
585 mutex_lock(&dbgdev->dbg_mutex);
586
587 rc = check_addr(dbgdev->addr, __func__);
588 if (rc)
589 goto done;
590
591 rc = pm8901_write(dbgdev->pm_chip, dbgdev->addr, &reg, 1);
592
593 if (rc)
594 pr_err("%s: FAIL pm8901_write(0x%03X)=0x%02X: rc=%d\n",
595 __func__, dbgdev->addr, reg, rc);
596done:
597 mutex_unlock(&dbgdev->dbg_mutex);
598 return rc;
599}
600
601static int data_get(void *data, u64 *val)
602{
603 struct pm8901_dbg_device *dbgdev = data;
604 int rc;
605 u8 reg;
606
607 mutex_lock(&dbgdev->dbg_mutex);
608
609 rc = check_addr(dbgdev->addr, __func__);
610 if (rc)
611 goto done;
612
613 rc = pm8901_read(dbgdev->pm_chip, dbgdev->addr, &reg, 1);
614
615 if (rc) {
616 pr_err("%s: FAIL pm8901_read(0x%03X)=0x%02X: rc=%d\n",
617 __func__, dbgdev->addr, reg, rc);
618 goto done;
619 }
620
621 *val = reg;
622done:
623 mutex_unlock(&dbgdev->dbg_mutex);
624 return rc;
625}
626
627DEFINE_SIMPLE_ATTRIBUTE(dbg_data_fops, data_get, data_set, "0x%02llX\n");
628
629static int addr_set(void *data, u64 val)
630{
631 struct pm8901_dbg_device *dbgdev = data;
632 int rc;
633
634 rc = check_addr(val, __func__);
635 if (rc)
636 return rc;
637
638 mutex_lock(&dbgdev->dbg_mutex);
639 dbgdev->addr = val;
640 mutex_unlock(&dbgdev->dbg_mutex);
641
642 return 0;
643}
644
645static int addr_get(void *data, u64 *val)
646{
647 struct pm8901_dbg_device *dbgdev = data;
648 int rc;
649
650 mutex_lock(&dbgdev->dbg_mutex);
651
652 rc = check_addr(dbgdev->addr, __func__);
653 if (rc) {
654 mutex_unlock(&dbgdev->dbg_mutex);
655 return rc;
656 }
657 *val = dbgdev->addr;
658
659 mutex_unlock(&dbgdev->dbg_mutex);
660
661 return 0;
662}
663
664DEFINE_SIMPLE_ATTRIBUTE(dbg_addr_fops, addr_get, addr_set, "0x%03llX\n");
665
666static int __devinit pmic8901_dbg_probe(struct pm8901_chip *chip)
667{
668 struct pm8901_dbg_device *dbgdev;
669 struct dentry *dent;
670 struct dentry *temp;
Jay Chokshi898340f2011-09-29 17:19:31 -0700671 int rc;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700672
673 if (chip == NULL) {
674 pr_err("%s: no parent data passed in.\n", __func__);
675 return -EINVAL;
676 }
677
678 dbgdev = kzalloc(sizeof *dbgdev, GFP_KERNEL);
679 if (dbgdev == NULL) {
680 pr_err("%s: kzalloc() failed.\n", __func__);
681 return -ENOMEM;
682 }
683
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700684 dbgdev->pm_chip = chip;
685 dbgdev->addr = -1;
686
687 dent = debugfs_create_dir("pm8901-dbg", NULL);
688 if (dent == NULL || IS_ERR(dent)) {
689 pr_err("%s: ERR debugfs_create_dir: dent=0x%X\n",
690 __func__, (unsigned)dent);
Jay Chokshi898340f2011-09-29 17:19:31 -0700691 rc = PTR_ERR(dent);
692 goto dir_error;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700693 }
694
695 temp = debugfs_create_file("addr", S_IRUSR | S_IWUSR, dent,
696 dbgdev, &dbg_addr_fops);
697 if (temp == NULL || IS_ERR(temp)) {
698 pr_err("%s: ERR debugfs_create_file: dent=0x%X\n",
699 __func__, (unsigned)temp);
Jay Chokshi898340f2011-09-29 17:19:31 -0700700 rc = PTR_ERR(temp);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700701 goto debug_error;
702 }
703
704 temp = debugfs_create_file("data", S_IRUSR | S_IWUSR, dent,
705 dbgdev, &dbg_data_fops);
706 if (temp == NULL || IS_ERR(temp)) {
707 pr_err("%s: ERR debugfs_create_file: dent=0x%X\n",
708 __func__, (unsigned)temp);
Jay Chokshi898340f2011-09-29 17:19:31 -0700709 rc = PTR_ERR(temp);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700710 goto debug_error;
711 }
712
Jay Chokshi898340f2011-09-29 17:19:31 -0700713 mutex_init(&dbgdev->dbg_mutex);
714
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700715 dbgdev->dent = dent;
716
717 pmic_dbg_device = dbgdev;
718
719 return 0;
720
721debug_error:
722 debugfs_remove_recursive(dent);
Jay Chokshi898340f2011-09-29 17:19:31 -0700723dir_error:
724 kfree(dbgdev);
725
726 return rc;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700727}
728
729static int __devexit pmic8901_dbg_remove(void)
730{
731 if (pmic_dbg_device) {
732 debugfs_remove_recursive(pmic_dbg_device->dent);
Jay Chokshi898340f2011-09-29 17:19:31 -0700733 mutex_destroy(&pmic_dbg_device->dbg_mutex);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700734 kfree(pmic_dbg_device);
735 }
736 return 0;
737}
738
739#else
740
741static int __devinit pmic8901_dbg_probe(struct pm8901_chip *chip)
742{
743 return 0;
744}
745
746static int __devexit pmic8901_dbg_remove(void)
747{
748 return 0;
749}
750
751#endif
752
753static struct irq_chip pm8901_irq_chip = {
754 .name = "pm8901",
755 .irq_ack = pm8901_irq_ack,
756 .irq_mask = pm8901_irq_mask,
757 .irq_unmask = pm8901_irq_unmask,
758 .irq_set_type = pm8901_irq_set_type,
759 .irq_set_wake = pm8901_irq_set_wake,
760};
761
Anirudh Ghayalc8051a82011-11-17 09:28:24 +0530762static int pm8901_probe(struct platform_device *pdev)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700763{
Anirudh Ghayalc8051a82011-11-17 09:28:24 +0530764 int i, rc;
765 struct pm8901_platform_data *pdata = pdev->dev.platform_data;
766 struct pm8901_chip *chip;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700767
Anirudh Ghayalc8051a82011-11-17 09:28:24 +0530768 if (pdata == NULL || pdata->irq <= 0) {
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700769 pr_err("%s: No platform_data or IRQ.\n", __func__);
770 return -ENODEV;
771 }
772
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700773 chip = kzalloc(sizeof *chip, GFP_KERNEL);
774 if (chip == NULL) {
775 pr_err("%s: kzalloc() failed.\n", __func__);
776 return -ENOMEM;
777 }
778
Anirudh Ghayalc8051a82011-11-17 09:28:24 +0530779 chip->dev = &pdev->dev;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700780
781 /* Read PMIC chip revision */
782 rc = ssbi_read(chip->dev, SSBI_REG_REV, &chip->revision, 1);
783 if (rc)
784 pr_err("%s: Failed on ssbi_read for revision: rc=%d.\n",
785 __func__, rc);
786 pr_info("%s: PMIC revision: %X\n", __func__, chip->revision);
787
788 (void) memcpy((void *)&chip->pdata, (const void *)pdata,
789 sizeof(chip->pdata));
790
Anirudh Ghayalc8051a82011-11-17 09:28:24 +0530791 irq_set_handler_data(pdata->irq, (void *)chip);
792 irq_set_irq_wake(pdata->irq, 1);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700793
794 chip->pm_max_irq = 0;
795 chip->pm_max_blocks = 0;
796 chip->pm_max_masters = 0;
797
Anirudh Ghayalc8051a82011-11-17 09:28:24 +0530798 platform_set_drvdata(pdev, chip);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700799
800 pmic_chip = chip;
801 spin_lock_init(&chip->pm_lock);
802
803 /* Register for all reserved IRQs */
804 for (i = pdata->irq_base; i < (pdata->irq_base + MAX_PM_IRQ); i++) {
805 irq_set_chip(i, &pm8901_irq_chip);
806 irq_set_handler(i, handle_edge_irq);
807 set_irq_flags(i, IRQF_VALID);
808 irq_set_handler_data(i, (void *)chip);
809 }
810
Anirudh Ghayalc8051a82011-11-17 09:28:24 +0530811 rc = mfd_add_devices(chip->dev, 0, pdata->sub_devices,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700812 pdata->num_subdevs, NULL, 0);
813 if (rc) {
814 pr_err("%s: could not add devices %d\n", __func__, rc);
815 return rc;
816 }
817
Anirudh Ghayalc8051a82011-11-17 09:28:24 +0530818 rc = request_threaded_irq(pdata->irq, NULL, pm8901_isr_thread,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700819 IRQF_ONESHOT | IRQF_DISABLED | pdata->irq_trigger_flags,
820 "pm8901-irq", chip);
821 if (rc)
822 pr_err("%s: could not request irq %d: %d\n", __func__,
Anirudh Ghayalc8051a82011-11-17 09:28:24 +0530823 pdata->irq, rc);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700824
825 rc = pmic8901_dbg_probe(chip);
826 if (rc < 0)
827 pr_err("%s: could not set up debugfs: %d\n", __func__, rc);
828
829 return rc;
830}
831
Anirudh Ghayalc8051a82011-11-17 09:28:24 +0530832static int __devexit pm8901_remove(struct platform_device *pdev)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700833{
834 struct pm8901_chip *chip;
835
Anirudh Ghayalc8051a82011-11-17 09:28:24 +0530836 chip = platform_get_drvdata(pdev);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700837 if (chip) {
838 if (chip->pm_max_irq) {
Anirudh Ghayalc8051a82011-11-17 09:28:24 +0530839 irq_set_irq_wake(chip->pdata.irq, 0);
840 free_irq(chip->pdata.irq, chip);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700841 }
842
Anirudh Ghayalc8051a82011-11-17 09:28:24 +0530843 mfd_remove_devices(chip->dev);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700844
845 chip->dev = NULL;
846
847 kfree(chip);
848 }
849
850 pmic8901_dbg_remove();
851
852 return 0;
853}
854
855#ifdef CONFIG_PM
Anirudh Ghayalc8051a82011-11-17 09:28:24 +0530856static int pm8901_suspend(struct platform_device *pdev, pm_message_t mesg)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700857{
858 struct pm8901_chip *chip;
859 int i;
860 unsigned long irqsave;
861
Anirudh Ghayalc8051a82011-11-17 09:28:24 +0530862 chip = platform_get_drvdata(pdev);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700863
864 for (i = 0; i < MAX_PM_IRQ; i++) {
865 spin_lock_irqsave(&chip->pm_lock, irqsave);
866 if (chip->config[i] && !chip->wake_enable[i]) {
867 if (!((chip->config[i] & PM8901_IRQF_MASK_ALL)
868 == PM8901_IRQF_MASK_ALL))
869 pm8901_irq_mask(irq_get_irq_data(i +
870 chip->pdata.irq_base));
871 }
872 spin_unlock_irqrestore(&chip->pm_lock, irqsave);
873 }
874
875 if (!chip->count_wakeable)
Anirudh Ghayalc8051a82011-11-17 09:28:24 +0530876 disable_irq(chip->pdata.irq);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700877
878 return 0;
879}
880
Anirudh Ghayalc8051a82011-11-17 09:28:24 +0530881static int pm8901_resume(struct platform_device *pdev)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700882{
883 struct pm8901_chip *chip;
884 int i;
885 unsigned long irqsave;
886
Anirudh Ghayalc8051a82011-11-17 09:28:24 +0530887 chip = platform_get_drvdata(pdev);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700888
889 for (i = 0; i < MAX_PM_IRQ; i++) {
890 spin_lock_irqsave(&chip->pm_lock, irqsave);
891 if (chip->config[i] && !chip->wake_enable[i]) {
892 if (!((chip->config[i] & PM8901_IRQF_MASK_ALL)
893 == PM8901_IRQF_MASK_ALL))
894 pm8901_irq_unmask(irq_get_irq_data(i +
895 chip->pdata.irq_base));
896 }
897 spin_unlock_irqrestore(&chip->pm_lock, irqsave);
898 }
899
900 if (!chip->count_wakeable)
Anirudh Ghayalc8051a82011-11-17 09:28:24 +0530901 enable_irq(chip->pdata.irq);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700902
903 return 0;
904}
905#else
906#define pm8901_suspend NULL
907#define pm8901_resume NULL
908#endif
909
Anirudh Ghayalc8051a82011-11-17 09:28:24 +0530910static struct platform_driver pm8901_driver = {
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700911 .probe = pm8901_probe,
912 .remove = __devexit_p(pm8901_remove),
Anirudh Ghayalc8051a82011-11-17 09:28:24 +0530913 .driver = {
914 .name = "pm8901-core",
915 .owner = THIS_MODULE,
916 },
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700917 .suspend = pm8901_suspend,
918 .resume = pm8901_resume,
919};
920
921static int __init pm8901_init(void)
922{
Anirudh Ghayalc8051a82011-11-17 09:28:24 +0530923 return platform_driver_register(&pm8901_driver);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700924}
Anirudh Ghayalc8051a82011-11-17 09:28:24 +0530925arch_initcall(pm8901_init);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700926
927static void __exit pm8901_exit(void)
928{
Anirudh Ghayalc8051a82011-11-17 09:28:24 +0530929 platform_driver_unregister(&pm8901_driver);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700930}
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700931module_exit(pm8901_exit);
932
933MODULE_LICENSE("GPL v2");
934MODULE_DESCRIPTION("PMIC8901 core driver");
935MODULE_VERSION("1.0");
936MODULE_ALIAS("platform:pmic8901-core");