Skip to content

Commit e659bab

Browse files
ryanhrobMarc Zyngier
authored andcommitted
KVM: selftests: arm64: Fix pte encode/decode for PA bits > 48
The high bits [51:48] of a physical address should appear at [15:12] in a 64K pte, not at [51:48] as was previously being programmed. Fix this with new helper functions that do the conversion correctly. This also sets us up nicely for adding LPA2 encodings in future. Fixes: 7a6629e ("kvm: selftests: add virt mem support for aarch64") Signed-off-by: Ryan Roberts <ryan.roberts@arm.com> Reviewed-by: Oliver Upton <oliver.upton@linux.dev> Signed-off-by: Marc Zyngier <maz@kernel.org> Link: https://lore.kernel.org/r/20230308110948.1820163-3-ryan.roberts@arm.com
1 parent a2bed39 commit e659bab

1 file changed

Lines changed: 24 additions & 8 deletions

File tree

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

Lines changed: 24 additions & 8 deletions
Original file line numberDiff line numberDiff line change
@@ -58,10 +58,27 @@ static uint64_t pte_index(struct kvm_vm *vm, vm_vaddr_t gva)
5858
return (gva >> vm->page_shift) & mask;
5959
}
6060

61-
static uint64_t pte_addr(struct kvm_vm *vm, uint64_t entry)
61+
static uint64_t addr_pte(struct kvm_vm *vm, uint64_t pa, uint64_t attrs)
6262
{
63-
uint64_t mask = ((1UL << (vm->va_bits - vm->page_shift)) - 1) << vm->page_shift;
64-
return entry & mask;
63+
uint64_t pte;
64+
65+
pte = pa & GENMASK(47, vm->page_shift);
66+
if (vm->page_shift == 16)
67+
pte |= FIELD_GET(GENMASK(51, 48), pa) << 12;
68+
pte |= attrs;
69+
70+
return pte;
71+
}
72+
73+
static uint64_t pte_addr(struct kvm_vm *vm, uint64_t pte)
74+
{
75+
uint64_t pa;
76+
77+
pa = pte & GENMASK(47, vm->page_shift);
78+
if (vm->page_shift == 16)
79+
pa |= FIELD_GET(GENMASK(15, 12), pte) << 48;
80+
81+
return pa;
6582
}
6683

6784
static uint64_t ptrs_per_pgd(struct kvm_vm *vm)
@@ -110,18 +127,18 @@ static void _virt_pg_map(struct kvm_vm *vm, uint64_t vaddr, uint64_t paddr,
110127

111128
ptep = addr_gpa2hva(vm, vm->pgd) + pgd_index(vm, vaddr) * 8;
112129
if (!*ptep)
113-
*ptep = vm_alloc_page_table(vm) | 3;
130+
*ptep = addr_pte(vm, vm_alloc_page_table(vm), 3);
114131

115132
switch (vm->pgtable_levels) {
116133
case 4:
117134
ptep = addr_gpa2hva(vm, pte_addr(vm, *ptep)) + pud_index(vm, vaddr) * 8;
118135
if (!*ptep)
119-
*ptep = vm_alloc_page_table(vm) | 3;
136+
*ptep = addr_pte(vm, vm_alloc_page_table(vm), 3);
120137
/* fall through */
121138
case 3:
122139
ptep = addr_gpa2hva(vm, pte_addr(vm, *ptep)) + pmd_index(vm, vaddr) * 8;
123140
if (!*ptep)
124-
*ptep = vm_alloc_page_table(vm) | 3;
141+
*ptep = addr_pte(vm, vm_alloc_page_table(vm), 3);
125142
/* fall through */
126143
case 2:
127144
ptep = addr_gpa2hva(vm, pte_addr(vm, *ptep)) + pte_index(vm, vaddr) * 8;
@@ -130,8 +147,7 @@ static void _virt_pg_map(struct kvm_vm *vm, uint64_t vaddr, uint64_t paddr,
130147
TEST_FAIL("Page table levels must be 2, 3, or 4");
131148
}
132149

133-
*ptep = paddr | 3;
134-
*ptep |= (attr_idx << 2) | (1 << 10) /* Access Flag */;
150+
*ptep = addr_pte(vm, paddr, (attr_idx << 2) | (1 << 10) | 3); /* AF */
135151
}
136152

137153
void virt_arch_pg_map(struct kvm_vm *vm, uint64_t vaddr, uint64_t paddr)

0 commit comments

Comments
 (0)