Commit 3881ee0b authored by Ard Biesheuvel's avatar Ard Biesheuvel

efi: avoid efivars layer when loading SSDTs from variables

The efivars intermediate variable access layer provides an abstraction
that permits the EFI variable store to be replaced by something else
that implements a compatible interface, and caches all variables in the
variable store for fast access via the efivarfs pseudo-filesystem.

The SSDT override feature does not take advantage of either feature, as
it is only used when the generic EFI implementation of efivars is used,
and it traverses all variables only once to find the ones it is
interested in, and frees all data structures that the efivars layer
keeps right after.

So in this case, let's just call EFI's code directly, using the function
pointers in struct efi.
Signed-off-by: default avatarArd Biesheuvel <ardb@kernel.org>
parent db01ea88
...@@ -202,7 +202,7 @@ static void generic_ops_unregister(void) ...@@ -202,7 +202,7 @@ static void generic_ops_unregister(void)
} }
#ifdef CONFIG_EFI_CUSTOM_SSDT_OVERLAYS #ifdef CONFIG_EFI_CUSTOM_SSDT_OVERLAYS
#define EFIVAR_SSDT_NAME_MAX 16 #define EFIVAR_SSDT_NAME_MAX 16UL
static char efivar_ssdt[EFIVAR_SSDT_NAME_MAX] __initdata; static char efivar_ssdt[EFIVAR_SSDT_NAME_MAX] __initdata;
static int __init efivar_ssdt_setup(char *str) static int __init efivar_ssdt_setup(char *str)
{ {
...@@ -219,83 +219,62 @@ static int __init efivar_ssdt_setup(char *str) ...@@ -219,83 +219,62 @@ static int __init efivar_ssdt_setup(char *str)
} }
__setup("efivar_ssdt=", efivar_ssdt_setup); __setup("efivar_ssdt=", efivar_ssdt_setup);
static __init int efivar_ssdt_iter(efi_char16_t *name, efi_guid_t vendor,
unsigned long name_size, void *data)
{
struct efivar_entry *entry;
struct list_head *list = data;
char utf8_name[EFIVAR_SSDT_NAME_MAX];
int limit = min_t(unsigned long, EFIVAR_SSDT_NAME_MAX, name_size);
ucs2_as_utf8(utf8_name, name, limit - 1);
if (strncmp(utf8_name, efivar_ssdt, limit) != 0)
return 0;
entry = kmalloc(sizeof(*entry), GFP_KERNEL);
if (!entry)
return 0;
memcpy(entry->var.VariableName, name, name_size);
memcpy(&entry->var.VendorGuid, &vendor, sizeof(efi_guid_t));
efivar_entry_add(entry, list);
return 0;
}
static __init int efivar_ssdt_load(void) static __init int efivar_ssdt_load(void)
{ {
LIST_HEAD(entries); unsigned long name_size = 256;
struct efivar_entry *entry, *aux; efi_char16_t *name = NULL;
unsigned long size; efi_status_t status;
void *data; efi_guid_t guid;
int ret;
if (!efivar_ssdt[0]) if (!efivar_ssdt[0])
return 0; return 0;
ret = efivar_init(efivar_ssdt_iter, &entries, true, &entries); name = kzalloc(name_size, GFP_KERNEL);
if (!name)
list_for_each_entry_safe(entry, aux, &entries, list) { return -ENOMEM;
pr_info("loading SSDT from variable %s-%pUl\n", efivar_ssdt,
&entry->var.VendorGuid);
list_del(&entry->list); for (;;) {
char utf8_name[EFIVAR_SSDT_NAME_MAX];
unsigned long data_size = 0;
void *data;
int limit;
ret = efivar_entry_size(entry, &size); status = efi.get_next_variable(&name_size, name, &guid);
if (ret) { if (status == EFI_NOT_FOUND) {
pr_err("failed to get var size\n"); break;
goto free_entry; } else if (status == EFI_BUFFER_TOO_SMALL) {
name = krealloc(name, name_size, GFP_KERNEL);
if (!name)
return -ENOMEM;
continue;
} }
data = kmalloc(size, GFP_KERNEL); limit = min(EFIVAR_SSDT_NAME_MAX, name_size);
if (!data) { ucs2_as_utf8(utf8_name, name, limit - 1);
ret = -ENOMEM; if (strncmp(utf8_name, efivar_ssdt, limit) != 0)
goto free_entry; continue;
}
ret = efivar_entry_get(entry, NULL, &size, data); pr_info("loading SSDT from variable %s-%pUl\n", efivar_ssdt, &guid);
if (ret) {
pr_err("failed to get var data\n");
goto free_data;
}
ret = acpi_load_table(data, NULL); status = efi.get_variable(name, &guid, NULL, &data_size, NULL);
if (ret) { if (status != EFI_BUFFER_TOO_SMALL || !data_size)
pr_err("failed to load table: %d\n", ret); return -EIO;
goto free_data;
}
goto free_entry; data = kmalloc(data_size, GFP_KERNEL);
if (!data)
return -ENOMEM;
free_data: status = efi.get_variable(name, &guid, NULL, &data_size, data);
if (status == EFI_SUCCESS) {
acpi_status ret = acpi_load_table(data, NULL);
if (ret)
pr_err("failed to load table: %u\n", ret);
} else {
pr_err("failed to get var data: 0x%lx\n", status);
}
kfree(data); kfree(data);
free_entry:
kfree(entry);
} }
return 0;
return ret;
} }
#else #else
static inline int efivar_ssdt_load(void) { return 0; } static inline int efivar_ssdt_load(void) { return 0; }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment