aboutsummaryrefslogtreecommitdiffstats
path: root/arch/x86/kvm
diff options
context:
space:
mode:
Diffstat (limited to 'arch/x86/kvm')
-rw-r--r--arch/x86/kvm/paging_tmpl.h59
1 files changed, 24 insertions, 35 deletions
diff --git a/arch/x86/kvm/paging_tmpl.h b/arch/x86/kvm/paging_tmpl.h
index 8cb85f9c8adb..d9a2742014e3 100644
--- a/arch/x86/kvm/paging_tmpl.h
+++ b/arch/x86/kvm/paging_tmpl.h
@@ -320,12 +320,10 @@ static u64 *FNAME(fetch)(struct kvm_vcpu *vcpu, gva_t addr,
320{ 320{
321 unsigned access = gw->pt_access; 321 unsigned access = gw->pt_access;
322 struct kvm_mmu_page *sp = NULL; 322 struct kvm_mmu_page *sp = NULL;
323 u64 *sptep = NULL;
324 int uninitialized_var(level);
325 bool dirty = is_dirty_gpte(gw->ptes[gw->level - 1]); 323 bool dirty = is_dirty_gpte(gw->ptes[gw->level - 1]);
326 int top_level; 324 int top_level;
327 unsigned direct_access; 325 unsigned direct_access;
328 struct kvm_shadow_walk_iterator iterator; 326 struct kvm_shadow_walk_iterator it;
329 327
330 if (!is_present_gpte(gw->ptes[gw->level - 1])) 328 if (!is_present_gpte(gw->ptes[gw->level - 1]))
331 return NULL; 329 return NULL;
@@ -346,68 +344,59 @@ static u64 *FNAME(fetch)(struct kvm_vcpu *vcpu, gva_t addr,
346 if (FNAME(gpte_changed)(vcpu, gw, top_level)) 344 if (FNAME(gpte_changed)(vcpu, gw, top_level))
347 goto out_gpte_changed; 345 goto out_gpte_changed;
348 346
349 for (shadow_walk_init(&iterator, vcpu, addr); 347 for (shadow_walk_init(&it, vcpu, addr);
350 shadow_walk_okay(&iterator) && iterator.level > gw->level; 348 shadow_walk_okay(&it) && it.level > gw->level;
351 shadow_walk_next(&iterator)) { 349 shadow_walk_next(&it)) {
352 gfn_t table_gfn; 350 gfn_t table_gfn;
353 351
354 level = iterator.level; 352 drop_large_spte(vcpu, it.sptep);
355 sptep = iterator.sptep;
356
357 drop_large_spte(vcpu, sptep);
358 353
359 sp = NULL; 354 sp = NULL;
360 if (!is_shadow_present_pte(*sptep)) { 355 if (!is_shadow_present_pte(*it.sptep)) {
361 table_gfn = gw->table_gfn[level - 2]; 356 table_gfn = gw->table_gfn[it.level - 2];
362 sp = kvm_mmu_get_page(vcpu, table_gfn, addr, level-1, 357 sp = kvm_mmu_get_page(vcpu, table_gfn, addr, it.level-1,
363 false, access, sptep); 358 false, access, it.sptep);
364 } 359 }
365 360
366 /* 361 /*
367 * Verify that the gpte in the page we've just write 362 * Verify that the gpte in the page we've just write
368 * protected is still there. 363 * protected is still there.
369 */ 364 */
370 if (FNAME(gpte_changed)(vcpu, gw, level - 1)) 365 if (FNAME(gpte_changed)(vcpu, gw, it.level - 1))
371 goto out_gpte_changed; 366 goto out_gpte_changed;
372 367
373 if (sp) 368 if (sp)
374 link_shadow_page(sptep, sp); 369 link_shadow_page(it.sptep, sp);
375 } 370 }
376 371
377 for (; 372 for (;
378 shadow_walk_okay(&iterator) && iterator.level > hlevel; 373 shadow_walk_okay(&it) && it.level > hlevel;
379 shadow_walk_next(&iterator)) { 374 shadow_walk_next(&it)) {
380 gfn_t direct_gfn; 375 gfn_t direct_gfn;
381 376
382 level = iterator.level; 377 validate_direct_spte(vcpu, it.sptep, direct_access);
383 sptep = iterator.sptep;
384 378
385 validate_direct_spte(vcpu, sptep, direct_access); 379 drop_large_spte(vcpu, it.sptep);
386 380
387 drop_large_spte(vcpu, sptep); 381 if (is_shadow_present_pte(*it.sptep))
388
389 if (is_shadow_present_pte(*sptep))
390 continue; 382 continue;
391 383
392 direct_gfn = gw->gfn & ~(KVM_PAGES_PER_HPAGE(level) - 1); 384 direct_gfn = gw->gfn & ~(KVM_PAGES_PER_HPAGE(it.level) - 1);
393 385
394 sp = kvm_mmu_get_page(vcpu, direct_gfn, addr, level-1, 386 sp = kvm_mmu_get_page(vcpu, direct_gfn, addr, it.level-1,
395 true, direct_access, sptep); 387 true, direct_access, it.sptep);
396 link_shadow_page(sptep, sp); 388 link_shadow_page(it.sptep, sp);
397 } 389 }
398 390
399 sptep = iterator.sptep; 391 mmu_set_spte(vcpu, it.sptep, access, gw->pte_access & access,
400 level = iterator.level; 392 user_fault, write_fault, dirty, ptwrite, it.level,
401
402 mmu_set_spte(vcpu, sptep, access, gw->pte_access & access,
403 user_fault, write_fault, dirty, ptwrite, level,
404 gw->gfn, pfn, false, true); 393 gw->gfn, pfn, false, true);
405 394
406 return sptep; 395 return it.sptep;
407 396
408out_gpte_changed: 397out_gpte_changed:
409 if (sp) 398 if (sp)
410 kvm_mmu_put_page(sp, sptep); 399 kvm_mmu_put_page(sp, it.sptep);
411 kvm_release_pfn_clean(pfn); 400 kvm_release_pfn_clean(pfn);
412 return NULL; 401 return NULL;
413} 402}