diff options
author | Suresh Siddha <suresh.b.siddha@intel.com> | 2009-03-16 20:05:03 -0400 |
---|---|---|
committer | H. Peter Anvin <hpa@linux.intel.com> | 2009-03-17 18:45:29 -0400 |
commit | 05c3dc2c4b60387769cbe73174347de4cf85f0c9 (patch) | |
tree | 7e5918518c5f22f6121ff885283bef5397a0a842 /arch/x86/kernel/apic/io_apic.c | |
parent | 29b61be65a33c95564fa82e7e8d60d97adb68ea8 (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.c | 34 |
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); | |||
853 | static struct IO_APIC_route_entry *early_ioapic_entries[MAX_IO_APICS]; | 853 | static 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 | */ |
858 | int save_mask_IO_APIC_setup(void) | 858 | int 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 | ||
898 | void 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 | |||
905 | void restore_IO_APIC_setup(void) | 917 | void restore_IO_APIC_setup(void) |
906 | { | 918 | { |
907 | int apic, pin; | 919 | int apic, pin; |