[PATCH v2 4/8] irqchip/apple-aic: Wire PMU interrupts
Hector Martin
marcan at marcan.st
Sat Dec 11 23:25:49 PST 2021
On 01/12/2021 22.49, Marc Zyngier wrote:
> Add the necessary code to configure and P and E-core PMU interrupts
> with their respective affinities. When such an interrupt fires, map
> it onto the right pseudo-interrupt.
>
> Signed-off-by: Marc Zyngier <maz at kernel.org>
> ---
> drivers/irqchip/irq-apple-aic.c | 34 +++++++++++++++++++++------------
> 1 file changed, 22 insertions(+), 12 deletions(-)
>
> diff --git a/drivers/irqchip/irq-apple-aic.c b/drivers/irqchip/irq-apple-aic.c
> index 30ca80ccda8b..23f5f10e974e 100644
> --- a/drivers/irqchip/irq-apple-aic.c
> +++ b/drivers/irqchip/irq-apple-aic.c
> @@ -155,7 +155,7 @@
> #define SYS_IMP_APL_UPMSR_EL1 sys_reg(3, 7, 15, 6, 4)
> #define UPMSR_IACT BIT(0)
>
> -#define AIC_NR_FIQ 4
> +#define AIC_NR_FIQ 6
> #define AIC_NR_SWIPI 32
>
> /*
> @@ -207,6 +207,11 @@ static bool __is_pcore(u64 mpidr)
> return MPIDR_AFFINITY_LEVEL(mpidr, 2) == 1;
> }
>
> +static bool is_pcore(void)
> +{
> + return __is_pcore(read_cpuid_mpidr());
> +}
> +
> /*
> * IRQ irqchip
> */
> @@ -420,16 +425,10 @@ static void __exception_irq_entry aic_handle_fiq(struct pt_regs *regs)
> aic_irqc->nr_hw + AIC_TMR_EL02_VIRT);
> }
>
> - if ((read_sysreg_s(SYS_IMP_APL_PMCR0_EL1) & (PMCR0_IMODE | PMCR0_IACT)) ==
> - (FIELD_PREP(PMCR0_IMODE, PMCR0_IMODE_FIQ) | PMCR0_IACT)) {
> - /*
> - * Not supported yet, let's figure out how to handle this when
> - * we implement these proprietary performance counters. For now,
> - * just mask it and move on.
> - */
> - pr_err_ratelimited("PMC FIQ fired. Masking.\n");
> - sysreg_clear_set_s(SYS_IMP_APL_PMCR0_EL1, PMCR0_IMODE | PMCR0_IACT,
> - FIELD_PREP(PMCR0_IMODE, PMCR0_IMODE_OFF));
> + if (read_sysreg_s(SYS_IMP_APL_PMCR0_EL1) & PMCR0_IACT) {
> + int irq = is_pcore() ? AIC_CPU_PMU_P : AIC_CPU_PMU_E;
> + generic_handle_domain_irq(aic_irqc->hw_domain,
> + aic_irqc->nr_hw + irq);
> }
>
> if (FIELD_GET(UPMCR0_IMODE, read_sysreg_s(SYS_IMP_APL_UPMCR0_EL1)) == UPMCR0_IMODE_FIQ &&
> @@ -469,7 +468,18 @@ static int aic_irq_domain_map(struct irq_domain *id, unsigned int irq,
> handle_fasteoi_irq, NULL, NULL);
> irqd_set_single_target(irq_desc_get_irq_data(irq_to_desc(irq)));
> } else {
> - irq_set_percpu_devid(irq);
> + switch (hw - ic->nr_hw) {
> + case AIC_CPU_PMU_P:
> + irq_set_percpu_devid_partition(irq, &ic->pcore_mask);
> + break;
> + case AIC_CPU_PMU_E:
> + irq_set_percpu_devid_partition(irq, &ic->ecore_mask);
> + break;
> + default:
> + irq_set_percpu_devid(irq);
> + break;
> + }
> +
> irq_domain_set_info(id, irq, hw, &fiq_chip, id->host_data,
> handle_percpu_devid_irq, NULL, NULL);
> }
>
I still find this idea that we need to split the IRQs virtually quite
bizarre, but if that's how bit.LITTLE systems do it...
Reviewed-by: Hector Martin <marcan at marcan.st>
--
Hector Martin (marcan at marcan.st)
Public Key: https://mrcn.st/pub
More information about the linux-arm-kernel
mailing list