paging_tmpl.h 14.7 KB
Newer Older
A
Avi Kivity 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27
/*
 * Kernel-based Virtual Machine driver for Linux
 *
 * This module enables machines with Intel VT-x extensions to run virtual
 * machines without emulation or binary translation.
 *
 * MMU support
 *
 * Copyright (C) 2006 Qumranet, Inc.
 *
 * Authors:
 *   Yaniv Kamay  <yaniv@qumranet.com>
 *   Avi Kivity   <avi@qumranet.com>
 *
 * This work is licensed under the terms of the GNU GPL, version 2.  See
 * the COPYING file in the top-level directory.
 *
 */

/*
 * We need the mmu code to access both 32-bit and 64-bit guest ptes,
 * so the code in this file is compiled twice, once per pte size.
 */

#if PTTYPE == 64
	#define pt_element_t u64
	#define guest_walker guest_walker64
28
	#define shadow_walker shadow_walker64
A
Avi Kivity 已提交
29 30 31 32 33
	#define FNAME(name) paging##64_##name
	#define PT_BASE_ADDR_MASK PT64_BASE_ADDR_MASK
	#define PT_DIR_BASE_ADDR_MASK PT64_DIR_BASE_ADDR_MASK
	#define PT_INDEX(addr, level) PT64_INDEX(addr, level)
	#define PT_LEVEL_MASK(level) PT64_LEVEL_MASK(level)
34
	#define PT_LEVEL_BITS PT64_LEVEL_BITS
35 36
	#ifdef CONFIG_X86_64
	#define PT_MAX_FULL_LEVELS 4
37
	#define CMPXCHG cmpxchg
38
	#else
39
	#define CMPXCHG cmpxchg64
40 41
	#define PT_MAX_FULL_LEVELS 2
	#endif
A
Avi Kivity 已提交
42 43 44
#elif PTTYPE == 32
	#define pt_element_t u32
	#define guest_walker guest_walker32
45
	#define shadow_walker shadow_walker32
A
Avi Kivity 已提交
46 47 48 49 50
	#define FNAME(name) paging##32_##name
	#define PT_BASE_ADDR_MASK PT32_BASE_ADDR_MASK
	#define PT_DIR_BASE_ADDR_MASK PT32_DIR_BASE_ADDR_MASK
	#define PT_INDEX(addr, level) PT32_INDEX(addr, level)
	#define PT_LEVEL_MASK(level) PT32_LEVEL_MASK(level)
51
	#define PT_LEVEL_BITS PT32_LEVEL_BITS
52
	#define PT_MAX_FULL_LEVELS 2
53
	#define CMPXCHG cmpxchg
A
Avi Kivity 已提交
54 55 56 57
#else
	#error Invalid PTTYPE value
#endif

58 59 60
#define gpte_to_gfn FNAME(gpte_to_gfn)
#define gpte_to_gfn_pde FNAME(gpte_to_gfn_pde)

A
Avi Kivity 已提交
61 62 63 64 65 66
/*
 * The guest_walker structure emulates the behavior of the hardware page
 * table walker.
 */
struct guest_walker {
	int level;
67
	gfn_t table_gfn[PT_MAX_FULL_LEVELS];
68 69
	pt_element_t ptes[PT_MAX_FULL_LEVELS];
	gpa_t pte_gpa[PT_MAX_FULL_LEVELS];
70 71
	unsigned pt_access;
	unsigned pte_access;
72
	gfn_t gfn;
73
	u32 error_code;
A
Avi Kivity 已提交
74 75
};

76 77 78 79 80 81 82 83 84 85 86
struct shadow_walker {
	struct kvm_shadow_walk walker;
	struct guest_walker *guest_walker;
	int user_fault;
	int write_fault;
	int largepage;
	int *ptwrite;
	pfn_t pfn;
	u64 *sptep;
};

87 88 89 90 91 92 93 94 95 96
static gfn_t gpte_to_gfn(pt_element_t gpte)
{
	return (gpte & PT_BASE_ADDR_MASK) >> PAGE_SHIFT;
}

static gfn_t gpte_to_gfn_pde(pt_element_t gpte)
{
	return (gpte & PT_DIR_BASE_ADDR_MASK) >> PAGE_SHIFT;
}

