162306a36Sopenharmony_ci/* SPDX-License-Identifier: GPL-2.0-only */ 262306a36Sopenharmony_ci/* 362306a36Sopenharmony_ci * arch/arm64/include/asm/arch_gicv3.h 462306a36Sopenharmony_ci * 562306a36Sopenharmony_ci * Copyright (C) 2015 ARM Ltd. 662306a36Sopenharmony_ci */ 762306a36Sopenharmony_ci#ifndef __ASM_ARCH_GICV3_H 862306a36Sopenharmony_ci#define __ASM_ARCH_GICV3_H 962306a36Sopenharmony_ci 1062306a36Sopenharmony_ci#include <asm/sysreg.h> 1162306a36Sopenharmony_ci 1262306a36Sopenharmony_ci#ifndef __ASSEMBLY__ 1362306a36Sopenharmony_ci 1462306a36Sopenharmony_ci#include <linux/irqchip/arm-gic-common.h> 1562306a36Sopenharmony_ci#include <linux/stringify.h> 1662306a36Sopenharmony_ci#include <asm/barrier.h> 1762306a36Sopenharmony_ci#include <asm/cacheflush.h> 1862306a36Sopenharmony_ci 1962306a36Sopenharmony_ci#define read_gicreg(r) read_sysreg_s(SYS_ ## r) 2062306a36Sopenharmony_ci#define write_gicreg(v, r) write_sysreg_s(v, SYS_ ## r) 2162306a36Sopenharmony_ci 2262306a36Sopenharmony_ci/* 2362306a36Sopenharmony_ci * Low-level accessors 2462306a36Sopenharmony_ci * 2562306a36Sopenharmony_ci * These system registers are 32 bits, but we make sure that the compiler 2662306a36Sopenharmony_ci * sets the GP register's most significant bits to 0 with an explicit cast. 2762306a36Sopenharmony_ci */ 2862306a36Sopenharmony_ci 2962306a36Sopenharmony_cistatic __always_inline void gic_write_dir(u32 irq) 3062306a36Sopenharmony_ci{ 3162306a36Sopenharmony_ci write_sysreg_s(irq, SYS_ICC_DIR_EL1); 3262306a36Sopenharmony_ci isb(); 3362306a36Sopenharmony_ci} 3462306a36Sopenharmony_ci 3562306a36Sopenharmony_cistatic inline u64 gic_read_iar_common(void) 3662306a36Sopenharmony_ci{ 3762306a36Sopenharmony_ci u64 irqstat; 3862306a36Sopenharmony_ci 3962306a36Sopenharmony_ci irqstat = read_sysreg_s(SYS_ICC_IAR1_EL1); 4062306a36Sopenharmony_ci dsb(sy); 4162306a36Sopenharmony_ci return irqstat; 4262306a36Sopenharmony_ci} 4362306a36Sopenharmony_ci 4462306a36Sopenharmony_ci/* 4562306a36Sopenharmony_ci * Cavium ThunderX erratum 23154 4662306a36Sopenharmony_ci * 4762306a36Sopenharmony_ci * The gicv3 of ThunderX requires a modified version for reading the 4862306a36Sopenharmony_ci * IAR status to ensure data synchronization (access to icc_iar1_el1 4962306a36Sopenharmony_ci * is not sync'ed before and after). 5062306a36Sopenharmony_ci * 5162306a36Sopenharmony_ci * Erratum 38545 5262306a36Sopenharmony_ci * 5362306a36Sopenharmony_ci * When a IAR register read races with a GIC interrupt RELEASE event, 5462306a36Sopenharmony_ci * GIC-CPU interface could wrongly return a valid INTID to the CPU 5562306a36Sopenharmony_ci * for an interrupt that is already released(non activated) instead of 0x3ff. 5662306a36Sopenharmony_ci * 5762306a36Sopenharmony_ci * To workaround this, return a valid interrupt ID only if there is a change 5862306a36Sopenharmony_ci * in the active priority list after the IAR read. 5962306a36Sopenharmony_ci * 6062306a36Sopenharmony_ci * Common function used for both the workarounds since, 6162306a36Sopenharmony_ci * 1. On Thunderx 88xx 1.x both erratas are applicable. 6262306a36Sopenharmony_ci * 2. Having extra nops doesn't add any side effects for Silicons where 6362306a36Sopenharmony_ci * erratum 23154 is not applicable. 6462306a36Sopenharmony_ci */ 6562306a36Sopenharmony_cistatic inline u64 gic_read_iar_cavium_thunderx(void) 6662306a36Sopenharmony_ci{ 6762306a36Sopenharmony_ci u64 irqstat, apr; 6862306a36Sopenharmony_ci 6962306a36Sopenharmony_ci apr = read_sysreg_s(SYS_ICC_AP1R0_EL1); 7062306a36Sopenharmony_ci nops(8); 7162306a36Sopenharmony_ci irqstat = read_sysreg_s(SYS_ICC_IAR1_EL1); 7262306a36Sopenharmony_ci nops(4); 7362306a36Sopenharmony_ci mb(); 7462306a36Sopenharmony_ci 7562306a36Sopenharmony_ci /* Max priority groups implemented is only 32 */ 7662306a36Sopenharmony_ci if (likely(apr != read_sysreg_s(SYS_ICC_AP1R0_EL1))) 7762306a36Sopenharmony_ci return irqstat; 7862306a36Sopenharmony_ci 7962306a36Sopenharmony_ci return 0x3ff; 8062306a36Sopenharmony_ci} 8162306a36Sopenharmony_ci 8262306a36Sopenharmony_cistatic inline void gic_write_ctlr(u32 val) 8362306a36Sopenharmony_ci{ 8462306a36Sopenharmony_ci write_sysreg_s(val, SYS_ICC_CTLR_EL1); 8562306a36Sopenharmony_ci isb(); 8662306a36Sopenharmony_ci} 8762306a36Sopenharmony_ci 8862306a36Sopenharmony_cistatic inline u32 gic_read_ctlr(void) 8962306a36Sopenharmony_ci{ 9062306a36Sopenharmony_ci return read_sysreg_s(SYS_ICC_CTLR_EL1); 9162306a36Sopenharmony_ci} 9262306a36Sopenharmony_ci 9362306a36Sopenharmony_cistatic inline void gic_write_grpen1(u32 val) 9462306a36Sopenharmony_ci{ 9562306a36Sopenharmony_ci write_sysreg_s(val, SYS_ICC_IGRPEN1_EL1); 9662306a36Sopenharmony_ci isb(); 9762306a36Sopenharmony_ci} 9862306a36Sopenharmony_ci 9962306a36Sopenharmony_cistatic inline void gic_write_sgi1r(u64 val) 10062306a36Sopenharmony_ci{ 10162306a36Sopenharmony_ci write_sysreg_s(val, SYS_ICC_SGI1R_EL1); 10262306a36Sopenharmony_ci} 10362306a36Sopenharmony_ci 10462306a36Sopenharmony_cistatic inline u32 gic_read_sre(void) 10562306a36Sopenharmony_ci{ 10662306a36Sopenharmony_ci return read_sysreg_s(SYS_ICC_SRE_EL1); 10762306a36Sopenharmony_ci} 10862306a36Sopenharmony_ci 10962306a36Sopenharmony_cistatic inline void gic_write_sre(u32 val) 11062306a36Sopenharmony_ci{ 11162306a36Sopenharmony_ci write_sysreg_s(val, SYS_ICC_SRE_EL1); 11262306a36Sopenharmony_ci isb(); 11362306a36Sopenharmony_ci} 11462306a36Sopenharmony_ci 11562306a36Sopenharmony_cistatic inline void gic_write_bpr1(u32 val) 11662306a36Sopenharmony_ci{ 11762306a36Sopenharmony_ci write_sysreg_s(val, SYS_ICC_BPR1_EL1); 11862306a36Sopenharmony_ci} 11962306a36Sopenharmony_ci 12062306a36Sopenharmony_cistatic inline u32 gic_read_pmr(void) 12162306a36Sopenharmony_ci{ 12262306a36Sopenharmony_ci return read_sysreg_s(SYS_ICC_PMR_EL1); 12362306a36Sopenharmony_ci} 12462306a36Sopenharmony_ci 12562306a36Sopenharmony_cistatic __always_inline void gic_write_pmr(u32 val) 12662306a36Sopenharmony_ci{ 12762306a36Sopenharmony_ci write_sysreg_s(val, SYS_ICC_PMR_EL1); 12862306a36Sopenharmony_ci} 12962306a36Sopenharmony_ci 13062306a36Sopenharmony_cistatic inline u32 gic_read_rpr(void) 13162306a36Sopenharmony_ci{ 13262306a36Sopenharmony_ci return read_sysreg_s(SYS_ICC_RPR_EL1); 13362306a36Sopenharmony_ci} 13462306a36Sopenharmony_ci 13562306a36Sopenharmony_ci#define gic_read_typer(c) readq_relaxed(c) 13662306a36Sopenharmony_ci#define gic_write_irouter(v, c) writeq_relaxed(v, c) 13762306a36Sopenharmony_ci#define gic_read_lpir(c) readq_relaxed(c) 13862306a36Sopenharmony_ci#define gic_write_lpir(v, c) writeq_relaxed(v, c) 13962306a36Sopenharmony_ci 14062306a36Sopenharmony_ci#define gic_flush_dcache_to_poc(a,l) \ 14162306a36Sopenharmony_ci dcache_clean_inval_poc((unsigned long)(a), (unsigned long)(a)+(l)) 14262306a36Sopenharmony_ci 14362306a36Sopenharmony_ci#define gits_read_baser(c) readq_relaxed(c) 14462306a36Sopenharmony_ci#define gits_write_baser(v, c) writeq_relaxed(v, c) 14562306a36Sopenharmony_ci 14662306a36Sopenharmony_ci#define gits_read_cbaser(c) readq_relaxed(c) 14762306a36Sopenharmony_ci#define gits_write_cbaser(v, c) writeq_relaxed(v, c) 14862306a36Sopenharmony_ci 14962306a36Sopenharmony_ci#define gits_write_cwriter(v, c) writeq_relaxed(v, c) 15062306a36Sopenharmony_ci 15162306a36Sopenharmony_ci#define gicr_read_propbaser(c) readq_relaxed(c) 15262306a36Sopenharmony_ci#define gicr_write_propbaser(v, c) writeq_relaxed(v, c) 15362306a36Sopenharmony_ci 15462306a36Sopenharmony_ci#define gicr_write_pendbaser(v, c) writeq_relaxed(v, c) 15562306a36Sopenharmony_ci#define gicr_read_pendbaser(c) readq_relaxed(c) 15662306a36Sopenharmony_ci 15762306a36Sopenharmony_ci#define gicr_write_vpropbaser(v, c) writeq_relaxed(v, c) 15862306a36Sopenharmony_ci#define gicr_read_vpropbaser(c) readq_relaxed(c) 15962306a36Sopenharmony_ci 16062306a36Sopenharmony_ci#define gicr_write_vpendbaser(v, c) writeq_relaxed(v, c) 16162306a36Sopenharmony_ci#define gicr_read_vpendbaser(c) readq_relaxed(c) 16262306a36Sopenharmony_ci 16362306a36Sopenharmony_cistatic inline bool gic_prio_masking_enabled(void) 16462306a36Sopenharmony_ci{ 16562306a36Sopenharmony_ci return system_uses_irq_prio_masking(); 16662306a36Sopenharmony_ci} 16762306a36Sopenharmony_ci 16862306a36Sopenharmony_cistatic inline void gic_pmr_mask_irqs(void) 16962306a36Sopenharmony_ci{ 17062306a36Sopenharmony_ci BUILD_BUG_ON(GICD_INT_DEF_PRI < (__GIC_PRIO_IRQOFF | 17162306a36Sopenharmony_ci GIC_PRIO_PSR_I_SET)); 17262306a36Sopenharmony_ci BUILD_BUG_ON(GICD_INT_DEF_PRI >= GIC_PRIO_IRQON); 17362306a36Sopenharmony_ci /* 17462306a36Sopenharmony_ci * Need to make sure IRQON allows IRQs when SCR_EL3.FIQ is cleared 17562306a36Sopenharmony_ci * and non-secure PMR accesses are not subject to the shifts that 17662306a36Sopenharmony_ci * are applied to IRQ priorities 17762306a36Sopenharmony_ci */ 17862306a36Sopenharmony_ci BUILD_BUG_ON((0x80 | (GICD_INT_DEF_PRI >> 1)) >= GIC_PRIO_IRQON); 17962306a36Sopenharmony_ci /* 18062306a36Sopenharmony_ci * Same situation as above, but now we make sure that we can mask 18162306a36Sopenharmony_ci * regular interrupts. 18262306a36Sopenharmony_ci */ 18362306a36Sopenharmony_ci BUILD_BUG_ON((0x80 | (GICD_INT_DEF_PRI >> 1)) < (__GIC_PRIO_IRQOFF_NS | 18462306a36Sopenharmony_ci GIC_PRIO_PSR_I_SET)); 18562306a36Sopenharmony_ci gic_write_pmr(GIC_PRIO_IRQOFF); 18662306a36Sopenharmony_ci} 18762306a36Sopenharmony_ci 18862306a36Sopenharmony_cistatic inline void gic_arch_enable_irqs(void) 18962306a36Sopenharmony_ci{ 19062306a36Sopenharmony_ci asm volatile ("msr daifclr, #3" : : : "memory"); 19162306a36Sopenharmony_ci} 19262306a36Sopenharmony_ci 19362306a36Sopenharmony_cistatic inline bool gic_has_relaxed_pmr_sync(void) 19462306a36Sopenharmony_ci{ 19562306a36Sopenharmony_ci return cpus_have_cap(ARM64_HAS_GIC_PRIO_RELAXED_SYNC); 19662306a36Sopenharmony_ci} 19762306a36Sopenharmony_ci 19862306a36Sopenharmony_ci#endif /* __ASSEMBLY__ */ 19962306a36Sopenharmony_ci#endif /* __ASM_ARCH_GICV3_H */ 200