Skip to content

Commit 4ee602e

Browse files
dmatlackbonzini
authored andcommitted
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: David Matlack <dmatlack@google.com> Message-Id: <20220520233249.3776001-2-dmatlack@google.com> Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
1 parent e3cdaab commit 4ee602e

4 files changed

Lines changed: 29 additions & 24 deletions

File tree

tools/testing/selftests/kvm/include/x86_64/processor.h

Lines changed: 12 additions & 6 deletions
Original file line numberDiff line numberDiff line change
@@ -482,13 +482,19 @@ void vcpu_set_hv_cpuid(struct kvm_vm *vm, uint32_t vcpuid);
482482
struct kvm_cpuid2 *vcpu_get_supported_hv_cpuid(struct kvm_vm *vm, uint32_t vcpuid);
483483
void vm_xsave_req_perm(int bit);
484484

485-
enum x86_page_size {
486-
X86_PAGE_SIZE_4K = 0,
487-
X86_PAGE_SIZE_2M,
488-
X86_PAGE_SIZE_1G,
485+
enum pg_level {
486+
PG_LEVEL_NONE,
487+
PG_LEVEL_4K,
488+
PG_LEVEL_2M,
489+
PG_LEVEL_1G,
490+
PG_LEVEL_512G,
491+
PG_LEVEL_NUM
489492
};
490-
void __virt_pg_map(struct kvm_vm *vm, uint64_t vaddr, uint64_t paddr,
491-
enum x86_page_size page_size);
493+
494+
#define PG_LEVEL_SHIFT(_level) ((_level - 1) * 9 + 12)
495+
#define PG_LEVEL_SIZE(_level) (1ull << PG_LEVEL_SHIFT(_level))
496+
497+
void __virt_pg_map(struct kvm_vm *vm, uint64_t vaddr, uint64_t paddr, int level);
492498

493499
/*
494500
* Basic CPU control in CR0

tools/testing/selftests/kvm/lib/x86_64/processor.c

Lines changed: 15 additions & 16 deletions
Original file line numberDiff line numberDiff line change
@@ -158,7 +158,7 @@ static void *virt_get_pte(struct kvm_vm *vm, uint64_t pt_pfn, uint64_t vaddr,
158158
int level)
159159
{
160160
uint64_t *page_table = addr_gpa2hva(vm, pt_pfn << vm->page_shift);
161-
int index = vaddr >> (vm->page_shift + level * 9) & 0x1ffu;
161+
int index = (vaddr >> PG_LEVEL_SHIFT(level)) & 0x1ffu;
162162

163163
return &page_table[index];
164164
}
@@ -167,14 +167,14 @@ static uint64_t *virt_create_upper_pte(struct kvm_vm *vm,
167167
uint64_t pt_pfn,
168168
uint64_t vaddr,
169169
uint64_t paddr,
170-
int level,
171-
enum x86_page_size page_size)
170+
int current_level,
171+
int target_level)
172172
{
173-
uint64_t *pte = virt_get_pte(vm, pt_pfn, vaddr, level);
173+
uint64_t *pte = virt_get_pte(vm, pt_pfn, vaddr, current_level);
174174

175175
if (!(*pte & PTE_PRESENT_MASK)) {
176176
*pte = PTE_PRESENT_MASK | PTE_WRITABLE_MASK;
177-
if (level == page_size)
177+
if (current_level == target_level)
178178
*pte |= PTE_LARGE_MASK | (paddr & PHYSICAL_PAGE_MASK);
179179
else
180180
*pte |= vm_alloc_page_table(vm) & PHYSICAL_PAGE_MASK;
@@ -184,20 +184,19 @@ static uint64_t *virt_create_upper_pte(struct kvm_vm *vm,
184184
* a hugepage at this level, and that there isn't a hugepage at
185185
* this level.
186186
*/
187-
TEST_ASSERT(level != page_size,
187+
TEST_ASSERT(current_level != target_level,
188188
"Cannot create hugepage at level: %u, vaddr: 0x%lx\n",
189-
page_size, vaddr);
189+
current_level, vaddr);
190190
TEST_ASSERT(!(*pte & PTE_LARGE_MASK),
191191
"Cannot create page table at level: %u, vaddr: 0x%lx\n",
192-
level, vaddr);
192+
current_level, vaddr);
193193
}
194194
return pte;
195195
}
196196

