mirror of
https://github.com/torvalds/linux.git
synced 2024-12-31 23:31:29 +00:00
KVM: x86: Use gpa_t for cr2/gpa to fix TDP support on 32-bit KVM
Convert a plethora of parameters and variables in the MMU and page fault flows from type gva_t to gpa_t to properly handle TDP on 32-bit KVM. Thanks to PSE and PAE paging, 32-bit kernels can access 64-bit physical addresses. When TDP is enabled, the fault address is a guest physical address and thus can be a 64-bit value, even when both KVM and its guest are using 32-bit virtual addressing, e.g. VMX's VMCS.GUEST_PHYSICAL is a 64-bit field, not a natural width field. Using a gva_t for the fault address means KVM will incorrectly drop the upper 32-bits of the GPA. Ditto for gva_to_gpa() when it is used to translate L2 GPAs to L1 GPAs. Opportunistically rename variables and parameters to better reflect the dual address modes, e.g. use "cr2_or_gpa" for fault addresses and plain "addr" instead of "vaddr" when the address may be either a GVA or an L2 GPA. Similarly, use "gpa" in the nonpaging_page_fault() flows to avoid a confusing "gpa_t gva" declaration; this also sets the stage for a future patch to combing nonpaging_page_fault() and tdp_page_fault() with minimal churn. Sprinkle in a few comments to document flows where an address is known to be a GVA and thus can be safely truncated to a 32-bit value. Add WARNs in kvm_handle_page_fault() and FNAME(gva_to_gpa_nested)() to help document such cases and detect bugs. Cc: stable@vger.kernel.org Signed-off-by: Sean Christopherson <sean.j.christopherson@intel.com> Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
This commit is contained in:
parent
95145c25a7
commit
736c291c9f
@ -378,12 +378,12 @@ struct kvm_mmu {
|
|||||||
void (*set_cr3)(struct kvm_vcpu *vcpu, unsigned long root);
|
void (*set_cr3)(struct kvm_vcpu *vcpu, unsigned long root);
|
||||||
unsigned long (*get_cr3)(struct kvm_vcpu *vcpu);
|
unsigned long (*get_cr3)(struct kvm_vcpu *vcpu);
|
||||||
u64 (*get_pdptr)(struct kvm_vcpu *vcpu, int index);
|
u64 (*get_pdptr)(struct kvm_vcpu *vcpu, int index);
|
||||||
int (*page_fault)(struct kvm_vcpu *vcpu, gva_t gva, u32 err,
|
int (*page_fault)(struct kvm_vcpu *vcpu, gpa_t cr2_or_gpa, u32 err,
|
||||||
bool prefault);
|
bool prefault);
|
||||||
void (*inject_page_fault)(struct kvm_vcpu *vcpu,
|
void (*inject_page_fault)(struct kvm_vcpu *vcpu,
|
||||||
struct x86_exception *fault);
|
struct x86_exception *fault);
|
||||||
gpa_t (*gva_to_gpa)(struct kvm_vcpu *vcpu, gva_t gva, u32 access,
|
gpa_t (*gva_to_gpa)(struct kvm_vcpu *vcpu, gpa_t gva_or_gpa,
|
||||||
struct x86_exception *exception);
|
u32 access, struct x86_exception *exception);
|
||||||
gpa_t (*translate_gpa)(struct kvm_vcpu *vcpu, gpa_t gpa, u32 access,
|
gpa_t (*translate_gpa)(struct kvm_vcpu *vcpu, gpa_t gpa, u32 access,
|
||||||
struct x86_exception *exception);
|
struct x86_exception *exception);
|
||||||
int (*sync_page)(struct kvm_vcpu *vcpu,
|
int (*sync_page)(struct kvm_vcpu *vcpu,
|
||||||
@ -1473,7 +1473,7 @@ void kvm_vcpu_deactivate_apicv(struct kvm_vcpu *vcpu);
|
|||||||
|
|
||||||
int kvm_emulate_hypercall(struct kvm_vcpu *vcpu);
|
int kvm_emulate_hypercall(struct kvm_vcpu *vcpu);
|
||||||
|
|
||||||
int kvm_mmu_page_fault(struct kvm_vcpu *vcpu, gva_t gva, u64 error_code,
|
int kvm_mmu_page_fault(struct kvm_vcpu *vcpu, gpa_t cr2_or_gpa, u64 error_code,
|
||||||
void *insn, int insn_len);
|
void *insn, int insn_len);
|
||||||
void kvm_mmu_invlpg(struct kvm_vcpu *vcpu, gva_t gva);
|
void kvm_mmu_invlpg(struct kvm_vcpu *vcpu, gva_t gva);
|
||||||
void kvm_mmu_invpcid_gva(struct kvm_vcpu *vcpu, gva_t gva, unsigned long pcid);
|
void kvm_mmu_invpcid_gva(struct kvm_vcpu *vcpu, gva_t gva, unsigned long pcid);
|
||||||
|
@ -3532,7 +3532,7 @@ static bool is_access_allowed(u32 fault_err_code, u64 spte)
|
|||||||
* - true: let the vcpu to access on the same address again.
|
* - true: let the vcpu to access on the same address again.
|
||||||
* - false: let the real page fault path to fix it.
|
* - false: let the real page fault path to fix it.
|
||||||
*/
|
*/
|
||||||
static bool fast_page_fault(struct kvm_vcpu *vcpu, gva_t gva, int level,
|
static bool fast_page_fault(struct kvm_vcpu *vcpu, gpa_t cr2_or_gpa, int level,
|
||||||
u32 error_code)
|
u32 error_code)
|
||||||
{
|
{
|
||||||
struct kvm_shadow_walk_iterator iterator;
|
struct kvm_shadow_walk_iterator iterator;
|
||||||
@ -3552,7 +3552,7 @@ static bool fast_page_fault(struct kvm_vcpu *vcpu, gva_t gva, int level,
|
|||||||
do {
|
do {
|
||||||
u64 new_spte;
|
u64 new_spte;
|
||||||
|
|
||||||
for_each_shadow_entry_lockless(vcpu, gva, iterator, spte)
|
for_each_shadow_entry_lockless(vcpu, cr2_or_gpa, iterator, spte)
|
||||||
if (!is_shadow_present_pte(spte) ||
|
if (!is_shadow_present_pte(spte) ||
|
||||||
iterator.level < level)
|
iterator.level < level)
|
||||||
break;
|
break;
|
||||||
@ -3630,7 +3630,7 @@ static bool fast_page_fault(struct kvm_vcpu *vcpu, gva_t gva, int level,
|
|||||||
|
|
||||||
} while (true);
|
} while (true);
|
||||||
|
|
||||||
trace_fast_page_fault(vcpu, gva, error_code, iterator.sptep,
|
trace_fast_page_fault(vcpu, cr2_or_gpa, error_code, iterator.sptep,
|
||||||
spte, fault_handled);
|
spte, fault_handled);
|
||||||
walk_shadow_page_lockless_end(vcpu);
|
walk_shadow_page_lockless_end(vcpu);
|
||||||
|
|
||||||
@ -3638,10 +3638,11 @@ static bool fast_page_fault(struct kvm_vcpu *vcpu, gva_t gva, int level,
|
|||||||
}
|
}
|
||||||
|
|
||||||
static bool try_async_pf(struct kvm_vcpu *vcpu, bool prefault, gfn_t gfn,
|
static bool try_async_pf(struct kvm_vcpu *vcpu, bool prefault, gfn_t gfn,
|
||||||
gva_t gva, kvm_pfn_t *pfn, bool write, bool *writable);
|
gpa_t cr2_or_gpa, kvm_pfn_t *pfn, bool write,
|
||||||
|
bool *writable);
|
||||||
static int make_mmu_pages_available(struct kvm_vcpu *vcpu);
|
static int make_mmu_pages_available(struct kvm_vcpu *vcpu);
|
||||||
|
|
||||||
static int nonpaging_map(struct kvm_vcpu *vcpu, gva_t v, u32 error_code,
|
static int nonpaging_map(struct kvm_vcpu *vcpu, gpa_t gpa, u32 error_code,
|
||||||
gfn_t gfn, bool prefault)
|
gfn_t gfn, bool prefault)
|
||||||
{
|
{
|
||||||
int r;
|
int r;
|
||||||
@ -3667,16 +3668,16 @@ static int nonpaging_map(struct kvm_vcpu *vcpu, gva_t v, u32 error_code,
|
|||||||
gfn &= ~(KVM_PAGES_PER_HPAGE(level) - 1);
|
gfn &= ~(KVM_PAGES_PER_HPAGE(level) - 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (fast_page_fault(vcpu, v, level, error_code))
|
if (fast_page_fault(vcpu, gpa, level, error_code))
|
||||||
return RET_PF_RETRY;
|
return RET_PF_RETRY;
|
||||||
|
|
||||||
mmu_seq = vcpu->kvm->mmu_notifier_seq;
|
mmu_seq = vcpu->kvm->mmu_notifier_seq;
|
||||||
smp_rmb();
|
smp_rmb();
|
||||||
|
|
||||||
if (try_async_pf(vcpu, prefault, gfn, v, &pfn, write, &map_writable))
|
if (try_async_pf(vcpu, prefault, gfn, gpa, &pfn, write, &map_writable))
|
||||||
return RET_PF_RETRY;
|
return RET_PF_RETRY;
|
||||||
|
|
||||||
if (handle_abnormal_pfn(vcpu, v, gfn, pfn, ACC_ALL, &r))
|
if (handle_abnormal_pfn(vcpu, gpa, gfn, pfn, ACC_ALL, &r))
|
||||||
return r;
|
return r;
|
||||||
|
|
||||||
r = RET_PF_RETRY;
|
r = RET_PF_RETRY;
|
||||||
@ -3687,7 +3688,7 @@ static int nonpaging_map(struct kvm_vcpu *vcpu, gva_t v, u32 error_code,
|
|||||||
goto out_unlock;
|
goto out_unlock;
|
||||||
if (likely(!force_pt_level))
|
if (likely(!force_pt_level))
|
||||||
transparent_hugepage_adjust(vcpu, gfn, &pfn, &level);
|
transparent_hugepage_adjust(vcpu, gfn, &pfn, &level);
|
||||||
r = __direct_map(vcpu, v, write, map_writable, level, pfn,
|
r = __direct_map(vcpu, gpa, write, map_writable, level, pfn,
|
||||||
prefault, false);
|
prefault, false);
|
||||||
out_unlock:
|
out_unlock:
|
||||||
spin_unlock(&vcpu->kvm->mmu_lock);
|
spin_unlock(&vcpu->kvm->mmu_lock);
|
||||||
@ -3985,7 +3986,7 @@ void kvm_mmu_sync_roots(struct kvm_vcpu *vcpu)
|
|||||||
}
|
}
|
||||||
EXPORT_SYMBOL_GPL(kvm_mmu_sync_roots);
|
EXPORT_SYMBOL_GPL(kvm_mmu_sync_roots);
|
||||||
|
|
||||||
static gpa_t nonpaging_gva_to_gpa(struct kvm_vcpu *vcpu, gva_t vaddr,
|
static gpa_t nonpaging_gva_to_gpa(struct kvm_vcpu *vcpu, gpa_t vaddr,
|
||||||
u32 access, struct x86_exception *exception)
|
u32 access, struct x86_exception *exception)
|
||||||
{
|
{
|
||||||
if (exception)
|
if (exception)
|
||||||
@ -3993,7 +3994,7 @@ static gpa_t nonpaging_gva_to_gpa(struct kvm_vcpu *vcpu, gva_t vaddr,
|
|||||||
return vaddr;
|
return vaddr;
|
||||||
}
|
}
|
||||||
|
|
||||||
static gpa_t nonpaging_gva_to_gpa_nested(struct kvm_vcpu *vcpu, gva_t vaddr,
|
static gpa_t nonpaging_gva_to_gpa_nested(struct kvm_vcpu *vcpu, gpa_t vaddr,
|
||||||
u32 access,
|
u32 access,
|
||||||
struct x86_exception *exception)
|
struct x86_exception *exception)
|
||||||
{
|
{
|
||||||
@ -4153,13 +4154,14 @@ static void shadow_page_table_clear_flood(struct kvm_vcpu *vcpu, gva_t addr)
|
|||||||
walk_shadow_page_lockless_end(vcpu);
|
walk_shadow_page_lockless_end(vcpu);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int nonpaging_page_fault(struct kvm_vcpu *vcpu, gva_t gva,
|
static int nonpaging_page_fault(struct kvm_vcpu *vcpu, gpa_t gpa,
|
||||||
u32 error_code, bool prefault)
|
u32 error_code, bool prefault)
|
||||||
{
|
{
|
||||||
gfn_t gfn = gva >> PAGE_SHIFT;
|
gfn_t gfn = gpa >> PAGE_SHIFT;
|
||||||
int r;
|
int r;
|
||||||
|
|
||||||
pgprintk("%s: gva %lx error %x\n", __func__, gva, error_code);
|
/* Note, paging is disabled, ergo gva == gpa. */
|
||||||
|
pgprintk("%s: gva %lx error %x\n", __func__, gpa, error_code);
|
||||||
|
|
||||||
if (page_fault_handle_page_track(vcpu, error_code, gfn))
|
if (page_fault_handle_page_track(vcpu, error_code, gfn))
|
||||||
return RET_PF_EMULATE;
|
return RET_PF_EMULATE;
|
||||||
@ -4171,11 +4173,12 @@ static int nonpaging_page_fault(struct kvm_vcpu *vcpu, gva_t gva,
|
|||||||
MMU_WARN_ON(!VALID_PAGE(vcpu->arch.mmu->root_hpa));
|
MMU_WARN_ON(!VALID_PAGE(vcpu->arch.mmu->root_hpa));
|
||||||
|
|
||||||
|
|
||||||
return nonpaging_map(vcpu, gva & PAGE_MASK,
|
return nonpaging_map(vcpu, gpa & PAGE_MASK,
|
||||||
error_code, gfn, prefault);
|
error_code, gfn, prefault);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int kvm_arch_setup_async_pf(struct kvm_vcpu *vcpu, gva_t gva, gfn_t gfn)
|
static int kvm_arch_setup_async_pf(struct kvm_vcpu *vcpu, gpa_t cr2_or_gpa,
|
||||||
|
gfn_t gfn)
|
||||||
{
|
{
|
||||||
struct kvm_arch_async_pf arch;
|
struct kvm_arch_async_pf arch;
|
||||||
|
|
||||||
@ -4184,11 +4187,13 @@ static int kvm_arch_setup_async_pf(struct kvm_vcpu *vcpu, gva_t gva, gfn_t gfn)
|
|||||||
arch.direct_map = vcpu->arch.mmu->direct_map;
|
arch.direct_map = vcpu->arch.mmu->direct_map;
|
||||||
arch.cr3 = vcpu->arch.mmu->get_cr3(vcpu);
|
arch.cr3 = vcpu->arch.mmu->get_cr3(vcpu);
|
||||||
|
|
||||||
return kvm_setup_async_pf(vcpu, gva, kvm_vcpu_gfn_to_hva(vcpu, gfn), &arch);
|
return kvm_setup_async_pf(vcpu, cr2_or_gpa,
|
||||||
|
kvm_vcpu_gfn_to_hva(vcpu, gfn), &arch);
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool try_async_pf(struct kvm_vcpu *vcpu, bool prefault, gfn_t gfn,
|
static bool try_async_pf(struct kvm_vcpu *vcpu, bool prefault, gfn_t gfn,
|
||||||
gva_t gva, kvm_pfn_t *pfn, bool write, bool *writable)
|
gpa_t cr2_or_gpa, kvm_pfn_t *pfn, bool write,
|
||||||
|
bool *writable)
|
||||||
{
|
{
|
||||||
struct kvm_memory_slot *slot;
|
struct kvm_memory_slot *slot;
|
||||||
bool async;
|
bool async;
|
||||||
@ -4208,12 +4213,12 @@ static bool try_async_pf(struct kvm_vcpu *vcpu, bool prefault, gfn_t gfn,
|
|||||||
return false; /* *pfn has correct page already */
|
return false; /* *pfn has correct page already */
|
||||||
|
|
||||||
if (!prefault && kvm_can_do_async_pf(vcpu)) {
|
if (!prefault && kvm_can_do_async_pf(vcpu)) {
|
||||||
trace_kvm_try_async_get_page(gva, gfn);
|
trace_kvm_try_async_get_page(cr2_or_gpa, gfn);
|
||||||
if (kvm_find_async_pf_gfn(vcpu, gfn)) {
|
if (kvm_find_async_pf_gfn(vcpu, gfn)) {
|
||||||
trace_kvm_async_pf_doublefault(gva, gfn);
|
trace_kvm_async_pf_doublefault(cr2_or_gpa, gfn);
|
||||||
kvm_make_request(KVM_REQ_APF_HALT, vcpu);
|
kvm_make_request(KVM_REQ_APF_HALT, vcpu);
|
||||||
return true;
|
return true;
|
||||||
} else if (kvm_arch_setup_async_pf(vcpu, gva, gfn))
|
} else if (kvm_arch_setup_async_pf(vcpu, cr2_or_gpa, gfn))
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -4226,6 +4231,12 @@ int kvm_handle_page_fault(struct kvm_vcpu *vcpu, u64 error_code,
|
|||||||
{
|
{
|
||||||
int r = 1;
|
int r = 1;
|
||||||
|
|
||||||
|
#ifndef CONFIG_X86_64
|
||||||
|
/* A 64-bit CR2 should be impossible on 32-bit KVM. */
|
||||||
|
if (WARN_ON_ONCE(fault_address >> 32))
|
||||||
|
return -EFAULT;
|
||||||
|
#endif
|
||||||
|
|
||||||
vcpu->arch.l1tf_flush_l1d = true;
|
vcpu->arch.l1tf_flush_l1d = true;
|
||||||
switch (vcpu->arch.apf.host_apf_reason) {
|
switch (vcpu->arch.apf.host_apf_reason) {
|
||||||
default:
|
default:
|
||||||
@ -4263,7 +4274,7 @@ check_hugepage_cache_consistency(struct kvm_vcpu *vcpu, gfn_t gfn, int level)
|
|||||||
return kvm_mtrr_check_gfn_range_consistency(vcpu, gfn, page_num);
|
return kvm_mtrr_check_gfn_range_consistency(vcpu, gfn, page_num);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int tdp_page_fault(struct kvm_vcpu *vcpu, gva_t gpa, u32 error_code,
|
static int tdp_page_fault(struct kvm_vcpu *vcpu, gpa_t gpa, u32 error_code,
|
||||||
bool prefault)
|
bool prefault)
|
||||||
{
|
{
|
||||||
kvm_pfn_t pfn;
|
kvm_pfn_t pfn;
|
||||||
@ -5520,7 +5531,7 @@ static int make_mmu_pages_available(struct kvm_vcpu *vcpu)
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
int kvm_mmu_page_fault(struct kvm_vcpu *vcpu, gva_t cr2, u64 error_code,
|
int kvm_mmu_page_fault(struct kvm_vcpu *vcpu, gpa_t cr2_or_gpa, u64 error_code,
|
||||||
void *insn, int insn_len)
|
void *insn, int insn_len)
|
||||||
{
|
{
|
||||||
int r, emulation_type = 0;
|
int r, emulation_type = 0;
|
||||||
@ -5529,18 +5540,18 @@ int kvm_mmu_page_fault(struct kvm_vcpu *vcpu, gva_t cr2, u64 error_code,
|
|||||||
/* With shadow page tables, fault_address contains a GVA or nGPA. */
|
/* With shadow page tables, fault_address contains a GVA or nGPA. */
|
||||||
if (vcpu->arch.mmu->direct_map) {
|
if (vcpu->arch.mmu->direct_map) {
|
||||||
vcpu->arch.gpa_available = true;
|
vcpu->arch.gpa_available = true;
|
||||||
vcpu->arch.gpa_val = cr2;
|
vcpu->arch.gpa_val = cr2_or_gpa;
|
||||||
}
|
}
|
||||||
|
|
||||||
r = RET_PF_INVALID;
|
r = RET_PF_INVALID;
|
||||||
if (unlikely(error_code & PFERR_RSVD_MASK)) {
|
if (unlikely(error_code & PFERR_RSVD_MASK)) {
|
||||||
r = handle_mmio_page_fault(vcpu, cr2, direct);
|
r = handle_mmio_page_fault(vcpu, cr2_or_gpa, direct);
|
||||||
if (r == RET_PF_EMULATE)
|
if (r == RET_PF_EMULATE)
|
||||||
goto emulate;
|
goto emulate;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (r == RET_PF_INVALID) {
|
if (r == RET_PF_INVALID) {
|
||||||
r = vcpu->arch.mmu->page_fault(vcpu, cr2,
|
r = vcpu->arch.mmu->page_fault(vcpu, cr2_or_gpa,
|
||||||
lower_32_bits(error_code),
|
lower_32_bits(error_code),
|
||||||
false);
|
false);
|
||||||
WARN_ON(r == RET_PF_INVALID);
|
WARN_ON(r == RET_PF_INVALID);
|
||||||
@ -5560,7 +5571,7 @@ int kvm_mmu_page_fault(struct kvm_vcpu *vcpu, gva_t cr2, u64 error_code,
|
|||||||
*/
|
*/
|
||||||
if (vcpu->arch.mmu->direct_map &&
|
if (vcpu->arch.mmu->direct_map &&
|
||||||
(error_code & PFERR_NESTED_GUEST_PAGE) == PFERR_NESTED_GUEST_PAGE) {
|
(error_code & PFERR_NESTED_GUEST_PAGE) == PFERR_NESTED_GUEST_PAGE) {
|
||||||
kvm_mmu_unprotect_page(vcpu->kvm, gpa_to_gfn(cr2));
|
kvm_mmu_unprotect_page(vcpu->kvm, gpa_to_gfn(cr2_or_gpa));
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -5575,7 +5586,7 @@ int kvm_mmu_page_fault(struct kvm_vcpu *vcpu, gva_t cr2, u64 error_code,
|
|||||||
* explicitly shadowing L1's page tables, i.e. unprotecting something
|
* explicitly shadowing L1's page tables, i.e. unprotecting something
|
||||||
* for L1 isn't going to magically fix whatever issue cause L2 to fail.
|
* for L1 isn't going to magically fix whatever issue cause L2 to fail.
|
||||||
*/
|
*/
|
||||||
if (!mmio_info_in_cache(vcpu, cr2, direct) && !is_guest_mode(vcpu))
|
if (!mmio_info_in_cache(vcpu, cr2_or_gpa, direct) && !is_guest_mode(vcpu))
|
||||||
emulation_type = EMULTYPE_ALLOW_RETRY;
|
emulation_type = EMULTYPE_ALLOW_RETRY;
|
||||||
emulate:
|
emulate:
|
||||||
/*
|
/*
|
||||||
@ -5590,7 +5601,7 @@ emulate:
|
|||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
return x86_emulate_instruction(vcpu, cr2, emulation_type, insn,
|
return x86_emulate_instruction(vcpu, cr2_or_gpa, emulation_type, insn,
|
||||||
insn_len);
|
insn_len);
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL_GPL(kvm_mmu_page_fault);
|
EXPORT_SYMBOL_GPL(kvm_mmu_page_fault);
|
||||||
|
@ -291,11 +291,11 @@ static inline unsigned FNAME(gpte_pkeys)(struct kvm_vcpu *vcpu, u64 gpte)
|
|||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Fetch a guest pte for a guest virtual address
|
* Fetch a guest pte for a guest virtual address, or for an L2's GPA.
|
||||||
*/
|
*/
|
||||||
static int FNAME(walk_addr_generic)(struct guest_walker *walker,
|
static int FNAME(walk_addr_generic)(struct guest_walker *walker,
|
||||||
struct kvm_vcpu *vcpu, struct kvm_mmu *mmu,
|
struct kvm_vcpu *vcpu, struct kvm_mmu *mmu,
|
||||||
gva_t addr, u32 access)
|
gpa_t addr, u32 access)
|
||||||
{
|
{
|
||||||
int ret;
|
int ret;
|
||||||
pt_element_t pte;
|
pt_element_t pte;
|
||||||
@ -496,7 +496,7 @@ error:
|
|||||||
}
|
}
|
||||||
|
|
||||||
static int FNAME(walk_addr)(struct guest_walker *walker,
|
static int FNAME(walk_addr)(struct guest_walker *walker,
|
||||||
struct kvm_vcpu *vcpu, gva_t addr, u32 access)
|
struct kvm_vcpu *vcpu, gpa_t addr, u32 access)
|
||||||
{
|
{
|
||||||
return FNAME(walk_addr_generic)(walker, vcpu, vcpu->arch.mmu, addr,
|
return FNAME(walk_addr_generic)(walker, vcpu, vcpu->arch.mmu, addr,
|
||||||
access);
|
access);
|
||||||
@ -611,7 +611,7 @@ static void FNAME(pte_prefetch)(struct kvm_vcpu *vcpu, struct guest_walker *gw,
|
|||||||
* If the guest tries to write a write-protected page, we need to
|
* If the guest tries to write a write-protected page, we need to
|
||||||
* emulate this operation, return 1 to indicate this case.
|
* emulate this operation, return 1 to indicate this case.
|
||||||
*/
|
*/
|
||||||
static int FNAME(fetch)(struct kvm_vcpu *vcpu, gva_t addr,
|
static int FNAME(fetch)(struct kvm_vcpu *vcpu, gpa_t addr,
|
||||||
struct guest_walker *gw,
|
struct guest_walker *gw,
|
||||||
int write_fault, int hlevel,
|
int write_fault, int hlevel,
|
||||||
kvm_pfn_t pfn, bool map_writable, bool prefault,
|
kvm_pfn_t pfn, bool map_writable, bool prefault,
|
||||||
@ -765,7 +765,7 @@ FNAME(is_self_change_mapping)(struct kvm_vcpu *vcpu,
|
|||||||
* Returns: 1 if we need to emulate the instruction, 0 otherwise, or
|
* Returns: 1 if we need to emulate the instruction, 0 otherwise, or
|
||||||
* a negative value on error.
|
* a negative value on error.
|
||||||
*/
|
*/
|
||||||
static int FNAME(page_fault)(struct kvm_vcpu *vcpu, gva_t addr, u32 error_code,
|
static int FNAME(page_fault)(struct kvm_vcpu *vcpu, gpa_t addr, u32 error_code,
|
||||||
bool prefault)
|
bool prefault)
|
||||||
{
|
{
|
||||||
int write_fault = error_code & PFERR_WRITE_MASK;
|
int write_fault = error_code & PFERR_WRITE_MASK;
|
||||||
@ -945,18 +945,19 @@ static void FNAME(invlpg)(struct kvm_vcpu *vcpu, gva_t gva, hpa_t root_hpa)
|
|||||||
spin_unlock(&vcpu->kvm->mmu_lock);
|
spin_unlock(&vcpu->kvm->mmu_lock);
|
||||||
}
|
}
|
||||||
|
|
||||||
static gpa_t FNAME(gva_to_gpa)(struct kvm_vcpu *vcpu, gva_t vaddr, u32 access,
|
/* Note, @addr is a GPA when gva_to_gpa() translates an L2 GPA to an L1 GPA. */
|
||||||
|
static gpa_t FNAME(gva_to_gpa)(struct kvm_vcpu *vcpu, gpa_t addr, u32 access,
|
||||||
struct x86_exception *exception)
|
struct x86_exception *exception)
|
||||||
{
|
{
|
||||||
struct guest_walker walker;
|
struct guest_walker walker;
|
||||||
gpa_t gpa = UNMAPPED_GVA;
|
gpa_t gpa = UNMAPPED_GVA;
|
||||||
int r;
|
int r;
|
||||||
|
|
||||||
r = FNAME(walk_addr)(&walker, vcpu, vaddr, access);
|
r = FNAME(walk_addr)(&walker, vcpu, addr, access);
|
||||||
|
|
||||||
if (r) {
|
if (r) {
|
||||||
gpa = gfn_to_gpa(walker.gfn);
|
gpa = gfn_to_gpa(walker.gfn);
|
||||||
gpa |= vaddr & ~PAGE_MASK;
|
gpa |= addr & ~PAGE_MASK;
|
||||||
} else if (exception)
|
} else if (exception)
|
||||||
*exception = walker.fault;
|
*exception = walker.fault;
|
||||||
|
|
||||||
@ -964,7 +965,8 @@ static gpa_t FNAME(gva_to_gpa)(struct kvm_vcpu *vcpu, gva_t vaddr, u32 access,
|
|||||||
}
|
}
|
||||||
|
|
||||||
#if PTTYPE != PTTYPE_EPT
|
#if PTTYPE != PTTYPE_EPT
|
||||||
static gpa_t FNAME(gva_to_gpa_nested)(struct kvm_vcpu *vcpu, gva_t vaddr,
|
/* Note, gva_to_gpa_nested() is only used to translate L2 GVAs. */
|
||||||
|
static gpa_t FNAME(gva_to_gpa_nested)(struct kvm_vcpu *vcpu, gpa_t vaddr,
|
||||||
u32 access,
|
u32 access,
|
||||||
struct x86_exception *exception)
|
struct x86_exception *exception)
|
||||||
{
|
{
|
||||||
@ -972,6 +974,11 @@ static gpa_t FNAME(gva_to_gpa_nested)(struct kvm_vcpu *vcpu, gva_t vaddr,
|
|||||||
gpa_t gpa = UNMAPPED_GVA;
|
gpa_t gpa = UNMAPPED_GVA;
|
||||||
int r;
|
int r;
|
||||||
|
|
||||||
|
#ifndef CONFIG_X86_64
|
||||||
|
/* A 64-bit GVA should be impossible on 32-bit KVM. */
|
||||||
|
WARN_ON_ONCE(vaddr >> 32);
|
||||||
|
#endif
|
||||||
|
|
||||||
r = FNAME(walk_addr_nested)(&walker, vcpu, vaddr, access);
|
r = FNAME(walk_addr_nested)(&walker, vcpu, vaddr, access);
|
||||||
|
|
||||||
if (r) {
|
if (r) {
|
||||||
|
@ -249,13 +249,13 @@ TRACE_EVENT(
|
|||||||
|
|
||||||
TRACE_EVENT(
|
TRACE_EVENT(
|
||||||
fast_page_fault,
|
fast_page_fault,
|
||||||
TP_PROTO(struct kvm_vcpu *vcpu, gva_t gva, u32 error_code,
|
TP_PROTO(struct kvm_vcpu *vcpu, gpa_t cr2_or_gpa, u32 error_code,
|
||||||
u64 *sptep, u64 old_spte, bool retry),
|
u64 *sptep, u64 old_spte, bool retry),
|
||||||
TP_ARGS(vcpu, gva, error_code, sptep, old_spte, retry),
|
TP_ARGS(vcpu, cr2_or_gpa, error_code, sptep, old_spte, retry),
|
||||||
|
|
||||||
TP_STRUCT__entry(
|
TP_STRUCT__entry(
|
||||||
__field(int, vcpu_id)
|
__field(int, vcpu_id)
|
||||||
__field(gva_t, gva)
|
__field(gpa_t, cr2_or_gpa)
|
||||||
__field(u32, error_code)
|
__field(u32, error_code)
|
||||||
__field(u64 *, sptep)
|
__field(u64 *, sptep)
|
||||||
__field(u64, old_spte)
|
__field(u64, old_spte)
|
||||||
@ -265,7 +265,7 @@ TRACE_EVENT(
|
|||||||
|
|
||||||
TP_fast_assign(
|
TP_fast_assign(
|
||||||
__entry->vcpu_id = vcpu->vcpu_id;
|
__entry->vcpu_id = vcpu->vcpu_id;
|
||||||
__entry->gva = gva;
|
__entry->cr2_or_gpa = cr2_or_gpa;
|
||||||
__entry->error_code = error_code;
|
__entry->error_code = error_code;
|
||||||
__entry->sptep = sptep;
|
__entry->sptep = sptep;
|
||||||
__entry->old_spte = old_spte;
|
__entry->old_spte = old_spte;
|
||||||
@ -273,9 +273,9 @@ TRACE_EVENT(
|
|||||||
__entry->retry = retry;
|
__entry->retry = retry;
|
||||||
),
|
),
|
||||||
|
|
||||||
TP_printk("vcpu %d gva %lx error_code %s sptep %p old %#llx"
|
TP_printk("vcpu %d gva %llx error_code %s sptep %p old %#llx"
|
||||||
" new %llx spurious %d fixed %d", __entry->vcpu_id,
|
" new %llx spurious %d fixed %d", __entry->vcpu_id,
|
||||||
__entry->gva, __print_flags(__entry->error_code, "|",
|
__entry->cr2_or_gpa, __print_flags(__entry->error_code, "|",
|
||||||
kvm_mmu_trace_pferr_flags), __entry->sptep,
|
kvm_mmu_trace_pferr_flags), __entry->sptep,
|
||||||
__entry->old_spte, __entry->new_spte,
|
__entry->old_spte, __entry->new_spte,
|
||||||
__spte_satisfied(old_spte), __spte_satisfied(new_spte)
|
__spte_satisfied(old_spte), __spte_satisfied(new_spte)
|
||||||
|
@ -6379,11 +6379,11 @@ static int handle_emulation_failure(struct kvm_vcpu *vcpu, int emulation_type)
|
|||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool reexecute_instruction(struct kvm_vcpu *vcpu, gva_t cr2,
|
static bool reexecute_instruction(struct kvm_vcpu *vcpu, gpa_t cr2_or_gpa,
|
||||||
bool write_fault_to_shadow_pgtable,
|
bool write_fault_to_shadow_pgtable,
|
||||||
int emulation_type)
|
int emulation_type)
|
||||||
{
|
{
|
||||||
gpa_t gpa = cr2;
|
gpa_t gpa = cr2_or_gpa;
|
||||||
kvm_pfn_t pfn;
|
kvm_pfn_t pfn;
|
||||||
|
|
||||||
if (!(emulation_type & EMULTYPE_ALLOW_RETRY))
|
if (!(emulation_type & EMULTYPE_ALLOW_RETRY))
|
||||||
@ -6397,7 +6397,7 @@ static bool reexecute_instruction(struct kvm_vcpu *vcpu, gva_t cr2,
|
|||||||
* Write permission should be allowed since only
|
* Write permission should be allowed since only
|
||||||
* write access need to be emulated.
|
* write access need to be emulated.
|
||||||
*/
|
*/
|
||||||
gpa = kvm_mmu_gva_to_gpa_write(vcpu, cr2, NULL);
|
gpa = kvm_mmu_gva_to_gpa_write(vcpu, cr2_or_gpa, NULL);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* If the mapping is invalid in guest, let cpu retry
|
* If the mapping is invalid in guest, let cpu retry
|
||||||
@ -6454,10 +6454,10 @@ static bool reexecute_instruction(struct kvm_vcpu *vcpu, gva_t cr2,
|
|||||||
}
|
}
|
||||||
|
|
||||||
static bool retry_instruction(struct x86_emulate_ctxt *ctxt,
|
static bool retry_instruction(struct x86_emulate_ctxt *ctxt,
|
||||||
unsigned long cr2, int emulation_type)
|
gpa_t cr2_or_gpa, int emulation_type)
|
||||||
{
|
{
|
||||||
struct kvm_vcpu *vcpu = emul_to_vcpu(ctxt);
|
struct kvm_vcpu *vcpu = emul_to_vcpu(ctxt);
|
||||||
unsigned long last_retry_eip, last_retry_addr, gpa = cr2;
|
unsigned long last_retry_eip, last_retry_addr, gpa = cr2_or_gpa;
|
||||||
|
|
||||||
last_retry_eip = vcpu->arch.last_retry_eip;
|
last_retry_eip = vcpu->arch.last_retry_eip;
|
||||||
last_retry_addr = vcpu->arch.last_retry_addr;
|
last_retry_addr = vcpu->arch.last_retry_addr;
|
||||||
@ -6486,14 +6486,14 @@ static bool retry_instruction(struct x86_emulate_ctxt *ctxt,
|
|||||||
if (x86_page_table_writing_insn(ctxt))
|
if (x86_page_table_writing_insn(ctxt))
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
if (ctxt->eip == last_retry_eip && last_retry_addr == cr2)
|
if (ctxt->eip == last_retry_eip && last_retry_addr == cr2_or_gpa)
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
vcpu->arch.last_retry_eip = ctxt->eip;
|
vcpu->arch.last_retry_eip = ctxt->eip;
|
||||||
vcpu->arch.last_retry_addr = cr2;
|
vcpu->arch.last_retry_addr = cr2_or_gpa;
|
||||||
|
|
||||||
if (!vcpu->arch.mmu->direct_map)
|
if (!vcpu->arch.mmu->direct_map)
|
||||||
gpa = kvm_mmu_gva_to_gpa_write(vcpu, cr2, NULL);
|
gpa = kvm_mmu_gva_to_gpa_write(vcpu, cr2_or_gpa, NULL);
|
||||||
|
|
||||||
kvm_mmu_unprotect_page(vcpu->kvm, gpa_to_gfn(gpa));
|
kvm_mmu_unprotect_page(vcpu->kvm, gpa_to_gfn(gpa));
|
||||||
|
|
||||||
@ -6639,11 +6639,8 @@ static bool is_vmware_backdoor_opcode(struct x86_emulate_ctxt *ctxt)
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
int x86_emulate_instruction(struct kvm_vcpu *vcpu,
|
int x86_emulate_instruction(struct kvm_vcpu *vcpu, gpa_t cr2_or_gpa,
|
||||||
unsigned long cr2,
|
int emulation_type, void *insn, int insn_len)
|
||||||
int emulation_type,
|
|
||||||
void *insn,
|
|
||||||
int insn_len)
|
|
||||||
{
|
{
|
||||||
int r;
|
int r;
|
||||||
struct x86_emulate_ctxt *ctxt = &vcpu->arch.emulate_ctxt;
|
struct x86_emulate_ctxt *ctxt = &vcpu->arch.emulate_ctxt;
|
||||||
@ -6689,8 +6686,9 @@ int x86_emulate_instruction(struct kvm_vcpu *vcpu,
|
|||||||
kvm_queue_exception(vcpu, UD_VECTOR);
|
kvm_queue_exception(vcpu, UD_VECTOR);
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
if (reexecute_instruction(vcpu, cr2, write_fault_to_spt,
|
if (reexecute_instruction(vcpu, cr2_or_gpa,
|
||||||
emulation_type))
|
write_fault_to_spt,
|
||||||
|
emulation_type))
|
||||||
return 1;
|
return 1;
|
||||||
if (ctxt->have_exception) {
|
if (ctxt->have_exception) {
|
||||||
/*
|
/*
|
||||||
@ -6724,7 +6722,7 @@ int x86_emulate_instruction(struct kvm_vcpu *vcpu,
|
|||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (retry_instruction(ctxt, cr2, emulation_type))
|
if (retry_instruction(ctxt, cr2_or_gpa, emulation_type))
|
||||||
return 1;
|
return 1;
|
||||||
|
|
||||||
/* this is needed for vmware backdoor interface to work since it
|
/* this is needed for vmware backdoor interface to work since it
|
||||||
@ -6736,7 +6734,7 @@ int x86_emulate_instruction(struct kvm_vcpu *vcpu,
|
|||||||
|
|
||||||
restart:
|
restart:
|
||||||
/* Save the faulting GPA (cr2) in the address field */
|
/* Save the faulting GPA (cr2) in the address field */
|
||||||
ctxt->exception.address = cr2;
|
ctxt->exception.address = cr2_or_gpa;
|
||||||
|
|
||||||
r = x86_emulate_insn(ctxt);
|
r = x86_emulate_insn(ctxt);
|
||||||
|
|
||||||
@ -6744,7 +6742,7 @@ restart:
|
|||||||
return 1;
|
return 1;
|
||||||
|
|
||||||
if (r == EMULATION_FAILED) {
|
if (r == EMULATION_FAILED) {
|
||||||
if (reexecute_instruction(vcpu, cr2, write_fault_to_spt,
|
if (reexecute_instruction(vcpu, cr2_or_gpa, write_fault_to_spt,
|
||||||
emulation_type))
|
emulation_type))
|
||||||
return 1;
|
return 1;
|
||||||
|
|
||||||
@ -10025,7 +10023,7 @@ void kvm_arch_async_page_ready(struct kvm_vcpu *vcpu, struct kvm_async_pf *work)
|
|||||||
work->arch.cr3 != vcpu->arch.mmu->get_cr3(vcpu))
|
work->arch.cr3 != vcpu->arch.mmu->get_cr3(vcpu))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
vcpu->arch.mmu->page_fault(vcpu, work->gva, 0, true);
|
vcpu->arch.mmu->page_fault(vcpu, work->cr2_or_gpa, 0, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline u32 kvm_async_pf_hash_fn(gfn_t gfn)
|
static inline u32 kvm_async_pf_hash_fn(gfn_t gfn)
|
||||||
@ -10138,7 +10136,7 @@ void kvm_arch_async_page_not_present(struct kvm_vcpu *vcpu,
|
|||||||
{
|
{
|
||||||
struct x86_exception fault;
|
struct x86_exception fault;
|
||||||
|
|
||||||
trace_kvm_async_pf_not_present(work->arch.token, work->gva);
|
trace_kvm_async_pf_not_present(work->arch.token, work->cr2_or_gpa);
|
||||||
kvm_add_async_pf_gfn(vcpu, work->arch.gfn);
|
kvm_add_async_pf_gfn(vcpu, work->arch.gfn);
|
||||||
|
|
||||||
if (kvm_can_deliver_async_pf(vcpu) &&
|
if (kvm_can_deliver_async_pf(vcpu) &&
|
||||||
@ -10173,7 +10171,7 @@ void kvm_arch_async_page_present(struct kvm_vcpu *vcpu,
|
|||||||
work->arch.token = ~0; /* broadcast wakeup */
|
work->arch.token = ~0; /* broadcast wakeup */
|
||||||
else
|
else
|
||||||
kvm_del_async_pf_gfn(vcpu, work->arch.gfn);
|
kvm_del_async_pf_gfn(vcpu, work->arch.gfn);
|
||||||
trace_kvm_async_pf_ready(work->arch.token, work->gva);
|
trace_kvm_async_pf_ready(work->arch.token, work->cr2_or_gpa);
|
||||||
|
|
||||||
if (vcpu->arch.apf.msr_val & KVM_ASYNC_PF_ENABLED &&
|
if (vcpu->arch.apf.msr_val & KVM_ASYNC_PF_ENABLED &&
|
||||||
!apf_get_user(vcpu, &val)) {
|
!apf_get_user(vcpu, &val)) {
|
||||||
|
@ -289,7 +289,7 @@ int kvm_mtrr_get_msr(struct kvm_vcpu *vcpu, u32 msr, u64 *pdata);
|
|||||||
bool kvm_mtrr_check_gfn_range_consistency(struct kvm_vcpu *vcpu, gfn_t gfn,
|
bool kvm_mtrr_check_gfn_range_consistency(struct kvm_vcpu *vcpu, gfn_t gfn,
|
||||||
int page_num);
|
int page_num);
|
||||||
bool kvm_vector_hashing_enabled(void);
|
bool kvm_vector_hashing_enabled(void);
|
||||||
int x86_emulate_instruction(struct kvm_vcpu *vcpu, unsigned long cr2,
|
int x86_emulate_instruction(struct kvm_vcpu *vcpu, gpa_t cr2_or_gpa,
|
||||||
int emulation_type, void *insn, int insn_len);
|
int emulation_type, void *insn, int insn_len);
|
||||||
|
|
||||||
#define KVM_SUPPORTED_XCR0 (XFEATURE_MASK_FP | XFEATURE_MASK_SSE \
|
#define KVM_SUPPORTED_XCR0 (XFEATURE_MASK_FP | XFEATURE_MASK_SSE \
|
||||||
|
@ -204,7 +204,7 @@ struct kvm_async_pf {
|
|||||||
struct list_head queue;
|
struct list_head queue;
|
||||||
struct kvm_vcpu *vcpu;
|
struct kvm_vcpu *vcpu;
|
||||||
struct mm_struct *mm;
|
struct mm_struct *mm;
|
||||||
gva_t gva;
|
gpa_t cr2_or_gpa;
|
||||||
unsigned long addr;
|
unsigned long addr;
|
||||||
struct kvm_arch_async_pf arch;
|
struct kvm_arch_async_pf arch;
|
||||||
bool wakeup_all;
|
bool wakeup_all;
|
||||||
@ -212,8 +212,8 @@ struct kvm_async_pf {
|
|||||||
|
|
||||||
void kvm_clear_async_pf_completion_queue(struct kvm_vcpu *vcpu);
|
void kvm_clear_async_pf_completion_queue(struct kvm_vcpu *vcpu);
|
||||||
void kvm_check_async_pf_completion(struct kvm_vcpu *vcpu);
|
void kvm_check_async_pf_completion(struct kvm_vcpu *vcpu);
|
||||||
int kvm_setup_async_pf(struct kvm_vcpu *vcpu, gva_t gva, unsigned long hva,
|
int kvm_setup_async_pf(struct kvm_vcpu *vcpu, gpa_t cr2_or_gpa,
|
||||||
struct kvm_arch_async_pf *arch);
|
unsigned long hva, struct kvm_arch_async_pf *arch);
|
||||||
int kvm_async_pf_wakeup_all(struct kvm_vcpu *vcpu);
|
int kvm_async_pf_wakeup_all(struct kvm_vcpu *vcpu);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
@ -64,7 +64,7 @@ static void async_pf_execute(struct work_struct *work)
|
|||||||
struct mm_struct *mm = apf->mm;
|
struct mm_struct *mm = apf->mm;
|
||||||
struct kvm_vcpu *vcpu = apf->vcpu;
|
struct kvm_vcpu *vcpu = apf->vcpu;
|
||||||
unsigned long addr = apf->addr;
|
unsigned long addr = apf->addr;
|
||||||
gva_t gva = apf->gva;
|
gpa_t cr2_or_gpa = apf->cr2_or_gpa;
|
||||||
int locked = 1;
|
int locked = 1;
|
||||||
|
|
||||||
might_sleep();
|
might_sleep();
|
||||||
@ -92,7 +92,7 @@ static void async_pf_execute(struct work_struct *work)
|
|||||||
* this point
|
* this point
|
||||||
*/
|
*/
|
||||||
|
|
||||||
trace_kvm_async_pf_completed(addr, gva);
|
trace_kvm_async_pf_completed(addr, cr2_or_gpa);
|
||||||
|
|
||||||
if (swq_has_sleeper(&vcpu->wq))
|
if (swq_has_sleeper(&vcpu->wq))
|
||||||
swake_up_one(&vcpu->wq);
|
swake_up_one(&vcpu->wq);
|
||||||
@ -165,8 +165,8 @@ void kvm_check_async_pf_completion(struct kvm_vcpu *vcpu)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
int kvm_setup_async_pf(struct kvm_vcpu *vcpu, gva_t gva, unsigned long hva,
|
int kvm_setup_async_pf(struct kvm_vcpu *vcpu, gpa_t cr2_or_gpa,
|
||||||
struct kvm_arch_async_pf *arch)
|
unsigned long hva, struct kvm_arch_async_pf *arch)
|
||||||
{
|
{
|
||||||
struct kvm_async_pf *work;
|
struct kvm_async_pf *work;
|
||||||
|
|
||||||
@ -185,7 +185,7 @@ int kvm_setup_async_pf(struct kvm_vcpu *vcpu, gva_t gva, unsigned long hva,
|
|||||||
|
|
||||||
work->wakeup_all = false;
|
work->wakeup_all = false;
|
||||||
work->vcpu = vcpu;
|
work->vcpu = vcpu;
|
||||||
work->gva = gva;
|
work->cr2_or_gpa = cr2_or_gpa;
|
||||||
work->addr = hva;
|
work->addr = hva;
|
||||||
work->arch = *arch;
|
work->arch = *arch;
|
||||||
work->mm = current->mm;
|
work->mm = current->mm;
|
||||||
|
Loading…
Reference in New Issue
Block a user