Commit 4ee602e7 authored by David Matlack's avatar David Matlack Committed by Paolo Bonzini

KVM: selftests: Replace x86_page_size with PG_LEVEL_XX

x86_page_size is an enum used to communicate the desired page size with
which to map a range of memory. Under the hood they just encode the
desired level at which to map the page. This ends up being clunky in a
few ways:

 - The name suggests it encodes the size of the page rather than the
   level.
 - In other places in x86_64/processor.c we just use a raw int to encode
   the level.

Simplify this by adopting the kernel style of PG_LEVEL_XX enums and pass
around raw ints when referring to the level. This makes the code easier
to understand since these macros are very common in KVM MMU code.
Signed-off-by: default avatarDavid Matlack <dmatlack@google.com>
Message-Id: <20220520233249.3776001-2-dmatlack@google.com>
Signed-off-by: default avatarPaolo Bonzini <pbonzini@redhat.com>
parent e3cdaab5
...@@ -482,13 +482,19 @@ void vcpu_set_hv_cpuid(struct kvm_vm *vm, uint32_t vcpuid); ...@@ -482,13 +482,19 @@ void vcpu_set_hv_cpuid(struct kvm_vm *vm, uint32_t vcpuid);
struct kvm_cpuid2 *vcpu_get_supported_hv_cpuid(struct kvm_vm *vm, uint32_t vcpuid); struct kvm_cpuid2 *vcpu_get_supported_hv_cpuid(struct kvm_vm *vm, uint32_t vcpuid);
void vm_xsave_req_perm(int bit); void vm_xsave_req_perm(int bit);
enum x86_page_size { enum pg_level {
X86_PAGE_SIZE_4K = 0, PG_LEVEL_NONE,
X86_PAGE_SIZE_2M, PG_LEVEL_4K,
X86_PAGE_SIZE_1G, PG_LEVEL_2M,
PG_LEVEL_1G,
PG_LEVEL_512G,
PG_LEVEL_NUM
}; };
void __virt_pg_map(struct kvm_vm *vm, uint64_t vaddr, uint64_t paddr,
enum x86_page_size page_size); #define PG_LEVEL_SHIFT(_level) ((_level - 1) * 9 + 12)
#define PG_LEVEL_SIZE(_level) (1ull << PG_LEVEL_SHIFT(_level))
void __virt_pg_map(struct kvm_vm *vm, uint64_t vaddr, uint64_t paddr, int level);
/* /*
* Basic CPU control in CR0 * Basic CPU control in CR0
......
...@@ -158,7 +158,7 @@ static void *virt_get_pte(struct kvm_vm *vm, uint64_t pt_pfn, uint64_t vaddr, ...@@ -158,7 +158,7 @@ static void *virt_get_pte(struct kvm_vm *vm, uint64_t pt_pfn, uint64_t vaddr,
int level) int level)
{ {
uint64_t *page_table = addr_gpa2hva(vm, pt_pfn << vm->page_shift); uint64_t *page_table = addr_gpa2hva(vm, pt_pfn << vm->page_shift);
int index = vaddr >> (vm->page_shift + level * 9) & 0x1ffu; int index = (vaddr >> PG_LEVEL_SHIFT(level)) & 0x1ffu;
return &page_table[index]; return &page_table[index];
} }
...@@ -167,14 +167,14 @@ static uint64_t *virt_create_upper_pte(struct kvm_vm *vm, ...@@ -167,14 +167,14 @@ static uint64_t *virt_create_upper_pte(struct kvm_vm *vm,
uint64_t pt_pfn, uint64_t pt_pfn,
uint64_t vaddr, uint64_t vaddr,
uint64_t paddr, uint64_t paddr,
int level, int current_level,
enum x86_page_size page_size) int target_level)
{ {
uint64_t *pte = virt_get_pte(vm, pt_pfn, vaddr, level); uint64_t *pte = virt_get_pte(vm, pt_pfn, vaddr, current_level);
if (!(*pte & PTE_PRESENT_MASK)) { if (!(*pte & PTE_PRESENT_MASK)) {
*pte = PTE_PRESENT_MASK | PTE_WRITABLE_MASK; *pte = PTE_PRESENT_MASK | PTE_WRITABLE_MASK;
if (level == page_size) if (current_level == target_level)
*pte |= PTE_LARGE_MASK | (paddr & PHYSICAL_PAGE_MASK); *pte |= PTE_LARGE_MASK | (paddr & PHYSICAL_PAGE_MASK);
else else
*pte |= vm_alloc_page_table(vm) & PHYSICAL_PAGE_MASK; *pte |= vm_alloc_page_table(vm) & PHYSICAL_PAGE_MASK;
...@@ -184,20 +184,19 @@ static uint64_t *virt_create_upper_pte(struct kvm_vm *vm, ...@@ -184,20 +184,19 @@ static uint64_t *virt_create_upper_pte(struct kvm_vm *vm,
* a hugepage at this level, and that there isn't a hugepage at * a hugepage at this level, and that there isn't a hugepage at
* this level. * this level.
*/ */
TEST_ASSERT(level != page_size, TEST_ASSERT(current_level != target_level,
"Cannot create hugepage at level: %u, vaddr: 0x%lx\n", "Cannot create hugepage at level: %u, vaddr: 0x%lx\n",
page_size, vaddr); current_level, vaddr);
TEST_ASSERT(!(*pte & PTE_LARGE_MASK), TEST_ASSERT(!(*pte & PTE_LARGE_MASK),
"Cannot create page table at level: %u, vaddr: 0x%lx\n", "Cannot create page table at level: %u, vaddr: 0x%lx\n",
level, vaddr); current_level, vaddr);
} }
return pte; return pte;
} }
void __virt_pg_map(struct kvm_vm *vm, uint64_t vaddr, uint64_t paddr, void __virt_pg_map(struct kvm_vm *vm, uint64_t vaddr, uint64_t paddr, int level)
enum x86_page_size page_size)
{ {
const uint64_t pg_size = 1ull << ((page_size * 9) + 12); const uint64_t pg_size = PG_LEVEL_SIZE(level);
uint64_t *pml4e, *pdpe, *pde; uint64_t *pml4e, *pdpe, *pde;
uint64_t *pte; uint64_t *pte;
...@@ -222,20 +221,20 @@ void __virt_pg_map(struct kvm_vm *vm, uint64_t vaddr, uint64_t paddr, ...@@ -222,20 +221,20 @@ void __virt_pg_map(struct kvm_vm *vm, uint64_t vaddr, uint64_t paddr,
* early if a hugepage was created. * early if a hugepage was created.
*/ */
pml4e = virt_create_upper_pte(vm, vm->pgd >> vm->page_shift, pml4e = virt_create_upper_pte(vm, vm->pgd >> vm->page_shift,
vaddr, paddr, 3, page_size); vaddr, paddr, PG_LEVEL_512G, level);
if (*pml4e & PTE_LARGE_MASK) if (*pml4e & PTE_LARGE_MASK)
return; return;
pdpe = virt_create_upper_pte(vm, PTE_GET_PFN(*pml4e), vaddr, paddr, 2, page_size); pdpe = virt_create_upper_pte(vm, PTE_GET_PFN(*pml4e), vaddr, paddr, PG_LEVEL_1G, level);
if (*pdpe & PTE_LARGE_MASK) if (*pdpe & PTE_LARGE_MASK)
return; return;
pde = virt_create_upper_pte(vm, PTE_GET_PFN(*pdpe), vaddr, paddr, 1, page_size); pde = virt_create_upper_pte(vm, PTE_GET_PFN(*pdpe), vaddr, paddr, PG_LEVEL_2M, level);
if (*pde & PTE_LARGE_MASK) if (*pde & PTE_LARGE_MASK)
return; return;
/* Fill in page table entry. */ /* Fill in page table entry. */
pte = virt_get_pte(vm, PTE_GET_PFN(*pde), vaddr, 0); pte = virt_get_pte(vm, PTE_GET_PFN(*pde), vaddr, PG_LEVEL_4K);
TEST_ASSERT(!(*pte & PTE_PRESENT_MASK), TEST_ASSERT(!(*pte & PTE_PRESENT_MASK),
"PTE already present for 4k page at vaddr: 0x%lx\n", vaddr); "PTE already present for 4k page at vaddr: 0x%lx\n", vaddr);
*pte = PTE_PRESENT_MASK | PTE_WRITABLE_MASK | (paddr & PHYSICAL_PAGE_MASK); *pte = PTE_PRESENT_MASK | PTE_WRITABLE_MASK | (paddr & PHYSICAL_PAGE_MASK);
...@@ -243,7 +242,7 @@ void __virt_pg_map(struct kvm_vm *vm, uint64_t vaddr, uint64_t paddr, ...@@ -243,7 +242,7 @@ void __virt_pg_map(struct kvm_vm *vm, uint64_t vaddr, uint64_t paddr,
void virt_pg_map(struct kvm_vm *vm, uint64_t vaddr, uint64_t paddr) void virt_pg_map(struct kvm_vm *vm, uint64_t vaddr, uint64_t paddr)
{ {
__virt_pg_map(vm, vaddr, paddr, X86_PAGE_SIZE_4K); __virt_pg_map(vm, vaddr, paddr, PG_LEVEL_4K);
} }
static uint64_t *_vm_get_page_table_entry(struct kvm_vm *vm, int vcpuid, static uint64_t *_vm_get_page_table_entry(struct kvm_vm *vm, int vcpuid,
......
...@@ -244,7 +244,7 @@ int main(int argc, char *argv[]) ...@@ -244,7 +244,7 @@ int main(int argc, char *argv[])
#ifdef __x86_64__ #ifdef __x86_64__
/* Identity map memory in the guest using 1gb pages. */ /* Identity map memory in the guest using 1gb pages. */
for (i = 0; i < slot_size; i += size_1gb) for (i = 0; i < slot_size; i += size_1gb)
__virt_pg_map(vm, gpa + i, gpa + i, X86_PAGE_SIZE_1G); __virt_pg_map(vm, gpa + i, gpa + i, PG_LEVEL_1G);
#else #else
for (i = 0; i < slot_size; i += vm_get_page_size(vm)) for (i = 0; i < slot_size; i += vm_get_page_size(vm))
virt_pg_map(vm, gpa + i, gpa + i); virt_pg_map(vm, gpa + i, gpa + i);
......
...@@ -35,7 +35,7 @@ static void mmu_role_test(u32 *cpuid_reg, u32 evil_cpuid_val) ...@@ -35,7 +35,7 @@ static void mmu_role_test(u32 *cpuid_reg, u32 evil_cpuid_val)
run = vcpu_state(vm, VCPU_ID); run = vcpu_state(vm, VCPU_ID);
/* Map 1gb page without a backing memlot. */ /* Map 1gb page without a backing memlot. */
__virt_pg_map(vm, MMIO_GPA, MMIO_GPA, X86_PAGE_SIZE_1G); __virt_pg_map(vm, MMIO_GPA, MMIO_GPA, PG_LEVEL_1G);
r = _vcpu_run(vm, VCPU_ID); r = _vcpu_run(vm, VCPU_ID);
......
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