KVM: PPC: Book3S HV: Handle non-present PTEs in page fault functions
Sincecd758a9b57
"KVM: PPC: Book3S HV: Use __gfn_to_pfn_memslot in HPT page fault handler", it's been possible in fairly rare circumstances to load a non-present PTE in kvmppc_book3s_hv_page_fault() when running a guest on a POWER8 host. Because that case wasn't checked for, we could misinterpret the non-present PTE as being a cache-inhibited PTE. That could mismatch with the corresponding hash PTE, which would cause the function to fail with -EFAULT a little further down. That would propagate up to the KVM_RUN ioctl() generally causing the KVM userspace (usually qemu) to fall over. This addresses the problem by catching that case and returning to the guest instead. For completeness, this fixes the radix page fault handler in the same way. For radix this didn't cause any obvious misbehaviour, because we ended up putting the non-present PTE into the guest's partition-scoped page tables, leading immediately to another hypervisor data/instruction storage interrupt, which would go through the page fault path again and fix things up. Fixes:cd758a9b57
"KVM: PPC: Book3S HV: Use __gfn_to_pfn_memslot in HPT page fault handler" Bugzilla: https://bugzilla.redhat.com/show_bug.cgi?id=1820402 Reported-by: David Gibson <david@gibson.dropbear.id.au> Tested-by: David Gibson <david@gibson.dropbear.id.au> Signed-off-by: Paul Mackerras <paulus@ozlabs.org>
This commit is contained in:
parent
dbef2808af
commit
ae49dedaa9
|
@ -604,18 +604,19 @@ int kvmppc_book3s_hv_page_fault(struct kvm_run *run, struct kvm_vcpu *vcpu,
|
||||||
*/
|
*/
|
||||||
local_irq_disable();
|
local_irq_disable();
|
||||||
ptep = __find_linux_pte(vcpu->arch.pgdir, hva, NULL, &shift);
|
ptep = __find_linux_pte(vcpu->arch.pgdir, hva, NULL, &shift);
|
||||||
|
pte = __pte(0);
|
||||||
|
if (ptep)
|
||||||
|
pte = *ptep;
|
||||||
|
local_irq_enable();
|
||||||
/*
|
/*
|
||||||
* If the PTE disappeared temporarily due to a THP
|
* If the PTE disappeared temporarily due to a THP
|
||||||
* collapse, just return and let the guest try again.
|
* collapse, just return and let the guest try again.
|
||||||
*/
|
*/
|
||||||
if (!ptep) {
|
if (!pte_present(pte)) {
|
||||||
local_irq_enable();
|
|
||||||
if (page)
|
if (page)
|
||||||
put_page(page);
|
put_page(page);
|
||||||
return RESUME_GUEST;
|
return RESUME_GUEST;
|
||||||
}
|
}
|
||||||
pte = *ptep;
|
|
||||||
local_irq_enable();
|
|
||||||
hpa = pte_pfn(pte) << PAGE_SHIFT;
|
hpa = pte_pfn(pte) << PAGE_SHIFT;
|
||||||
pte_size = PAGE_SIZE;
|
pte_size = PAGE_SIZE;
|
||||||
if (shift)
|
if (shift)
|
||||||
|
|
|
@ -815,18 +815,19 @@ int kvmppc_book3s_instantiate_page(struct kvm_vcpu *vcpu,
|
||||||
*/
|
*/
|
||||||
local_irq_disable();
|
local_irq_disable();
|
||||||
ptep = __find_linux_pte(vcpu->arch.pgdir, hva, NULL, &shift);
|
ptep = __find_linux_pte(vcpu->arch.pgdir, hva, NULL, &shift);
|
||||||
|
pte = __pte(0);
|
||||||
|
if (ptep)
|
||||||
|
pte = *ptep;
|
||||||
|
local_irq_enable();
|
||||||
/*
|
/*
|
||||||
* If the PTE disappeared temporarily due to a THP
|
* If the PTE disappeared temporarily due to a THP
|
||||||
* collapse, just return and let the guest try again.
|
* collapse, just return and let the guest try again.
|
||||||
*/
|
*/
|
||||||
if (!ptep) {
|
if (!pte_present(pte)) {
|
||||||
local_irq_enable();
|
|
||||||
if (page)
|
if (page)
|
||||||
put_page(page);
|
put_page(page);
|
||||||
return RESUME_GUEST;
|
return RESUME_GUEST;
|
||||||
}
|
}
|
||||||
pte = *ptep;
|
|
||||||
local_irq_enable();
|
|
||||||
|
|
||||||
/* If we're logging dirty pages, always map single pages */
|
/* If we're logging dirty pages, always map single pages */
|
||||||
large_enable = !(memslot->flags & KVM_MEM_LOG_DIRTY_PAGES);
|
large_enable = !(memslot->flags & KVM_MEM_LOG_DIRTY_PAGES);
|
||||||
|
|
Loading…
Reference in New Issue