| // SPDX-License-Identifier: GPL-2.0 |
| |
| #include <linux/efi_embedded_fw.h> |
| #include <linux/property.h> |
| #include <linux/security.h> |
| #include <linux/vmalloc.h> |
| |
| #include "fallback.h" |
| #include "firmware.h" |
| |
| int firmware_fallback_platform(struct fw_priv *fw_priv) |
| { |
| const u8 *data; |
| size_t size; |
| int rc; |
| |
| if (!(fw_priv->opt_flags & FW_OPT_FALLBACK_PLATFORM)) |
| return -ENOENT; |
| |
| rc = security_kernel_load_data(LOADING_FIRMWARE, true); |
| if (rc) |
| return rc; |
| |
| rc = efi_get_embedded_fw(fw_priv->fw_name, &data, &size); |
| if (rc) |
| return rc; /* rc == -ENOENT when the fw was not found */ |
| |
| if (fw_priv->data && size > fw_priv->allocated_size) |
| return -ENOMEM; |
| |
| rc = security_kernel_post_load_data((u8 *)data, size, LOADING_FIRMWARE, |
| "platform"); |
| if (rc) |
| return rc; |
| |
| if (!fw_priv->data) |
| fw_priv->data = vmalloc(size); |
| if (!fw_priv->data) |
| return -ENOMEM; |
| |
| memcpy(fw_priv->data, data, size); |
| fw_priv->size = size; |
| fw_state_done(fw_priv); |
| return 0; |
| } |