From 78683c970f4c113a7fc9f34916fefae3deab77d2 Mon Sep 17 00:00:00 2001 From: Pawel Wieczorkiewicz Date: Tue, 21 Nov 2023 16:13:13 +0100 Subject: [PATCH] mm,vmm: improve get_free_pages() implementation Use vmap_range() to handle multiple mapping areas automatically and consistently. Standardize rules for virtual address returned by get_free_pages(). Signed-off-by: Pawel Wieczorkiewicz --- include/arch/x86/page.h | 12 +++++ include/mm/vmm.h | 9 ++-- mm/vmm.c | 98 ++++++++++++++++++++++++++++------------- 3 files changed, 86 insertions(+), 33 deletions(-) diff --git a/include/arch/x86/page.h b/include/arch/x86/page.h index 35cfd688..3de15428 100644 --- a/include/arch/x86/page.h +++ b/include/arch/x86/page.h @@ -275,6 +275,18 @@ static inline mfn_t virt_to_mfn(const void *va) { return paddr_to_mfn(virt_to_paddr(va)); } +static inline unsigned long order_to_flags(unsigned int order) { + switch (order) { + case PAGE_ORDER_2M: + return L2_PROT; + case PAGE_ORDER_1G: + return L3_PROT; + case PAGE_ORDER_4K: + default: + return L1_PROT; + } +} + #endif /* __ASSEMBLY__ */ #endif /* KTF_PAGE_H */ diff --git a/include/mm/vmm.h b/include/mm/vmm.h index 3d14b4e8..1baf0587 100644 --- a/include/mm/vmm.h +++ b/include/mm/vmm.h @@ -27,12 +27,15 @@ #include +/* clang-format off */ enum gfp_flags { - GFP_KERNEL = 0x00000001, - GFP_USER = 0x00000002, - GFP_IDENT = 0x00000004, + GFP_NONE = 0x00000000, + GFP_IDENT = 0x00000001, + GFP_USER = 0x00000002, + GFP_KERNEL = 0x00000004, GFP_KERNEL_MAP = 0x00000008, }; +/* clang-format on */ typedef enum gfp_flags gfp_flags_t; /* External definitions */ diff --git a/mm/vmm.c b/mm/vmm.c index 193f63b1..ff3ae70d 100644 --- a/mm/vmm.c +++ b/mm/vmm.c @@ -33,10 +33,70 @@ /* Used by higher level mmap_range() functions - must be taken before vmap_lock */ static spinlock_t mmap_lock = SPINLOCK_INIT; -void *get_free_pages(unsigned int order, gfp_flags_t flags) { - frame_t *frame; +static inline vmap_flags_t gfp_to_vmap_flags(gfp_flags_t gfp_flags) { + vmap_flags_t vmap_flags = VMAP_NONE; + + if (gfp_flags == GFP_USER) + return VMAP_KERNEL_USER | VMAP_USER; + + if (gfp_flags & GFP_IDENT) { + vmap_flags |= VMAP_IDENT; + if (gfp_flags & GFP_USER) + vmap_flags |= VMAP_USER_IDENT; + } + + if (gfp_flags & GFP_KERNEL) { + vmap_flags |= VMAP_KERNEL; + if (gfp_flags & GFP_USER) + vmap_flags |= VMAP_USER_KERNEL; + } + + if (gfp_flags & GFP_KERNEL_MAP) { + vmap_flags |= VMAP_KERNEL_MAP; + if (gfp_flags & GFP_USER) + vmap_flags |= VMAP_USER_KERNEL_MAP; + } + + return vmap_flags; +} + +static inline void *gfp_mfn_to_virt(gfp_flags_t gfp_flags, mfn_t mfn) { + /* Return virtual address if a single area is specified ... */ + switch (gfp_flags) { + case GFP_IDENT: + return mfn_to_virt(mfn); + case GFP_KERNEL_MAP: + return mfn_to_virt_map(mfn); + case GFP_USER: + return mfn_to_virt_user(mfn); + case GFP_KERNEL: + return mfn_to_virt_kern(mfn); + default: + /* Otherwise, return kernel addresses if specified before identity + * mapping or user. The below order reflects most common uses. + */ + if (gfp_flags & GFP_KERNEL_MAP) + return mfn_to_virt_map(mfn); + else if (gfp_flags & GFP_KERNEL) + return mfn_to_virt_kern(mfn); + else if (gfp_flags & GFP_IDENT) + return mfn_to_virt(mfn); + else if (gfp_flags & GFP_USER) + return mfn_to_virt_user(mfn); + } + + return NULL; +} + +void *get_free_pages(unsigned int order, gfp_flags_t gfp_flags) { void *va = NULL; + frame_t *frame; mfn_t mfn; + size_t size; + unsigned long pt_flags; + vmap_flags_t vmap_flags; + + ASSERT(gfp_flags != GFP_NONE); if (!boot_flags.virt) panic("Unable to use %s() before final page tables are set", __func__); @@ -46,35 +106,13 @@ void *get_free_pages(unsigned int order, gfp_flags_t flags) { return va; mfn = frame->mfn; - spin_lock(&mmap_lock); - if (flags == GFP_USER) { - va = vmap_kern(mfn_to_virt_user(mfn), mfn, order, L4_PROT, L3_PROT, L2_PROT, - L1_PROT); - vmap_user(mfn_to_virt_user(mfn), mfn, order, L4_PROT_USER, L3_PROT_USER, - L2_PROT_USER, L1_PROT_USER); - } - - if (flags & GFP_IDENT) { - va = vmap_kern(mfn_to_virt(mfn), mfn, order, L4_PROT, L3_PROT, L2_PROT, L1_PROT); - if (flags & GFP_USER) - vmap_user(mfn_to_virt(mfn), mfn, order, L4_PROT, L3_PROT, L2_PROT, L1_PROT); - } - - if (flags & GFP_KERNEL) { - va = vmap_kern(mfn_to_virt_kern(mfn), mfn, order, L4_PROT, L3_PROT, L2_PROT, - L1_PROT); - if (flags & GFP_USER) - vmap_user(mfn_to_virt_kern(mfn), mfn, order, L4_PROT, L3_PROT, L2_PROT, - L1_PROT); - } + size = ORDER_TO_SIZE(order); + pt_flags = order_to_flags(order); + vmap_flags = gfp_to_vmap_flags(gfp_flags); - if (flags & GFP_KERNEL_MAP) { - va = vmap_kern(mfn_to_virt_map(mfn), mfn, order, L4_PROT, L3_PROT, L2_PROT, - L1_PROT); - if (flags & GFP_USER) - vmap_user(mfn_to_virt_map(mfn), mfn, order, L4_PROT, L3_PROT, L2_PROT, - L1_PROT); - } + spin_lock(&mmap_lock); + if (vmap_range(mfn_to_paddr(mfn), size, pt_flags, vmap_flags) == 0) + va = gfp_mfn_to_virt(gfp_flags, mfn); spin_unlock(&mmap_lock); return va;