Lines Matching +full:edge +full:- +full:offset

1 // SPDX-License-Identifier: GPL-2.0-only
22 #include <linux/pinctrl/pinconf-generic.h>
30 #include "gpio-tangier.h"
37 #define GRER 0x064 /* Rising edge detect */
38 #define GFER 0x07c /* Falling edge detect */
46 * struct tng_gpio_context - Context to be saved during suspend-resume
49 * @grer: Rising edge detect enable
50 * @gfer: Falling edge detect enable
63 static void __iomem *gpio_reg(struct gpio_chip *chip, unsigned int offset, in gpio_reg() argument
67 u8 reg_offset = offset / 32; in gpio_reg()
69 return priv->reg_base + reg + reg_offset * 4; in gpio_reg()
72 static void __iomem *gpio_reg_and_bit(struct gpio_chip *chip, unsigned int offset, in gpio_reg_and_bit() argument
76 u8 reg_offset = offset / 32; in gpio_reg_and_bit()
77 u8 shift = offset % 32; in gpio_reg_and_bit()
80 return priv->reg_base + reg + reg_offset * 4; in gpio_reg_and_bit()
83 static int tng_gpio_get(struct gpio_chip *chip, unsigned int offset) in tng_gpio_get() argument
88 gplr = gpio_reg_and_bit(chip, offset, GPLR, &shift); in tng_gpio_get()
93 static void tng_gpio_set(struct gpio_chip *chip, unsigned int offset, int value) in tng_gpio_set() argument
99 reg = gpio_reg_and_bit(chip, offset, value ? GPSR : GPCR, &shift); in tng_gpio_set()
101 guard(raw_spinlock_irqsave)(&priv->lock); in tng_gpio_set()
106 static int tng_gpio_direction_input(struct gpio_chip *chip, unsigned int offset) in tng_gpio_direction_input() argument
113 gpdr = gpio_reg_and_bit(chip, offset, GPDR, &shift); in tng_gpio_direction_input()
115 guard(raw_spinlock_irqsave)(&priv->lock); in tng_gpio_direction_input()
124 static int tng_gpio_direction_output(struct gpio_chip *chip, unsigned int offset, in tng_gpio_direction_output() argument
131 gpdr = gpio_reg_and_bit(chip, offset, GPDR, &shift); in tng_gpio_direction_output()
132 tng_gpio_set(chip, offset, value); in tng_gpio_direction_output()
134 guard(raw_spinlock_irqsave)(&priv->lock); in tng_gpio_direction_output()
143 static int tng_gpio_get_direction(struct gpio_chip *chip, unsigned int offset) in tng_gpio_get_direction() argument
148 gpdr = gpio_reg_and_bit(chip, offset, GPDR, &shift); in tng_gpio_get_direction()
156 static int tng_gpio_set_debounce(struct gpio_chip *chip, unsigned int offset, in tng_gpio_set_debounce() argument
164 gfbr = gpio_reg_and_bit(chip, offset, GFBR, &shift); in tng_gpio_set_debounce()
166 guard(raw_spinlock_irqsave)(&priv->lock); in tng_gpio_set_debounce()
178 static int tng_gpio_set_config(struct gpio_chip *chip, unsigned int offset, in tng_gpio_set_config() argument
187 return gpiochip_generic_config(chip, offset, config); in tng_gpio_set_config()
190 return tng_gpio_set_debounce(chip, offset, debounce); in tng_gpio_set_config()
192 return -ENOTSUPP; in tng_gpio_set_config()
204 gisr = gpio_reg_and_bit(&priv->chip, gpio, GISR, &shift); in tng_irq_ack()
206 guard(raw_spinlock_irqsave)(&priv->lock); in tng_irq_ack()
217 gimr = gpio_reg_and_bit(&priv->chip, gpio, GIMR, &shift); in tng_irq_unmask_mask()
219 guard(raw_spinlock_irqsave)(&priv->lock); in tng_irq_unmask_mask()
236 gpiochip_disable_irq(&priv->chip, gpio); in tng_irq_mask()
245 gpiochip_enable_irq(&priv->chip, gpio); in tng_irq_unmask()
254 void __iomem *grer = gpio_reg(&priv->chip, gpio, GRER); in tng_irq_set_type()
255 void __iomem *gfer = gpio_reg(&priv->chip, gpio, GFER); in tng_irq_set_type()
256 void __iomem *gitr = gpio_reg(&priv->chip, gpio, GITR); in tng_irq_set_type()
257 void __iomem *glpr = gpio_reg(&priv->chip, gpio, GLPR); in tng_irq_set_type()
261 guard(raw_spinlock_irqsave)(&priv->lock); in tng_irq_set_type()
310 void __iomem *gwmr = gpio_reg(&priv->chip, gpio, priv->wake_regs.gwmr); in tng_irq_set_wake()
311 void __iomem *gwsr = gpio_reg(&priv->chip, gpio, priv->wake_regs.gwsr); in tng_irq_set_wake()
315 dev_dbg(priv->dev, "%s wake for gpio %lu\n", str_enable_disable(on), gpio); in tng_irq_set_wake()
317 guard(raw_spinlock_irqsave)(&priv->lock); in tng_irq_set_wake()
333 .name = "gpio-tangier",
353 for (base = 0; base < priv->chip.ngpio; base += 32) { in tng_irq_handler()
354 void __iomem *gisr = gpio_reg(&priv->chip, base, GISR); in tng_irq_handler()
355 void __iomem *gimr = gpio_reg(&priv->chip, base, GIMR); in tng_irq_handler()
365 generic_handle_domain_irq(gc->irq.domain, base + gpio); in tng_irq_handler()
377 for (base = 0; base < priv->chip.ngpio; base += 32) { in tng_irq_init_hw()
378 /* Clear the rising-edge detect register */ in tng_irq_init_hw()
379 reg = gpio_reg(&priv->chip, base, GRER); in tng_irq_init_hw()
382 /* Clear the falling-edge detect register */ in tng_irq_init_hw()
383 reg = gpio_reg(&priv->chip, base, GFER); in tng_irq_init_hw()
397 for (i = 0; i < priv->pin_info.nranges; i++) { in tng_gpio_add_pin_ranges()
398 range = &priv->pin_info.pin_ranges[i]; in tng_gpio_add_pin_ranges()
399 ret = gpiochip_add_pin_range(&priv->chip, in tng_gpio_add_pin_ranges()
400 priv->pin_info.name, in tng_gpio_add_pin_ranges()
401 range->gpio_base, in tng_gpio_add_pin_ranges()
402 range->pin_base, in tng_gpio_add_pin_ranges()
403 range->npins); in tng_gpio_add_pin_ranges()
405 dev_err(priv->dev, "failed to add GPIO pin range\n"); in tng_gpio_add_pin_ranges()
415 const struct tng_gpio_info *info = &gpio->info; in devm_tng_gpio_probe()
416 size_t nctx = DIV_ROUND_UP(info->ngpio, 32); in devm_tng_gpio_probe()
420 gpio->ctx = devm_kcalloc(dev, nctx, sizeof(*gpio->ctx), GFP_KERNEL); in devm_tng_gpio_probe()
421 if (!gpio->ctx) in devm_tng_gpio_probe()
422 return -ENOMEM; in devm_tng_gpio_probe()
424 gpio->chip.label = dev_name(dev); in devm_tng_gpio_probe()
425 gpio->chip.parent = dev; in devm_tng_gpio_probe()
426 gpio->chip.request = gpiochip_generic_request; in devm_tng_gpio_probe()
427 gpio->chip.free = gpiochip_generic_free; in devm_tng_gpio_probe()
428 gpio->chip.direction_input = tng_gpio_direction_input; in devm_tng_gpio_probe()
429 gpio->chip.direction_output = tng_gpio_direction_output; in devm_tng_gpio_probe()
430 gpio->chip.get = tng_gpio_get; in devm_tng_gpio_probe()
431 gpio->chip.set = tng_gpio_set; in devm_tng_gpio_probe()
432 gpio->chip.get_direction = tng_gpio_get_direction; in devm_tng_gpio_probe()
433 gpio->chip.set_config = tng_gpio_set_config; in devm_tng_gpio_probe()
434 gpio->chip.base = info->base; in devm_tng_gpio_probe()
435 gpio->chip.ngpio = info->ngpio; in devm_tng_gpio_probe()
436 gpio->chip.can_sleep = false; in devm_tng_gpio_probe()
437 gpio->chip.add_pin_ranges = tng_gpio_add_pin_ranges; in devm_tng_gpio_probe()
439 raw_spin_lock_init(&gpio->lock); in devm_tng_gpio_probe()
441 girq = &gpio->chip.irq; in devm_tng_gpio_probe()
443 girq->init_hw = tng_irq_init_hw; in devm_tng_gpio_probe()
444 girq->parent_handler = tng_irq_handler; in devm_tng_gpio_probe()
445 girq->num_parents = 1; in devm_tng_gpio_probe()
446 girq->parents = devm_kcalloc(dev, girq->num_parents, in devm_tng_gpio_probe()
447 sizeof(*girq->parents), GFP_KERNEL); in devm_tng_gpio_probe()
448 if (!girq->parents) in devm_tng_gpio_probe()
449 return -ENOMEM; in devm_tng_gpio_probe()
451 girq->parents[0] = gpio->irq; in devm_tng_gpio_probe()
452 girq->first = info->first; in devm_tng_gpio_probe()
453 girq->default_type = IRQ_TYPE_NONE; in devm_tng_gpio_probe()
454 girq->handler = handle_bad_irq; in devm_tng_gpio_probe()
456 ret = devm_gpiochip_add_data(dev, &gpio->chip, gpio); in devm_tng_gpio_probe()
467 struct tng_gpio_context *ctx = priv->ctx; in tng_gpio_suspend()
470 guard(raw_spinlock_irqsave)(&priv->lock); in tng_gpio_suspend()
472 for (base = 0; base < priv->chip.ngpio; base += 32, ctx++) { in tng_gpio_suspend()
474 ctx->level = readl(gpio_reg(&priv->chip, base, GPLR)); in tng_gpio_suspend()
476 ctx->gpdr = readl(gpio_reg(&priv->chip, base, GPDR)); in tng_gpio_suspend()
477 ctx->grer = readl(gpio_reg(&priv->chip, base, GRER)); in tng_gpio_suspend()
478 ctx->gfer = readl(gpio_reg(&priv->chip, base, GFER)); in tng_gpio_suspend()
479 ctx->gimr = readl(gpio_reg(&priv->chip, base, GIMR)); in tng_gpio_suspend()
481 ctx->gwmr = readl(gpio_reg(&priv->chip, base, priv->wake_regs.gwmr)); in tng_gpio_suspend()
490 struct tng_gpio_context *ctx = priv->ctx; in tng_gpio_resume()
493 guard(raw_spinlock_irqsave)(&priv->lock); in tng_gpio_resume()
495 for (base = 0; base < priv->chip.ngpio; base += 32, ctx++) { in tng_gpio_resume()
497 writel(ctx->level, gpio_reg(&priv->chip, base, GPSR)); in tng_gpio_resume()
499 writel(ctx->gpdr, gpio_reg(&priv->chip, base, GPDR)); in tng_gpio_resume()
500 writel(ctx->grer, gpio_reg(&priv->chip, base, GRER)); in tng_gpio_resume()
501 writel(ctx->gfer, gpio_reg(&priv->chip, base, GFER)); in tng_gpio_resume()
502 writel(ctx->gimr, gpio_reg(&priv->chip, base, GIMR)); in tng_gpio_resume()
504 writel(ctx->gwmr, gpio_reg(&priv->chip, base, priv->wake_regs.gwmr)); in tng_gpio_resume()