The config of LPIs need to be same in each prop_page. So if one prop_page is modified, other prop_page should be updated too.
Signed-off-by: Yang Yingliang <yangyingli...@huawei.com> --- drivers/irqchip/irq-gic-v3-its.c | 39 ++++++++++++++++++++++++++++++--------- 1 file changed, 30 insertions(+), 9 deletions(-) diff --git a/drivers/irqchip/irq-gic-v3-its.c b/drivers/irqchip/irq-gic-v3-its.c index 02a5d95..27306a5 100644 --- a/drivers/irqchip/irq-gic-v3-its.c +++ b/drivers/irqchip/irq-gic-v3-its.c @@ -1051,11 +1051,25 @@ static inline u32 its_get_event_id(struct irq_data *d) return d->hwirq - its_dev->event_map.lpi_base; } +static inline void lpi_flush_config(u8 *cfg) +{ + /* + * Make the above write visible to the redistributors. + * And yes, we're flushing exactly: One. Single. Byte. + * Humpf... + */ + if (gic_rdists->flags & RDIST_FLAGS_PROPBASE_NEEDS_FLUSHING) + gic_flush_dcache_to_poc(cfg, sizeof(*cfg)); + else + dsb(ishst); +} + static void lpi_write_config(struct irq_data *d, u8 clr, u8 set) { irq_hw_number_t hwirq; struct page *prop_page; u8 *cfg; + int cpu; if (irqd_is_forwarded_to_vcpu(d)) { struct its_device *its_dev = irq_data_get_irq_chip_data(d); @@ -1078,15 +1092,22 @@ static void lpi_write_config(struct irq_data *d, u8 clr, u8 set) *cfg &= ~clr; *cfg |= set | LPI_PROP_GROUP1; - /* - * Make the above write visible to the redistributors. - * And yes, we're flushing exactly: One. Single. Byte. - * Humpf... - */ - if (gic_rdists->flags & RDIST_FLAGS_PROPBASE_NEEDS_FLUSHING) - gic_flush_dcache_to_poc(cfg, sizeof(*cfg)); - else - dsb(ishst); + if (gic_rdists->common_aff_lpi) { + /* It's protected by desc->lock, don't need lock any more. */ + for_each_cpu(cpu, cpu_possible_mask) { + u8 *other_cfg; + struct page *other_prop_page = per_cpu_ptr(gic_rdists->rdist, cpu)->prop_page; + + if (other_prop_page == prop_page) + continue; + + other_cfg = page_address(other_prop_page) + hwirq - 8192; + *other_cfg = *cfg; + lpi_flush_config(other_cfg); + } + } else { + lpi_flush_config(cfg); + } } static void lpi_update_config(struct irq_data *d, u8 clr, u8 set) -- 1.8.3