Commit 96de80c1 authored by Florian Fainelli's avatar Florian Fainelli Committed by Marc Zyngier

irqchip/irq-bcm7038-l1: Support brcm,int-fwd-mask

On some specific chips like 7211 we need to leave some interrupts
untouched/forwarded to the VPU which is another agent in the system
making use of that interrupt controller hardware (goes to both ARM GIC
and VPU L1 interrupt controller). Make that possible by using the
existing brcm,int-fwd-mask property and take necessary actions to avoid
masking that interrupt as well as not allowing Linux to map them.
Signed-off-by: default avatarFlorian Fainelli <f.fainelli@gmail.com>
Signed-off-by: default avatarMarc Zyngier <maz@kernel.org>
Link: https://lore.kernel.org/r/20191024201415.23454-6-f.fainelli@gmail.com
parent e14b5e5f
...@@ -44,6 +44,7 @@ struct bcm7038_l1_chip { ...@@ -44,6 +44,7 @@ struct bcm7038_l1_chip {
struct list_head list; struct list_head list;
u32 wake_mask[MAX_WORDS]; u32 wake_mask[MAX_WORDS];
#endif #endif
u32 irq_fwd_mask[MAX_WORDS];
u8 affinity[MAX_WORDS * IRQS_PER_WORD]; u8 affinity[MAX_WORDS * IRQS_PER_WORD];
}; };
...@@ -254,6 +255,7 @@ static int __init bcm7038_l1_init_one(struct device_node *dn, ...@@ -254,6 +255,7 @@ static int __init bcm7038_l1_init_one(struct device_node *dn,
resource_size_t sz; resource_size_t sz;
struct bcm7038_l1_cpu *cpu; struct bcm7038_l1_cpu *cpu;
unsigned int i, n_words, parent_irq; unsigned int i, n_words, parent_irq;
int ret;
if (of_address_to_resource(dn, idx, &res)) if (of_address_to_resource(dn, idx, &res))
return -EINVAL; return -EINVAL;
...@@ -267,6 +269,14 @@ static int __init bcm7038_l1_init_one(struct device_node *dn, ...@@ -267,6 +269,14 @@ static int __init bcm7038_l1_init_one(struct device_node *dn,
else if (intc->n_words != n_words) else if (intc->n_words != n_words)
return -EINVAL; return -EINVAL;
ret = of_property_read_u32_array(dn , "brcm,int-fwd-mask",
intc->irq_fwd_mask, n_words);
if (ret != 0 && ret != -EINVAL) {
/* property exists but has the wrong number of words */
pr_err("invalid brcm,int-fwd-mask property\n");
return -EINVAL;
}
cpu = intc->cpus[idx] = kzalloc(sizeof(*cpu) + n_words * sizeof(u32), cpu = intc->cpus[idx] = kzalloc(sizeof(*cpu) + n_words * sizeof(u32),
GFP_KERNEL); GFP_KERNEL);
if (!cpu) if (!cpu)
...@@ -277,8 +287,11 @@ static int __init bcm7038_l1_init_one(struct device_node *dn, ...@@ -277,8 +287,11 @@ static int __init bcm7038_l1_init_one(struct device_node *dn,
return -ENOMEM; return -ENOMEM;
for (i = 0; i < n_words; i++) { for (i = 0; i < n_words; i++) {
l1_writel(0xffffffff, cpu->map_base + reg_mask_set(intc, i)); l1_writel(~intc->irq_fwd_mask[i],
cpu->mask_cache[i] = 0xffffffff; cpu->map_base + reg_mask_set(intc, i));
l1_writel(intc->irq_fwd_mask[i],
cpu->map_base + reg_mask_clr(intc, i));
cpu->mask_cache[i] = ~intc->irq_fwd_mask[i];
} }
parent_irq = irq_of_parse_and_map(dn, idx); parent_irq = irq_of_parse_and_map(dn, idx);
...@@ -311,15 +324,17 @@ static int bcm7038_l1_suspend(void) ...@@ -311,15 +324,17 @@ static int bcm7038_l1_suspend(void)
{ {
struct bcm7038_l1_chip *intc; struct bcm7038_l1_chip *intc;
int boot_cpu, word; int boot_cpu, word;
u32 val;
/* Wakeup interrupt should only come from the boot cpu */ /* Wakeup interrupt should only come from the boot cpu */
boot_cpu = cpu_logical_map(0); boot_cpu = cpu_logical_map(0);
list_for_each_entry(intc, &bcm7038_l1_intcs_list, list) { list_for_each_entry(intc, &bcm7038_l1_intcs_list, list) {
for (word = 0; word < intc->n_words; word++) { for (word = 0; word < intc->n_words; word++) {
l1_writel(~intc->wake_mask[word], val = intc->wake_mask[word] | intc->irq_fwd_mask[word];
l1_writel(~val,
intc->cpus[boot_cpu]->map_base + reg_mask_set(intc, word)); intc->cpus[boot_cpu]->map_base + reg_mask_set(intc, word));
l1_writel(intc->wake_mask[word], l1_writel(val,
intc->cpus[boot_cpu]->map_base + reg_mask_clr(intc, word)); intc->cpus[boot_cpu]->map_base + reg_mask_clr(intc, word));
} }
} }
...@@ -383,6 +398,13 @@ static struct irq_chip bcm7038_l1_irq_chip = { ...@@ -383,6 +398,13 @@ static struct irq_chip bcm7038_l1_irq_chip = {
static int bcm7038_l1_map(struct irq_domain *d, unsigned int virq, static int bcm7038_l1_map(struct irq_domain *d, unsigned int virq,
irq_hw_number_t hw_irq) irq_hw_number_t hw_irq)
{ {
struct bcm7038_l1_chip *intc = d->host_data;
u32 mask = BIT(hw_irq % IRQS_PER_WORD);
u32 word = hw_irq / IRQS_PER_WORD;
if (intc->irq_fwd_mask[word] & mask)
return -EPERM;
irq_set_chip_and_handler(virq, &bcm7038_l1_irq_chip, handle_level_irq); irq_set_chip_and_handler(virq, &bcm7038_l1_irq_chip, handle_level_irq);
irq_set_chip_data(virq, d->host_data); irq_set_chip_data(virq, d->host_data);
irqd_set_single_target(irq_desc_get_irq_data(irq_to_desc(virq))); irqd_set_single_target(irq_desc_get_irq_data(irq_to_desc(virq)));
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment