x86, microcode, AMD: Do not use smp_processor_id() in preemtible context
authorBorislav Petkov <bp@suse.de>
Mon, 1 Dec 2014 10:12:21 +0000 (11:12 +0100)
committerThomas Gleixner <tglx@linutronix.de>
Mon, 1 Dec 2014 10:51:05 +0000 (11:51 +0100)
Hand down the cpu number instead, otherwise lockdep screams when doing

echo 1 > /sys/devices/system/cpu/microcode/reload.

BUG: using smp_processor_id() in preemptible [00000000] code: amd64-microcode/2470
caller is debug_smp_processor_id+0x12/0x20
CPU: 1 PID: 2470 Comm: amd64-microcode Not tainted 3.18.0-rc6+ #26
...

Signed-off-by: Borislav Petkov <bp@suse.de>
Link: http://lkml.kernel.org/r/1417428741-4501-1-git-send-email-bp@alien8.de
Signed-off-by: Thomas Gleixner <tglx@linutronix.de>
arch/x86/include/asm/microcode_amd.h
arch/x86/kernel/cpu/microcode/amd.c
arch/x86/kernel/cpu/microcode/amd_early.c

index b7b10b82d3e510bcad0faf2d2b82675b635a6b7d..48a48024ebe45ff2db7d393b539b80c4eac7d828 100644 (file)
@@ -59,7 +59,7 @@ static inline u16 find_equiv_id(struct equiv_cpu_entry *equiv_cpu_table,
 
 extern int __apply_microcode_amd(struct microcode_amd *mc_amd);
 extern int apply_microcode_amd(int cpu);
-extern enum ucode_state load_microcode_amd(u8 family, const u8 *data, size_t size);
+extern enum ucode_state load_microcode_amd(int cpu, u8 family, const u8 *data, size_t size);
 
 #define PATCH_MAX_SIZE PAGE_SIZE
 extern u8 amd_ucode_patch[PATCH_MAX_SIZE];
index 8fffd845e22b9e609107b53b5428b96ae1a6aec4..bfbbe6195e2da7e798323d014cfc5ae2d406d0f3 100644 (file)
@@ -376,7 +376,7 @@ static enum ucode_state __load_microcode_amd(u8 family, const u8 *data,
        return UCODE_OK;
 }
 
-enum ucode_state load_microcode_amd(u8 family, const u8 *data, size_t size)
+enum ucode_state load_microcode_amd(int cpu, u8 family, const u8 *data, size_t size)
 {
        enum ucode_state ret;
 
@@ -390,8 +390,8 @@ enum ucode_state load_microcode_amd(u8 family, const u8 *data, size_t size)
 
 #if defined(CONFIG_MICROCODE_AMD_EARLY) && defined(CONFIG_X86_32)
        /* save BSP's matching patch for early load */
-       if (cpu_data(smp_processor_id()).cpu_index == boot_cpu_data.cpu_index) {
-               struct ucode_patch *p = find_patch(smp_processor_id());
+       if (cpu_data(cpu).cpu_index == boot_cpu_data.cpu_index) {
+               struct ucode_patch *p = find_patch(cpu);
                if (p) {
                        memset(amd_ucode_patch, 0, PATCH_MAX_SIZE);
                        memcpy(amd_ucode_patch, p->data, min_t(u32, ksize(p->data),
@@ -444,7 +444,7 @@ static enum ucode_state request_microcode_amd(int cpu, struct device *device,
                goto fw_release;
        }
 
-       ret = load_microcode_amd(c->x86, fw->data, fw->size);
+       ret = load_microcode_amd(cpu, c->x86, fw->data, fw->size);
 
  fw_release:
        release_firmware(fw);
index 06674473b0e66736d0a64c235c1e9fe7e774deb9..3d988a30a21d13d02ceb1e709fe9ea7862c8a072 100644 (file)
@@ -389,7 +389,7 @@ int __init save_microcode_in_initrd_amd(void)
        eax   = cpuid_eax(0x00000001);
        eax   = ((eax >> 8) & 0xf) + ((eax >> 20) & 0xff);
 
-       ret = load_microcode_amd(eax, container, container_size);
+       ret = load_microcode_amd(smp_processor_id(), eax, container, container_size);
        if (ret != UCODE_OK)
                retval = -EINVAL;