aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorKyle Evans <kevans@FreeBSD.org>2023-05-15 17:21:45 +0000
committerKyle Evans <kevans@FreeBSD.org>2023-05-22 15:23:53 +0000
commit172af24449cd8d34339172d125832b7ecd274213 (patch)
treeb2c462645fb501f63343475e1c9b20a81848382f
parent26056fa8d348b8ec03b4b2d0c9d086c4232f2711 (diff)
downloadsrc-172af24449cd8d34339172d125832b7ecd274213.tar.gz
src-172af24449cd8d34339172d125832b7ecd274213.zip
arm64: gicv3: setup PPIs on all APs after they're online
For all PPIs setup earlier than SI_SUB_SMP, PIC_INIT_SECONDARY ends up cleaning these up for each AP as it comes online. Once they're online, we don't currently do anything to make sure they're configured for other APs. Fix it by using smp_rendezvous for the meaty bits of configuring a PPI, which will just do single-thread behavior before APs are online but do the right thing for other CPUs after. While we're here, make sure redistributor config is correct for other APs as they come online in gic_v3_init_secondary. Reported/Tested by: Souradeep Chakrabarti (Microsoft/Hyper-V) Reviewed by: andrew (before slight refactor) Differential Revision: https://reviews.freebsd.org/D40112
-rw-r--r--sys/arm64/arm64/gic_v3.c157
1 files changed, 109 insertions, 48 deletions
diff --git a/sys/arm64/arm64/gic_v3.c b/sys/arm64/arm64/gic_v3.c
index dfa0f6c7cad1..e9754797d095 100644
--- a/sys/arm64/arm64/gic_v3.c
+++ b/sys/arm64/arm64/gic_v3.c
@@ -835,15 +835,66 @@ gic_v3_map_intr(device_t dev, struct intr_map_data *data,
return (error);
}
+struct gic_v3_setup_periph_args {
+ device_t dev;
+ struct intr_irqsrc *isrc;
+};
+
+static void
+gic_v3_setup_intr_periph(void *argp)
+{
+ struct gic_v3_setup_periph_args *args = argp;
+ struct intr_irqsrc *isrc = args->isrc;
+ struct gic_v3_irqsrc *gi = (struct gic_v3_irqsrc *)isrc;
+ device_t dev = args->dev;
+ u_int irq = gi->gi_irq;
+ struct gic_v3_softc *sc = device_get_softc(dev);
+ uint32_t reg;
+
+ MPASS(irq <= GIC_LAST_SPI);
+
+ /*
+ * We need the lock for both SGIs and PPIs for an atomic CPU_SET() at a
+ * minimum, but we also need it below for SPIs.
+ */
+ mtx_lock_spin(&sc->gic_mtx);
+
+ if (isrc->isrc_flags & INTR_ISRCF_PPI)
+ CPU_SET(PCPU_GET(cpuid), &isrc->isrc_cpu);
+
+ if (irq >= GIC_FIRST_PPI && irq <= GIC_LAST_SPI) {
+ /* Set the trigger and polarity */
+ if (irq <= GIC_LAST_PPI)
+ reg = gic_r_read(sc, 4,
+ GICR_SGI_BASE_SIZE + GICD_ICFGR(irq));
+ else
+ reg = gic_d_read(sc, 4, GICD_ICFGR(irq));
+ if (gi->gi_trig == INTR_TRIGGER_LEVEL)
+ reg &= ~(2 << ((irq % 16) * 2));
+ else
+ reg |= 2 << ((irq % 16) * 2);
+
+ if (irq <= GIC_LAST_PPI) {
+ gic_r_write(sc, 4,
+ GICR_SGI_BASE_SIZE + GICD_ICFGR(irq), reg);
+ gic_v3_wait_for_rwp(sc, REDIST);
+ } else {
+ gic_d_write(sc, 4, GICD_ICFGR(irq), reg);
+ gic_v3_wait_for_rwp(sc, DIST);
+ }
+ }
+
+ mtx_unlock_spin(&sc->gic_mtx);
+}
+
static int
gic_v3_setup_intr(device_t dev, struct intr_irqsrc *isrc,
struct resource *res, struct intr_map_data *data)
{
- struct gic_v3_softc *sc = device_get_softc(dev);
struct gic_v3_irqsrc *gi = (struct gic_v3_irqsrc *)isrc;
+ struct gic_v3_setup_periph_args pargs;
enum intr_trigger trig;
enum intr_polarity pol;
- uint32_t reg;
u_int irq;
int error;
@@ -872,41 +923,18 @@ gic_v3_setup_intr(device_t dev, struct intr_irqsrc *isrc,
gi->gi_trig = trig;
}
- /*
- * XXX - In case that per CPU interrupt is going to be enabled in time
- * when SMP is already started, we need some IPI call which
- * enables it on others CPUs. Further, it's more complicated as
- * pic_enable_source() and pic_disable_source() should act on
- * per CPU basis only. Thus, it should be solved here somehow.
- */
- if (isrc->isrc_flags & INTR_ISRCF_PPI)
- CPU_SET(PCPU_GET(cpuid), &isrc->isrc_cpu);
-
- if (irq >= GIC_FIRST_PPI && irq <= GIC_LAST_SPI) {
- mtx_lock_spin(&sc->gic_mtx);
-
- /* Set the trigger and polarity */
- if (irq <= GIC_LAST_PPI)
- reg = gic_r_read(sc, 4,
- GICR_SGI_BASE_SIZE + GICD_ICFGR(irq));
- else
- reg = gic_d_read(sc, 4, GICD_ICFGR(irq));
- if (trig == INTR_TRIGGER_LEVEL)
- reg &= ~(2 << ((irq % 16) * 2));
- else
- reg |= 2 << ((irq % 16) * 2);
-
- if (irq <= GIC_LAST_PPI) {
- gic_r_write(sc, 4,
- GICR_SGI_BASE_SIZE + GICD_ICFGR(irq), reg);
- gic_v3_wait_for_rwp(sc, REDIST);
- } else {
- gic_d_write(sc, 4, GICD_ICFGR(irq), reg);
- gic_v3_wait_for_rwp(sc, DIST);
- }
-
- mtx_unlock_spin(&sc->gic_mtx);
+ pargs.dev = dev;
+ pargs.isrc = isrc;
+ if (isrc->isrc_flags & INTR_ISRCF_PPI) {
+ /*
+ * If APs haven't been fired up yet, smp_rendezvous() will just
+ * execute it on the single CPU and gic_v3_init_secondary() will
+ * clean up afterwards.
+ */
+ smp_rendezvous(NULL, gic_v3_setup_intr_periph, NULL, &pargs);
+ } else if (irq >= GIC_FIRST_SPI && irq <= GIC_LAST_SPI) {
+ gic_v3_setup_intr_periph(&pargs);
gic_v3_bind_intr(dev, isrc);
}
@@ -952,22 +980,49 @@ gic_v3_disable_intr(device_t dev, struct intr_irqsrc *isrc)
}
static void
+gic_v3_enable_intr_periph(void *argp)
+{
+ struct gic_v3_setup_periph_args *args = argp;
+ struct gic_v3_irqsrc *gi = (struct gic_v3_irqsrc *)args->isrc;
+ device_t dev = args->dev;
+ struct gic_v3_softc *sc = device_get_softc(dev);
+ u_int irq = gi->gi_irq;
+
+ /* SGIs and PPIs in corresponding Re-Distributor */
+ gic_r_write(sc, 4, GICR_SGI_BASE_SIZE + GICD_ISENABLER(irq),
+ GICD_I_MASK(irq));
+ gic_v3_wait_for_rwp(sc, REDIST);
+}
+
+static void
gic_v3_enable_intr(device_t dev, struct intr_irqsrc *isrc)
{
+ struct gic_v3_setup_periph_args pargs;
struct gic_v3_softc *sc;
struct gic_v3_irqsrc *gi;
u_int irq;
- sc = device_get_softc(dev);
gi = (struct gic_v3_irqsrc *)isrc;
irq = gi->gi_irq;
+ pargs.isrc = isrc;
+ pargs.dev = dev;
if (irq <= GIC_LAST_PPI) {
- /* SGIs and PPIs in corresponding Re-Distributor */
- gic_r_write(sc, 4, GICR_SGI_BASE_SIZE + GICD_ISENABLER(irq),
- GICD_I_MASK(irq));
- gic_v3_wait_for_rwp(sc, REDIST);
- } else if (irq >= GIC_FIRST_SPI && irq <= GIC_LAST_SPI) {
+ /*
+ * SGIs only need configured on the current AP. We'll setup and
+ * enable IPIs as APs come online.
+ */
+ if (irq <= GIC_LAST_SGI)
+ gic_v3_enable_intr_periph(&pargs);
+ else
+ smp_rendezvous(NULL, gic_v3_enable_intr_periph, NULL,
+ &pargs);
+ return;
+ }
+
+ sc = device_get_softc(dev);
+
+ if (irq >= GIC_FIRST_SPI && irq <= GIC_LAST_SPI) {
/* SPIs in distributor */
gic_d_write(sc, 4, GICD_ISENABLER(irq), GICD_I_MASK(irq));
gic_v3_wait_for_rwp(sc, DIST);
@@ -1010,8 +1065,6 @@ gic_v3_bind_intr(device_t dev, struct intr_irqsrc *isrc)
int cpu;
gi = (struct gic_v3_irqsrc *)isrc;
- if (gi->gi_irq <= GIC_LAST_PPI)
- return (EINVAL);
KASSERT(gi->gi_irq >= GIC_FIRST_SPI && gi->gi_irq <= GIC_LAST_SPI,
("%s: Attempting to bind an invalid IRQ", __func__));
@@ -1039,6 +1092,7 @@ gic_v3_bind_intr(device_t dev, struct intr_irqsrc *isrc)
static void
gic_v3_init_secondary(device_t dev)
{
+ struct gic_v3_setup_periph_args pargs;
device_t child;
struct gic_v3_softc *sc;
gic_v3_initseq_t *init_func;
@@ -1060,18 +1114,25 @@ gic_v3_init_secondary(device_t dev)
}
}
+ pargs.dev = dev;
+
/* Unmask attached SGI interrupts. */
for (irq = GIC_FIRST_SGI; irq <= GIC_LAST_SGI; irq++) {
isrc = GIC_INTR_ISRC(sc, irq);
- if (intr_isrc_init_on_cpu(isrc, cpu))
- gic_v3_enable_intr(dev, isrc);
+ if (intr_isrc_init_on_cpu(isrc, cpu)) {
+ pargs.isrc = isrc;
+ gic_v3_enable_intr_periph(&pargs);
+ }
}
/* Unmask attached PPI interrupts. */
for (irq = GIC_FIRST_PPI; irq <= GIC_LAST_PPI; irq++) {
isrc = GIC_INTR_ISRC(sc, irq);
- if (intr_isrc_init_on_cpu(isrc, cpu))
- gic_v3_enable_intr(dev, isrc);
+ if (intr_isrc_init_on_cpu(isrc, cpu)) {
+ pargs.isrc = isrc;
+ gic_v3_setup_intr_periph(&pargs);
+ gic_v3_enable_intr_periph(&pargs);
+ }
}
for (i = 0; i < sc->gic_nchildren; i++) {