97 98 99 100 101 102 103 104 105
static bool FNAME(cmpxchg_gpte)(struct kvm *kvm,
			 gfn_t table_gfn, unsigned index,
			 pt_element_t orig_pte, pt_element_t new_pte)
{
	pt_element_t ret;
	pt_element_t *table;
	struct page *page;

	page = gfn_to_page(kvm, table_gfn);
106

107 108 109 110 111 112 113 114 115
	table = kmap_atomic(page, KM_USER0);
	ret = CMPXCHG(&table[index], orig_pte, new_pte);
	kunmap_atomic(table, KM_USER0);

	kvm_release_page_dirty(page);

	return (ret != orig_pte);
}

116 117 118 119 120 121 122 123 124 125 126 127
static unsigned FNAME(gpte_access)(struct kvm_vcpu *vcpu, pt_element_t gpte)
{
	unsigned access;

	access = (gpte & (PT_WRITABLE_MASK | PT_USER_MASK)) | ACC_EXEC_MASK;
#if PTTYPE == 64
	if (is_nx(vcpu))
		access &= ~(gpte >> PT64_NX_SHIFT);
#endif
	return access;
}

128 129 130
/*
 * Fetch a guest pte for a guest virtual address
 */
131 132
static int FNAME(walk_addr)(struct guest_walker *walker,
			    struct kvm_vcpu *vcpu, gva_t addr,
133
			    int write_fault, int user_fault, int fetch_fault)
A
Avi Kivity 已提交
134
{
135
	pt_element_t pte;
136
	gfn_t table_gfn;
137
	unsigned index, pt_access, pte_access;
138
	gpa_t pte_gpa;
A
Avi Kivity 已提交
139

140
	pgprintk("%s: addr %lx\n", __func__, addr);
141
walk:
142 143
	walker->level = vcpu->arch.mmu.root_level;
	pte = vcpu->arch.cr3;
144 145
#if PTTYPE == 64
	if (!is_long_mode(vcpu)) {
146
		pte = vcpu->arch.pdptrs[(addr >> 30) & 3];
147
		if (!is_present_pte(pte))
148
			goto not_present;
149 150 151
		--walker->level;
	}
#endif
A
Avi Kivity 已提交
152
	ASSERT((!is_long_mode(vcpu) && is_pae(vcpu)) ||
153
	       (vcpu->arch.cr3 & CR3_NONPAE_RESERVED_BITS) == 0);
A
Avi Kivity 已提交
154

155
	pt_access = ACC_ALL;
156 157

	for (;;) {
158
		index = PT_INDEX(addr, walker->level);
159

160
		table_gfn = gpte_to_gfn(pte);
A
Avi Kivity 已提交
161
		pte_gpa = gfn_to_gpa(table_gfn);
162
		pte_gpa += index * sizeof(pt_element_t);
163
		walker->table_gfn[walker->level - 1] = table_gfn;
164
		walker->pte_gpa[walker->level - 1] = pte_gpa;
165
		pgprintk("%s: table_gfn[%d] %lx\n", __func__,
166 167
			 walker->level - 1, table_gfn);

168
		kvm_read_guest(vcpu->kvm, pte_gpa, &pte, sizeof(pte));
169 170

		if (!is_present_pte(pte))
171 172
			goto not_present;

173
		if (write_fault && !is_writeble_pte(pte))
174 175 176
			if (user_fault || is_write_protection(vcpu))
				goto access_error;

177
		if (user_fault && !(pte & PT_USER_MASK))
178 179
			goto access_error;

180
#if PTTYPE == 64
181
		if (fetch_fault && is_nx(vcpu) && (pte & PT64_NX_MASK))
182 183 184
			goto access_error;
#endif

185
		if (!(pte & PT_ACCESSED_MASK)) {
186
			mark_page_dirty(vcpu->kvm, table_gfn);
187 188 189
			if (FNAME(cmpxchg_gpte)(vcpu->kvm, table_gfn,
			    index, pte, pte|PT_ACCESSED_MASK))
				goto walk;
190
			pte |= PT_ACCESSED_MASK;
191
		}
192

193
		pte_access = pt_access & FNAME(gpte_access)(vcpu, pte);
194

195 196
		walker->ptes[walker->level - 1] = pte;

197
		if (walker->level == PT_PAGE_TABLE_LEVEL) {
198
			walker->gfn = gpte_to_gfn(pte);
199 200 201 202
			break;
		}

		if (walker->level == PT_DIRECTORY_LEVEL
203
		    && (pte & PT_PAGE_SIZE_MASK)
204
		    && (PTTYPE == 64 || is_pse(vcpu))) {
205
			walker->gfn = gpte_to_gfn_pde(pte);
206
			walker->gfn += PT_INDEX(addr, PT_PAGE_TABLE_LEVEL);
207 208
			if (PTTYPE == 32 && is_cpuid_PSE36())
				walker->gfn += pse36_gfn_delta(pte);
209
			break;
210
		}
211

212
		pt_access = pte_access;
213 214
		--walker->level;
	}
215 216

	if (write_fault && !is_dirty_pte(pte)) {
217 218
		bool ret;

219
		mark_page_dirty(vcpu->kvm, table_gfn);
220 221 222 223
		ret = FNAME(cmpxchg_gpte)(vcpu->kvm, table_gfn, index, pte,
			    pte|PT_DIRTY_MASK);
		if (ret)
			goto walk;
224 225
		pte |= PT_DIRTY_MASK;
		kvm_mmu_pte_write(vcpu, pte_gpa, (u8 *)&pte, sizeof(pte));
226
		walker->ptes[walker->level - 1] = pte;
227 228
	}

229 230 231
	walker->pt_access = pt_access;
	walker->pte_access = pte_access;
	pgprintk("%s: pte %llx pte_access %x pt_access %x\n",
232
		 __func__, (u64)pte, pt_access, pte_access);
233 234 235 236 237 238 239 240 241 242 243 244 245 246
	return 1;

not_present:
	walker->error_code = 0;
	goto err;

access_error:
	walker->error_code = PFERR_PRESENT_MASK;

err:
	if (write_fault)
		walker->error_code |= PFERR_WRITE_MASK;
	if (user_fault)
		walker->error_code |= PFERR_USER_MASK;
247 248
	if (fetch_fault)
		walker->error_code |= PFERR_FETCH_MASK;
249
	return 0;
A
Avi Kivity 已提交
250 251
}

