mirror of
https://github.com/AuxXxilium/linux_dsm_epyc7002.git
synced 2024-12-16 20:36:43 +07:00
KVM: PPC: Book3S HV: Radix changes for secure guest
- After the guest becomes secure, when we handle a page fault of a page belonging to SVM in HV, send that page to UV via UV_PAGE_IN. - Whenever a page is unmapped on the HV side, inform UV via UV_PAGE_INVAL. - Ensure all those routines that walk the secondary page tables of the guest don't do so in case of secure VM. For secure guest, the active secondary page tables are in secure memory and the secondary page tables in HV are freed when guest becomes secure. Signed-off-by: Bharata B Rao <bharata@linux.ibm.com> Signed-off-by: Paul Mackerras <paulus@ozlabs.org>
This commit is contained in:
parent
60f0a643aa
commit
008e359c76
@ -18,6 +18,7 @@ unsigned long kvmppc_h_svm_page_out(struct kvm *kvm,
|
|||||||
unsigned long page_shift);
|
unsigned long page_shift);
|
||||||
unsigned long kvmppc_h_svm_init_start(struct kvm *kvm);
|
unsigned long kvmppc_h_svm_init_start(struct kvm *kvm);
|
||||||
unsigned long kvmppc_h_svm_init_done(struct kvm *kvm);
|
unsigned long kvmppc_h_svm_init_done(struct kvm *kvm);
|
||||||
|
int kvmppc_send_page_to_uv(struct kvm *kvm, unsigned long gfn);
|
||||||
#else
|
#else
|
||||||
static inline int kvmppc_uvmem_init(void)
|
static inline int kvmppc_uvmem_init(void)
|
||||||
{
|
{
|
||||||
@ -58,5 +59,10 @@ static inline unsigned long kvmppc_h_svm_init_done(struct kvm *kvm)
|
|||||||
{
|
{
|
||||||
return H_UNSUPPORTED;
|
return H_UNSUPPORTED;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static inline int kvmppc_send_page_to_uv(struct kvm *kvm, unsigned long gfn)
|
||||||
|
{
|
||||||
|
return -EFAULT;
|
||||||
|
}
|
||||||
#endif /* CONFIG_PPC_UV */
|
#endif /* CONFIG_PPC_UV */
|
||||||
#endif /* __ASM_KVM_BOOK3S_UVMEM_H__ */
|
#endif /* __ASM_KVM_BOOK3S_UVMEM_H__ */
|
||||||
|
@ -32,5 +32,6 @@
|
|||||||
#define UV_SHARE_PAGE 0xF130
|
#define UV_SHARE_PAGE 0xF130
|
||||||
#define UV_UNSHARE_PAGE 0xF134
|
#define UV_UNSHARE_PAGE 0xF134
|
||||||
#define UV_UNSHARE_ALL_PAGES 0xF140
|
#define UV_UNSHARE_ALL_PAGES 0xF140
|
||||||
|
#define UV_PAGE_INVAL 0xF138
|
||||||
|
|
||||||
#endif /* _ASM_POWERPC_ULTRAVISOR_API_H */
|
#endif /* _ASM_POWERPC_ULTRAVISOR_API_H */
|
||||||
|
@ -67,4 +67,9 @@ static inline int uv_register_mem_slot(u64 lpid, u64 start_gpa, u64 size,
|
|||||||
size, flags, slotid);
|
size, flags, slotid);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static inline int uv_page_inval(u64 lpid, u64 gpa, u64 page_shift)
|
||||||
|
{
|
||||||
|
return ucall_norets(UV_PAGE_INVAL, lpid, gpa, page_shift);
|
||||||
|
}
|
||||||
|
|
||||||
#endif /* _ASM_POWERPC_ULTRAVISOR_H */
|
#endif /* _ASM_POWERPC_ULTRAVISOR_H */
|
||||||
|
@ -19,6 +19,8 @@
|
|||||||
#include <asm/pgtable.h>
|
#include <asm/pgtable.h>
|
||||||
#include <asm/pgalloc.h>
|
#include <asm/pgalloc.h>
|
||||||
#include <asm/pte-walk.h>
|
#include <asm/pte-walk.h>
|
||||||
|
#include <asm/ultravisor.h>
|
||||||
|
#include <asm/kvm_book3s_uvmem.h>
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Supported radix tree geometry.
|
* Supported radix tree geometry.
|
||||||
@ -915,6 +917,9 @@ int kvmppc_book3s_radix_page_fault(struct kvm_run *run, struct kvm_vcpu *vcpu,
|
|||||||
if (!(dsisr & DSISR_PRTABLE_FAULT))
|
if (!(dsisr & DSISR_PRTABLE_FAULT))
|
||||||
gpa |= ea & 0xfff;
|
gpa |= ea & 0xfff;
|
||||||
|
|
||||||
|
if (kvm->arch.secure_guest & KVMPPC_SECURE_INIT_DONE)
|
||||||
|
return kvmppc_send_page_to_uv(kvm, gfn);
|
||||||
|
|
||||||
/* Get the corresponding memslot */
|
/* Get the corresponding memslot */
|
||||||
memslot = gfn_to_memslot(kvm, gfn);
|
memslot = gfn_to_memslot(kvm, gfn);
|
||||||
|
|
||||||
@ -972,6 +977,11 @@ int kvm_unmap_radix(struct kvm *kvm, struct kvm_memory_slot *memslot,
|
|||||||
unsigned long gpa = gfn << PAGE_SHIFT;
|
unsigned long gpa = gfn << PAGE_SHIFT;
|
||||||
unsigned int shift;
|
unsigned int shift;
|
||||||
|
|
||||||
|
if (kvm->arch.secure_guest & KVMPPC_SECURE_INIT_DONE) {
|
||||||
|
uv_page_inval(kvm->arch.lpid, gpa, PAGE_SHIFT);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
ptep = __find_linux_pte(kvm->arch.pgtable, gpa, NULL, &shift);
|
ptep = __find_linux_pte(kvm->arch.pgtable, gpa, NULL, &shift);
|
||||||
if (ptep && pte_present(*ptep))
|
if (ptep && pte_present(*ptep))
|
||||||
kvmppc_unmap_pte(kvm, ptep, gpa, shift, memslot,
|
kvmppc_unmap_pte(kvm, ptep, gpa, shift, memslot,
|
||||||
@ -989,6 +999,9 @@ int kvm_age_radix(struct kvm *kvm, struct kvm_memory_slot *memslot,
|
|||||||
int ref = 0;
|
int ref = 0;
|
||||||
unsigned long old, *rmapp;
|
unsigned long old, *rmapp;
|
||||||
|
|
||||||
|
if (kvm->arch.secure_guest & KVMPPC_SECURE_INIT_DONE)
|
||||||
|
return ref;
|
||||||
|
|
||||||
ptep = __find_linux_pte(kvm->arch.pgtable, gpa, NULL, &shift);
|
ptep = __find_linux_pte(kvm->arch.pgtable, gpa, NULL, &shift);
|
||||||
if (ptep && pte_present(*ptep) && pte_young(*ptep)) {
|
if (ptep && pte_present(*ptep) && pte_young(*ptep)) {
|
||||||
old = kvmppc_radix_update_pte(kvm, ptep, _PAGE_ACCESSED, 0,
|
old = kvmppc_radix_update_pte(kvm, ptep, _PAGE_ACCESSED, 0,
|
||||||
@ -1013,6 +1026,9 @@ int kvm_test_age_radix(struct kvm *kvm, struct kvm_memory_slot *memslot,
|
|||||||
unsigned int shift;
|
unsigned int shift;
|
||||||
int ref = 0;
|
int ref = 0;
|
||||||
|
|
||||||
|
if (kvm->arch.secure_guest & KVMPPC_SECURE_INIT_DONE)
|
||||||
|
return ref;
|
||||||
|
|
||||||
ptep = __find_linux_pte(kvm->arch.pgtable, gpa, NULL, &shift);
|
ptep = __find_linux_pte(kvm->arch.pgtable, gpa, NULL, &shift);
|
||||||
if (ptep && pte_present(*ptep) && pte_young(*ptep))
|
if (ptep && pte_present(*ptep) && pte_young(*ptep))
|
||||||
ref = 1;
|
ref = 1;
|
||||||
@ -1030,6 +1046,9 @@ static int kvm_radix_test_clear_dirty(struct kvm *kvm,
|
|||||||
int ret = 0;
|
int ret = 0;
|
||||||
unsigned long old, *rmapp;
|
unsigned long old, *rmapp;
|
||||||
|
|
||||||
|
if (kvm->arch.secure_guest & KVMPPC_SECURE_INIT_DONE)
|
||||||
|
return ret;
|
||||||
|
|
||||||
ptep = __find_linux_pte(kvm->arch.pgtable, gpa, NULL, &shift);
|
ptep = __find_linux_pte(kvm->arch.pgtable, gpa, NULL, &shift);
|
||||||
if (ptep && pte_present(*ptep) && pte_dirty(*ptep)) {
|
if (ptep && pte_present(*ptep) && pte_dirty(*ptep)) {
|
||||||
ret = 1;
|
ret = 1;
|
||||||
@ -1082,6 +1101,9 @@ void kvmppc_radix_flush_memslot(struct kvm *kvm,
|
|||||||
unsigned long gpa;
|
unsigned long gpa;
|
||||||
unsigned int shift;
|
unsigned int shift;
|
||||||
|
|
||||||
|
if (kvm->arch.secure_guest & KVMPPC_SECURE_INIT_DONE)
|
||||||
|
return;
|
||||||
|
|
||||||
gpa = memslot->base_gfn << PAGE_SHIFT;
|
gpa = memslot->base_gfn << PAGE_SHIFT;
|
||||||
spin_lock(&kvm->mmu_lock);
|
spin_lock(&kvm->mmu_lock);
|
||||||
for (n = memslot->npages; n; --n) {
|
for (n = memslot->npages; n; --n) {
|
||||||
|
@ -69,6 +69,17 @@
|
|||||||
* Shared pages: Whenever guest shares a secure page, UV will split and
|
* Shared pages: Whenever guest shares a secure page, UV will split and
|
||||||
* remap the 2MB page if required and issue H_SVM_PAGE_IN with 64K page size.
|
* remap the 2MB page if required and issue H_SVM_PAGE_IN with 64K page size.
|
||||||
*
|
*
|
||||||
|
* HV invalidating a page: When a regular page belonging to secure
|
||||||
|
* guest gets unmapped, HV informs UV with UV_PAGE_INVAL of 64K
|
||||||
|
* page size. Using 64K page size is correct here because any non-secure
|
||||||
|
* page will essentially be of 64K page size. Splitting by UV during sharing
|
||||||
|
* and page-out ensures this.
|
||||||
|
*
|
||||||
|
* Page fault handling: When HV handles page fault of a page belonging
|
||||||
|
* to secure guest, it sends that to UV with a 64K UV_PAGE_IN request.
|
||||||
|
* Using 64K size is correct here too as UV would have split the 2MB page
|
||||||
|
* into 64k mappings and would have done page-outs earlier.
|
||||||
|
*
|
||||||
* In summary, the current secure pages handling code in HV assumes
|
* In summary, the current secure pages handling code in HV assumes
|
||||||
* 64K page size and in fact fails any page-in/page-out requests of
|
* 64K page size and in fact fails any page-in/page-out requests of
|
||||||
* non-64K size upfront. If and when UV starts supporting multiple
|
* non-64K size upfront. If and when UV starts supporting multiple
|
||||||
@ -630,6 +641,27 @@ kvmppc_h_svm_page_out(struct kvm *kvm, unsigned long gpa,
|
|||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int kvmppc_send_page_to_uv(struct kvm *kvm, unsigned long gfn)
|
||||||
|
{
|
||||||
|
unsigned long pfn;
|
||||||
|
int ret = U_SUCCESS;
|
||||||
|
|
||||||
|
pfn = gfn_to_pfn(kvm, gfn);
|
||||||
|
if (is_error_noslot_pfn(pfn))
|
||||||
|
return -EFAULT;
|
||||||
|
|
||||||
|
mutex_lock(&kvm->arch.uvmem_lock);
|
||||||
|
if (kvmppc_gfn_is_uvmem_pfn(gfn, kvm, NULL))
|
||||||
|
goto out;
|
||||||
|
|
||||||
|
ret = uv_page_in(kvm->arch.lpid, pfn << PAGE_SHIFT, gfn << PAGE_SHIFT,
|
||||||
|
0, PAGE_SHIFT);
|
||||||
|
out:
|
||||||
|
kvm_release_pfn_clean(pfn);
|
||||||
|
mutex_unlock(&kvm->arch.uvmem_lock);
|
||||||
|
return (ret == U_SUCCESS) ? RESUME_GUEST : -EFAULT;
|
||||||
|
}
|
||||||
|
|
||||||
static u64 kvmppc_get_secmem_size(void)
|
static u64 kvmppc_get_secmem_size(void)
|
||||||
{
|
{
|
||||||
struct device_node *np;
|
struct device_node *np;
|
||||||
|
Loading…
Reference in New Issue
Block a user