Path: blob/master/tools/testing/selftests/kvm/lib/arm64/gic_v3.c
51074 views
// SPDX-License-Identifier: GPL-2.01/*2* ARM Generic Interrupt Controller (GIC) v3 support3*/45#include <linux/sizes.h>67#include "kvm_util.h"8#include "processor.h"9#include "delay.h"1011#include "gic.h"12#include "gic_v3.h"13#include "gic_private.h"1415#define GICV3_MAX_CPUS 5121617#define GICD_INT_DEF_PRI 0xa018#define GICD_INT_DEF_PRI_X4 ((GICD_INT_DEF_PRI << 24) |\19(GICD_INT_DEF_PRI << 16) |\20(GICD_INT_DEF_PRI << 8) |\21GICD_INT_DEF_PRI)2223#define ICC_PMR_DEF_PRIO 0xf02425struct gicv3_data {26unsigned int nr_cpus;27unsigned int nr_spis;28};2930#define sgi_base_from_redist(redist_base) (redist_base + SZ_64K)31#define DIST_BIT (1U << 31)3233enum gicv3_intid_range {34SGI_RANGE,35PPI_RANGE,36SPI_RANGE,37INVALID_RANGE,38};3940static struct gicv3_data gicv3_data;4142static void gicv3_gicd_wait_for_rwp(void)43{44unsigned int count = 100000; /* 1s */4546while (readl(GICD_BASE_GVA + GICD_CTLR) & GICD_CTLR_RWP) {47GUEST_ASSERT(count--);48udelay(10);49}50}5152static inline volatile void *gicr_base_cpu(uint32_t cpu)53{54/* Align all the redistributors sequentially */55return GICR_BASE_GVA + cpu * SZ_64K * 2;56}5758static void gicv3_gicr_wait_for_rwp(uint32_t cpu)59{60unsigned int count = 100000; /* 1s */6162while (readl(gicr_base_cpu(cpu) + GICR_CTLR) & GICR_CTLR_RWP) {63GUEST_ASSERT(count--);64udelay(10);65}66}6768static void gicv3_wait_for_rwp(uint32_t cpu_or_dist)69{70if (cpu_or_dist & DIST_BIT)71gicv3_gicd_wait_for_rwp();72else73gicv3_gicr_wait_for_rwp(cpu_or_dist);74}7576static enum gicv3_intid_range get_intid_range(unsigned int intid)77{78switch (intid) {79case 0 ... 15:80return SGI_RANGE;81case 16 ... 31:82return PPI_RANGE;83case 32 ... 1019:84return SPI_RANGE;85}8687/* We should not be reaching here */88GUEST_ASSERT(0);8990return INVALID_RANGE;91}9293static uint64_t gicv3_read_iar(void)94{95uint64_t irqstat = read_sysreg_s(SYS_ICC_IAR1_EL1);9697dsb(sy);98return irqstat;99}100101static void gicv3_write_eoir(uint32_t irq)102{103write_sysreg_s(irq, SYS_ICC_EOIR1_EL1);104isb();105}106107static void gicv3_write_dir(uint32_t irq)108{109write_sysreg_s(irq, SYS_ICC_DIR_EL1);110isb();111}112113static void gicv3_set_priority_mask(uint64_t mask)114{115write_sysreg_s(mask, SYS_ICC_PMR_EL1);116}117118static void gicv3_set_eoi_split(bool split)119{120uint32_t val;121122/*123* All other fields are read-only, so no need to read CTLR first. In124* fact, the kernel does the same.125*/126val = split ? (1U << 1) : 0;127write_sysreg_s(val, SYS_ICC_CTLR_EL1);128isb();129}130131uint32_t gicv3_reg_readl(uint32_t cpu_or_dist, uint64_t offset)132{133volatile void *base = cpu_or_dist & DIST_BIT ? GICD_BASE_GVA134: sgi_base_from_redist(gicr_base_cpu(cpu_or_dist));135return readl(base + offset);136}137138void gicv3_reg_writel(uint32_t cpu_or_dist, uint64_t offset, uint32_t reg_val)139{140volatile void *base = cpu_or_dist & DIST_BIT ? GICD_BASE_GVA141: sgi_base_from_redist(gicr_base_cpu(cpu_or_dist));142writel(reg_val, base + offset);143}144145uint32_t gicv3_getl_fields(uint32_t cpu_or_dist, uint64_t offset, uint32_t mask)146{147return gicv3_reg_readl(cpu_or_dist, offset) & mask;148}149150void gicv3_setl_fields(uint32_t cpu_or_dist, uint64_t offset,151uint32_t mask, uint32_t reg_val)152{153uint32_t tmp = gicv3_reg_readl(cpu_or_dist, offset) & ~mask;154155tmp |= (reg_val & mask);156gicv3_reg_writel(cpu_or_dist, offset, tmp);157}158159/*160* We use a single offset for the distributor and redistributor maps as they161* have the same value in both. The only exceptions are registers that only162* exist in one and not the other, like GICR_WAKER that doesn't exist in the163* distributor map. Such registers are conveniently marked as reserved in the164* map that doesn't implement it; like GICR_WAKER's offset of 0x0014 being165* marked as "Reserved" in the Distributor map.166*/167static void gicv3_access_reg(uint32_t intid, uint64_t offset,168uint32_t reg_bits, uint32_t bits_per_field,169bool write, uint32_t *val)170{171uint32_t cpu = guest_get_vcpuid();172enum gicv3_intid_range intid_range = get_intid_range(intid);173uint32_t fields_per_reg, index, mask, shift;174uint32_t cpu_or_dist;175176GUEST_ASSERT(bits_per_field <= reg_bits);177GUEST_ASSERT(!write || *val < (1U << bits_per_field));178/*179* This function does not support 64 bit accesses. Just asserting here180* until we implement readq/writeq.181*/182GUEST_ASSERT(reg_bits == 32);183184fields_per_reg = reg_bits / bits_per_field;185index = intid % fields_per_reg;186shift = index * bits_per_field;187mask = ((1U << bits_per_field) - 1) << shift;188189/* Set offset to the actual register holding intid's config. */190offset += (intid / fields_per_reg) * (reg_bits / 8);191192cpu_or_dist = (intid_range == SPI_RANGE) ? DIST_BIT : cpu;193194if (write)195gicv3_setl_fields(cpu_or_dist, offset, mask, *val << shift);196*val = gicv3_getl_fields(cpu_or_dist, offset, mask) >> shift;197}198199static void gicv3_write_reg(uint32_t intid, uint64_t offset,200uint32_t reg_bits, uint32_t bits_per_field, uint32_t val)201{202gicv3_access_reg(intid, offset, reg_bits,203bits_per_field, true, &val);204}205206static uint32_t gicv3_read_reg(uint32_t intid, uint64_t offset,207uint32_t reg_bits, uint32_t bits_per_field)208{209uint32_t val;210211gicv3_access_reg(intid, offset, reg_bits,212bits_per_field, false, &val);213return val;214}215216static void gicv3_set_priority(uint32_t intid, uint32_t prio)217{218gicv3_write_reg(intid, GICD_IPRIORITYR, 32, 8, prio);219}220221/* Sets the intid to be level-sensitive or edge-triggered. */222static void gicv3_irq_set_config(uint32_t intid, bool is_edge)223{224uint32_t val;225226/* N/A for private interrupts. */227GUEST_ASSERT(get_intid_range(intid) == SPI_RANGE);228val = is_edge ? 2 : 0;229gicv3_write_reg(intid, GICD_ICFGR, 32, 2, val);230}231232static void gicv3_irq_enable(uint32_t intid)233{234bool is_spi = get_intid_range(intid) == SPI_RANGE;235uint32_t cpu = guest_get_vcpuid();236237gicv3_write_reg(intid, GICD_ISENABLER, 32, 1, 1);238gicv3_wait_for_rwp(is_spi ? DIST_BIT : cpu);239}240241static void gicv3_irq_disable(uint32_t intid)242{243bool is_spi = get_intid_range(intid) == SPI_RANGE;244uint32_t cpu = guest_get_vcpuid();245246gicv3_write_reg(intid, GICD_ICENABLER, 32, 1, 1);247gicv3_wait_for_rwp(is_spi ? DIST_BIT : cpu);248}249250static void gicv3_irq_set_active(uint32_t intid)251{252gicv3_write_reg(intid, GICD_ISACTIVER, 32, 1, 1);253}254255static void gicv3_irq_clear_active(uint32_t intid)256{257gicv3_write_reg(intid, GICD_ICACTIVER, 32, 1, 1);258}259260static bool gicv3_irq_get_active(uint32_t intid)261{262return gicv3_read_reg(intid, GICD_ISACTIVER, 32, 1);263}264265static void gicv3_irq_set_pending(uint32_t intid)266{267gicv3_write_reg(intid, GICD_ISPENDR, 32, 1, 1);268}269270static void gicv3_irq_clear_pending(uint32_t intid)271{272gicv3_write_reg(intid, GICD_ICPENDR, 32, 1, 1);273}274275static bool gicv3_irq_get_pending(uint32_t intid)276{277return gicv3_read_reg(intid, GICD_ISPENDR, 32, 1);278}279280static void gicv3_enable_redist(volatile void *redist_base)281{282uint32_t val = readl(redist_base + GICR_WAKER);283unsigned int count = 100000; /* 1s */284285val &= ~GICR_WAKER_ProcessorSleep;286writel(val, redist_base + GICR_WAKER);287288/* Wait until the processor is 'active' */289while (readl(redist_base + GICR_WAKER) & GICR_WAKER_ChildrenAsleep) {290GUEST_ASSERT(count--);291udelay(10);292}293}294295static void gicv3_set_group(uint32_t intid, bool grp)296{297uint32_t cpu_or_dist;298uint32_t val;299300cpu_or_dist = (get_intid_range(intid) == SPI_RANGE) ? DIST_BIT : guest_get_vcpuid();301val = gicv3_reg_readl(cpu_or_dist, GICD_IGROUPR + (intid / 32) * 4);302if (grp)303val |= BIT(intid % 32);304else305val &= ~BIT(intid % 32);306gicv3_reg_writel(cpu_or_dist, GICD_IGROUPR + (intid / 32) * 4, val);307}308309static void gicv3_cpu_init(unsigned int cpu)310{311volatile void *sgi_base;312unsigned int i;313volatile void *redist_base_cpu;314u64 typer;315316GUEST_ASSERT(cpu < gicv3_data.nr_cpus);317318redist_base_cpu = gicr_base_cpu(cpu);319sgi_base = sgi_base_from_redist(redist_base_cpu);320321/* Verify assumption that GICR_TYPER.Processor_number == cpu */322typer = readq_relaxed(redist_base_cpu + GICR_TYPER);323GUEST_ASSERT_EQ(GICR_TYPER_CPU_NUMBER(typer), cpu);324325gicv3_enable_redist(redist_base_cpu);326327/*328* Mark all the SGI and PPI interrupts as non-secure Group-1.329* Also, deactivate and disable them.330*/331writel(~0, sgi_base + GICR_IGROUPR0);332writel(~0, sgi_base + GICR_ICACTIVER0);333writel(~0, sgi_base + GICR_ICENABLER0);334335/* Set a default priority for all the SGIs and PPIs */336for (i = 0; i < 32; i += 4)337writel(GICD_INT_DEF_PRI_X4,338sgi_base + GICR_IPRIORITYR0 + i);339340gicv3_gicr_wait_for_rwp(cpu);341342/* Enable the GIC system register (ICC_*) access */343write_sysreg_s(read_sysreg_s(SYS_ICC_SRE_EL1) | ICC_SRE_EL1_SRE,344SYS_ICC_SRE_EL1);345346/* Set a default priority threshold */347write_sysreg_s(ICC_PMR_DEF_PRIO, SYS_ICC_PMR_EL1);348349/* Disable Group-0 interrupts */350write_sysreg_s(ICC_IGRPEN0_EL1_MASK, SYS_ICC_IGRPEN1_EL1);351/* Enable non-secure Group-1 interrupts */352write_sysreg_s(ICC_IGRPEN1_EL1_MASK, SYS_ICC_IGRPEN1_EL1);353}354355static void gicv3_dist_init(void)356{357unsigned int i;358359/* Disable the distributor until we set things up */360writel(0, GICD_BASE_GVA + GICD_CTLR);361gicv3_gicd_wait_for_rwp();362363/*364* Mark all the SPI interrupts as non-secure Group-1.365* Also, deactivate and disable them.366*/367for (i = 32; i < gicv3_data.nr_spis; i += 32) {368writel(~0, GICD_BASE_GVA + GICD_IGROUPR + i / 8);369writel(~0, GICD_BASE_GVA + GICD_ICACTIVER + i / 8);370writel(~0, GICD_BASE_GVA + GICD_ICENABLER + i / 8);371}372373/* Set a default priority for all the SPIs */374for (i = 32; i < gicv3_data.nr_spis; i += 4)375writel(GICD_INT_DEF_PRI_X4,376GICD_BASE_GVA + GICD_IPRIORITYR + i);377378/* Wait for the settings to sync-in */379gicv3_gicd_wait_for_rwp();380381/* Finally, enable the distributor globally with ARE */382writel(GICD_CTLR_ARE_NS | GICD_CTLR_ENABLE_G1A |383GICD_CTLR_ENABLE_G1, GICD_BASE_GVA + GICD_CTLR);384gicv3_gicd_wait_for_rwp();385}386387static void gicv3_init(unsigned int nr_cpus)388{389GUEST_ASSERT(nr_cpus <= GICV3_MAX_CPUS);390391gicv3_data.nr_cpus = nr_cpus;392gicv3_data.nr_spis = GICD_TYPER_SPIS(393readl(GICD_BASE_GVA + GICD_TYPER));394if (gicv3_data.nr_spis > 1020)395gicv3_data.nr_spis = 1020;396397/*398* Initialize only the distributor for now.399* The redistributor and CPU interfaces are initialized400* later for every PE.401*/402gicv3_dist_init();403}404405const struct gic_common_ops gicv3_ops = {406.gic_init = gicv3_init,407.gic_cpu_init = gicv3_cpu_init,408.gic_irq_enable = gicv3_irq_enable,409.gic_irq_disable = gicv3_irq_disable,410.gic_read_iar = gicv3_read_iar,411.gic_write_eoir = gicv3_write_eoir,412.gic_write_dir = gicv3_write_dir,413.gic_set_priority_mask = gicv3_set_priority_mask,414.gic_set_eoi_split = gicv3_set_eoi_split,415.gic_set_priority = gicv3_set_priority,416.gic_irq_set_active = gicv3_irq_set_active,417.gic_irq_clear_active = gicv3_irq_clear_active,418.gic_irq_get_active = gicv3_irq_get_active,419.gic_irq_set_pending = gicv3_irq_set_pending,420.gic_irq_clear_pending = gicv3_irq_clear_pending,421.gic_irq_get_pending = gicv3_irq_get_pending,422.gic_irq_set_config = gicv3_irq_set_config,423.gic_irq_set_group = gicv3_set_group,424};425426void gic_rdist_enable_lpis(vm_paddr_t cfg_table, size_t cfg_table_size,427vm_paddr_t pend_table)428{429volatile void *rdist_base = gicr_base_cpu(guest_get_vcpuid());430431u32 ctlr;432u64 val;433434val = (cfg_table |435GICR_PROPBASER_InnerShareable |436GICR_PROPBASER_RaWaWb |437((ilog2(cfg_table_size) - 1) & GICR_PROPBASER_IDBITS_MASK));438writeq_relaxed(val, rdist_base + GICR_PROPBASER);439440val = (pend_table |441GICR_PENDBASER_InnerShareable |442GICR_PENDBASER_RaWaWb);443writeq_relaxed(val, rdist_base + GICR_PENDBASER);444445ctlr = readl_relaxed(rdist_base + GICR_CTLR);446ctlr |= GICR_CTLR_ENABLE_LPIS;447writel_relaxed(ctlr, rdist_base + GICR_CTLR);448}449450451