Lines Matching +full:int +full:- +full:fwd +full:- +full:mask
1 // SPDX-License-Identifier: GPL-2.0-only
42 unsigned int n_words;
45 int en_offset[MAX_WORDS];
46 int stat_offset[MAX_WORDS];
51 int num_parent_irqs;
58 struct bcm7120_l2_intc_data *b = data->b; in bcm7120_l2_intc_irq_handle()
60 unsigned int idx; in bcm7120_l2_intc_irq_handle()
64 for (idx = 0; idx < b->n_words; idx++) { in bcm7120_l2_intc_irq_handle()
65 int base = idx * IRQS_PER_WORD; in bcm7120_l2_intc_irq_handle()
67 irq_get_domain_generic_chip(b->domain, base); in bcm7120_l2_intc_irq_handle()
69 int hwirq; in bcm7120_l2_intc_irq_handle()
72 pending = irq_reg_readl(gc, b->stat_offset[idx]) & in bcm7120_l2_intc_irq_handle()
73 gc->mask_cache & in bcm7120_l2_intc_irq_handle()
74 data->irq_map_mask[idx]; in bcm7120_l2_intc_irq_handle()
78 generic_handle_irq(irq_find_mapping(b->domain, in bcm7120_l2_intc_irq_handle()
88 struct bcm7120_l2_intc_data *b = gc->private; in bcm7120_l2_intc_suspend()
89 struct irq_chip_type *ct = gc->chip_types; in bcm7120_l2_intc_suspend()
92 if (b->can_wake) in bcm7120_l2_intc_suspend()
93 irq_reg_writel(gc, gc->mask_cache | gc->wake_active, in bcm7120_l2_intc_suspend()
94 ct->regs.mask); in bcm7120_l2_intc_suspend()
100 struct irq_chip_type *ct = gc->chip_types; in bcm7120_l2_intc_resume()
102 /* Restore the saved mask */ in bcm7120_l2_intc_resume()
104 irq_reg_writel(gc, gc->mask_cache, ct->regs.mask); in bcm7120_l2_intc_resume()
108 static int bcm7120_l2_intc_init_one(struct device_node *dn, in bcm7120_l2_intc_init_one()
110 int irq, u32 *valid_mask) in bcm7120_l2_intc_init_one()
112 struct bcm7120_l1_intc_data *l1_data = &data->l1_data[irq]; in bcm7120_l2_intc_init_one()
113 int parent_irq; in bcm7120_l2_intc_init_one()
114 unsigned int idx; in bcm7120_l2_intc_init_one()
119 return -EINVAL; in bcm7120_l2_intc_init_one()
126 * map_mask in order to mask the status register with it because we in bcm7120_l2_intc_init_one()
131 for (idx = 0; idx < data->n_words; idx++) { in bcm7120_l2_intc_init_one()
132 if (data->map_mask_prop) { in bcm7120_l2_intc_init_one()
133 l1_data->irq_map_mask[idx] |= in bcm7120_l2_intc_init_one()
134 be32_to_cpup(data->map_mask_prop + in bcm7120_l2_intc_init_one()
135 irq * data->n_words + idx); in bcm7120_l2_intc_init_one()
137 l1_data->irq_map_mask[idx] = 0xffffffff; in bcm7120_l2_intc_init_one()
139 valid_mask[idx] |= l1_data->irq_map_mask[idx]; in bcm7120_l2_intc_init_one()
142 l1_data->b = data; in bcm7120_l2_intc_init_one()
146 if (data->can_wake) in bcm7120_l2_intc_init_one()
152 static int __init bcm7120_l2_intc_iomap_7120(struct device_node *dn, in bcm7120_l2_intc_iomap_7120()
155 int ret; in bcm7120_l2_intc_iomap_7120()
157 data->map_base[0] = of_iomap(dn, 0); in bcm7120_l2_intc_iomap_7120()
158 if (!data->map_base[0]) { in bcm7120_l2_intc_iomap_7120()
160 return -ENOMEM; in bcm7120_l2_intc_iomap_7120()
163 data->pair_base[0] = data->map_base[0]; in bcm7120_l2_intc_iomap_7120()
164 data->en_offset[0] = IRQEN; in bcm7120_l2_intc_iomap_7120()
165 data->stat_offset[0] = IRQSTAT; in bcm7120_l2_intc_iomap_7120()
166 data->n_words = 1; in bcm7120_l2_intc_iomap_7120()
168 ret = of_property_read_u32_array(dn, "brcm,int-fwd-mask", in bcm7120_l2_intc_iomap_7120()
169 data->irq_fwd_mask, data->n_words); in bcm7120_l2_intc_iomap_7120()
170 if (ret != 0 && ret != -EINVAL) { in bcm7120_l2_intc_iomap_7120()
172 pr_err("invalid brcm,int-fwd-mask property\n"); in bcm7120_l2_intc_iomap_7120()
173 return -EINVAL; in bcm7120_l2_intc_iomap_7120()
176 data->map_mask_prop = of_get_property(dn, "brcm,int-map-mask", &ret); in bcm7120_l2_intc_iomap_7120()
177 if (!data->map_mask_prop || in bcm7120_l2_intc_iomap_7120()
178 (ret != (sizeof(__be32) * data->num_parent_irqs * data->n_words))) { in bcm7120_l2_intc_iomap_7120()
179 pr_err("invalid brcm,int-map-mask property\n"); in bcm7120_l2_intc_iomap_7120()
180 return -EINVAL; in bcm7120_l2_intc_iomap_7120()
186 static int __init bcm7120_l2_intc_iomap_3380(struct device_node *dn, in bcm7120_l2_intc_iomap_3380()
189 unsigned int gc_idx; in bcm7120_l2_intc_iomap_3380()
192 unsigned int map_idx = gc_idx * 2; in bcm7120_l2_intc_iomap_3380()
197 data->map_base[map_idx + 0] = en; in bcm7120_l2_intc_iomap_3380()
198 data->map_base[map_idx + 1] = stat; in bcm7120_l2_intc_iomap_3380()
203 data->pair_base[gc_idx] = base; in bcm7120_l2_intc_iomap_3380()
204 data->en_offset[gc_idx] = en - base; in bcm7120_l2_intc_iomap_3380()
205 data->stat_offset[gc_idx] = stat - base; in bcm7120_l2_intc_iomap_3380()
210 return -EINVAL; in bcm7120_l2_intc_iomap_3380()
213 data->n_words = gc_idx; in bcm7120_l2_intc_iomap_3380()
217 static int __init bcm7120_l2_intc_probe(struct device_node *dn, in bcm7120_l2_intc_probe()
219 int (*iomap_regs_fn)(struct device_node *, in bcm7120_l2_intc_probe()
223 unsigned int clr = IRQ_NOREQUEST | IRQ_NOPROBE | IRQ_NOAUTOEN; in bcm7120_l2_intc_probe()
227 int ret = 0; in bcm7120_l2_intc_probe()
228 unsigned int idx, irq, flags; in bcm7120_l2_intc_probe()
233 return -ENOMEM; in bcm7120_l2_intc_probe()
235 data->num_parent_irqs = of_irq_count(dn); in bcm7120_l2_intc_probe()
236 if (data->num_parent_irqs <= 0) { in bcm7120_l2_intc_probe()
238 ret = -ENOMEM; in bcm7120_l2_intc_probe()
242 data->l1_data = kcalloc(data->num_parent_irqs, sizeof(*data->l1_data), in bcm7120_l2_intc_probe()
244 if (!data->l1_data) { in bcm7120_l2_intc_probe()
245 ret = -ENOMEM; in bcm7120_l2_intc_probe()
253 data->can_wake = of_property_read_bool(dn, "brcm,irq-can-wake"); in bcm7120_l2_intc_probe()
255 for (irq = 0; irq < data->num_parent_irqs; irq++) { in bcm7120_l2_intc_probe()
261 data->domain = irq_domain_add_linear(dn, IRQS_PER_WORD * data->n_words, in bcm7120_l2_intc_probe()
263 if (!data->domain) { in bcm7120_l2_intc_probe()
264 ret = -ENOMEM; in bcm7120_l2_intc_probe()
269 * peripheral registers for CPU-native byte order. in bcm7120_l2_intc_probe()
275 ret = irq_alloc_domain_generic_chips(data->domain, IRQS_PER_WORD, 1, in bcm7120_l2_intc_probe()
276 dn->full_name, handle_level_irq, clr, in bcm7120_l2_intc_probe()
283 for (idx = 0; idx < data->n_words; idx++) { in bcm7120_l2_intc_probe()
285 gc = irq_get_domain_generic_chip(data->domain, irq); in bcm7120_l2_intc_probe()
287 gc->unused = 0xffffffff & ~valid_mask[idx]; in bcm7120_l2_intc_probe()
288 gc->private = data; in bcm7120_l2_intc_probe()
289 ct = gc->chip_types; in bcm7120_l2_intc_probe()
291 gc->reg_base = data->pair_base[idx]; in bcm7120_l2_intc_probe()
292 ct->regs.mask = data->en_offset[idx]; in bcm7120_l2_intc_probe()
294 /* gc->reg_base is defined and so is gc->writel */ in bcm7120_l2_intc_probe()
295 irq_reg_writel(gc, data->irq_fwd_mask[idx], in bcm7120_l2_intc_probe()
296 data->en_offset[idx]); in bcm7120_l2_intc_probe()
298 ct->chip.irq_mask = irq_gc_mask_clr_bit; in bcm7120_l2_intc_probe()
299 ct->chip.irq_unmask = irq_gc_mask_set_bit; in bcm7120_l2_intc_probe()
300 ct->chip.irq_ack = irq_gc_noop; in bcm7120_l2_intc_probe()
301 gc->suspend = bcm7120_l2_intc_suspend; in bcm7120_l2_intc_probe()
302 gc->resume = bcm7120_l2_intc_resume; in bcm7120_l2_intc_probe()
305 * Initialize mask-cache, in case we need it for in bcm7120_l2_intc_probe()
306 * saving/restoring fwd mask even w/o any child interrupts in bcm7120_l2_intc_probe()
309 gc->mask_cache = irq_reg_readl(gc, ct->regs.mask); in bcm7120_l2_intc_probe()
311 if (data->can_wake) { in bcm7120_l2_intc_probe()
313 * relevant child interrupts in wake_enabled mask in bcm7120_l2_intc_probe()
315 gc->wake_enabled = 0xffffffff; in bcm7120_l2_intc_probe()
316 gc->wake_enabled &= ~gc->unused; in bcm7120_l2_intc_probe()
317 ct->chip.irq_set_wake = irq_gc_set_wake; in bcm7120_l2_intc_probe()
322 intc_name, dn, data->num_parent_irqs); in bcm7120_l2_intc_probe()
327 irq_domain_remove(data->domain); in bcm7120_l2_intc_probe()
329 kfree(data->l1_data); in bcm7120_l2_intc_probe()
332 if (data->map_base[idx]) in bcm7120_l2_intc_probe()
333 iounmap(data->map_base[idx]); in bcm7120_l2_intc_probe()
339 static int __init bcm7120_l2_intc_probe_7120(struct device_node *dn, in bcm7120_l2_intc_probe_7120()
346 static int __init bcm7120_l2_intc_probe_3380(struct device_node *dn, in bcm7120_l2_intc_probe_3380()
353 IRQCHIP_DECLARE(bcm7120_l2_intc, "brcm,bcm7120-l2-intc",
356 IRQCHIP_DECLARE(bcm3380_l2_intc, "brcm,bcm3380-l2-intc",