252
static void FNAME(update_pte)(struct kvm_vcpu *vcpu, struct kvm_mmu_page *page,
253
			      u64 *spte, const void *pte)
254 255
{
	pt_element_t gpte;
256
	unsigned pte_access;
257
	pfn_t pfn;
M
Marcelo Tosatti 已提交
258
	int largepage = vcpu->arch.update_pte.largepage;
259 260

	gpte = *(const pt_element_t *)pte;
261
	if (~gpte & (PT_PRESENT_MASK | PT_ACCESSED_MASK)) {
262
		if (!is_present_pte(gpte))
263 264 265
			set_shadow_pte(spte, shadow_notrap_nonpresent_pte);
		return;
	}
266
	pgprintk("%s: gpte %llx spte %p\n", __func__, (u64)gpte, spte);
267
	pte_access = page->role.access & FNAME(gpte_access)(vcpu, gpte);
268 269
	if (gpte_to_gfn(gpte) != vcpu->arch.update_pte.gfn)
		return;
270 271
	pfn = vcpu->arch.update_pte.pfn;
	if (is_error_pfn(pfn))
272
		return;
273 274
	if (mmu_notifier_retry(vcpu, vcpu->arch.update_pte.mmu_seq))
		return;
275
	kvm_get_pfn(pfn);
276
	mmu_set_spte(vcpu, spte, page->role.access, pte_access, 0, 0,
M
Marcelo Tosatti 已提交
277
		     gpte & PT_DIRTY_MASK, NULL, largepage, gpte_to_gfn(gpte),
278
		     pfn, true);
279 280
}

A
Avi Kivity 已提交
281 282 283
/*
 * Fetch a shadow pte for a specific level in the paging hierarchy.
 */
284
static int FNAME(shadow_walk_entry)(struct kvm_shadow_walk *_sw,
285
				    struct kvm_vcpu *vcpu, u64 addr,
286
				    u64 *sptep, int level)
