diff options
Diffstat (limited to 'drivers/kvm/paging_tmpl.h')
-rw-r--r-- | drivers/kvm/paging_tmpl.h | 68 |
1 files changed, 38 insertions, 30 deletions
diff --git a/drivers/kvm/paging_tmpl.h b/drivers/kvm/paging_tmpl.h index 6bc41950fbb3..afcd2a8f45bb 100644 --- a/drivers/kvm/paging_tmpl.h +++ b/drivers/kvm/paging_tmpl.h | |||
@@ -63,13 +63,15 @@ struct guest_walker { | |||
63 | pt_element_t *ptep; | 63 | pt_element_t *ptep; |
64 | pt_element_t inherited_ar; | 64 | pt_element_t inherited_ar; |
65 | gfn_t gfn; | 65 | gfn_t gfn; |
66 | u32 error_code; | ||
66 | }; | 67 | }; |
67 | 68 | ||
68 | /* | 69 | /* |
69 | * Fetch a guest pte for a guest virtual address | 70 | * Fetch a guest pte for a guest virtual address |
70 | */ | 71 | */ |
71 | static void FNAME(walk_addr)(struct guest_walker *walker, | 72 | static int FNAME(walk_addr)(struct guest_walker *walker, |
72 | struct kvm_vcpu *vcpu, gva_t addr) | 73 | struct kvm_vcpu *vcpu, gva_t addr, |
74 | int write_fault, int user_fault) | ||
73 | { | 75 | { |
74 | hpa_t hpa; | 76 | hpa_t hpa; |
75 | struct kvm_memory_slot *slot; | 77 | struct kvm_memory_slot *slot; |
@@ -86,7 +88,7 @@ static void FNAME(walk_addr)(struct guest_walker *walker, | |||
86 | walker->ptep = &vcpu->pdptrs[(addr >> 30) & 3]; | 88 | walker->ptep = &vcpu->pdptrs[(addr >> 30) & 3]; |
87 | root = *walker->ptep; | 89 | root = *walker->ptep; |
88 | if (!(root & PT_PRESENT_MASK)) | 90 | if (!(root & PT_PRESENT_MASK)) |
89 | return; | 91 | goto not_present; |
90 | --walker->level; | 92 | --walker->level; |
91 | } | 93 | } |
92 | #endif | 94 | #endif |
@@ -111,11 +113,18 @@ static void FNAME(walk_addr)(struct guest_walker *walker, | |||
111 | ASSERT(((unsigned long)walker->table & PAGE_MASK) == | 113 | ASSERT(((unsigned long)walker->table & PAGE_MASK) == |
112 | ((unsigned long)ptep & PAGE_MASK)); | 114 | ((unsigned long)ptep & PAGE_MASK)); |
113 | 115 | ||
114 | if (is_present_pte(*ptep) && !(*ptep & PT_ACCESSED_MASK)) | ||
115 | *ptep |= PT_ACCESSED_MASK; | ||
116 | |||
117 | if (!is_present_pte(*ptep)) | 116 | if (!is_present_pte(*ptep)) |
118 | break; | 117 | goto not_present; |
118 | |||
119 | if (write_fault && !is_writeble_pte(*ptep)) | ||
120 | if (user_fault || is_write_protection(vcpu)) | ||
121 | goto access_error; | ||
122 | |||
123 | if (user_fault && !(*ptep & PT_USER_MASK)) | ||
124 | goto access_error; | ||
125 | |||
126 | if (!(*ptep & PT_ACCESSED_MASK)) | ||
127 | *ptep |= PT_ACCESSED_MASK; /* avoid rmw */ | ||
119 | 128 | ||
120 | if (walker->level == PT_PAGE_TABLE_LEVEL) { | 129 | if (walker->level == PT_PAGE_TABLE_LEVEL) { |
121 | walker->gfn = (*ptep & PT_BASE_ADDR_MASK) | 130 | walker->gfn = (*ptep & PT_BASE_ADDR_MASK) |
@@ -146,6 +155,21 @@ static void FNAME(walk_addr)(struct guest_walker *walker, | |||
146 | } | 155 | } |
147 | walker->ptep = ptep; | 156 | walker->ptep = ptep; |
148 | pgprintk("%s: pte %llx\n", __FUNCTION__, (u64)*ptep); | 157 | pgprintk("%s: pte %llx\n", __FUNCTION__, (u64)*ptep); |
158 | return 1; | ||
159 | |||
160 | not_present: | ||
161 | walker->error_code = 0; | ||
162 | goto err; | ||
163 | |||
164 | access_error: | ||
165 | walker->error_code = PFERR_PRESENT_MASK; | ||
166 | |||
167 | err: | ||
168 | if (write_fault) | ||
169 | walker->error_code |= PFERR_WRITE_MASK; | ||
170 | if (user_fault) | ||
171 | walker->error_code |= PFERR_USER_MASK; | ||
172 | return 0; | ||
149 | } | 173 | } |
150 | 174 | ||
151 | static void FNAME(release_walker)(struct guest_walker *walker) | 175 | static void FNAME(release_walker)(struct guest_walker *walker) |
@@ -347,7 +371,6 @@ static int FNAME(page_fault)(struct kvm_vcpu *vcpu, gva_t addr, | |||
347 | u32 error_code) | 371 | u32 error_code) |
348 | { | 372 | { |
349 | int write_fault = error_code & PFERR_WRITE_MASK; | 373 | int write_fault = error_code & PFERR_WRITE_MASK; |
350 | int pte_present = error_code & PFERR_PRESENT_MASK; | ||
351 | int user_fault = error_code & PFERR_USER_MASK; | 374 | int user_fault = error_code & PFERR_USER_MASK; |
352 | struct guest_walker walker; | 375 | struct guest_walker walker; |
353 | u64 *shadow_pte; | 376 | u64 *shadow_pte; |
@@ -365,19 +388,19 @@ static int FNAME(page_fault)(struct kvm_vcpu *vcpu, gva_t addr, | |||
365 | /* | 388 | /* |
366 | * Look up the shadow pte for the faulting address. | 389 | * Look up the shadow pte for the faulting address. |
367 | */ | 390 | */ |
368 | FNAME(walk_addr)(&walker, vcpu, addr); | 391 | r = FNAME(walk_addr)(&walker, vcpu, addr, write_fault, user_fault); |
369 | shadow_pte = FNAME(fetch)(vcpu, addr, &walker); | ||
370 | 392 | ||
371 | /* | 393 | /* |
372 | * The page is not mapped by the guest. Let the guest handle it. | 394 | * The page is not mapped by the guest. Let the guest handle it. |
373 | */ | 395 | */ |
374 | if (!shadow_pte) { | 396 | if (!r) { |
375 | pgprintk("%s: not mapped\n", __FUNCTION__); | 397 | pgprintk("%s: guest page fault\n", __FUNCTION__); |
376 | inject_page_fault(vcpu, addr, error_code); | 398 | inject_page_fault(vcpu, addr, walker.error_code); |
377 | FNAME(release_walker)(&walker); | 399 | FNAME(release_walker)(&walker); |
378 | return 0; | 400 | return 0; |
379 | } | 401 | } |
380 | 402 | ||
403 | shadow_pte = FNAME(fetch)(vcpu, addr, &walker); | ||
381 | pgprintk("%s: shadow pte %p %llx\n", __FUNCTION__, | 404 | pgprintk("%s: shadow pte %p %llx\n", __FUNCTION__, |
382 | shadow_pte, *shadow_pte); | 405 | shadow_pte, *shadow_pte); |
383 | 406 | ||
@@ -399,22 +422,7 @@ static int FNAME(page_fault)(struct kvm_vcpu *vcpu, gva_t addr, | |||
399 | * mmio: emulate if accessible, otherwise its a guest fault. | 422 | * mmio: emulate if accessible, otherwise its a guest fault. |
400 | */ | 423 | */ |
401 | if (is_io_pte(*shadow_pte)) { | 424 | if (is_io_pte(*shadow_pte)) { |
402 | if (may_access(*shadow_pte, write_fault, user_fault)) | 425 | return 1; |
403 | return 1; | ||
404 | pgprintk("%s: io work, no access\n", __FUNCTION__); | ||
405 | inject_page_fault(vcpu, addr, | ||
406 | error_code | PFERR_PRESENT_MASK); | ||
407 | kvm_mmu_audit(vcpu, "post page fault (io)"); | ||
408 | return 0; | ||
409 | } | ||
410 | |||
411 | /* | ||
412 | * pte not present, guest page fault. | ||
413 | */ | ||
414 | if (pte_present && !fixed && !write_pt) { | ||
415 | inject_page_fault(vcpu, addr, error_code); | ||
416 | kvm_mmu_audit(vcpu, "post page fault (guest)"); | ||
417 | return 0; | ||
418 | } | 426 | } |
419 | 427 | ||
420 | ++kvm_stat.pf_fixed; | 428 | ++kvm_stat.pf_fixed; |
@@ -429,7 +437,7 @@ static gpa_t FNAME(gva_to_gpa)(struct kvm_vcpu *vcpu, gva_t vaddr) | |||
429 | pt_element_t guest_pte; | 437 | pt_element_t guest_pte; |
430 | gpa_t gpa; | 438 | gpa_t gpa; |
431 | 439 | ||
432 | FNAME(walk_addr)(&walker, vcpu, vaddr); | 440 | FNAME(walk_addr)(&walker, vcpu, vaddr, 0, 0); |
433 | guest_pte = *walker.ptep; | 441 | guest_pte = *walker.ptep; |
434 | FNAME(release_walker)(&walker); | 442 | FNAME(release_walker)(&walker); |
435 | 443 | ||