diff options
| -rw-r--r-- | drivers/firmware/efi/efi.c | 21 | ||||
| -rw-r--r-- | include/linux/efi.h | 3 |
2 files changed, 20 insertions, 4 deletions
diff --git a/drivers/firmware/efi/efi.c b/drivers/firmware/efi/efi.c index 80b11521627a..4c46ff6f2242 100644 --- a/drivers/firmware/efi/efi.c +++ b/drivers/firmware/efi/efi.c | |||
| @@ -997,8 +997,8 @@ static int __init efi_memreserve_map_root(void) | |||
| 997 | int __ref efi_mem_reserve_persistent(phys_addr_t addr, u64 size) | 997 | int __ref efi_mem_reserve_persistent(phys_addr_t addr, u64 size) |
| 998 | { | 998 | { |
| 999 | struct linux_efi_memreserve *rsv; | 999 | struct linux_efi_memreserve *rsv; |
| 1000 | int rsvsize = EFI_MEMRESERVE_SIZE(1); | 1000 | unsigned long prsv; |
| 1001 | int rc; | 1001 | int rc, index; |
| 1002 | 1002 | ||
| 1003 | if (efi_memreserve_root == (void *)ULONG_MAX) | 1003 | if (efi_memreserve_root == (void *)ULONG_MAX) |
| 1004 | return -ENODEV; | 1004 | return -ENODEV; |
| @@ -1009,11 +1009,24 @@ int __ref efi_mem_reserve_persistent(phys_addr_t addr, u64 size) | |||
| 1009 | return rc; | 1009 | return rc; |
| 1010 | } | 1010 | } |
| 1011 | 1011 | ||
| 1012 | rsv = kmalloc(rsvsize, GFP_ATOMIC); | 1012 | /* first try to find a slot in an existing linked list entry */ |
| 1013 | for (prsv = efi_memreserve_root->next; prsv; prsv = rsv->next) { | ||
| 1014 | rsv = __va(prsv); | ||
| 1015 | index = atomic_fetch_add_unless(&rsv->count, 1, rsv->size); | ||
| 1016 | if (index < rsv->size) { | ||
| 1017 | rsv->entry[index].base = addr; | ||
| 1018 | rsv->entry[index].size = size; | ||
| 1019 | |||
| 1020 | return 0; | ||
| 1021 | } | ||
| 1022 | } | ||
| 1023 | |||
| 1024 | /* no slot found - allocate a new linked list entry */ | ||
| 1025 | rsv = (struct linux_efi_memreserve *)__get_free_page(GFP_ATOMIC); | ||
| 1013 | if (!rsv) | 1026 | if (!rsv) |
| 1014 | return -ENOMEM; | 1027 | return -ENOMEM; |
| 1015 | 1028 | ||
| 1016 | rsv->size = 1; | 1029 | rsv->size = EFI_MEMRESERVE_COUNT(PAGE_SIZE); |
| 1017 | atomic_set(&rsv->count, 1); | 1030 | atomic_set(&rsv->count, 1); |
| 1018 | rsv->entry[0].base = addr; | 1031 | rsv->entry[0].base = addr; |
| 1019 | rsv->entry[0].size = size; | 1032 | rsv->entry[0].size = size; |
diff --git a/include/linux/efi.h b/include/linux/efi.h index 4f27640fdcdc..becd5d76a207 100644 --- a/include/linux/efi.h +++ b/include/linux/efi.h | |||
| @@ -1724,4 +1724,7 @@ struct linux_efi_memreserve { | |||
| 1724 | #define EFI_MEMRESERVE_SIZE(count) (sizeof(struct linux_efi_memreserve) + \ | 1724 | #define EFI_MEMRESERVE_SIZE(count) (sizeof(struct linux_efi_memreserve) + \ |
| 1725 | (count) * sizeof(((struct linux_efi_memreserve *)0)->entry[0])) | 1725 | (count) * sizeof(((struct linux_efi_memreserve *)0)->entry[0])) |
| 1726 | 1726 | ||
| 1727 | #define EFI_MEMRESERVE_COUNT(size) (((size) - sizeof(struct linux_efi_memreserve)) \ | ||
| 1728 | / sizeof(((struct linux_efi_memreserve *)0)->entry[0])) | ||
| 1729 | |||
| 1727 | #endif /* _LINUX_EFI_H */ | 1730 | #endif /* _LINUX_EFI_H */ |
