aboutsummaryrefslogtreecommitdiffstats
path: root/arch/x86/kernel/apic/io_apic.c
diff options
context:
space:
mode:
authorSuresh Siddha <suresh.b.siddha@intel.com>2009-03-16 20:05:03 -0400
committerH. Peter Anvin <hpa@linux.intel.com>2009-03-17 18:45:29 -0400
commit05c3dc2c4b60387769cbe73174347de4cf85f0c9 (patch)
tree7e5918518c5f22f6121ff885283bef5397a0a842 /arch/x86/kernel/apic/io_apic.c
parent29b61be65a33c95564fa82e7e8d60d97adb68ea8 (diff)
x86, ioapic: Fix non atomic allocation with interrupts disabled
Impact: fix possible race save_mask_IO_APIC_setup() was using non atomic memory allocation while getting called with interrupts disabled. Fix this by splitting this into two different function. Allocation part save_IO_APIC_setup() now happens before disabling interrupts. Signed-off-by: Suresh Siddha <suresh.b.siddha@intel.com> Signed-off-by: H. Peter Anvin <hpa@linux.intel.com>
Diffstat (limited to 'arch/x86/kernel/apic/io_apic.c')
-rw-r--r--arch/x86/kernel/apic/io_apic.c34
1 files changed, 23 insertions, 11 deletions
diff --git a/arch/x86/kernel/apic/io_apic.c b/arch/x86/kernel/apic/io_apic.c
index cf27795c641..ff1759a1128 100644
--- a/arch/x86/kernel/apic/io_apic.c
+++ b/arch/x86/kernel/apic/io_apic.c
@@ -853,9 +853,9 @@ __setup("pirq=", ioapic_pirq_setup);
853static struct IO_APIC_route_entry *early_ioapic_entries[MAX_IO_APICS]; 853static struct IO_APIC_route_entry *early_ioapic_entries[MAX_IO_APICS];
854 854
855/* 855/*
856 * Saves and masks all the unmasked IO-APIC RTE's 856 * Saves all the IO-APIC RTE's
857 */ 857 */
858int save_mask_IO_APIC_setup(void) 858int save_IO_APIC_setup(void)
859{ 859{
860 union IO_APIC_reg_01 reg_01; 860 union IO_APIC_reg_01 reg_01;
861 unsigned long flags; 861 unsigned long flags;
@@ -880,16 +880,9 @@ int save_mask_IO_APIC_setup(void)
880 } 880 }
881 881
882 for (apic = 0; apic < nr_ioapics; apic++) 882 for (apic = 0; apic < nr_ioapics; apic++)
883 for (pin = 0; pin < nr_ioapic_registers[apic]; pin++) { 883 for (pin = 0; pin < nr_ioapic_registers[apic]; pin++)
884 struct IO_APIC_route_entry entry; 884 early_ioapic_entries[apic][pin] =
885
886 entry = early_ioapic_entries[apic][pin] =
887 ioapic_read_entry(apic, pin); 885 ioapic_read_entry(apic, pin);
888 if (!entry.mask) {
889 entry.mask = 1;
890 ioapic_write_entry(apic, pin, entry);
891 }
892 }
893 886
894 return 0; 887 return 0;
895 888
@@ -902,6 +895,25 @@ nomem:
902 return -ENOMEM; 895 return -ENOMEM;
903} 896}
904 897
898void mask_IO_APIC_setup(void)
899{
900 int apic, pin;
901
902 for (apic = 0; apic < nr_ioapics; apic++) {
903 if (!early_ioapic_entries[apic])
904 break;
905 for (pin = 0; pin < nr_ioapic_registers[apic]; pin++) {
906 struct IO_APIC_route_entry entry;
907
908 entry = early_ioapic_entries[apic][pin];
909 if (!entry.mask) {
910 entry.mask = 1;
911 ioapic_write_entry(apic, pin, entry);
912 }
913 }
914 }
915}
916
905void restore_IO_APIC_setup(void) 917void restore_IO_APIC_setup(void)
906{ 918{
907 int apic, pin; 919 int apic, pin;