197-
void __virt_pg_map(struct kvm_vm *vm, uint64_t vaddr, uint64_t paddr,
198-
enum x86_page_size page_size)
197+
void __virt_pg_map(struct kvm_vm *vm, uint64_t vaddr, uint64_t paddr, int level)
199198
{
200-
const uint64_t pg_size = 1ull << ((page_size * 9) + 12);
199+
const uint64_t pg_size = PG_LEVEL_SIZE(level);
201200
uint64_t *pml4e, *pdpe, *pde;
202201
uint64_t *pte;
203202

@@ -222,28 +221,28 @@ void __virt_pg_map(struct kvm_vm *vm, uint64_t vaddr, uint64_t paddr,
222221
* early if a hugepage was created.
223222
*/
224223
pml4e = virt_create_upper_pte(vm, vm->pgd >> vm->page_shift,
225-
vaddr, paddr, 3, page_size);
224+
vaddr, paddr, PG_LEVEL_512G, level);
226225
if (*pml4e & PTE_LARGE_MASK)
227226
return;
228227

229-
pdpe = virt_create_upper_pte(vm, PTE_GET_PFN(*pml4e), vaddr, paddr, 2, page_size);
228+
pdpe = virt_create_upper_pte(vm, PTE_GET_PFN(*pml4e), vaddr, paddr, PG_LEVEL_1G, level);
230229
if (*pdpe & PTE_LARGE_MASK)
231230
return;
232231

233-
pde = virt_create_upper_pte(vm, PTE_GET_PFN(*pdpe), vaddr, paddr, 1, page_size);
232+
pde = virt_create_upper_pte(vm, PTE_GET_PFN(*pdpe), vaddr, paddr, PG_LEVEL_2M, level);
234233
if (*pde & PTE_LARGE_MASK)
235234
return;
236235

237236
/* Fill in page table entry. */
238-
pte = virt_get_pte(vm, PTE_GET_PFN(*pde), vaddr, 0);
237+
pte = virt_get_pte(vm, PTE_GET_PFN(*pde), vaddr, PG_LEVEL_4K);
239238
TEST_ASSERT(!(*pte & PTE_PRESENT_MASK),
240239
"PTE already present for 4k page at vaddr: 0x%lx\n", vaddr);
241240
*pte = PTE_PRESENT_MASK | PTE_WRITABLE_MASK | (paddr & PHYSICAL_PAGE_MASK);
242241
}
243242

244243
void virt_pg_map(struct kvm_vm *vm, uint64_t vaddr, uint64_t paddr)
245244
{
246-
__virt_pg_map(vm, vaddr, paddr, X86_PAGE_SIZE_4K);
245+
__virt_pg_map(vm, vaddr, paddr, PG_LEVEL_4K);
247246
}
248247

249248
static uint64_t *_vm_get_page_table_entry(struct kvm_vm *vm, int vcpuid,

tools/testing/selftests/kvm/max_guest_memory_test.c

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -244,7 +244,7 @@ int main(int argc, char *argv[])
244244
#ifdef __x86_64__
245245
/* Identity map memory in the guest using 1gb pages. */
246246
for (i = 0; i < slot_size; i += size_1gb)
247-
__virt_pg_map(vm, gpa + i, gpa + i, X86_PAGE_SIZE_1G);
247+
__virt_pg_map(vm, gpa + i, gpa + i, PG_LEVEL_1G);
248248
#else
249249
for (i = 0; i < slot_size; i += vm_get_page_size(vm))
250250
virt_pg_map(vm, gpa + i, gpa + i);

tools/testing/selftests/kvm/x86_64/mmu_role_test.c

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -35,7 +35,7 @@ static void mmu_role_test(u32 *cpuid_reg, u32 evil_cpuid_val)
3535
run = vcpu_state(vm, VCPU_ID);
3636

3737
/* Map 1gb page without a backing memlot. */
38-
__virt_pg_map(vm, MMIO_GPA, MMIO_GPA, X86_PAGE_SIZE_1G);
38+
__virt_pg_map(vm, MMIO_GPA, MMIO_GPA, PG_LEVEL_1G);
3939

4040
r = _vcpu_run(vm, VCPU_ID);
4141

0 commit comments

Comments
 (0)