A
Avi Kivity 已提交
287
{
288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307
	struct shadow_walker *sw =
		container_of(_sw, struct shadow_walker, walker);
	struct guest_walker *gw = sw->guest_walker;
	unsigned access = gw->pt_access;
	struct kvm_mmu_page *shadow_page;
	u64 spte;
	int metaphysical;
	gfn_t table_gfn;
	int r;
	pt_element_t curr_pte;

	if (level == PT_PAGE_TABLE_LEVEL
	    || (sw->largepage && level == PT_DIRECTORY_LEVEL)) {
		mmu_set_spte(vcpu, sptep, access, gw->pte_access & access,
			     sw->user_fault, sw->write_fault,
			     gw->ptes[gw->level-1] & PT_DIRTY_MASK,
			     sw->ptwrite, sw->largepage, gw->gfn, sw->pfn,
			     false);
		sw->sptep = sptep;
		return 1;
308
	}
A
Avi Kivity 已提交
309

310 311
	if (is_shadow_present_pte(*sptep) && !is_large_pte(*sptep))
		return 0;
A
Avi Kivity 已提交
312

313 314 315
	if (is_large_pte(*sptep)) {
		set_shadow_pte(sptep, shadow_trap_nonpresent_pte);
		kvm_flush_remote_tlbs(vcpu->kvm);
316
		rmap_remove(vcpu->kvm, sptep);
317
	}
318 319 320 321 322 323 324 325 326 327

	if (level == PT_DIRECTORY_LEVEL && gw->level == PT_DIRECTORY_LEVEL) {
		metaphysical = 1;
		if (!is_dirty_pte(gw->ptes[level - 1]))
			access &= ~ACC_WRITE_MASK;
		table_gfn = gpte_to_gfn(gw->ptes[level - 1]);
	} else {
		metaphysical = 0;
		table_gfn = gw->table_gfn[level - 2];
	}
328
	shadow_page = kvm_mmu_get_page(vcpu, table_gfn, (gva_t)addr, level-1,
329 330 331 332 333 334 335 336
				       metaphysical, access, sptep);
	if (!metaphysical) {
		r = kvm_read_guest_atomic(vcpu->kvm, gw->pte_gpa[level - 2],
					  &curr_pte, sizeof(curr_pte));
		if (r || curr_pte != gw->ptes[level - 2]) {
			kvm_release_pfn_clean(sw->pfn);
			sw->sptep = NULL;
			return 1;
337
		}
A
Avi Kivity 已提交
338
	}
339

340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364
	spte = __pa(shadow_page->spt) | PT_PRESENT_MASK | PT_ACCESSED_MASK
		| PT_WRITABLE_MASK | PT_USER_MASK;
	*sptep = spte;
	return 0;
}

static u64 *FNAME(fetch)(struct kvm_vcpu *vcpu, gva_t addr,
			 struct guest_walker *guest_walker,
			 int user_fault, int write_fault, int largepage,
			 int *ptwrite, pfn_t pfn)
{
	struct shadow_walker walker = {
		.walker = { .entry = FNAME(shadow_walk_entry), },
		.guest_walker = guest_walker,
		.user_fault = user_fault,
		.write_fault = write_fault,
		.largepage = largepage,
		.ptwrite = ptwrite,
		.pfn = pfn,
	};

	if (!is_present_pte(guest_walker->ptes[guest_walker->level - 1]))
		return NULL;

	walk_shadow(&walker.walker, vcpu, addr);
A
Avi Kivity 已提交
365

366
	return walker.sptep;
A
Avi Kivity 已提交
367 368 369 370 371 372 373 374 375 376 377 378 379
}

/*
 * Page fault handler.  There are several causes for a page fault:
 *   - there is no shadow pte for the guest pte
 *   - write access through a shadow pte marked read only so that we can set
 *     the dirty bit
 *   - write access to a shadow pte marked read only so we can update the page
 *     dirty bitmap, when userspace requests it
 *   - mmio access; in this case we will never install a present shadow pte
 *   - normal guest page fault due to the guest pte marked not present, not
 *     writable, or not executable
 *
380 381
 *  Returns: 1 if we need to emulate the instruction, 0 otherwise, or
 *           a negative value on error.
A
Avi Kivity 已提交
382 383 384 385 386 387
 */
