diff options
Diffstat (limited to 'drivers/pinctrl')
-rw-r--r-- | drivers/pinctrl/intel/pinctrl-baytrail.c | 254 | ||||
-rw-r--r-- | drivers/pinctrl/intel/pinctrl-cherryview.c | 1 | ||||
-rw-r--r-- | drivers/pinctrl/pinctrl-at91.c | 17 | ||||
-rw-r--r-- | drivers/pinctrl/sunxi/pinctrl-sun4i-a10.c | 1 | ||||
-rw-r--r-- | drivers/pinctrl/sunxi/pinctrl-sunxi.c | 14 | ||||
-rw-r--r-- | drivers/pinctrl/sunxi/pinctrl-sunxi.h | 4 |
6 files changed, 213 insertions, 78 deletions
diff --git a/drivers/pinctrl/intel/pinctrl-baytrail.c b/drivers/pinctrl/intel/pinctrl-baytrail.c index 5afe03e28b91..2062c224e32f 100644 --- a/drivers/pinctrl/intel/pinctrl-baytrail.c +++ b/drivers/pinctrl/intel/pinctrl-baytrail.c | |||
@@ -66,6 +66,10 @@ | |||
66 | #define BYT_DIR_MASK (BIT(1) | BIT(2)) | 66 | #define BYT_DIR_MASK (BIT(1) | BIT(2)) |
67 | #define BYT_TRIG_MASK (BIT(26) | BIT(25) | BIT(24)) | 67 | #define BYT_TRIG_MASK (BIT(26) | BIT(25) | BIT(24)) |
68 | 68 | ||
69 | #define BYT_CONF0_RESTORE_MASK (BYT_DIRECT_IRQ_EN | BYT_TRIG_MASK | \ | ||
70 | BYT_PIN_MUX) | ||
71 | #define BYT_VAL_RESTORE_MASK (BYT_DIR_MASK | BYT_LEVEL) | ||
72 | |||
69 | #define BYT_NGPIO_SCORE 102 | 73 | #define BYT_NGPIO_SCORE 102 |
70 | #define BYT_NGPIO_NCORE 28 | 74 | #define BYT_NGPIO_NCORE 28 |
71 | #define BYT_NGPIO_SUS 44 | 75 | #define BYT_NGPIO_SUS 44 |
@@ -134,12 +138,18 @@ static struct pinctrl_gpio_range byt_ranges[] = { | |||
134 | }, | 138 | }, |
135 | }; | 139 | }; |
136 | 140 | ||
141 | struct byt_gpio_pin_context { | ||
142 | u32 conf0; | ||
143 | u32 val; | ||
144 | }; | ||
145 | |||
137 | struct byt_gpio { | 146 | struct byt_gpio { |
138 | struct gpio_chip chip; | 147 | struct gpio_chip chip; |
139 | struct platform_device *pdev; | 148 | struct platform_device *pdev; |
140 | spinlock_t lock; | 149 | spinlock_t lock; |
141 | void __iomem *reg_base; | 150 | void __iomem *reg_base; |
142 | struct pinctrl_gpio_range *range; | 151 | struct pinctrl_gpio_range *range; |
152 | struct byt_gpio_pin_context *saved_context; | ||
143 | }; | 153 | }; |
144 | 154 | ||
145 | #define to_byt_gpio(c) container_of(c, struct byt_gpio, chip) | 155 | #define to_byt_gpio(c) container_of(c, struct byt_gpio, chip) |
@@ -158,40 +168,62 @@ static void __iomem *byt_gpio_reg(struct gpio_chip *chip, unsigned offset, | |||
158 | return vg->reg_base + reg_offset + reg; | 168 | return vg->reg_base + reg_offset + reg; |
159 | } | 169 | } |
160 | 170 | ||
161 | static bool is_special_pin(struct byt_gpio *vg, unsigned offset) | 171 | static void byt_gpio_clear_triggering(struct byt_gpio *vg, unsigned offset) |
172 | { | ||
173 | void __iomem *reg = byt_gpio_reg(&vg->chip, offset, BYT_CONF0_REG); | ||
174 | unsigned long flags; | ||
175 | u32 value; | ||
176 | |||
177 | spin_lock_irqsave(&vg->lock, flags); | ||
178 | value = readl(reg); | ||
179 | value &= ~(BYT_TRIG_POS | BYT_TRIG_NEG | BYT_TRIG_LVL); | ||
180 | writel(value, reg); | ||
181 | spin_unlock_irqrestore(&vg->lock, flags); | ||
182 | } | ||
183 | |||
184 | static u32 byt_get_gpio_mux(struct byt_gpio *vg, unsigned offset) | ||
162 | { | 185 | { |
163 | /* SCORE pin 92-93 */ | 186 | /* SCORE pin 92-93 */ |
164 | if (!strcmp(vg->range->name, BYT_SCORE_ACPI_UID) && | 187 | if (!strcmp(vg->range->name, BYT_SCORE_ACPI_UID) && |
165 | offset >= 92 && offset <= 93) | 188 | offset >= 92 && offset <= 93) |
166 | return true; | 189 | return 1; |
167 | 190 | ||
168 | /* SUS pin 11-21 */ | 191 | /* SUS pin 11-21 */ |
169 | if (!strcmp(vg->range->name, BYT_SUS_ACPI_UID) && | 192 | if (!strcmp(vg->range->name, BYT_SUS_ACPI_UID) && |
170 | offset >= 11 && offset <= 21) | 193 | offset >= 11 && offset <= 21) |
171 | return true; | 194 | return 1; |
172 | 195 | ||
173 | return false; | 196 | return 0; |
174 | } | 197 | } |
175 | 198 | ||
176 | static int byt_gpio_request(struct gpio_chip *chip, unsigned offset) | 199 | static int byt_gpio_request(struct gpio_chip *chip, unsigned offset) |
177 | { | 200 | { |
178 | struct byt_gpio *vg = to_byt_gpio(chip); | 201 | struct byt_gpio *vg = to_byt_gpio(chip); |
179 | void __iomem *reg = byt_gpio_reg(chip, offset, BYT_CONF0_REG); | 202 | void __iomem *reg = byt_gpio_reg(chip, offset, BYT_CONF0_REG); |
180 | u32 value; | 203 | u32 value, gpio_mux; |
181 | bool special; | ||
182 | 204 | ||
183 | /* | 205 | /* |
184 | * In most cases, func pin mux 000 means GPIO function. | 206 | * In most cases, func pin mux 000 means GPIO function. |
185 | * But, some pins may have func pin mux 001 represents | 207 | * But, some pins may have func pin mux 001 represents |
186 | * GPIO function. Only allow user to export pin with | 208 | * GPIO function. |
187 | * func pin mux preset as GPIO function by BIOS/FW. | 209 | * |
210 | * Because there are devices out there where some pins were not | ||
211 | * configured correctly we allow changing the mux value from | ||
212 | * request (but print out warning about that). | ||
188 | */ | 213 | */ |
189 | value = readl(reg) & BYT_PIN_MUX; | 214 | value = readl(reg) & BYT_PIN_MUX; |
190 | special = is_special_pin(vg, offset); | 215 | gpio_mux = byt_get_gpio_mux(vg, offset); |
191 | if ((special && value != 1) || (!special && value)) { | 216 | if (WARN_ON(gpio_mux != value)) { |
192 | dev_err(&vg->pdev->dev, | 217 | unsigned long flags; |
193 | "pin %u cannot be used as GPIO.\n", offset); | 218 | |
194 | return -EINVAL; | 219 | spin_lock_irqsave(&vg->lock, flags); |
220 | value = readl(reg) & ~BYT_PIN_MUX; | ||
221 | value |= gpio_mux; | ||
222 | writel(value, reg); | ||
223 | spin_unlock_irqrestore(&vg->lock, flags); | ||
224 | |||
225 | dev_warn(&vg->pdev->dev, | ||
226 | "pin %u forcibly re-configured as GPIO\n", offset); | ||
195 | } | 227 | } |
196 | 228 | ||
197 | pm_runtime_get(&vg->pdev->dev); | 229 | pm_runtime_get(&vg->pdev->dev); |
@@ -202,14 +234,8 @@ static int byt_gpio_request(struct gpio_chip *chip, unsigned offset) | |||
202 | static void byt_gpio_free(struct gpio_chip *chip, unsigned offset) | 234 | static void byt_gpio_free(struct gpio_chip *chip, unsigned offset) |
203 | { | 235 | { |
204 | struct byt_gpio *vg = to_byt_gpio(chip); | 236 | struct byt_gpio *vg = to_byt_gpio(chip); |
205 | void __iomem *reg = byt_gpio_reg(&vg->chip, offset, BYT_CONF0_REG); | ||
206 | u32 value; | ||
207 | |||
208 | /* clear interrupt triggering */ | ||
209 | value = readl(reg); | ||
210 | value &= ~(BYT_TRIG_POS | BYT_TRIG_NEG | BYT_TRIG_LVL); | ||
211 | writel(value, reg); | ||
212 | 237 | ||
238 | byt_gpio_clear_triggering(vg, offset); | ||
213 | pm_runtime_put(&vg->pdev->dev); | 239 | pm_runtime_put(&vg->pdev->dev); |
214 | } | 240 | } |
215 | 241 | ||
@@ -236,23 +262,13 @@ static int byt_irq_type(struct irq_data *d, unsigned type) | |||
236 | value &= ~(BYT_DIRECT_IRQ_EN | BYT_TRIG_POS | BYT_TRIG_NEG | | 262 | value &= ~(BYT_DIRECT_IRQ_EN | BYT_TRIG_POS | BYT_TRIG_NEG | |
237 | BYT_TRIG_LVL); | 263 | BYT_TRIG_LVL); |
238 | 264 | ||
239 | switch (type) { | ||
240 | case IRQ_TYPE_LEVEL_HIGH: | ||
241 | value |= BYT_TRIG_LVL; | ||
242 | case IRQ_TYPE_EDGE_RISING: | ||
243 | value |= BYT_TRIG_POS; | ||
244 | break; | ||
245 | case IRQ_TYPE_LEVEL_LOW: | ||
246 | value |= BYT_TRIG_LVL; | ||
247 | case IRQ_TYPE_EDGE_FALLING: | ||
248 | value |= BYT_TRIG_NEG; | ||
249 | break; | ||
250 | case IRQ_TYPE_EDGE_BOTH: | ||
251 | value |= (BYT_TRIG_NEG | BYT_TRIG_POS); | ||
252 | break; | ||
253 | } | ||
254 | writel(value, reg); | 265 | writel(value, reg); |
255 | 266 | ||
267 | if (type & IRQ_TYPE_EDGE_BOTH) | ||
268 | __irq_set_handler_locked(d->irq, handle_edge_irq); | ||
269 | else if (type & IRQ_TYPE_LEVEL_MASK) | ||
270 | __irq_set_handler_locked(d->irq, handle_level_irq); | ||
271 | |||
256 | spin_unlock_irqrestore(&vg->lock, flags); | 272 | spin_unlock_irqrestore(&vg->lock, flags); |
257 | 273 | ||
258 | return 0; | 274 | return 0; |
@@ -410,58 +426,80 @@ static void byt_gpio_irq_handler(unsigned irq, struct irq_desc *desc) | |||
410 | struct irq_data *data = irq_desc_get_irq_data(desc); | 426 | struct irq_data *data = irq_desc_get_irq_data(desc); |
411 | struct byt_gpio *vg = to_byt_gpio(irq_desc_get_handler_data(desc)); | 427 | struct byt_gpio *vg = to_byt_gpio(irq_desc_get_handler_data(desc)); |
412 | struct irq_chip *chip = irq_data_get_irq_chip(data); | 428 | struct irq_chip *chip = irq_data_get_irq_chip(data); |
413 | u32 base, pin, mask; | 429 | u32 base, pin; |
414 | void __iomem *reg; | 430 | void __iomem *reg; |
415 | u32 pending; | 431 | unsigned long pending; |
416 | unsigned virq; | 432 | unsigned virq; |
417 | int looplimit = 0; | ||
418 | 433 | ||
419 | /* check from GPIO controller which pin triggered the interrupt */ | 434 | /* check from GPIO controller which pin triggered the interrupt */ |
420 | for (base = 0; base < vg->chip.ngpio; base += 32) { | 435 | for (base = 0; base < vg->chip.ngpio; base += 32) { |
421 | |||
422 | reg = byt_gpio_reg(&vg->chip, base, BYT_INT_STAT_REG); | 436 | reg = byt_gpio_reg(&vg->chip, base, BYT_INT_STAT_REG); |
423 | 437 | pending = readl(reg); | |
424 | while ((pending = readl(reg))) { | 438 | for_each_set_bit(pin, &pending, 32) { |
425 | pin = __ffs(pending); | ||
426 | mask = BIT(pin); | ||
427 | /* Clear before handling so we can't lose an edge */ | ||
428 | writel(mask, reg); | ||
429 | |||
430 | virq = irq_find_mapping(vg->chip.irqdomain, base + pin); | 439 | virq = irq_find_mapping(vg->chip.irqdomain, base + pin); |
431 | generic_handle_irq(virq); | 440 | generic_handle_irq(virq); |
432 | |||
433 | /* In case bios or user sets triggering incorretly a pin | ||
434 | * might remain in "interrupt triggered" state. | ||
435 | */ | ||
436 | if (looplimit++ > 32) { | ||
437 | dev_err(&vg->pdev->dev, | ||
438 | "Gpio %d interrupt flood, disabling\n", | ||
439 | base + pin); | ||
440 | |||
441 | reg = byt_gpio_reg(&vg->chip, base + pin, | ||
442 | BYT_CONF0_REG); | ||
443 | mask = readl(reg); | ||
444 | mask &= ~(BYT_TRIG_NEG | BYT_TRIG_POS | | ||
445 | BYT_TRIG_LVL); | ||
446 | writel(mask, reg); | ||
447 | mask = readl(reg); /* flush */ | ||
448 | break; | ||
449 | } | ||
450 | } | 441 | } |
451 | } | 442 | } |
452 | chip->irq_eoi(data); | 443 | chip->irq_eoi(data); |
453 | } | 444 | } |
454 | 445 | ||
446 | static void byt_irq_ack(struct irq_data *d) | ||
447 | { | ||
448 | struct gpio_chip *gc = irq_data_get_irq_chip_data(d); | ||
449 | struct byt_gpio *vg = to_byt_gpio(gc); | ||
450 | unsigned offset = irqd_to_hwirq(d); | ||
451 | void __iomem *reg; | ||
452 | |||
453 | reg = byt_gpio_reg(&vg->chip, offset, BYT_INT_STAT_REG); | ||
454 | writel(BIT(offset % 32), reg); | ||
455 | } | ||
456 | |||
455 | static void byt_irq_unmask(struct irq_data *d) | 457 | static void byt_irq_unmask(struct irq_data *d) |
456 | { | 458 | { |
459 | struct gpio_chip *gc = irq_data_get_irq_chip_data(d); | ||
460 | struct byt_gpio *vg = to_byt_gpio(gc); | ||
461 | unsigned offset = irqd_to_hwirq(d); | ||
462 | unsigned long flags; | ||
463 | void __iomem *reg; | ||
464 | u32 value; | ||
465 | |||
466 | spin_lock_irqsave(&vg->lock, flags); | ||
467 | |||
468 | reg = byt_gpio_reg(&vg->chip, offset, BYT_CONF0_REG); | ||
469 | value = readl(reg); | ||
470 | |||
471 | switch (irqd_get_trigger_type(d)) { | ||
472 | case IRQ_TYPE_LEVEL_HIGH: | ||
473 | value |= BYT_TRIG_LVL; | ||
474 | case IRQ_TYPE_EDGE_RISING: | ||
475 | value |= BYT_TRIG_POS; | ||
476 | break; | ||
477 | case IRQ_TYPE_LEVEL_LOW: | ||
478 | value |= BYT_TRIG_LVL; | ||
479 | case IRQ_TYPE_EDGE_FALLING: | ||
480 | value |= BYT_TRIG_NEG; | ||
481 | break; | ||
482 | case IRQ_TYPE_EDGE_BOTH: | ||
483 | value |= (BYT_TRIG_NEG | BYT_TRIG_POS); | ||
484 | break; | ||
485 | } | ||
486 | |||
487 | writel(value, reg); | ||
488 | |||
489 | spin_unlock_irqrestore(&vg->lock, flags); | ||
457 | } | 490 | } |
458 | 491 | ||
459 | static void byt_irq_mask(struct irq_data *d) | 492 | static void byt_irq_mask(struct irq_data *d) |
460 | { | 493 | { |
494 | struct gpio_chip *gc = irq_data_get_irq_chip_data(d); | ||
495 | struct byt_gpio *vg = to_byt_gpio(gc); | ||
496 | |||
497 | byt_gpio_clear_triggering(vg, irqd_to_hwirq(d)); | ||
461 | } | 498 | } |
462 | 499 | ||
463 | static struct irq_chip byt_irqchip = { | 500 | static struct irq_chip byt_irqchip = { |
464 | .name = "BYT-GPIO", | 501 | .name = "BYT-GPIO", |
502 | .irq_ack = byt_irq_ack, | ||
465 | .irq_mask = byt_irq_mask, | 503 | .irq_mask = byt_irq_mask, |
466 | .irq_unmask = byt_irq_unmask, | 504 | .irq_unmask = byt_irq_unmask, |
467 | .irq_set_type = byt_irq_type, | 505 | .irq_set_type = byt_irq_type, |
@@ -472,6 +510,21 @@ static void byt_gpio_irq_init_hw(struct byt_gpio *vg) | |||
472 | { | 510 | { |
473 | void __iomem *reg; | 511 | void __iomem *reg; |
474 | u32 base, value; | 512 | u32 base, value; |
513 | int i; | ||
514 | |||
515 | /* | ||
516 | * Clear interrupt triggers for all pins that are GPIOs and | ||
517 | * do not use direct IRQ mode. This will prevent spurious | ||
518 | * interrupts from misconfigured pins. | ||
519 | */ | ||
520 | for (i = 0; i < vg->chip.ngpio; i++) { | ||
521 | value = readl(byt_gpio_reg(&vg->chip, i, BYT_CONF0_REG)); | ||
522 | if ((value & BYT_PIN_MUX) == byt_get_gpio_mux(vg, i) && | ||
523 | !(value & BYT_DIRECT_IRQ_EN)) { | ||
524 | byt_gpio_clear_triggering(vg, i); | ||
525 | dev_dbg(&vg->pdev->dev, "disabling GPIO %d\n", i); | ||
526 | } | ||
527 | } | ||
475 | 528 | ||
476 | /* clear interrupt status trigger registers */ | 529 | /* clear interrupt status trigger registers */ |
477 | for (base = 0; base < vg->chip.ngpio; base += 32) { | 530 | for (base = 0; base < vg->chip.ngpio; base += 32) { |
@@ -541,6 +594,11 @@ static int byt_gpio_probe(struct platform_device *pdev) | |||
541 | gc->can_sleep = false; | 594 | gc->can_sleep = false; |
542 | gc->dev = dev; | 595 | gc->dev = dev; |
543 | 596 | ||
597 | #ifdef CONFIG_PM_SLEEP | ||
598 | vg->saved_context = devm_kcalloc(&pdev->dev, gc->ngpio, | ||
599 | sizeof(*vg->saved_context), GFP_KERNEL); | ||
600 | #endif | ||
601 | |||
544 | ret = gpiochip_add(gc); | 602 | ret = gpiochip_add(gc); |
545 | if (ret) { | 603 | if (ret) { |
546 | dev_err(&pdev->dev, "failed adding byt-gpio chip\n"); | 604 | dev_err(&pdev->dev, "failed adding byt-gpio chip\n"); |
@@ -569,6 +627,69 @@ static int byt_gpio_probe(struct platform_device *pdev) | |||
569 | return 0; | 627 | return 0; |
570 | } | 628 | } |
571 | 629 | ||
630 | #ifdef CONFIG_PM_SLEEP | ||
631 | static int byt_gpio_suspend(struct device *dev) | ||
632 | { | ||
633 | struct platform_device *pdev = to_platform_device(dev); | ||
634 | struct byt_gpio *vg = platform_get_drvdata(pdev); | ||
635 | int i; | ||
636 | |||
637 | for (i = 0; i < vg->chip.ngpio; i++) { | ||
638 | void __iomem *reg; | ||
639 | u32 value; | ||
640 | |||
641 | reg = byt_gpio_reg(&vg->chip, i, BYT_CONF0_REG); | ||
642 | value = readl(reg) & BYT_CONF0_RESTORE_MASK; | ||
643 | vg->saved_context[i].conf0 = value; | ||
644 | |||
645 | reg = byt_gpio_reg(&vg->chip, i, BYT_VAL_REG); | ||
646 | value = readl(reg) & BYT_VAL_RESTORE_MASK; | ||
647 | vg->saved_context[i].val = value; | ||
648 | } | ||
649 | |||
650 | return 0; | ||
651 | } | ||
652 | |||
653 | static int byt_gpio_resume(struct device *dev) | ||
654 | { | ||
655 | struct platform_device *pdev = to_platform_device(dev); | ||
656 | struct byt_gpio *vg = platform_get_drvdata(pdev); | ||
657 | int i; | ||
658 | |||
659 | for (i = 0; i < vg->chip.ngpio; i++) { | ||
660 | void __iomem *reg; | ||
661 | u32 value; | ||
662 | |||
663 | reg = byt_gpio_reg(&vg->chip, i, BYT_CONF0_REG); | ||
664 | value = readl(reg); | ||
665 | if ((value & BYT_CONF0_RESTORE_MASK) != | ||
666 | vg->saved_context[i].conf0) { | ||
667 | value &= ~BYT_CONF0_RESTORE_MASK; | ||
668 | value |= vg->saved_context[i].conf0; | ||
669 | writel(value, reg); | ||
670 | dev_info(dev, "restored pin %d conf0 %#08x", i, value); | ||
671 | } | ||
672 | |||
673 | reg = byt_gpio_reg(&vg->chip, i, BYT_VAL_REG); | ||
674 | value = readl(reg); | ||
675 | if ((value & BYT_VAL_RESTORE_MASK) != | ||
676 | vg->saved_context[i].val) { | ||
677 | u32 v; | ||
678 | |||
679 | v = value & ~BYT_VAL_RESTORE_MASK; | ||
680 | v |= vg->saved_context[i].val; | ||
681 | if (v != value) { | ||
682 | writel(v, reg); | ||
683 | dev_dbg(dev, "restored pin %d val %#08x\n", | ||
684 | i, v); | ||
685 | } | ||
686 | } | ||
687 | } | ||
688 | |||
689 | return 0; | ||
690 | } | ||
691 | #endif | ||
692 | |||
572 | static int byt_gpio_runtime_suspend(struct device *dev) | 693 | static int byt_gpio_runtime_suspend(struct device *dev) |
573 | { | 694 | { |
574 | return 0; | 695 | return 0; |
@@ -580,8 +701,9 @@ static int byt_gpio_runtime_resume(struct device *dev) | |||
580 | } | 701 | } |
581 | 702 | ||
582 | static const struct dev_pm_ops byt_gpio_pm_ops = { | 703 | static const struct dev_pm_ops byt_gpio_pm_ops = { |
583 | .runtime_suspend = byt_gpio_runtime_suspend, | 704 | SET_LATE_SYSTEM_SLEEP_PM_OPS(byt_gpio_suspend, byt_gpio_resume) |
584 | .runtime_resume = byt_gpio_runtime_resume, | 705 | SET_RUNTIME_PM_OPS(byt_gpio_runtime_suspend, byt_gpio_runtime_resume, |
706 | NULL) | ||
585 | }; | 707 | }; |
586 | 708 | ||
587 | static const struct acpi_device_id byt_gpio_acpi_match[] = { | 709 | static const struct acpi_device_id byt_gpio_acpi_match[] = { |
diff --git a/drivers/pinctrl/intel/pinctrl-cherryview.c b/drivers/pinctrl/intel/pinctrl-cherryview.c index 3034fd03bced..82f691eeeec4 100644 --- a/drivers/pinctrl/intel/pinctrl-cherryview.c +++ b/drivers/pinctrl/intel/pinctrl-cherryview.c | |||
@@ -1226,6 +1226,7 @@ static int chv_gpio_direction_input(struct gpio_chip *chip, unsigned offset) | |||
1226 | static int chv_gpio_direction_output(struct gpio_chip *chip, unsigned offset, | 1226 | static int chv_gpio_direction_output(struct gpio_chip *chip, unsigned offset, |
1227 | int value) | 1227 | int value) |
1228 | { | 1228 | { |
1229 | chv_gpio_set(chip, offset, value); | ||
1229 | return pinctrl_gpio_direction_output(chip->base + offset); | 1230 | return pinctrl_gpio_direction_output(chip->base + offset); |
1230 | } | 1231 | } |
1231 | 1232 | ||
diff --git a/drivers/pinctrl/pinctrl-at91.c b/drivers/pinctrl/pinctrl-at91.c index f4cd0b9b2438..a4814066ea08 100644 --- a/drivers/pinctrl/pinctrl-at91.c +++ b/drivers/pinctrl/pinctrl-at91.c | |||
@@ -1477,28 +1477,25 @@ static void gpio_irq_ack(struct irq_data *d) | |||
1477 | /* the interrupt is already cleared before by reading ISR */ | 1477 | /* the interrupt is already cleared before by reading ISR */ |
1478 | } | 1478 | } |
1479 | 1479 | ||
1480 | static unsigned int gpio_irq_startup(struct irq_data *d) | 1480 | static int gpio_irq_request_res(struct irq_data *d) |
1481 | { | 1481 | { |
1482 | struct at91_gpio_chip *at91_gpio = irq_data_get_irq_chip_data(d); | 1482 | struct at91_gpio_chip *at91_gpio = irq_data_get_irq_chip_data(d); |
1483 | unsigned pin = d->hwirq; | 1483 | unsigned pin = d->hwirq; |
1484 | int ret; | 1484 | int ret; |
1485 | 1485 | ||
1486 | ret = gpiochip_lock_as_irq(&at91_gpio->chip, pin); | 1486 | ret = gpiochip_lock_as_irq(&at91_gpio->chip, pin); |
1487 | if (ret) { | 1487 | if (ret) |
1488 | dev_err(at91_gpio->chip.dev, "unable to lock pind %lu IRQ\n", | 1488 | dev_err(at91_gpio->chip.dev, "unable to lock pind %lu IRQ\n", |
1489 | d->hwirq); | 1489 | d->hwirq); |
1490 | return ret; | 1490 | |
1491 | } | 1491 | return ret; |
1492 | gpio_irq_unmask(d); | ||
1493 | return 0; | ||
1494 | } | 1492 | } |
1495 | 1493 | ||
1496 | static void gpio_irq_shutdown(struct irq_data *d) | 1494 | static void gpio_irq_release_res(struct irq_data *d) |
1497 | { | 1495 | { |
1498 | struct at91_gpio_chip *at91_gpio = irq_data_get_irq_chip_data(d); | 1496 | struct at91_gpio_chip *at91_gpio = irq_data_get_irq_chip_data(d); |
1499 | unsigned pin = d->hwirq; | 1497 | unsigned pin = d->hwirq; |
1500 | 1498 | ||
1501 | gpio_irq_mask(d); | ||
1502 | gpiochip_unlock_as_irq(&at91_gpio->chip, pin); | 1499 | gpiochip_unlock_as_irq(&at91_gpio->chip, pin); |
1503 | } | 1500 | } |
1504 | 1501 | ||
@@ -1577,8 +1574,8 @@ void at91_pinctrl_gpio_resume(void) | |||
1577 | static struct irq_chip gpio_irqchip = { | 1574 | static struct irq_chip gpio_irqchip = { |
1578 | .name = "GPIO", | 1575 | .name = "GPIO", |
1579 | .irq_ack = gpio_irq_ack, | 1576 | .irq_ack = gpio_irq_ack, |
1580 | .irq_startup = gpio_irq_startup, | 1577 | .irq_request_resources = gpio_irq_request_res, |
1581 | .irq_shutdown = gpio_irq_shutdown, | 1578 | .irq_release_resources = gpio_irq_release_res, |
1582 | .irq_disable = gpio_irq_mask, | 1579 | .irq_disable = gpio_irq_mask, |
1583 | .irq_mask = gpio_irq_mask, | 1580 | .irq_mask = gpio_irq_mask, |
1584 | .irq_unmask = gpio_irq_unmask, | 1581 | .irq_unmask = gpio_irq_unmask, |
diff --git a/drivers/pinctrl/sunxi/pinctrl-sun4i-a10.c b/drivers/pinctrl/sunxi/pinctrl-sun4i-a10.c index 24c5d88f943f..3c68a8e5e0dd 100644 --- a/drivers/pinctrl/sunxi/pinctrl-sun4i-a10.c +++ b/drivers/pinctrl/sunxi/pinctrl-sun4i-a10.c | |||
@@ -1011,6 +1011,7 @@ static const struct sunxi_pinctrl_desc sun4i_a10_pinctrl_data = { | |||
1011 | .pins = sun4i_a10_pins, | 1011 | .pins = sun4i_a10_pins, |
1012 | .npins = ARRAY_SIZE(sun4i_a10_pins), | 1012 | .npins = ARRAY_SIZE(sun4i_a10_pins), |
1013 | .irq_banks = 1, | 1013 | .irq_banks = 1, |
1014 | .irq_read_needs_mux = true, | ||
1014 | }; | 1015 | }; |
1015 | 1016 | ||
1016 | static int sun4i_a10_pinctrl_probe(struct platform_device *pdev) | 1017 | static int sun4i_a10_pinctrl_probe(struct platform_device *pdev) |
diff --git a/drivers/pinctrl/sunxi/pinctrl-sunxi.c b/drivers/pinctrl/sunxi/pinctrl-sunxi.c index 3d0744337736..f8e171b76693 100644 --- a/drivers/pinctrl/sunxi/pinctrl-sunxi.c +++ b/drivers/pinctrl/sunxi/pinctrl-sunxi.c | |||
@@ -29,6 +29,7 @@ | |||
29 | #include <linux/slab.h> | 29 | #include <linux/slab.h> |
30 | 30 | ||
31 | #include "../core.h" | 31 | #include "../core.h" |
32 | #include "../../gpio/gpiolib.h" | ||
32 | #include "pinctrl-sunxi.h" | 33 | #include "pinctrl-sunxi.h" |
33 | 34 | ||
34 | static struct irq_chip sunxi_pinctrl_edge_irq_chip; | 35 | static struct irq_chip sunxi_pinctrl_edge_irq_chip; |
@@ -464,10 +465,19 @@ static int sunxi_pinctrl_gpio_direction_input(struct gpio_chip *chip, | |||
464 | static int sunxi_pinctrl_gpio_get(struct gpio_chip *chip, unsigned offset) | 465 | static int sunxi_pinctrl_gpio_get(struct gpio_chip *chip, unsigned offset) |
465 | { | 466 | { |
466 | struct sunxi_pinctrl *pctl = dev_get_drvdata(chip->dev); | 467 | struct sunxi_pinctrl *pctl = dev_get_drvdata(chip->dev); |
467 | |||
468 | u32 reg = sunxi_data_reg(offset); | 468 | u32 reg = sunxi_data_reg(offset); |
469 | u8 index = sunxi_data_offset(offset); | 469 | u8 index = sunxi_data_offset(offset); |
470 | u32 val = (readl(pctl->membase + reg) >> index) & DATA_PINS_MASK; | 470 | u32 set_mux = pctl->desc->irq_read_needs_mux && |
471 | test_bit(FLAG_USED_AS_IRQ, &chip->desc[offset].flags); | ||
472 | u32 val; | ||
473 | |||
474 | if (set_mux) | ||
475 | sunxi_pmx_set(pctl->pctl_dev, offset, SUN4I_FUNC_INPUT); | ||
476 | |||
477 | val = (readl(pctl->membase + reg) >> index) & DATA_PINS_MASK; | ||
478 | |||
479 | if (set_mux) | ||
480 | sunxi_pmx_set(pctl->pctl_dev, offset, SUN4I_FUNC_IRQ); | ||
471 | 481 | ||
472 | return val; | 482 | return val; |
473 | } | 483 | } |
diff --git a/drivers/pinctrl/sunxi/pinctrl-sunxi.h b/drivers/pinctrl/sunxi/pinctrl-sunxi.h index 5a51523a3459..e248e81a0f9e 100644 --- a/drivers/pinctrl/sunxi/pinctrl-sunxi.h +++ b/drivers/pinctrl/sunxi/pinctrl-sunxi.h | |||
@@ -77,6 +77,9 @@ | |||
77 | #define IRQ_LEVEL_LOW 0x03 | 77 | #define IRQ_LEVEL_LOW 0x03 |
78 | #define IRQ_EDGE_BOTH 0x04 | 78 | #define IRQ_EDGE_BOTH 0x04 |
79 | 79 | ||
80 | #define SUN4I_FUNC_INPUT 0 | ||
81 | #define SUN4I_FUNC_IRQ 6 | ||
82 | |||
80 | struct sunxi_desc_function { | 83 | struct sunxi_desc_function { |
81 | const char *name; | 84 | const char *name; |
82 | u8 muxval; | 85 | u8 muxval; |
@@ -94,6 +97,7 @@ struct sunxi_pinctrl_desc { | |||
94 | int npins; | 97 | int npins; |
95 | unsigned pin_base; | 98 | unsigned pin_base; |
96 | unsigned irq_banks; | 99 | unsigned irq_banks; |
100 | bool irq_read_needs_mux; | ||
97 | }; | 101 | }; |
98 | 102 | ||
99 | struct sunxi_pinctrl_function { | 103 | struct sunxi_pinctrl_function { |