12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758 |
- #include <dragonstub/dragonstub.h>
- #include <dragonstub/linux/align.h>
- #include <dragonstub/linux/math.h>
- /**
- * efi_allocate_pages_aligned() - Allocate memory pages
- * @size: minimum number of bytes to allocate
- * @addr: On return the address of the first allocated page. The first
- * allocated page has alignment EFI_ALLOC_ALIGN which is an
- * architecture dependent multiple of the page size.
- * @max: the address that the last allocated memory page shall not
- * exceed
- * @align: minimum alignment of the base of the allocation
- *
- * Allocate pages as EFI_LOADER_DATA. The allocated pages are aligned according
- * to @align, which should be >= EFI_ALLOC_ALIGN. The last allocated page will
- * not exceed the address given by @max.
- *
- * Return: status code
- */
- efi_status_t efi_allocate_pages_aligned(unsigned long size, unsigned long *addr,
- unsigned long max, unsigned long align,
- int memory_type)
- {
- efi_physical_addr_t alloc_addr;
- efi_status_t status;
- int slack;
- max = min(max, EFI_ALLOC_LIMIT);
- if (align < EFI_ALLOC_ALIGN)
- align = EFI_ALLOC_ALIGN;
- alloc_addr = ALIGN_DOWN(max + 1, align) - 1;
- size = round_up(size, EFI_ALLOC_ALIGN);
- slack = align / EFI_PAGE_SIZE - 1;
- status = efi_bs_call(AllocatePages, EFI_ALLOCATE_MAX_ADDRESS,
- memory_type, size / EFI_PAGE_SIZE + slack,
- &alloc_addr);
- if (status != EFI_SUCCESS)
- return status;
- *addr = ALIGN_UP((unsigned long)alloc_addr, align);
- if (slack > 0) {
- int l = (alloc_addr & (align - 1)) / EFI_PAGE_SIZE;
- if (l) {
- efi_bs_call(FreePages, alloc_addr, slack - l + 1);
- slack = l - 1;
- }
- if (slack)
- efi_bs_call(FreePages, *addr + size, slack);
- }
- return EFI_SUCCESS;
- }
|