static int FNAME(page_fault)(struct kvm_vcpu *vcpu, gva_t addr,
			       u32 error_code)
{
	int write_fault = error_code & PFERR_WRITE_MASK;
	int user_fault = error_code & PFERR_USER_MASK;
388
	int fetch_fault = error_code & PFERR_FETCH_MASK;
A
Avi Kivity 已提交
389 390
	struct guest_walker walker;
	u64 *shadow_pte;
391
	int write_pt = 0;
392
	int r;
393
	pfn_t pfn;
M
Marcelo Tosatti 已提交
394
	int largepage = 0;
395
	unsigned long mmu_seq;
A
Avi Kivity 已提交
396

397
	pgprintk("%s: addr %lx err %x\n", __func__, addr, error_code);
398
	kvm_mmu_audit(vcpu, "pre page fault");
399

400 401 402
	r = mmu_topup_memory_caches(vcpu);
	if (r)
		return r;
403

A
Avi Kivity 已提交
404 405 406
	/*
	 * Look up the shadow pte for the faulting address.
	 */
407 408
	r = FNAME(walk_addr)(&walker, vcpu, addr, write_fault, user_fault,
			     fetch_fault);
A
Avi Kivity 已提交
409 410 411 412

	/*
	 * The page is not mapped by the guest.  Let the guest handle it.
	 */
413
	if (!r) {
414
		pgprintk("%s: guest page fault\n", __func__);
415
		inject_page_fault(vcpu, addr, walker.error_code);
416
		vcpu->arch.last_pt_write_count = 0; /* reset fork detector */
A
Avi Kivity 已提交
417 418 419
		return 0;
	}

M
Marcelo Tosatti 已提交
420 421 422 423 424 425 426 427
	if (walker.level == PT_DIRECTORY_LEVEL) {
		gfn_t large_gfn;
		large_gfn = walker.gfn & ~(KVM_PAGES_PER_HPAGE-1);
		if (is_largepage_backed(vcpu, large_gfn)) {
			walker.gfn = large_gfn;
			largepage = 1;
		}
	}
428
	mmu_seq = vcpu->kvm->mmu_notifier_seq;
429
	smp_rmb();
430
	pfn = gfn_to_pfn(vcpu->kvm, walker.gfn);
431

432
	/* mmio */
433
	if (is_error_pfn(pfn)) {
434
		pgprintk("gfn %lx is mmio\n", walker.gfn);
435
		kvm_release_pfn_clean(pfn);
436 437 438
		return 1;
	}

439
	spin_lock(&vcpu->kvm->mmu_lock);
440 441
	if (mmu_notifier_retry(vcpu, mmu_seq))
		goto out_unlock;
442
	kvm_mmu_free_some_pages(vcpu);
443
	shadow_pte = FNAME(fetch)(vcpu, addr, &walker, user_fault, write_fault,
444
				  largepage, &write_pt, pfn);
M
Marcelo Tosatti 已提交
445

446
	pgprintk("%s: shadow pte %p %llx ptwrite %d\n", __func__,
447
		 shadow_pte, *shadow_pte, write_pt);
448

449
	if (!write_pt)
450
		vcpu->arch.last_pt_write_count = 0; /* reset fork detector */
451

A
Avi Kivity 已提交
452
	++vcpu->stat.pf_fixed;
453
	kvm_mmu_audit(vcpu, "post page fault (fixed)");
454
	spin_unlock(&vcpu->kvm->mmu_lock);
A
Avi Kivity 已提交
455

456
	return write_pt;
457 458 459 460 461

out_unlock:
	spin_unlock(&vcpu->kvm->mmu_lock);
	kvm_release_pfn_clean(pfn);
	return 0;
A
Avi Kivity 已提交
462 463 464 465 466
}

static gpa_t FNAME(gva_to_gpa)(struct kvm_vcpu *vcpu, gva_t vaddr)
{
	struct guest_walker walker;
A
Avi Kivity 已提交
467 468
	gpa_t gpa = UNMAPPED_GVA;
	int r;
A
Avi Kivity 已提交
469

A
Avi Kivity 已提交
470
	r = FNAME(walk_addr)(&walker, vcpu, vaddr, 0, 0, 0);
A
Avi Kivity 已提交
471

A
Avi Kivity 已提交
472
	if (r) {
A
Avi Kivity 已提交
473
		gpa = gfn_to_gpa(walker.gfn);
A
Avi Kivity 已提交
474
		gpa |= vaddr & ~PAGE_MASK;
A
Avi Kivity 已提交
475 476 477 478 479
	}

	return gpa;
}

