blob: 2a85e5f15bc9e6bb530c85fb267bea49516e628e [file] [log] [blame]
Eric Holmberg6275b602012-11-19 13:05:04 -07001/* arch/arm/mach-msm/smp2p_gpio.c
2 *
3 * Copyright (c) 2013, The Linux Foundation. All rights reserved.
4 *
5 * This program is free software; you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License version 2 and
7 * only version 2 as published by the Free Software Foundation.
8 *
9 * This program is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 * GNU General Public License for more details.
13 */
14#include <linux/module.h>
15#include <linux/platform_device.h>
16#include <linux/bitmap.h>
17#include <linux/of.h>
18#include <linux/gpio.h>
19#include <linux/irq.h>
20#include <linux/irqdomain.h>
21#include <linux/slab.h>
22#include <linux/list.h>
23#include <mach/msm_ipc_logging.h>
24#include "smp2p_private_api.h"
25#include "smp2p_private.h"
26
27/* GPIO device - one per SMP2P entry. */
28struct smp2p_chip_dev {
29 struct list_head entry_list;
30 char name[SMP2P_MAX_ENTRY_NAME];
31 int remote_pid;
32 bool is_inbound;
33 bool is_open;
34 struct notifier_block out_notifier;
35 struct notifier_block in_notifier;
36 struct msm_smp2p_out *out_handle;
37
38 struct gpio_chip gpio;
39 struct irq_domain *irq_domain;
40 int irq_base;
41
42 spinlock_t irq_lock;
43 DECLARE_BITMAP(irq_enabled, SMP2P_BITS_PER_ENTRY);
44 DECLARE_BITMAP(irq_rising_edge, SMP2P_BITS_PER_ENTRY);
45 DECLARE_BITMAP(irq_falling_edge, SMP2P_BITS_PER_ENTRY);
46};
47
48static struct platform_driver smp2p_gpio_driver;
49static struct lock_class_key smp2p_gpio_lock_class;
50static struct irq_chip smp2p_gpio_irq_chip;
51static DEFINE_SPINLOCK(smp2p_entry_lock_lha1);
52static LIST_HEAD(smp2p_entry_list);
53
54/* Used for mapping edge to name for logging. */
55static const char * const edge_names[] = {
56 "-",
57 "0->1",
58 "1->0",
59 "-",
60};
61
62/* Used for mapping edge to value for logging. */
63static const char * const edge_name_rising[] = {
64 "-",
65 "0->1",
66};
67
68/* Used for mapping edge to value for logging. */
69static const char * const edge_name_falling[] = {
70 "-",
71 "1->0",
72};
73
74static int smp2p_gpio_to_irq(struct gpio_chip *cp,
75 unsigned offset);
76
77/**
78 * smp2p_get_value - Retrieves GPIO value.
79 *
80 * @cp: GPIO chip pointer
81 * @offset: Pin offset
82 * @returns: >=0: value of GPIO Pin; < 0 for error
83 *
84 * Error codes:
85 * -ENODEV - chip/entry invalid
86 * -ENETDOWN - valid entry, but entry not yet created
87 */
88static int smp2p_get_value(struct gpio_chip *cp,
89 unsigned offset)
90{
91 struct smp2p_chip_dev *chip;
92 int ret = 0;
93 uint32_t data;
94
95 if (!cp)
96 return -ENODEV;
97
98 chip = container_of(cp, struct smp2p_chip_dev, gpio);
99 if (!chip->is_open)
100 return -ENETDOWN;
101
102 if (chip->is_inbound)
103 ret = msm_smp2p_in_read(chip->remote_pid, chip->name, &data);
104 else
105 ret = msm_smp2p_out_read(chip->out_handle, &data);
106
107 if (!ret)
108 ret = (data & (1 << offset)) ? 1 : 0;
109
110 return ret;
111}
112
113/**
114 * smp2p_set_value - Sets GPIO value.
115 *
116 * @cp: GPIO chip pointer
117 * @offset: Pin offset
118 * @value: New value
119 */
120static void smp2p_set_value(struct gpio_chip *cp, unsigned offset, int value)
121{
122 struct smp2p_chip_dev *chip;
123 uint32_t data_set;
124 uint32_t data_clear;
125 int ret;
126
127 if (!cp)
128 return;
129
130 chip = container_of(cp, struct smp2p_chip_dev, gpio);
131 if (!chip->is_open)
132 return;
133
134 if (chip->is_inbound) {
135 SMP2P_ERR("%s: '%s':%d virq %d invalid operation\n",
136 __func__, chip->name, chip->remote_pid,
137 chip->irq_base + offset);
138 return;
139 }
140
141 if (value) {
142 data_set = 1 << offset;
143 data_clear = 0;
144 } else {
145 data_set = 0;
146 data_clear = 1 << offset;
147 }
148
149 ret = msm_smp2p_out_modify(chip->out_handle,
150 data_set, data_clear);
151
152 if (ret)
153 SMP2P_GPIO("'%s':%d gpio %d set to %d failed (%d)\n",
154 chip->name, chip->remote_pid,
155 chip->gpio.base + offset, value, ret);
156 else
157 SMP2P_GPIO("'%s':%d gpio %d set to %d\n",
158 chip->name, chip->remote_pid,
159 chip->gpio.base + offset, value);
160}
161
162/**
163 * smp2p_direction_input - Sets GPIO direction to input.
164 *
165 * @cp: GPIO chip pointer
166 * @offset: Pin offset
167 * @returns: 0 for success; < 0 for failure
168 */
169static int smp2p_direction_input(struct gpio_chip *cp, unsigned offset)
170{
171 struct smp2p_chip_dev *chip;
172
173 if (!cp)
174 return -ENODEV;
175
176 chip = container_of(cp, struct smp2p_chip_dev, gpio);
177 if (!chip->is_inbound)
178 return -EPERM;
179
180 return 0;
181}
182
183/**
184 * smp2p_direction_output - Sets GPIO direction to output.
185 *
186 * @cp: GPIO chip pointer
187 * @offset: Pin offset
188 * @value: Direction
189 * @returns: 0 for success; < 0 for failure
190 */
191static int smp2p_direction_output(struct gpio_chip *cp,
192 unsigned offset, int value)
193{
194 struct smp2p_chip_dev *chip;
195
196 if (!cp)
197 return -ENODEV;
198
199 chip = container_of(cp, struct smp2p_chip_dev, gpio);
200 if (chip->is_inbound)
201 return -EPERM;
202
203 return 0;
204}
205
206/**
207 * smp2p_gpio_to_irq - Convert GPIO pin to virtual IRQ pin.
208 *
209 * @cp: GPIO chip pointer
210 * @offset: Pin offset
211 * @returns: >0 for virtual irq value; < 0 for failure
212 */
213static int smp2p_gpio_to_irq(struct gpio_chip *cp, unsigned offset)
214{
215 struct smp2p_chip_dev *chip;
216
217 chip = container_of(cp, struct smp2p_chip_dev, gpio);
218 if (!cp || chip->irq_base <= 0)
219 return -ENODEV;
220
221 return chip->irq_base + offset;
222}
223
224/**
225 * smp2p_gpio_irq_mask_helper - Mask/Unmask interrupt.
226 *
227 * @d: IRQ data
228 * @mask: true to mask (disable), false to unmask (enable)
229 */
230static void smp2p_gpio_irq_mask_helper(struct irq_data *d, bool mask)
231{
232 struct smp2p_chip_dev *chip;
233 int offset;
234 unsigned long flags;
235
236 chip = (struct smp2p_chip_dev *)irq_get_chip_data(d->irq);
237 if (!chip || chip->irq_base <= 0)
238 return;
239
240 offset = d->irq - chip->irq_base;
241 spin_lock_irqsave(&chip->irq_lock, flags);
242 if (mask)
243 clear_bit(offset, chip->irq_enabled);
244 else
245 set_bit(offset, chip->irq_enabled);
246 spin_unlock_irqrestore(&chip->irq_lock, flags);
247}
248
249/**
250 * smp2p_gpio_irq_mask - Mask interrupt.
251 *
252 * @d: IRQ data
253 */
254static void smp2p_gpio_irq_mask(struct irq_data *d)
255{
256 smp2p_gpio_irq_mask_helper(d, true);
257}
258
259/**
260 * smp2p_gpio_irq_unmask - Unmask interrupt.
261 *
262 * @d: IRQ data
263 */
264static void smp2p_gpio_irq_unmask(struct irq_data *d)
265{
266 smp2p_gpio_irq_mask_helper(d, false);
267}
268
269/**
270 * smp2p_gpio_irq_set_type - Set interrupt edge type.
271 *
272 * @d: IRQ data
273 * @type: Edge type for interrupt
274 * @returns 0 for success; < 0 for failure
275 */
276static int smp2p_gpio_irq_set_type(struct irq_data *d, unsigned int type)
277{
278 struct smp2p_chip_dev *chip;
279 int offset;
280 unsigned long flags;
281 int ret = 0;
282
283 chip = (struct smp2p_chip_dev *)irq_get_chip_data(d->irq);
284 if (!chip)
285 return -ENODEV;
286
287 if (chip->irq_base <= 0) {
288 SMP2P_ERR("%s: '%s':%d virqbase %d invalid\n",
289 __func__, chip->name, chip->remote_pid,
290 chip->irq_base);
291 return -ENODEV;
292 }
293
294 offset = d->irq - chip->irq_base;
295
296 spin_lock_irqsave(&chip->irq_lock, flags);
297 clear_bit(offset, chip->irq_rising_edge);
298 clear_bit(offset, chip->irq_falling_edge);
299 switch (type) {
300 case IRQ_TYPE_EDGE_RISING:
301 set_bit(offset, chip->irq_rising_edge);
302 break;
303
304 case IRQ_TYPE_EDGE_FALLING:
305 set_bit(offset, chip->irq_falling_edge);
306 break;
307
308 case IRQ_TYPE_NONE:
309 case IRQ_TYPE_DEFAULT:
310 case IRQ_TYPE_EDGE_BOTH:
311 set_bit(offset, chip->irq_rising_edge);
312 set_bit(offset, chip->irq_falling_edge);
313 break;
314
315 default:
316 SMP2P_ERR("%s: unsupported interrupt type 0x%x\n",
317 __func__, type);
318 ret = -EINVAL;
319 break;
320 }
321 spin_unlock_irqrestore(&chip->irq_lock, flags);
322 return ret;
323}
324
325/**
326 * smp2p_irq_map - Creates or updates binding of virtual IRQ
327 *
328 * @domain_ptr: Interrupt domain pointer
329 * @virq: Virtual IRQ
330 * @hw: Hardware IRQ (same as virq for nomap)
331 * @returns: 0 for success
332 */
333static int smp2p_irq_map(struct irq_domain *domain_ptr, unsigned int virq,
334 irq_hw_number_t hw)
335{
336 struct smp2p_chip_dev *chip;
337
338 chip = domain_ptr->host_data;
339 if (!chip) {
340 SMP2P_ERR("%s: invalid domain ptr %p\n", __func__, domain_ptr);
341 return -ENODEV;
342 }
343
344 /* map chip structures to device */
345 irq_set_lockdep_class(virq, &smp2p_gpio_lock_class);
346 irq_set_chip_and_handler(virq, &smp2p_gpio_irq_chip,
347 handle_level_irq);
348 irq_set_chip_data(virq, chip);
349 set_irq_flags(virq, IRQF_VALID);
350
351 return 0;
352}
353
354static struct irq_chip smp2p_gpio_irq_chip = {
355 .name = "smp2p_gpio",
356 .irq_mask = smp2p_gpio_irq_mask,
357 .irq_unmask = smp2p_gpio_irq_unmask,
358 .irq_set_type = smp2p_gpio_irq_set_type,
359};
360
361/* No-map interrupt Domain */
362static const struct irq_domain_ops smp2p_irq_domain_ops = {
363 .map = smp2p_irq_map,
364};
365
366/**
367 * msm_summary_irq_handler - Handles inbound entry change notification.
368 *
369 * @chip: GPIO chip pointer
370 * @entry: Change notification data
371 *
372 * Whenever an entry changes, this callback is triggered to determine
373 * which bits changed and if the corresponding interrupts need to be
374 * triggered.
375 */
376static void msm_summary_irq_handler(struct smp2p_chip_dev *chip,
377 struct msm_smp2p_update_notif *entry)
378{
379 int i;
380 uint32_t cur_val;
381 uint32_t prev_val;
382 uint32_t edge;
383 unsigned long flags;
384 bool trigger_interrrupt;
385 bool irq_rising;
386 bool irq_falling;
387
388 cur_val = entry->current_value;
389 prev_val = entry->previous_value;
390
391 if (chip->irq_base <= 0)
392 return;
393
394 SMP2P_GPIO("'%s':%d GPIO Summary IRQ Change %08x->%08x\n",
395 chip->name, chip->remote_pid, prev_val, cur_val);
396
397 for (i = 0; i < SMP2P_BITS_PER_ENTRY; ++i) {
398 spin_lock_irqsave(&chip->irq_lock, flags);
399 trigger_interrrupt = false;
400 edge = (prev_val & 0x1) << 1 | (cur_val & 0x1);
401 irq_rising = test_bit(i, chip->irq_rising_edge);
402 irq_falling = test_bit(i, chip->irq_falling_edge);
403
404 if (test_bit(i, chip->irq_enabled)) {
405 if (edge == 0x1 && irq_rising)
406 /* 0->1 transition */
407 trigger_interrrupt = true;
408 else if (edge == 0x2 && irq_falling)
409 /* 1->0 transition */
410 trigger_interrrupt = true;
411 } else {
412 SMP2P_GPIO(
413 "'%s':%d GPIO bit %d virq %d (%s,%s) - edge %s disabled\n",
414 chip->name, chip->remote_pid, i,
415 chip->irq_base + i,
416 edge_name_rising[irq_rising],
417 edge_name_falling[irq_falling],
418 edge_names[edge]);
419 }
420 spin_unlock_irqrestore(&chip->irq_lock, flags);
421
422 if (trigger_interrrupt) {
423 SMP2P_GPIO(
424 "'%s':%d GPIO bit %d virq %d (%s,%s) - edge %s triggering\n",
425 chip->name, chip->remote_pid, i,
426 chip->irq_base + i,
427 edge_name_rising[irq_rising],
428 edge_name_falling[irq_falling],
429 edge_names[edge]);
430 (void)generic_handle_irq(chip->irq_base + i);
431 }
432
433 cur_val >>= 1;
434 prev_val >>= 1;
435 }
436}
437
438/**
439 * Adds an interrupt domain based upon the DT node.
440 *
441 * @chip: pointer to GPIO chip
442 * @node: pointer to Device Tree node
443 */
444static void smp2p_add_irq_domain(struct smp2p_chip_dev *chip,
445 struct device_node *node)
446{
447 int i;
448
449 /* map GPIO pins to interrupts */
450 chip->irq_domain = irq_domain_add_nomap(node, 0,
451 &smp2p_irq_domain_ops, chip);
452 if (!chip->irq_domain) {
453 SMP2P_ERR("%s: unable to create interrupt domain '%s':%d\n",
454 __func__, chip->name, chip->remote_pid);
455 return;
456 }
457
458 for (i = 0; i < SMP2P_BITS_PER_ENTRY; ++i) {
459 unsigned int virt_irq;
460
461 virt_irq = irq_create_direct_mapping(chip->irq_domain);
462 if (virt_irq == NO_IRQ) {
463 SMP2P_ERR("%s: gpio->virt IRQ mapping failed '%s':%d\n",
464 __func__, chip->name, chip->remote_pid);
465 } else if (!chip->irq_base) {
466 chip->irq_base = virt_irq;
467 }
468 }
469}
470
471/**
472 * Notifier function passed into smp2p API for out bound entries.
473 *
474 * @self: Pointer to calling notifier block
475 * @event: Event
476 * @data: Event-specific data
477 * @returns: 0
478 */
479static int smp2p_gpio_out_notify(struct notifier_block *self,
480 unsigned long event, void *data)
481{
482 struct smp2p_chip_dev *chip;
483
484 chip = container_of(self, struct smp2p_chip_dev, out_notifier);
485
486 switch (event) {
487 case SMP2P_OPEN:
488 chip->is_open = 1;
489 SMP2P_GPIO("%s: Opened out '%s':%d\n", __func__,
490 chip->name, chip->remote_pid);
491 break;
492 case SMP2P_ENTRY_UPDATE:
493 break;
494 default:
495 SMP2P_ERR("%s: Unknown event\n", __func__);
496 break;
497 }
498 return 0;
499}
500
501/**
502 * Notifier function passed into smp2p API for in bound entries.
503 *
504 * @self: Pointer to calling notifier block
505 * @event: Event
506 * @data: Event-specific data
507 * @returns: 0
508 */
509static int smp2p_gpio_in_notify(struct notifier_block *self,
510 unsigned long event, void *data)
511{
512 struct smp2p_chip_dev *chip;
513
514 chip = container_of(self, struct smp2p_chip_dev, in_notifier);
515
516 switch (event) {
517 case SMP2P_OPEN:
518 chip->is_open = 1;
519 SMP2P_GPIO("%s: Opened in '%s':%d\n", __func__,
520 chip->name, chip->remote_pid);
521 break;
522 case SMP2P_ENTRY_UPDATE:
523 msm_summary_irq_handler(chip, data);
524 break;
525 default:
526 SMP2P_ERR("%s: Unknown event\n", __func__);
527 break;
528 }
529 return 0;
530}
531
532/**
533 * Device tree probe function.
534 *
535 * @pdev: Pointer to device tree data.
536 * @returns: 0 on success; -ENODEV otherwise
537 *
538 * Called for each smp2pgpio entry in the device tree.
539 */
540static int __devinit smp2p_gpio_probe(struct platform_device *pdev)
541{
542 struct device_node *node;
543 char *key;
544 struct smp2p_chip_dev *chip;
545 const char *name_tmp;
546 unsigned long flags;
547 bool is_test_entry = false;
548 int ret;
549
550 chip = kzalloc(sizeof(struct smp2p_chip_dev), GFP_KERNEL);
551 if (!chip) {
552 SMP2P_ERR("%s: out of memory\n", __func__);
553 ret = -ENOMEM;
554 goto fail;
555 }
556 spin_lock_init(&chip->irq_lock);
557
558 /* parse device tree */
559 node = pdev->dev.of_node;
560 key = "qcom,entry-name";
561 ret = of_property_read_string(node, key, &name_tmp);
562 if (ret) {
563 SMP2P_ERR("%s: missing DT key '%s'\n", __func__, key);
564 goto fail;
565 }
566 strlcpy(chip->name, name_tmp, sizeof(chip->name));
567
568 key = "qcom,remote-pid";
569 ret = of_property_read_u32(node, key, &chip->remote_pid);
570 if (ret) {
571 SMP2P_ERR("%s: missing DT key '%s'\n", __func__, key);
572 goto fail;
573 }
574
575 key = "qcom,is-inbound";
576 chip->is_inbound = of_property_read_bool(node, key);
577
578 /* create virtual GPIO controller */
579 chip->gpio.label = chip->name;
580 chip->gpio.dev = &pdev->dev;
581 chip->gpio.owner = THIS_MODULE;
582 chip->gpio.direction_input = smp2p_direction_input,
583 chip->gpio.get = smp2p_get_value;
584 chip->gpio.direction_output = smp2p_direction_output,
585 chip->gpio.set = smp2p_set_value;
586 chip->gpio.to_irq = smp2p_gpio_to_irq,
587 chip->gpio.base = -1; /* use dynamic GPIO pin allocation */
588 chip->gpio.ngpio = SMP2P_BITS_PER_ENTRY;
589 ret = gpiochip_add(&chip->gpio);
590 if (ret) {
591 SMP2P_ERR("%s: unable to register GPIO '%s' ret %d\n",
592 __func__, chip->name, ret);
593 goto fail;
594 }
595
596 /*
597 * Test entries opened by GPIO Test conflict with loopback
598 * support, so the test entries must be explicitly opened
599 * in the unit test framework.
600 */
601 if (strncmp("smp2p", chip->name, SMP2P_MAX_ENTRY_NAME) == 0)
602 is_test_entry = true;
603
604 if (!chip->is_inbound) {
605 chip->out_notifier.notifier_call = smp2p_gpio_out_notify;
606 if (!is_test_entry) {
607 ret = msm_smp2p_out_open(chip->remote_pid, chip->name,
608 &chip->out_notifier,
609 &chip->out_handle);
610 if (ret < 0)
611 goto fail;
612 }
613 } else {
614 chip->in_notifier.notifier_call = smp2p_gpio_in_notify;
615 if (!is_test_entry) {
616 ret = msm_smp2p_in_register(chip->remote_pid,
617 chip->name,
618 &chip->in_notifier);
619 if (ret < 0)
620 goto fail;
621 }
622 }
623
624 spin_lock_irqsave(&smp2p_entry_lock_lha1, flags);
625 list_add(&chip->entry_list, &smp2p_entry_list);
626 spin_unlock_irqrestore(&smp2p_entry_lock_lha1, flags);
627
628 /*
629 * Create interrupt domain - note that chip can't be removed from the
630 * interrupt domain, so chip cannot be deleted after this point.
631 */
632 if (chip->is_inbound)
633 smp2p_add_irq_domain(chip, node);
634 else
635 chip->irq_base = -1;
636
637 SMP2P_GPIO("%s: added %s%s entry '%s':%d gpio %d irq %d",
638 __func__,
639 is_test_entry ? "test " : "",
640 chip->is_inbound ? "in" : "out",
641 chip->name, chip->remote_pid,
642 chip->gpio.base, chip->irq_base);
643
644 return 0;
645
646fail:
647 kfree(chip);
648 return ret;
649}
650
651/**
652 * smp2p_gpio_open_close - Opens or closes entry.
653 *
654 * @entry: Entry to open or close
655 * @do_open: true = open port; false = close
656 */
657static void smp2p_gpio_open_close(struct smp2p_chip_dev *entry,
658 bool do_open)
659{
660 int ret;
661
662 if (do_open) {
663 /* open entry */
664 if (entry->is_inbound)
665 ret = msm_smp2p_in_register(entry->remote_pid,
666 entry->name, &entry->in_notifier);
667 else
668 ret = msm_smp2p_out_open(entry->remote_pid,
669 entry->name, &entry->out_notifier,
670 &entry->out_handle);
671 SMP2P_GPIO("%s: opened %s '%s':%d ret %d\n",
672 __func__,
673 entry->is_inbound ? "in" : "out",
674 entry->name, entry->remote_pid,
675 ret);
676 } else {
677 /* close entry */
678 if (entry->is_inbound)
679 ret = msm_smp2p_in_unregister(entry->remote_pid,
680 entry->name, &entry->in_notifier);
681 else
682 ret = msm_smp2p_out_close(&entry->out_handle);
683 entry->is_open = false;
684 SMP2P_GPIO("%s: closed %s '%s':%d ret %d\n",
685 __func__,
686 entry->is_inbound ? "in" : "out",
687 entry->name, entry->remote_pid, ret);
688 }
689}
690
691/**
692 * smp2p_gpio_open_test_entry - Opens or closes test entries for unit testing.
693 *
694 * @name: Name of the entry
695 * @remote_pid: Remote processor ID
696 * @do_open: true = open port; false = close
697 */
698void smp2p_gpio_open_test_entry(const char *name, int remote_pid, bool do_open)
699{
700 struct smp2p_chip_dev *entry;
701 struct smp2p_chip_dev *start_entry;
702 unsigned long flags;
703
704 spin_lock_irqsave(&smp2p_entry_lock_lha1, flags);
705 if (list_empty(&smp2p_entry_list)) {
706 spin_unlock_irqrestore(&smp2p_entry_lock_lha1, flags);
707 return;
708 }
709 start_entry = list_first_entry(&smp2p_entry_list,
710 struct smp2p_chip_dev,
711 entry_list);
712 entry = start_entry;
713 do {
714 if (!strncmp(entry->name, name, SMP2P_MAX_ENTRY_NAME)
715 && entry->remote_pid == remote_pid) {
716 /* found entry to change */
717 spin_unlock_irqrestore(&smp2p_entry_lock_lha1, flags);
718 smp2p_gpio_open_close(entry, do_open);
719 spin_lock_irqsave(&smp2p_entry_lock_lha1, flags);
720 }
721 list_rotate_left(&smp2p_entry_list);
722 entry = list_first_entry(&smp2p_entry_list,
723 struct smp2p_chip_dev,
724 entry_list);
725 } while (entry != start_entry);
726 spin_unlock_irqrestore(&smp2p_entry_lock_lha1, flags);
727}
728
729static struct of_device_id msm_smp2p_match_table[] __devinitdata = {
730 {.compatible = "qcom,smp2pgpio", },
731 {},
732};
733
734static struct platform_driver smp2p_gpio_driver = {
735 .probe = smp2p_gpio_probe,
736 .driver = {
737 .name = "smp2pgpio",
738 .owner = THIS_MODULE,
739 .of_match_table = msm_smp2p_match_table,
740 },
741};
742
743static int __devinit smp2p_init(void)
744{
745 INIT_LIST_HEAD(&smp2p_entry_list);
746 return platform_driver_register(&smp2p_gpio_driver);
747}
748module_init(smp2p_init);
749
750static void __exit smp2p_exit(void)
751{
752 platform_driver_unregister(&smp2p_gpio_driver);
753}
754module_exit(smp2p_exit);
755
756MODULE_DESCRIPTION("SMP2P GPIO");
757MODULE_LICENSE("GPL v2");