aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--drivers/firmware/efi/efi.c21
-rw-r--r--include/linux/efi.h3
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)
997int __ref efi_mem_reserve_persistent(phys_addr_t addr, u64 size) 997int __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 */