480 481 482
static void FNAME(prefetch_page)(struct kvm_vcpu *vcpu,
				 struct kvm_mmu_page *sp)
{
A
Avi Kivity 已提交
483 484 485
	int i, j, offset, r;
	pt_element_t pt[256 / sizeof(pt_element_t)];
	gpa_t pte_gpa;
486

487 488
	if (sp->role.metaphysical
	    || (PTTYPE == 32 && sp->role.level > PT_PAGE_TABLE_LEVEL)) {
489 490 491 492
		nonpaging_prefetch_page(vcpu, sp);
		return;
	}

A
Avi Kivity 已提交
493 494
	pte_gpa = gfn_to_gpa(sp->gfn);
	if (PTTYPE == 32) {
495
		offset = sp->role.quadrant << PT64_LEVEL_BITS;
A
Avi Kivity 已提交
496 497
		pte_gpa += offset * sizeof(pt_element_t);
	}
498

A
Avi Kivity 已提交
499 500 501 502 503 504 505 506
	for (i = 0; i < PT64_ENT_PER_PAGE; i += ARRAY_SIZE(pt)) {
		r = kvm_read_guest_atomic(vcpu->kvm, pte_gpa, pt, sizeof pt);
		pte_gpa += ARRAY_SIZE(pt) * sizeof(pt_element_t);
		for (j = 0; j < ARRAY_SIZE(pt); ++j)
			if (r || is_present_pte(pt[j]))
				sp->spt[i+j] = shadow_trap_nonpresent_pte;
			else
				sp->spt[i+j] = shadow_notrap_nonpresent_pte;
507
	}
508 509
}

510 511 512 513 514 515 516 517 518 519 520 521 522 523 524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539 540 541 542 543 544 545 546 547 548 549 550 551 552 553 554 555 556 557 558 559 560 561 562 563
/*
 * Using the cached information from sp->gfns is safe because:
 * - The spte has a reference to the struct page, so the pfn for a given gfn
 *   can't change unless all sptes pointing to it are nuked first.
 * - Alias changes zap the entire shadow cache.
 */
static int FNAME(sync_page)(struct kvm_vcpu *vcpu, struct kvm_mmu_page *sp)
{
	int i, offset, nr_present;

	offset = nr_present = 0;

	if (PTTYPE == 32)
		offset = sp->role.quadrant << PT64_LEVEL_BITS;

	for (i = 0; i < PT64_ENT_PER_PAGE; i++) {
		unsigned pte_access;
		pt_element_t gpte;
		gpa_t pte_gpa;
		gfn_t gfn = sp->gfns[i];

		if (!is_shadow_present_pte(sp->spt[i]))
			continue;

		pte_gpa = gfn_to_gpa(sp->gfn);
		pte_gpa += (i+offset) * sizeof(pt_element_t);

		if (kvm_read_guest_atomic(vcpu->kvm, pte_gpa, &gpte,
					  sizeof(pt_element_t)))
			return -EINVAL;

		if (gpte_to_gfn(gpte) != gfn || !is_present_pte(gpte) ||
		    !(gpte & PT_ACCESSED_MASK)) {
			u64 nonpresent;

			rmap_remove(vcpu->kvm, &sp->spt[i]);
			if (is_present_pte(gpte))
				nonpresent = shadow_trap_nonpresent_pte;
			else
				nonpresent = shadow_notrap_nonpresent_pte;
			set_shadow_pte(&sp->spt[i], nonpresent);
			continue;
		}

		nr_present++;
		pte_access = sp->role.access & FNAME(gpte_access)(vcpu, gpte);
		set_spte(vcpu, &sp->spt[i], pte_access, 0, 0,
			 is_dirty_pte(gpte), 0, gfn,
			 spte_to_pfn(sp->spt[i]), true);
	}

	return !nr_present;
}

A
Avi Kivity 已提交
564 565
#undef pt_element_t
#undef guest_walker
566
#undef shadow_walker
A
Avi Kivity 已提交
567 568 569 570 571
#undef FNAME
#undef PT_BASE_ADDR_MASK
#undef PT_INDEX
#undef PT_LEVEL_MASK
#undef PT_DIR_BASE_ADDR_MASK
572
#undef PT_LEVEL_BITS
573
#undef PT_MAX_FULL_LEVELS
574 575
#undef gpte_to_gfn
#undef gpte_to_gfn_pde
576
#undef CMPXCHG