This uses the new flexible firmware API, since we don't have to keep the firmware around the sysdata API does the freeing for us safely.
Signed-off-by: Luis R. Rodriguez <mcg...@kernel.org> --- arch/x86/kernel/cpu/microcode/amd.c | 56 +++++++++++++++++++++++++------------ 1 file changed, 38 insertions(+), 18 deletions(-) diff --git a/arch/x86/kernel/cpu/microcode/amd.c b/arch/x86/kernel/cpu/microcode/amd.c index 27a0228c9cae..df4351ce4256 100644 --- a/arch/x86/kernel/cpu/microcode/amd.c +++ b/arch/x86/kernel/cpu/microcode/amd.c @@ -23,7 +23,7 @@ #define pr_fmt(fmt) "microcode: " fmt #include <linux/earlycpio.h> -#include <linux/firmware.h> +#include <linux/sysdata.h> #include <linux/uaccess.h> #include <linux/vmalloc.h> #include <linux/initrd.h> @@ -872,6 +872,31 @@ enum ucode_state load_microcode_amd(int cpu, u8 family, const u8 *data, size_t s return ret; } +struct amd_ucode_req { + int cpu; + struct cpuinfo_x86 *c; + enum ucode_state state; + const char *name; +}; + +static int request_microcode_amd_cb(void *context, + const struct sysdata_file *sysdata) +{ + struct amd_ucode_req *req = context; + + if (*(u32 *)sysdata->data != UCODE_MAGIC) { + pr_err("invalid magic value (0x%08x)\n", + *(u32 *)sysdata->data); + req->state = UCODE_ERROR; + return -EINVAL; + } + + req->state = load_microcode_amd(req->cpu, req->c->x86, + sysdata->data, sysdata->size); + + return 0; +} + /* * AMD microcode firmware naming convention, up to family 15h they are in * the legacy file: @@ -891,10 +916,13 @@ enum ucode_state load_microcode_amd(int cpu, u8 family, const u8 *data, size_t s static enum ucode_state request_microcode_amd(int cpu, struct device *device, bool refresh_fw) { + struct amd_ucode_req req; char fw_name[36] = "amd-ucode/microcode_amd.bin"; struct cpuinfo_x86 *c = &cpu_data(cpu); - enum ucode_state ret = UCODE_NFOUND; - const struct firmware *fw; + const struct sysdata_file_desc sysdata_desc = { + SYSDATA_DEFAULT_SYNC(request_microcode_amd_cb, &req), + .optional = true, + }; /* reload ucode container only on the boot cpu */ if (!refresh_fw || c->cpu_index != boot_cpu_data.cpu_index) @@ -903,24 +931,16 @@ static enum ucode_state request_microcode_amd(int cpu, struct device *device, if (c->x86 >= 0x15) snprintf(fw_name, sizeof(fw_name), "amd-ucode/microcode_amd_fam%.2xh.bin", c->x86); - if (request_firmware_direct(&fw, (const char *)fw_name, device)) { - pr_debug("failed to load file %s\n", fw_name); - goto out; - } + req.cpu = cpu; + req.c = c; + req.name = fw_name; + req.state = UCODE_NFOUND; - ret = UCODE_ERROR; - if (*(u32 *)fw->data != UCODE_MAGIC) { - pr_err("invalid magic value (0x%08x)\n", *(u32 *)fw->data); - goto fw_release; + if (sysdata_file_request((const char *)fw_name, &sysdata_desc, device)) { + pr_debug("failed to load file %s\n", fw_name); } - ret = load_microcode_amd(cpu, c->x86, fw->data, fw->size); - - fw_release: - release_firmware(fw); - - out: - return ret; + return req.state; } static enum ucode_state -- 2.8.2