[coreboot] New patch to review for coreboot: e0518c2 Divide setup_ioapic() in two parts.
Kyösti Mälkki (kyosti.malkki@gmail.com)
gerrit at coreboot.org
Wed Oct 19 12:57:40 CEST 2011
Kyösti Mälkki (kyosti.malkki at gmail.com) just uploaded a new patch set to gerrit, which you can find at http://review.coreboot.org/300
-gerrit
commit e0518c23887e545eb9a8894aaeed4ffbb02ae9e5
Author: Kyösti Mälkki <kyosti.malkki at gmail.com>
Date: Wed Oct 19 01:29:03 2011 +0300
Divide setup_ioapic() in two parts.
Currently some southbridge codes implement the init_ioapic() part
locally and do not implement the load_vectors() part at all.
I suspect that load_vectors() should always be called.
I can push cleaned i82801 variants for testing.
Revised init_ioapic() to log on a critical level if one tries to
change an already assigned (non-zero) APIC ID, or when the APIC ID
is not assigned.
Change-Id: Ic5e860b9b669ecd1e9ddac4bbb92d80bdb9c2fca
Signed-off-by: Kyösti Mälkki <kyosti.malkki at gmail.com>
---
src/arch/x86/lib/ioapic.c | 148 +++++++++++++++++++++++++++++++++------------
1 files changed, 110 insertions(+), 38 deletions(-)
diff --git a/src/arch/x86/lib/ioapic.c b/src/arch/x86/lib/ioapic.c
index 3f64209..94f796b 100644
--- a/src/arch/x86/lib/ioapic.c
+++ b/src/arch/x86/lib/ioapic.c
@@ -34,6 +34,11 @@ static void io_apic_write(u32 ioapic_base, u32 reg, u32 value)
write32(ioapic_base + 0x10, value);
}
+/**
+ * Clear all IOAPIC vectors.
+ *
+ * @ioapic_base IOAPIC base address
+ */
void clear_ioapic(u32 ioapic_base)
{
u32 low, high;
@@ -64,24 +69,82 @@ void clear_ioapic(u32 ioapic_base)
}
}
-void setup_ioapic(u32 ioapic_base, u8 ioapic_id)
+/**
+ * Assign IOAPIC with an ID and set delivery type.
+ *
+ * @ioapic_base IOAPIC base address
+ * @new_id If non-zero (1-15), assign new apic ID.
+ * If zero, use previously assigned apic ID.
+ * @delivery If 0x0, deliver interrupts on APIC serial bus
+ * If 0x1, deliver interrupts on FSB
+ */
+static void init_ioapic(u32 ioapic_base, u8 new_id, u8 delivery)
{
- u32 bsp_lapicid = lapicid();
- u32 low, high;
- u32 i, ioapic_interrupts;
+ u8 loglvl = BIOS_DEBUG;
+ u8 active_id;
+ u32 reg32;
- printk(BIOS_DEBUG, "IOAPIC: Initializing IOAPIC at 0x%08x\n",
- ioapic_base);
- printk(BIOS_DEBUG, "IOAPIC: Bootstrap Processor Local APIC = 0x%02x\n",
- bsp_lapicid);
+ reg32 = io_apic_read(ioapic_base, 0x00);
+ active_id = (reg32 >> 24) & 0xf;
- if (ioapic_id) {
- printk(BIOS_DEBUG, "IOAPIC: ID = 0x%02x\n", ioapic_id);
- /* Set IOAPIC ID if it has been specified. */
- io_apic_write(ioapic_base, 0x00,
- (io_apic_read(ioapic_base, 0x00) & 0xfff0ffff) |
- (ioapic_id << 24));
+ /* Changing a previously assigned ID and attempting
+ * to use ID=0 here are logged at a critical level.
+ */
+ if ( ((active_id && new_id) && (active_id != new_id)) ||
+ !(active_id || new_id))
+ loglvl=BIOS_CRIT;
+
+ if (!new_id) {
+ printk(loglvl, "IOAPIC: 0x%08x using old ID = %d\n",
+ ioapic_base, active_id);
+
+ } else {
+ reg32 &= 0xf0ffffff;
+ reg32 |= (new_id & 0xf) << 24;
+ io_apic_write(ioapic_base, 0x00, reg32);
+
+ printk(loglvl, "IOAPIC: 0x%08x changing ID = %d->%d\n",
+ ioapic_base, active_id, new_id);
+
+ reg32 = io_apic_read(ioapic_base, 0x00);
+ active_id = (reg32 >> 24) & 0xf;
+ if (active_id != new_id)
+ printk(loglvl, "IOAPIC: 0x%08x changing ID failed (%d!=%d)\n",
+ ioapic_base, active_id, new_id);
+ }
+
+ /* Assign interrupt delivery type */
+ reg32 = io_apic_read(ioapic_base, 0x03);
+ switch (delivery) {
+ case 0:
+ printk(BIOS_DEBUG, "IOAPIC: Delivery is on APIC serial bus\n");
+ reg32 = 0x00;
+ break;
+ case 1:
+ printk(BIOS_DEBUG, "IOAPIC: Delivery is on FSB\n");
+ reg32 = 0x01;
+ break;
+ default:
+ printk(BIOS_CRIT, "IOAPIC: Delivery is reverted to FSB\n");
+ reg32 = 0x01;
+ break;
}
+ io_apic_write(ioapic_base, 0x03, reg32);
+}
+
+/**
+ * Fill IOAPIC vectors all targeting the same processor.
+ * Virtual Wire Mode on vector 0 is enabled, others remain disabled.
+ *
+ *
+ * @ioapic_base IOAPIC base address
+ * @bsp_lapicid APIC ID of a CPU to receive the interrupts
+ *
+ */
+static void load_vectors(u32 ioapic_base, u8 bsp_lapicid)
+{
+ u32 low, high;
+ u32 i, ioapic_interrupts;
/* Read the available number of interrupts. */
ioapic_interrupts = (io_apic_read(ioapic_base, 0x01) >> 16) & 0xff;
@@ -89,28 +152,6 @@ void setup_ioapic(u32 ioapic_base, u8 ioapic_id)
ioapic_interrupts = 24;
printk(BIOS_DEBUG, "IOAPIC: %d interrupts\n", ioapic_interrupts);
-// XXX this decision should probably be made elsewhere, and
-// it's the C3, not the EPIA this depends on.
-#if CONFIG_EPIA_VT8237R_INIT
-#define IOAPIC_INTERRUPTS_ON_APIC_SERIAL_BUS
-#else
-#define IOAPIC_INTERRUPTS_ON_FSB
-#endif
-
-#ifdef IOAPIC_INTERRUPTS_ON_FSB
- /*
- * For the Pentium 4 and above APICs deliver their interrupts
- * on the front side bus, enable that.
- */
- printk(BIOS_DEBUG, "IOAPIC: Enabling interrupts on FSB\n");
- io_apic_write(ioapic_base, 0x03,
- io_apic_read(ioapic_base, 0x03) | (1 << 0));
-#endif
-#ifdef IOAPIC_INTERRUPTS_ON_APIC_SERIAL_BUS
- printk(BIOS_DEBUG, "IOAPIC: Enabling interrupts on APIC serial bus\n");
- io_apic_write(ioapic_base, 0x03, 0);
-#endif
-
/* Enable Virtual Wire Mode. */
low = ENABLED | TRIGGER_EDGE | POLARITY_HIGH | PHYSICAL_DEST | ExtINT;
high = bsp_lapicid << (56 - 32);
@@ -125,10 +166,8 @@ void setup_ioapic(u32 ioapic_base, u8 ioapic_id)
printk(BIOS_SPEW, "IOAPIC: reg 0x%08x value 0x%08x 0x%08x\n",
0, high, low);
-
low = DISABLED;
high = NONE;
-
for (i = 1; i < ioapic_interrupts; i++) {
io_apic_write(ioapic_base, i * 2 + 0x10, low);
io_apic_write(ioapic_base, i * 2 + 0x11, high);
@@ -137,3 +176,36 @@ void setup_ioapic(u32 ioapic_base, u8 ioapic_id)
i, high, low);
}
}
+
+// XXX this decision should probably be made elsewhere, and
+// it's the C3, not the EPIA this depends on.
+
+#if CONFIG_EPIA_VT8237R_INIT
+#define CONFIG_IOAPIC_DELIVERY_TYPE 0
+#else
+#define CONFIG_IOAPIC_DELIVERY_TYPE 1
+#endif
+
+/**
+ * Assign IOAPIC with an ID
+ *
+ * Compile-time options from mainboard Kconfig can affect the
+ * chosen IOAPIC operational mode.
+ *
+ * @ioapic_base IOAPIC base address
+ * @new_id If non-zero (1-15), assign new apic ID.
+ * If zero, use previously assigned apic ID.
+ */
+void setup_ioapic(u32 ioapic_base, u8 new_id)
+{
+ u8 bsp_lapicid = lapicid();
+
+ printk(BIOS_DEBUG, "IOAPIC: Initializing IOAPIC at 0x%08x\n",
+ ioapic_base);
+ init_ioapic(ioapic_base, new_id, CONFIG_IOAPIC_DELIVERY_TYPE);
+
+ printk(BIOS_DEBUG, "IOAPIC: Bootstrap Processor Local APIC ID = %d\n",
+ bsp_lapicid);
+ load_vectors(ioapic_base, bsp_lapicid);
+}
+
More information about the coreboot
mailing list