]> Zhao Yanbai Git Server - minix.git/commitdiff
SMP - clock calibration spurious IRQ deadlock fix
authorTomas Hruby <tom@minix3.org>
Thu, 26 Jan 2012 11:39:40 +0000 (11:39 +0000)
committerTomas Hruby <tom@minix3.org>
Thu, 26 Jan 2012 11:39:40 +0000 (11:39 +0000)
 - this patch fixes a deadlock which may occur if we get a
   spurious interrupt while calibrating clocks during the boot
   time. Since we never handle interrupts while in the kernel
   (BKL locked) the interrupt code locks the lock. This is a
   different situation, a corner case, boot time only. We do not
   return to userspace but to the kernel, so the BKL is not
   unlocked. So we need irq handler which leaves the BKL
   unlocked.  The clock handler does it already, this patch adds
   a dummy spurious irq handler for the same reason. It is better
   to handle the situation this way to keep the normal runtime
   code simple.

common/include/arch/i386/interrupt.h
kernel/arch/i386/apic.c

index 86b9447ff64ee346167a4662c9086d347c07e236..c13a1e422352d43299ca894269b50be30acc0e06 100644 (file)
@@ -49,6 +49,7 @@
 #define XT_WINI_IRQ        5   /* xt winchester */
 #define FLOPPY_IRQ         6   /* floppy disk */
 #define PRINTER_IRQ        7
+#define SPURIOUS_IRQ       7
 #define CMOS_CLOCK_IRQ     8
 #define KBD_AUX_IRQ       12   /* AUX (PS/2 mouse) port in kbd controller */
 #define AT_WINI_0_IRQ     14   /* at winchester controller 0 */
index b73b98f1ac8799518b82d4bb3269b7f6d25d77af..05602da62272bff2b60d30d970f6f8c021a9dc38 100644 (file)
@@ -428,13 +428,22 @@ PRIVATE int calib_clk_handler(irq_hook_t * UNUSED(hook))
        return 1;
 }
 
+PRIVATE int spurious_irq_handler(irq_hook_t * UNUSED(hook))
+{
+       /*
+        * Do nothing, only unlock the kernel so we do not deadlock!
+        */
+       BKL_UNLOCK();
+       return 1;
+}
+
 PRIVATE void apic_calibrate_clocks(unsigned cpu)
 {
        u32_t lvtt, val, lapic_delta;
        u64_t tsc_delta;
        u64_t cpu_freq;
 
-       irq_hook_t calib_clk;
+       irq_hook_t calib_clk, spurious_irq;
 
        BOOT_VERBOSE(printf("Calibrating clock\n"));
        /*
@@ -467,6 +476,14 @@ PRIVATE void apic_calibrate_clocks(unsigned cpu)
        /* set the probe, we use the legacy timer, IRQ 0 */
        put_irq_handler(&calib_clk, CLOCK_IRQ, calib_clk_handler);
 
+       /*
+        * A spurious interrupt may occur during the clock calibration. Since we
+        * do this calibration in kernel, we need a special handler which will
+        * leave the BKL unlocked like the clock handler. This is a corner case,
+        * boot time only situation
+        */
+       put_irq_handler(&spurious_irq, SPURIOUS_IRQ, spurious_irq_handler);
+
        /* set the PIC timer to get some time */
        init_8253A_timer(system_hz);
 
@@ -489,6 +506,7 @@ PRIVATE void apic_calibrate_clocks(unsigned cpu)
 
        /* remove the probe */
        rm_irq_handler(&calib_clk);
+       rm_irq_handler(&spurious_irq);
 
        lapic_delta = lapic_tctr0 - lapic_tctr1;
        tsc_delta = sub64(tsc1, tsc0);