Lines Matching +full:edge +full:- +full:offset
1 // SPDX-License-Identifier: GPL-2.0
3 * Renesas R-Car GPIO Support
61 #define EDGLEVEL 0x24 /* Edge/level Select Register */
64 #define BOTHEDGE 0x4c /* One Edge/Both Edge Select Register */
71 return ioread32(p->base + offs); in gpio_rcar_read()
77 iowrite32(value, p->base + offs); in gpio_rcar_write()
122 * "Setting Edge-Sensitive Interrupt Input Mode" and in gpio_rcar_config_interrupt_input_mode()
123 * "Setting Level-Sensitive Interrupt Input Mode" in gpio_rcar_config_interrupt_input_mode()
126 spin_lock_irqsave(&p->lock, flags); in gpio_rcar_config_interrupt_input_mode()
131 /* Configure edge or level trigger in EDGLEVEL */ in gpio_rcar_config_interrupt_input_mode()
134 /* Select one edge or both edges in BOTHEDGE */ in gpio_rcar_config_interrupt_input_mode()
135 if (p->info.has_both_edge_trigger) in gpio_rcar_config_interrupt_input_mode()
141 /* Write INTCLR in case of edge trigger */ in gpio_rcar_config_interrupt_input_mode()
145 spin_unlock_irqrestore(&p->lock, flags); in gpio_rcar_config_interrupt_input_mode()
154 dev_dbg(p->dev, "sense irq = %d, type = %d\n", hwirq, type); in gpio_rcar_irq_set_type()
174 if (!p->info.has_both_edge_trigger) in gpio_rcar_irq_set_type()
175 return -EINVAL; in gpio_rcar_irq_set_type()
180 return -EINVAL; in gpio_rcar_irq_set_type()
191 if (p->irq_parent) { in gpio_rcar_irq_set_wake()
192 error = irq_set_irq_wake(p->irq_parent, on); in gpio_rcar_irq_set_wake()
194 dev_dbg(p->dev, "irq %u doesn't support irq_set_wake\n", in gpio_rcar_irq_set_wake()
195 p->irq_parent); in gpio_rcar_irq_set_wake()
196 p->irq_parent = 0; in gpio_rcar_irq_set_wake()
201 atomic_inc(&p->wakeup_path); in gpio_rcar_irq_set_wake()
203 atomic_dec(&p->wakeup_path); in gpio_rcar_irq_set_wake()
209 .name = "gpio-rcar",
223 unsigned int offset, irqs_handled = 0; in gpio_rcar_irq_handler() local
227 offset = __ffs(pending); in gpio_rcar_irq_handler()
228 gpio_rcar_write(p, INTCLR, BIT(offset)); in gpio_rcar_irq_handler()
229 generic_handle_domain_irq(p->gpio_chip.irq.domain, in gpio_rcar_irq_handler()
230 offset); in gpio_rcar_irq_handler()
249 spin_lock_irqsave(&p->lock, flags); in gpio_rcar_config_general_input_output_mode()
261 if (p->info.has_outdtsel && output) in gpio_rcar_config_general_input_output_mode()
264 spin_unlock_irqrestore(&p->lock, flags); in gpio_rcar_config_general_input_output_mode()
267 static int gpio_rcar_request(struct gpio_chip *chip, unsigned offset) in gpio_rcar_request() argument
272 error = pm_runtime_get_sync(p->dev); in gpio_rcar_request()
274 pm_runtime_put(p->dev); in gpio_rcar_request()
278 error = pinctrl_gpio_request(chip, offset); in gpio_rcar_request()
280 pm_runtime_put(p->dev); in gpio_rcar_request()
285 static void gpio_rcar_free(struct gpio_chip *chip, unsigned offset) in gpio_rcar_free() argument
289 pinctrl_gpio_free(chip, offset); in gpio_rcar_free()
295 gpio_rcar_config_general_input_output_mode(chip, offset, false); in gpio_rcar_free()
297 pm_runtime_put(p->dev); in gpio_rcar_free()
300 static int gpio_rcar_get_direction(struct gpio_chip *chip, unsigned int offset) in gpio_rcar_get_direction() argument
304 if (gpio_rcar_read(p, INOUTSEL) & BIT(offset)) in gpio_rcar_get_direction()
310 static int gpio_rcar_direction_input(struct gpio_chip *chip, unsigned offset) in gpio_rcar_direction_input() argument
312 gpio_rcar_config_general_input_output_mode(chip, offset, false); in gpio_rcar_direction_input()
316 static int gpio_rcar_get(struct gpio_chip *chip, unsigned offset) in gpio_rcar_get() argument
319 u32 bit = BIT(offset); in gpio_rcar_get()
322 * Before R-Car Gen3, INDT does not show correct pin state when in gpio_rcar_get()
325 if (!p->info.has_always_in && (gpio_rcar_read(p, INOUTSEL) & bit)) in gpio_rcar_get()
338 bankmask = mask[0] & GENMASK(chip->ngpio - 1, 0); in gpio_rcar_get_multiple()
339 if (chip->valid_mask) in gpio_rcar_get_multiple()
340 bankmask &= chip->valid_mask[0]; in gpio_rcar_get_multiple()
345 if (p->info.has_always_in) { in gpio_rcar_get_multiple()
350 spin_lock_irqsave(&p->lock, flags); in gpio_rcar_get_multiple()
359 spin_unlock_irqrestore(&p->lock, flags); in gpio_rcar_get_multiple()
365 static void gpio_rcar_set(struct gpio_chip *chip, unsigned offset, int value) in gpio_rcar_set() argument
370 spin_lock_irqsave(&p->lock, flags); in gpio_rcar_set()
371 gpio_rcar_modify_bit(p, OUTDT, offset, value); in gpio_rcar_set()
372 spin_unlock_irqrestore(&p->lock, flags); in gpio_rcar_set()
382 bankmask = mask[0] & GENMASK(chip->ngpio - 1, 0); in gpio_rcar_set_multiple()
383 if (chip->valid_mask) in gpio_rcar_set_multiple()
384 bankmask &= chip->valid_mask[0]; in gpio_rcar_set_multiple()
389 spin_lock_irqsave(&p->lock, flags); in gpio_rcar_set_multiple()
394 spin_unlock_irqrestore(&p->lock, flags); in gpio_rcar_set_multiple()
397 static int gpio_rcar_direction_output(struct gpio_chip *chip, unsigned offset, in gpio_rcar_direction_output() argument
401 gpio_rcar_set(chip, offset, value); in gpio_rcar_direction_output()
402 gpio_rcar_config_general_input_output_mode(chip, offset, true); in gpio_rcar_direction_output()
436 .compatible = "renesas,gpio-r8a779a0",
439 .compatible = "renesas,rcar-gen1-gpio",
442 .compatible = "renesas,rcar-gen2-gpio",
445 .compatible = "renesas,rcar-gen3-gpio",
448 .compatible = "renesas,rcar-gen4-gpio",
451 .compatible = "renesas,gpio-rcar",
462 struct device_node *np = p->dev->of_node; in gpio_rcar_parse_dt()
467 info = of_device_get_match_data(p->dev); in gpio_rcar_parse_dt()
468 p->info = *info; in gpio_rcar_parse_dt()
470 ret = of_parse_phandle_with_fixed_args(np, "gpio-ranges", 3, 0, &args); in gpio_rcar_parse_dt()
474 dev_warn(p->dev, "Invalid number of gpio lines %u, using %u\n", in gpio_rcar_parse_dt()
484 u32 mask = GENMASK(p->gpio_chip.ngpio - 1, 0); in gpio_rcar_enable_inputs()
487 if (p->gpio_chip.valid_mask) in gpio_rcar_enable_inputs()
488 mask &= p->gpio_chip.valid_mask[0]; in gpio_rcar_enable_inputs()
498 struct device *dev = &pdev->dev; in gpio_rcar_probe()
505 return -ENOMEM; in gpio_rcar_probe()
507 p->dev = dev; in gpio_rcar_probe()
508 spin_lock_init(&p->lock); in gpio_rcar_probe()
522 p->irq_parent = ret; in gpio_rcar_probe()
524 p->base = devm_platform_ioremap_resource(pdev, 0); in gpio_rcar_probe()
525 if (IS_ERR(p->base)) { in gpio_rcar_probe()
526 ret = PTR_ERR(p->base); in gpio_rcar_probe()
530 gpio_chip = &p->gpio_chip; in gpio_rcar_probe()
531 gpio_chip->request = gpio_rcar_request; in gpio_rcar_probe()
532 gpio_chip->free = gpio_rcar_free; in gpio_rcar_probe()
533 gpio_chip->get_direction = gpio_rcar_get_direction; in gpio_rcar_probe()
534 gpio_chip->direction_input = gpio_rcar_direction_input; in gpio_rcar_probe()
535 gpio_chip->get = gpio_rcar_get; in gpio_rcar_probe()
536 gpio_chip->get_multiple = gpio_rcar_get_multiple; in gpio_rcar_probe()
537 gpio_chip->direction_output = gpio_rcar_direction_output; in gpio_rcar_probe()
538 gpio_chip->set = gpio_rcar_set; in gpio_rcar_probe()
539 gpio_chip->set_multiple = gpio_rcar_set_multiple; in gpio_rcar_probe()
540 gpio_chip->label = name; in gpio_rcar_probe()
541 gpio_chip->parent = dev; in gpio_rcar_probe()
542 gpio_chip->owner = THIS_MODULE; in gpio_rcar_probe()
543 gpio_chip->base = -1; in gpio_rcar_probe()
544 gpio_chip->ngpio = npins; in gpio_rcar_probe()
546 girq = &gpio_chip->irq; in gpio_rcar_probe()
549 girq->parent_handler = NULL; in gpio_rcar_probe()
550 girq->num_parents = 0; in gpio_rcar_probe()
551 girq->parents = NULL; in gpio_rcar_probe()
552 girq->default_type = IRQ_TYPE_NONE; in gpio_rcar_probe()
553 girq->handler = handle_level_irq; in gpio_rcar_probe()
561 irq_domain_set_pm_device(gpio_chip->irq.domain, dev); in gpio_rcar_probe()
562 ret = devm_request_irq(dev, p->irq_parent, gpio_rcar_irq_handler, in gpio_rcar_probe()
569 if (p->info.has_inen) { in gpio_rcar_probe()
590 gpiochip_remove(&p->gpio_chip); in gpio_rcar_remove()
592 pm_runtime_disable(&pdev->dev); in gpio_rcar_remove()
600 p->bank_info.iointsel = gpio_rcar_read(p, IOINTSEL); in gpio_rcar_suspend()
601 p->bank_info.inoutsel = gpio_rcar_read(p, INOUTSEL); in gpio_rcar_suspend()
602 p->bank_info.outdt = gpio_rcar_read(p, OUTDT); in gpio_rcar_suspend()
603 p->bank_info.intmsk = gpio_rcar_read(p, INTMSK); in gpio_rcar_suspend()
604 p->bank_info.posneg = gpio_rcar_read(p, POSNEG); in gpio_rcar_suspend()
605 p->bank_info.edglevel = gpio_rcar_read(p, EDGLEVEL); in gpio_rcar_suspend()
606 if (p->info.has_both_edge_trigger) in gpio_rcar_suspend()
607 p->bank_info.bothedge = gpio_rcar_read(p, BOTHEDGE); in gpio_rcar_suspend()
609 if (atomic_read(&p->wakeup_path)) in gpio_rcar_suspend()
618 unsigned int offset; in gpio_rcar_resume() local
621 for (offset = 0; offset < p->gpio_chip.ngpio; offset++) { in gpio_rcar_resume()
622 if (!gpiochip_line_is_valid(&p->gpio_chip, offset)) in gpio_rcar_resume()
625 mask = BIT(offset); in gpio_rcar_resume()
627 if (!(p->bank_info.iointsel & mask)) { in gpio_rcar_resume()
628 if (p->bank_info.inoutsel & mask) in gpio_rcar_resume()
630 &p->gpio_chip, offset, in gpio_rcar_resume()
631 !!(p->bank_info.outdt & mask)); in gpio_rcar_resume()
633 gpio_rcar_direction_input(&p->gpio_chip, in gpio_rcar_resume()
634 offset); in gpio_rcar_resume()
639 offset, in gpio_rcar_resume()
640 !(p->bank_info.posneg & mask), in gpio_rcar_resume()
641 !(p->bank_info.edglevel & mask), in gpio_rcar_resume()
642 !!(p->bank_info.bothedge & mask)); in gpio_rcar_resume()
644 if (p->bank_info.intmsk & mask) in gpio_rcar_resume()
649 if (p->info.has_inen) in gpio_rcar_resume()
671 MODULE_DESCRIPTION("Renesas R-Car GPIO Driver");