diff options
author | Damien Zammit via Bug reports for the GNU Hurd <bug-hurd@gnu.org> | 2024-12-22 01:43:36 +0000 |
---|---|---|
committer | Samuel Thibault <samuel.thibault@ens-lyon.org> | 2024-12-22 02:45:54 +0100 |
commit | 3f0bf8a31af1deca435888ad9edcd0fdab3b4e3d (patch) | |
tree | 0d67dba51bafa4b4a0f7309d58cf24f95d9f6a22 | |
parent | 608c720a8ef9ed80d650dc4a38582267e2a6c258 (diff) | |
download | gnumach-3f0bf8a31af1deca435888ad9edcd0fdab3b4e3d.tar.gz gnumach-3f0bf8a31af1deca435888ad9edcd0fdab3b4e3d.tar.bz2 gnumach-3f0bf8a31af1deca435888ad9edcd0fdab3b4e3d.zip |
smp: Use deassert for startup IPI not assert
Fixes ESR==0x8 error on AMD fam15h. Fixed timings.
Cannot locate documentation on correct mode for STARTUP IPI,
but this patch works on AMD hw and qemu.
Message-ID: <20241222014306.430098-3-damien@zamaudio.com>
-rw-r--r-- | i386/i386/smp.c | 62 |
1 files changed, 40 insertions, 22 deletions
diff --git a/i386/i386/smp.c b/i386/i386/smp.c index befa0510..e3e4cc82 100644 --- a/i386/i386/smp.c +++ b/i386/i386/smp.c @@ -95,6 +95,7 @@ smp_send_ipi_init(int bsp_apic_id) int err; lapic->error_status.r = 0; + err = lapic->error_status.r; /* Assert INIT IPI: * @@ -104,7 +105,6 @@ smp_send_ipi_init(int bsp_apic_id) /* Wait for delivery */ wait_for_ipi(); - hpet_mdelay(10); /* Deassert INIT IPI: * @@ -126,32 +126,49 @@ smp_send_ipi_init(int bsp_apic_id) } static int -smp_send_ipi_startup(int bsp_apic_id, int vector) +smp_send_ipi_startup_twice(int bsp_apic_id, int vector) { - int err; + int i, accept_err, send_err; + volatile int err; - lapic->error_status.r = 0; + for (i = 0; i < 2; i++) { + lapic->error_status.r = 0; + err = lapic->error_status.r; - /* StartUp IPI: - * - * Have not seen any documentation for trigger mode for this IPI - * but it seems to work with EDGE. (AMD BKDG FAM16h document specifies dont care) - */ - apic_send_ipi(ALL_EXCLUDING_SELF, STARTUP, PHYSICAL, ASSERT, EDGE, vector, bsp_apic_id); + /* StartUp IPI: + * + * Have not seen any documentation for trigger mode for this IPI + * but it seems to work with EDGE. (AMD BKDG FAM16h document specifies dont care) + */ + apic_send_ipi(ALL_EXCLUDING_SELF, STARTUP, PHYSICAL, DE_ASSERT, EDGE, vector, bsp_apic_id); - /* Wait for delivery */ - wait_for_ipi(); + hpet_udelay(10); - err = lapic->error_status.r; - if (err) { - printf("ESR error upon STARTUP 0x%x\n", err); + /* Wait for other cpu to accept IPI */ + wait_for_ipi(); + send_err = lapic->error_status.r; + + hpet_udelay(10); + + lapic->error_status.r = 0; + accept_err = lapic->error_status.r & 0xef; + + if (send_err || accept_err) + break; } - return 0; + + if (send_err) + printf("ESR error: DID NOT SEND? 0x%x\n", send_err); + if (accept_err) + printf("ESR error: delivery 0x%x\n", accept_err); + + return send_err | accept_err; } /* See Intel IA32/64 Software Developer's Manual 3A Section 8.4.4.1 */ int smp_startup_cpus(unsigned bsp_apic_id, phys_addr_t start_eip) { + int err; #if 0 /* This block goes with a legacy method of INIT that only works with * old hardware that does not support SIPIs. @@ -177,12 +194,13 @@ int smp_startup_cpus(unsigned bsp_apic_id, phys_addr_t start_eip) printf("Sending IPIs from BSP APIC ID %u...\n", bsp_apic_id); smp_send_ipi_init(bsp_apic_id); - hpet_mdelay(10); - smp_send_ipi_startup(bsp_apic_id, start_eip >> STARTUP_VECTOR_SHIFT); - hpet_udelay(200); - smp_send_ipi_startup(bsp_apic_id, start_eip >> STARTUP_VECTOR_SHIFT); - hpet_udelay(200); - + err = smp_send_ipi_startup_twice(bsp_apic_id, start_eip >> STARTUP_VECTOR_SHIFT); + if (err) { + printf("FATAL: APs failed to start\n"); + while(1) { + cpu_pause(); + } + } printf("done\n"); return 0; } |