xref: /freebsd/sys/amd64/vmm/vmm.c (revision f352ff0ca87a81ce2a766b0aba049e777fb172f3)
1366f6083SPeter Grehan /*-
2366f6083SPeter Grehan  * Copyright (c) 2011 NetApp, Inc.
3366f6083SPeter Grehan  * All rights reserved.
4366f6083SPeter Grehan  *
5366f6083SPeter Grehan  * Redistribution and use in source and binary forms, with or without
6366f6083SPeter Grehan  * modification, are permitted provided that the following conditions
7366f6083SPeter Grehan  * are met:
8366f6083SPeter Grehan  * 1. Redistributions of source code must retain the above copyright
9366f6083SPeter Grehan  *    notice, this list of conditions and the following disclaimer.
10366f6083SPeter Grehan  * 2. Redistributions in binary form must reproduce the above copyright
11366f6083SPeter Grehan  *    notice, this list of conditions and the following disclaimer in the
12366f6083SPeter Grehan  *    documentation and/or other materials provided with the distribution.
13366f6083SPeter Grehan  *
14366f6083SPeter Grehan  * THIS SOFTWARE IS PROVIDED BY NETAPP, INC ``AS IS'' AND
15366f6083SPeter Grehan  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
16366f6083SPeter Grehan  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
17366f6083SPeter Grehan  * ARE DISCLAIMED.  IN NO EVENT SHALL NETAPP, INC OR CONTRIBUTORS BE LIABLE
18366f6083SPeter Grehan  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
19366f6083SPeter Grehan  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
20366f6083SPeter Grehan  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
21366f6083SPeter Grehan  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
22366f6083SPeter Grehan  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
23366f6083SPeter Grehan  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
24366f6083SPeter Grehan  * SUCH DAMAGE.
25366f6083SPeter Grehan  *
26366f6083SPeter Grehan  * $FreeBSD$
27366f6083SPeter Grehan  */
28366f6083SPeter Grehan 
29366f6083SPeter Grehan #include <sys/cdefs.h>
30366f6083SPeter Grehan __FBSDID("$FreeBSD$");
31366f6083SPeter Grehan 
32366f6083SPeter Grehan #include <sys/param.h>
3338f1b189SPeter Grehan #include <sys/systm.h>
34366f6083SPeter Grehan #include <sys/kernel.h>
35366f6083SPeter Grehan #include <sys/module.h>
36366f6083SPeter Grehan #include <sys/sysctl.h>
37366f6083SPeter Grehan #include <sys/malloc.h>
38366f6083SPeter Grehan #include <sys/pcpu.h>
39366f6083SPeter Grehan #include <sys/lock.h>
40366f6083SPeter Grehan #include <sys/mutex.h>
41366f6083SPeter Grehan #include <sys/proc.h>
42366f6083SPeter Grehan #include <sys/sched.h>
43366f6083SPeter Grehan #include <sys/smp.h>
44366f6083SPeter Grehan #include <sys/systm.h>
45366f6083SPeter Grehan 
46366f6083SPeter Grehan #include <vm/vm.h>
47366f6083SPeter Grehan 
48366f6083SPeter Grehan #include <machine/vm.h>
49366f6083SPeter Grehan #include <machine/pcb.h>
5075dd3366SNeel Natu #include <machine/smp.h>
5134a6b2d6SJohn Baldwin #include <x86/apicreg.h>
52366f6083SPeter Grehan 
53366f6083SPeter Grehan #include <machine/vmm.h>
54366f6083SPeter Grehan #include "vmm_mem.h"
55366f6083SPeter Grehan #include "vmm_util.h"
56366f6083SPeter Grehan #include <machine/vmm_dev.h>
57366f6083SPeter Grehan #include "vlapic.h"
58366f6083SPeter Grehan #include "vmm_msr.h"
59366f6083SPeter Grehan #include "vmm_ipi.h"
60366f6083SPeter Grehan #include "vmm_stat.h"
61366f6083SPeter Grehan 
62366f6083SPeter Grehan #include "io/ppt.h"
63366f6083SPeter Grehan #include "io/iommu.h"
64366f6083SPeter Grehan 
65366f6083SPeter Grehan struct vlapic;
66366f6083SPeter Grehan 
67366f6083SPeter Grehan struct vcpu {
68366f6083SPeter Grehan 	int		flags;
6975dd3366SNeel Natu 	enum vcpu_state	state;
7075dd3366SNeel Natu 	struct mtx	mtx;
71366f6083SPeter Grehan 	int		pincpu;		/* host cpuid this vcpu is bound to */
72366f6083SPeter Grehan 	int		hostcpu;	/* host cpuid this vcpu last ran on */
73366f6083SPeter Grehan 	uint64_t	guest_msrs[VMM_MSR_NUM];
74366f6083SPeter Grehan 	struct vlapic	*vlapic;
75366f6083SPeter Grehan 	int		 vcpuid;
7638f1b189SPeter Grehan 	struct savefpu	*guestfpu;	/* guest fpu state */
77366f6083SPeter Grehan 	void		*stats;
7898ed632cSNeel Natu 	struct vm_exit	exitinfo;
79e9027382SNeel Natu 	enum x2apic_state x2apic_state;
80*f352ff0cSNeel Natu 	int		nmi_pending;
81366f6083SPeter Grehan };
82366f6083SPeter Grehan #define	VCPU_F_PINNED	0x0001
83366f6083SPeter Grehan 
84366f6083SPeter Grehan #define	VCPU_PINCPU(vm, vcpuid)	\
85366f6083SPeter Grehan     ((vm->vcpu[vcpuid].flags & VCPU_F_PINNED) ? vm->vcpu[vcpuid].pincpu : -1)
86366f6083SPeter Grehan 
87366f6083SPeter Grehan #define	VCPU_UNPIN(vm, vcpuid)	(vm->vcpu[vcpuid].flags &= ~VCPU_F_PINNED)
88366f6083SPeter Grehan 
89366f6083SPeter Grehan #define	VCPU_PIN(vm, vcpuid, host_cpuid)				\
90366f6083SPeter Grehan do {									\
91366f6083SPeter Grehan 	vm->vcpu[vcpuid].flags |= VCPU_F_PINNED;			\
92366f6083SPeter Grehan 	vm->vcpu[vcpuid].pincpu = host_cpuid;				\
93366f6083SPeter Grehan } while(0)
94366f6083SPeter Grehan 
9575dd3366SNeel Natu #define	vcpu_lock_init(v)	mtx_init(&((v)->mtx), "vcpu lock", 0, MTX_DEF)
9675dd3366SNeel Natu #define	vcpu_lock(v)		mtx_lock(&((v)->mtx))
9775dd3366SNeel Natu #define	vcpu_unlock(v)		mtx_unlock(&((v)->mtx))
9875dd3366SNeel Natu 
99366f6083SPeter Grehan #define	VM_MAX_MEMORY_SEGMENTS	2
100366f6083SPeter Grehan 
101366f6083SPeter Grehan struct vm {
102366f6083SPeter Grehan 	void		*cookie;	/* processor-specific data */
103366f6083SPeter Grehan 	void		*iommu;		/* iommu-specific data */
104366f6083SPeter Grehan 	struct vcpu	vcpu[VM_MAXCPU];
105366f6083SPeter Grehan 	int		num_mem_segs;
106366f6083SPeter Grehan 	struct vm_memory_segment mem_segs[VM_MAX_MEMORY_SEGMENTS];
107366f6083SPeter Grehan 	char		name[VM_MAX_NAMELEN];
108366f6083SPeter Grehan 
109366f6083SPeter Grehan 	/*
110a5615c90SPeter Grehan 	 * Set of active vcpus.
111366f6083SPeter Grehan 	 * An active vcpu is one that has been started implicitly (BSP) or
112366f6083SPeter Grehan 	 * explicitly (AP) by sending it a startup ipi.
113366f6083SPeter Grehan 	 */
114a5615c90SPeter Grehan 	cpuset_t	active_cpus;
115366f6083SPeter Grehan };
116366f6083SPeter Grehan 
117366f6083SPeter Grehan static struct vmm_ops *ops;
118366f6083SPeter Grehan #define	VMM_INIT()	(ops != NULL ? (*ops->init)() : 0)
119366f6083SPeter Grehan #define	VMM_CLEANUP()	(ops != NULL ? (*ops->cleanup)() : 0)
120366f6083SPeter Grehan 
121366f6083SPeter Grehan #define	VMINIT(vm)	(ops != NULL ? (*ops->vminit)(vm): NULL)
12298ed632cSNeel Natu #define	VMRUN(vmi, vcpu, rip) \
12398ed632cSNeel Natu 	(ops != NULL ? (*ops->vmrun)(vmi, vcpu, rip) : ENXIO)
124366f6083SPeter Grehan #define	VMCLEANUP(vmi)	(ops != NULL ? (*ops->vmcleanup)(vmi) : NULL)
125bda273f2SNeel Natu #define	VMMMAP_SET(vmi, gpa, hpa, len, attr, prot, spm)			\
126bda273f2SNeel Natu     	(ops != NULL ? 							\
127bda273f2SNeel Natu     	(*ops->vmmmap_set)(vmi, gpa, hpa, len, attr, prot, spm) :	\
128bda273f2SNeel Natu 	ENXIO)
129bda273f2SNeel Natu #define	VMMMAP_GET(vmi, gpa) \
130bda273f2SNeel Natu 	(ops != NULL ? (*ops->vmmmap_get)(vmi, gpa) : ENXIO)
131366f6083SPeter Grehan #define	VMGETREG(vmi, vcpu, num, retval)		\
132366f6083SPeter Grehan 	(ops != NULL ? (*ops->vmgetreg)(vmi, vcpu, num, retval) : ENXIO)
133366f6083SPeter Grehan #define	VMSETREG(vmi, vcpu, num, val)		\
134366f6083SPeter Grehan 	(ops != NULL ? (*ops->vmsetreg)(vmi, vcpu, num, val) : ENXIO)
135366f6083SPeter Grehan #define	VMGETDESC(vmi, vcpu, num, desc)		\
136366f6083SPeter Grehan 	(ops != NULL ? (*ops->vmgetdesc)(vmi, vcpu, num, desc) : ENXIO)
137366f6083SPeter Grehan #define	VMSETDESC(vmi, vcpu, num, desc)		\
138366f6083SPeter Grehan 	(ops != NULL ? (*ops->vmsetdesc)(vmi, vcpu, num, desc) : ENXIO)
139366f6083SPeter Grehan #define	VMINJECT(vmi, vcpu, type, vec, ec, ecv)	\
140366f6083SPeter Grehan 	(ops != NULL ? (*ops->vminject)(vmi, vcpu, type, vec, ec, ecv) : ENXIO)
141366f6083SPeter Grehan #define	VMGETCAP(vmi, vcpu, num, retval)	\
142366f6083SPeter Grehan 	(ops != NULL ? (*ops->vmgetcap)(vmi, vcpu, num, retval) : ENXIO)
143366f6083SPeter Grehan #define	VMSETCAP(vmi, vcpu, num, val)		\
144366f6083SPeter Grehan 	(ops != NULL ? (*ops->vmsetcap)(vmi, vcpu, num, val) : ENXIO)
145366f6083SPeter Grehan 
14638f1b189SPeter Grehan #define	fpu_start_emulating()	start_emulating()
14738f1b189SPeter Grehan #define	fpu_stop_emulating()	stop_emulating()
148366f6083SPeter Grehan 
149366f6083SPeter Grehan static MALLOC_DEFINE(M_VM, "vm", "vm");
150366f6083SPeter Grehan CTASSERT(VMM_MSR_NUM <= 64);	/* msr_mask can keep track of up to 64 msrs */
151366f6083SPeter Grehan 
152366f6083SPeter Grehan /* statistics */
153366f6083SPeter Grehan static VMM_STAT_DEFINE(VCPU_TOTAL_RUNTIME, "vcpu total runtime");
154366f6083SPeter Grehan 
155366f6083SPeter Grehan static void
156366f6083SPeter Grehan vcpu_cleanup(struct vcpu *vcpu)
157366f6083SPeter Grehan {
158366f6083SPeter Grehan 	vlapic_cleanup(vcpu->vlapic);
159366f6083SPeter Grehan 	vmm_stat_free(vcpu->stats);
16038f1b189SPeter Grehan 	fpu_save_area_free(vcpu->guestfpu);
161366f6083SPeter Grehan }
162366f6083SPeter Grehan 
163366f6083SPeter Grehan static void
164366f6083SPeter Grehan vcpu_init(struct vm *vm, uint32_t vcpu_id)
165366f6083SPeter Grehan {
166366f6083SPeter Grehan 	struct vcpu *vcpu;
167366f6083SPeter Grehan 
168366f6083SPeter Grehan 	vcpu = &vm->vcpu[vcpu_id];
169366f6083SPeter Grehan 
17075dd3366SNeel Natu 	vcpu_lock_init(vcpu);
17175dd3366SNeel Natu 	vcpu->hostcpu = NOCPU;
172366f6083SPeter Grehan 	vcpu->vcpuid = vcpu_id;
173366f6083SPeter Grehan 	vcpu->vlapic = vlapic_init(vm, vcpu_id);
17473820fb0SNeel Natu 	vm_set_x2apic_state(vm, vcpu_id, X2APIC_ENABLED);
17538f1b189SPeter Grehan 	vcpu->guestfpu = fpu_save_area_alloc();
17638f1b189SPeter Grehan 	fpu_save_area_reset(vcpu->guestfpu);
177366f6083SPeter Grehan 	vcpu->stats = vmm_stat_alloc();
178366f6083SPeter Grehan }
179366f6083SPeter Grehan 
18098ed632cSNeel Natu struct vm_exit *
18198ed632cSNeel Natu vm_exitinfo(struct vm *vm, int cpuid)
18298ed632cSNeel Natu {
18398ed632cSNeel Natu 	struct vcpu *vcpu;
18498ed632cSNeel Natu 
18598ed632cSNeel Natu 	if (cpuid < 0 || cpuid >= VM_MAXCPU)
18698ed632cSNeel Natu 		panic("vm_exitinfo: invalid cpuid %d", cpuid);
18798ed632cSNeel Natu 
18898ed632cSNeel Natu 	vcpu = &vm->vcpu[cpuid];
18998ed632cSNeel Natu 
19098ed632cSNeel Natu 	return (&vcpu->exitinfo);
19198ed632cSNeel Natu }
19298ed632cSNeel Natu 
193366f6083SPeter Grehan static int
194366f6083SPeter Grehan vmm_init(void)
195366f6083SPeter Grehan {
196366f6083SPeter Grehan 	int error;
197366f6083SPeter Grehan 
198366f6083SPeter Grehan 	vmm_ipi_init();
199366f6083SPeter Grehan 
200366f6083SPeter Grehan 	error = vmm_mem_init();
201366f6083SPeter Grehan 	if (error)
202366f6083SPeter Grehan 		return (error);
203366f6083SPeter Grehan 
204366f6083SPeter Grehan 	if (vmm_is_intel())
205366f6083SPeter Grehan 		ops = &vmm_ops_intel;
206366f6083SPeter Grehan 	else if (vmm_is_amd())
207366f6083SPeter Grehan 		ops = &vmm_ops_amd;
208366f6083SPeter Grehan 	else
209366f6083SPeter Grehan 		return (ENXIO);
210366f6083SPeter Grehan 
211366f6083SPeter Grehan 	vmm_msr_init();
212366f6083SPeter Grehan 
213366f6083SPeter Grehan 	return (VMM_INIT());
214366f6083SPeter Grehan }
215366f6083SPeter Grehan 
216366f6083SPeter Grehan static int
217366f6083SPeter Grehan vmm_handler(module_t mod, int what, void *arg)
218366f6083SPeter Grehan {
219366f6083SPeter Grehan 	int error;
220366f6083SPeter Grehan 
221366f6083SPeter Grehan 	switch (what) {
222366f6083SPeter Grehan 	case MOD_LOAD:
223366f6083SPeter Grehan 		vmmdev_init();
224366f6083SPeter Grehan 		iommu_init();
225366f6083SPeter Grehan 		error = vmm_init();
226366f6083SPeter Grehan 		break;
227366f6083SPeter Grehan 	case MOD_UNLOAD:
228cdc5b9e7SNeel Natu 		error = vmmdev_cleanup();
229cdc5b9e7SNeel Natu 		if (error == 0) {
230366f6083SPeter Grehan 			iommu_cleanup();
231366f6083SPeter Grehan 			vmm_ipi_cleanup();
232366f6083SPeter Grehan 			error = VMM_CLEANUP();
233cdc5b9e7SNeel Natu 		}
234366f6083SPeter Grehan 		break;
235366f6083SPeter Grehan 	default:
236366f6083SPeter Grehan 		error = 0;
237366f6083SPeter Grehan 		break;
238366f6083SPeter Grehan 	}
239366f6083SPeter Grehan 	return (error);
240366f6083SPeter Grehan }
241366f6083SPeter Grehan 
242366f6083SPeter Grehan static moduledata_t vmm_kmod = {
243366f6083SPeter Grehan 	"vmm",
244366f6083SPeter Grehan 	vmm_handler,
245366f6083SPeter Grehan 	NULL
246366f6083SPeter Grehan };
247366f6083SPeter Grehan 
248366f6083SPeter Grehan /*
249366f6083SPeter Grehan  * Execute the module load handler after the pci passthru driver has had
250366f6083SPeter Grehan  * a chance to claim devices. We need this information at the time we do
251366f6083SPeter Grehan  * iommu initialization.
252366f6083SPeter Grehan  */
253366f6083SPeter Grehan DECLARE_MODULE(vmm, vmm_kmod, SI_SUB_CONFIGURE + 1, SI_ORDER_ANY);
254366f6083SPeter Grehan MODULE_VERSION(vmm, 1);
255366f6083SPeter Grehan 
256366f6083SPeter Grehan SYSCTL_NODE(_hw, OID_AUTO, vmm, CTLFLAG_RW, NULL, NULL);
257366f6083SPeter Grehan 
258366f6083SPeter Grehan struct vm *
259366f6083SPeter Grehan vm_create(const char *name)
260366f6083SPeter Grehan {
261366f6083SPeter Grehan 	int i;
262366f6083SPeter Grehan 	struct vm *vm;
263366f6083SPeter Grehan 	vm_paddr_t maxaddr;
264366f6083SPeter Grehan 
265366f6083SPeter Grehan 	const int BSP = 0;
266366f6083SPeter Grehan 
267366f6083SPeter Grehan 	if (name == NULL || strlen(name) >= VM_MAX_NAMELEN)
268366f6083SPeter Grehan 		return (NULL);
269366f6083SPeter Grehan 
270366f6083SPeter Grehan 	vm = malloc(sizeof(struct vm), M_VM, M_WAITOK | M_ZERO);
271366f6083SPeter Grehan 	strcpy(vm->name, name);
272366f6083SPeter Grehan 	vm->cookie = VMINIT(vm);
273366f6083SPeter Grehan 
274366f6083SPeter Grehan 	for (i = 0; i < VM_MAXCPU; i++) {
275366f6083SPeter Grehan 		vcpu_init(vm, i);
276366f6083SPeter Grehan 		guest_msrs_init(vm, i);
277366f6083SPeter Grehan 	}
278366f6083SPeter Grehan 
279366f6083SPeter Grehan 	maxaddr = vmm_mem_maxaddr();
280366f6083SPeter Grehan 	vm->iommu = iommu_create_domain(maxaddr);
281366f6083SPeter Grehan 	vm_activate_cpu(vm, BSP);
282366f6083SPeter Grehan 
283366f6083SPeter Grehan 	return (vm);
284366f6083SPeter Grehan }
285366f6083SPeter Grehan 
286f7d51510SNeel Natu static void
287f7d51510SNeel Natu vm_free_mem_seg(struct vm *vm, struct vm_memory_segment *seg)
288f7d51510SNeel Natu {
289f7d51510SNeel Natu 	size_t len;
290f7d51510SNeel Natu 	vm_paddr_t hpa;
2917ce04d0aSNeel Natu 	void *host_domain;
2927ce04d0aSNeel Natu 
2937ce04d0aSNeel Natu 	host_domain = iommu_host_domain();
294f7d51510SNeel Natu 
295f7d51510SNeel Natu 	len = 0;
296f7d51510SNeel Natu 	while (len < seg->len) {
297f7d51510SNeel Natu 		hpa = vm_gpa2hpa(vm, seg->gpa + len, PAGE_SIZE);
298f7d51510SNeel Natu 		if (hpa == (vm_paddr_t)-1) {
299f7d51510SNeel Natu 			panic("vm_free_mem_segs: cannot free hpa "
300f7d51510SNeel Natu 			      "associated with gpa 0x%016lx", seg->gpa + len);
301f7d51510SNeel Natu 		}
302f7d51510SNeel Natu 
3037ce04d0aSNeel Natu 		/*
3047ce04d0aSNeel Natu 		 * Remove the 'gpa' to 'hpa' mapping in VMs domain.
3057ce04d0aSNeel Natu 		 * And resurrect the 1:1 mapping for 'hpa' in 'host_domain'.
3067ce04d0aSNeel Natu 		 */
3077ce04d0aSNeel Natu 		iommu_remove_mapping(vm->iommu, seg->gpa + len, PAGE_SIZE);
3087ce04d0aSNeel Natu 		iommu_create_mapping(host_domain, hpa, hpa, PAGE_SIZE);
3097ce04d0aSNeel Natu 
310f7d51510SNeel Natu 		vmm_mem_free(hpa, PAGE_SIZE);
311f7d51510SNeel Natu 
312f7d51510SNeel Natu 		len += PAGE_SIZE;
313f7d51510SNeel Natu 	}
314f7d51510SNeel Natu 
3157ce04d0aSNeel Natu 	/*
3167ce04d0aSNeel Natu 	 * Invalidate cached translations associated with 'vm->iommu' since
3177ce04d0aSNeel Natu 	 * we have now moved some pages from it.
3187ce04d0aSNeel Natu 	 */
3197ce04d0aSNeel Natu 	iommu_invalidate_tlb(vm->iommu);
3207ce04d0aSNeel Natu 
321f7d51510SNeel Natu 	bzero(seg, sizeof(struct vm_memory_segment));
322f7d51510SNeel Natu }
323f7d51510SNeel Natu 
324366f6083SPeter Grehan void
325366f6083SPeter Grehan vm_destroy(struct vm *vm)
326366f6083SPeter Grehan {
327366f6083SPeter Grehan 	int i;
328366f6083SPeter Grehan 
329366f6083SPeter Grehan 	ppt_unassign_all(vm);
330366f6083SPeter Grehan 
331366f6083SPeter Grehan 	for (i = 0; i < vm->num_mem_segs; i++)
332f7d51510SNeel Natu 		vm_free_mem_seg(vm, &vm->mem_segs[i]);
333f7d51510SNeel Natu 
334f7d51510SNeel Natu 	vm->num_mem_segs = 0;
335366f6083SPeter Grehan 
336366f6083SPeter Grehan 	for (i = 0; i < VM_MAXCPU; i++)
337366f6083SPeter Grehan 		vcpu_cleanup(&vm->vcpu[i]);
338366f6083SPeter Grehan 
339366f6083SPeter Grehan 	iommu_destroy_domain(vm->iommu);
340366f6083SPeter Grehan 
341366f6083SPeter Grehan 	VMCLEANUP(vm->cookie);
342366f6083SPeter Grehan 
343366f6083SPeter Grehan 	free(vm, M_VM);
344366f6083SPeter Grehan }
345366f6083SPeter Grehan 
346366f6083SPeter Grehan const char *
347366f6083SPeter Grehan vm_name(struct vm *vm)
348366f6083SPeter Grehan {
349366f6083SPeter Grehan 	return (vm->name);
350366f6083SPeter Grehan }
351366f6083SPeter Grehan 
352366f6083SPeter Grehan int
353366f6083SPeter Grehan vm_map_mmio(struct vm *vm, vm_paddr_t gpa, size_t len, vm_paddr_t hpa)
354366f6083SPeter Grehan {
355366f6083SPeter Grehan 	const boolean_t spok = TRUE;	/* superpage mappings are ok */
356366f6083SPeter Grehan 
357bda273f2SNeel Natu 	return (VMMMAP_SET(vm->cookie, gpa, hpa, len, VM_MEMATTR_UNCACHEABLE,
358366f6083SPeter Grehan 			   VM_PROT_RW, spok));
359366f6083SPeter Grehan }
360366f6083SPeter Grehan 
361366f6083SPeter Grehan int
362366f6083SPeter Grehan vm_unmap_mmio(struct vm *vm, vm_paddr_t gpa, size_t len)
363366f6083SPeter Grehan {
364366f6083SPeter Grehan 	const boolean_t spok = TRUE;	/* superpage mappings are ok */
365366f6083SPeter Grehan 
366bda273f2SNeel Natu 	return (VMMMAP_SET(vm->cookie, gpa, 0, len, 0,
367366f6083SPeter Grehan 			   VM_PROT_NONE, spok));
368366f6083SPeter Grehan }
369366f6083SPeter Grehan 
370341f19c9SNeel Natu /*
371341f19c9SNeel Natu  * Returns TRUE if 'gpa' is available for allocation and FALSE otherwise
372341f19c9SNeel Natu  */
373341f19c9SNeel Natu static boolean_t
374341f19c9SNeel Natu vm_gpa_available(struct vm *vm, vm_paddr_t gpa)
375366f6083SPeter Grehan {
376341f19c9SNeel Natu 	int i;
377341f19c9SNeel Natu 	vm_paddr_t gpabase, gpalimit;
378341f19c9SNeel Natu 
379341f19c9SNeel Natu 	if (gpa & PAGE_MASK)
380341f19c9SNeel Natu 		panic("vm_gpa_available: gpa (0x%016lx) not page aligned", gpa);
381341f19c9SNeel Natu 
382341f19c9SNeel Natu 	for (i = 0; i < vm->num_mem_segs; i++) {
383341f19c9SNeel Natu 		gpabase = vm->mem_segs[i].gpa;
384341f19c9SNeel Natu 		gpalimit = gpabase + vm->mem_segs[i].len;
385341f19c9SNeel Natu 		if (gpa >= gpabase && gpa < gpalimit)
386341f19c9SNeel Natu 			return (FALSE);
387341f19c9SNeel Natu 	}
388341f19c9SNeel Natu 
389341f19c9SNeel Natu 	return (TRUE);
390341f19c9SNeel Natu }
391341f19c9SNeel Natu 
392341f19c9SNeel Natu int
393341f19c9SNeel Natu vm_malloc(struct vm *vm, vm_paddr_t gpa, size_t len)
394341f19c9SNeel Natu {
395341f19c9SNeel Natu 	int error, available, allocated;
396f7d51510SNeel Natu 	struct vm_memory_segment *seg;
397341f19c9SNeel Natu 	vm_paddr_t g, hpa;
3987ce04d0aSNeel Natu 	void *host_domain;
399366f6083SPeter Grehan 
400366f6083SPeter Grehan 	const boolean_t spok = TRUE;	/* superpage mappings are ok */
401366f6083SPeter Grehan 
402341f19c9SNeel Natu 	if ((gpa & PAGE_MASK) || (len & PAGE_MASK) || len == 0)
403341f19c9SNeel Natu 		return (EINVAL);
404341f19c9SNeel Natu 
405341f19c9SNeel Natu 	available = allocated = 0;
406341f19c9SNeel Natu 	g = gpa;
407341f19c9SNeel Natu 	while (g < gpa + len) {
408341f19c9SNeel Natu 		if (vm_gpa_available(vm, g))
409341f19c9SNeel Natu 			available++;
410341f19c9SNeel Natu 		else
411341f19c9SNeel Natu 			allocated++;
412341f19c9SNeel Natu 
413341f19c9SNeel Natu 		g += PAGE_SIZE;
414341f19c9SNeel Natu 	}
415341f19c9SNeel Natu 
416366f6083SPeter Grehan 	/*
417341f19c9SNeel Natu 	 * If there are some allocated and some available pages in the address
418341f19c9SNeel Natu 	 * range then it is an error.
419366f6083SPeter Grehan 	 */
420341f19c9SNeel Natu 	if (allocated && available)
421341f19c9SNeel Natu 		return (EINVAL);
422341f19c9SNeel Natu 
423341f19c9SNeel Natu 	/*
424341f19c9SNeel Natu 	 * If the entire address range being requested has already been
425341f19c9SNeel Natu 	 * allocated then there isn't anything more to do.
426341f19c9SNeel Natu 	 */
427341f19c9SNeel Natu 	if (allocated && available == 0)
428341f19c9SNeel Natu 		return (0);
429366f6083SPeter Grehan 
430366f6083SPeter Grehan 	if (vm->num_mem_segs >= VM_MAX_MEMORY_SEGMENTS)
431366f6083SPeter Grehan 		return (E2BIG);
432366f6083SPeter Grehan 
4337ce04d0aSNeel Natu 	host_domain = iommu_host_domain();
4347ce04d0aSNeel Natu 
435f7d51510SNeel Natu 	seg = &vm->mem_segs[vm->num_mem_segs];
436366f6083SPeter Grehan 
4377ce04d0aSNeel Natu 	error = 0;
438f7d51510SNeel Natu 	seg->gpa = gpa;
439f7d51510SNeel Natu 	seg->len = 0;
440f7d51510SNeel Natu 	while (seg->len < len) {
441f7d51510SNeel Natu 		hpa = vmm_mem_alloc(PAGE_SIZE);
442f7d51510SNeel Natu 		if (hpa == 0) {
443f7d51510SNeel Natu 			error = ENOMEM;
444f7d51510SNeel Natu 			break;
445f7d51510SNeel Natu 		}
446f7d51510SNeel Natu 
447f7d51510SNeel Natu 		error = VMMMAP_SET(vm->cookie, gpa + seg->len, hpa, PAGE_SIZE,
448f7d51510SNeel Natu 				   VM_MEMATTR_WRITE_BACK, VM_PROT_ALL, spok);
449f7d51510SNeel Natu 		if (error)
450f7d51510SNeel Natu 			break;
451f7d51510SNeel Natu 
4527ce04d0aSNeel Natu 		/*
4537ce04d0aSNeel Natu 		 * Remove the 1:1 mapping for 'hpa' from the 'host_domain'.
4547ce04d0aSNeel Natu 		 * Add mapping for 'gpa + seg->len' to 'hpa' in the VMs domain.
4557ce04d0aSNeel Natu 		 */
4567ce04d0aSNeel Natu 		iommu_remove_mapping(host_domain, hpa, PAGE_SIZE);
457f7d51510SNeel Natu 		iommu_create_mapping(vm->iommu, gpa + seg->len, hpa, PAGE_SIZE);
458f7d51510SNeel Natu 
459f7d51510SNeel Natu 		seg->len += PAGE_SIZE;
460f7d51510SNeel Natu 	}
461f7d51510SNeel Natu 
4627ce04d0aSNeel Natu 	if (error) {
463f7d51510SNeel Natu 		vm_free_mem_seg(vm, seg);
464366f6083SPeter Grehan 		return (error);
465366f6083SPeter Grehan 	}
466366f6083SPeter Grehan 
4677ce04d0aSNeel Natu 	/*
4687ce04d0aSNeel Natu 	 * Invalidate cached translations associated with 'host_domain' since
4697ce04d0aSNeel Natu 	 * we have now moved some pages from it.
4707ce04d0aSNeel Natu 	 */
4717ce04d0aSNeel Natu 	iommu_invalidate_tlb(host_domain);
4727ce04d0aSNeel Natu 
473366f6083SPeter Grehan 	vm->num_mem_segs++;
474341f19c9SNeel Natu 
475366f6083SPeter Grehan 	return (0);
476366f6083SPeter Grehan }
477366f6083SPeter Grehan 
478366f6083SPeter Grehan vm_paddr_t
479366f6083SPeter Grehan vm_gpa2hpa(struct vm *vm, vm_paddr_t gpa, size_t len)
480366f6083SPeter Grehan {
4814db4fb2cSNeel Natu 	vm_paddr_t nextpage;
4824db4fb2cSNeel Natu 
4834db4fb2cSNeel Natu 	nextpage = rounddown(gpa + PAGE_SIZE, PAGE_SIZE);
4844db4fb2cSNeel Natu 	if (len > nextpage - gpa)
4854db4fb2cSNeel Natu 		panic("vm_gpa2hpa: invalid gpa/len: 0x%016lx/%lu", gpa, len);
486366f6083SPeter Grehan 
487bda273f2SNeel Natu 	return (VMMMAP_GET(vm->cookie, gpa));
488366f6083SPeter Grehan }
489366f6083SPeter Grehan 
490366f6083SPeter Grehan int
491366f6083SPeter Grehan vm_gpabase2memseg(struct vm *vm, vm_paddr_t gpabase,
492366f6083SPeter Grehan 		  struct vm_memory_segment *seg)
493366f6083SPeter Grehan {
494366f6083SPeter Grehan 	int i;
495366f6083SPeter Grehan 
496366f6083SPeter Grehan 	for (i = 0; i < vm->num_mem_segs; i++) {
497366f6083SPeter Grehan 		if (gpabase == vm->mem_segs[i].gpa) {
498366f6083SPeter Grehan 			*seg = vm->mem_segs[i];
499366f6083SPeter Grehan 			return (0);
500366f6083SPeter Grehan 		}
501366f6083SPeter Grehan 	}
502366f6083SPeter Grehan 	return (-1);
503366f6083SPeter Grehan }
504366f6083SPeter Grehan 
505366f6083SPeter Grehan int
506366f6083SPeter Grehan vm_get_register(struct vm *vm, int vcpu, int reg, uint64_t *retval)
507366f6083SPeter Grehan {
508366f6083SPeter Grehan 
509366f6083SPeter Grehan 	if (vcpu < 0 || vcpu >= VM_MAXCPU)
510366f6083SPeter Grehan 		return (EINVAL);
511366f6083SPeter Grehan 
512366f6083SPeter Grehan 	if (reg >= VM_REG_LAST)
513366f6083SPeter Grehan 		return (EINVAL);
514366f6083SPeter Grehan 
515366f6083SPeter Grehan 	return (VMGETREG(vm->cookie, vcpu, reg, retval));
516366f6083SPeter Grehan }
517366f6083SPeter Grehan 
518366f6083SPeter Grehan int
519366f6083SPeter Grehan vm_set_register(struct vm *vm, int vcpu, int reg, uint64_t val)
520366f6083SPeter Grehan {
521366f6083SPeter Grehan 
522366f6083SPeter Grehan 	if (vcpu < 0 || vcpu >= VM_MAXCPU)
523366f6083SPeter Grehan 		return (EINVAL);
524366f6083SPeter Grehan 
525366f6083SPeter Grehan 	if (reg >= VM_REG_LAST)
526366f6083SPeter Grehan 		return (EINVAL);
527366f6083SPeter Grehan 
528366f6083SPeter Grehan 	return (VMSETREG(vm->cookie, vcpu, reg, val));
529366f6083SPeter Grehan }
530366f6083SPeter Grehan 
531366f6083SPeter Grehan static boolean_t
532366f6083SPeter Grehan is_descriptor_table(int reg)
533366f6083SPeter Grehan {
534366f6083SPeter Grehan 
535366f6083SPeter Grehan 	switch (reg) {
536366f6083SPeter Grehan 	case VM_REG_GUEST_IDTR:
537366f6083SPeter Grehan 	case VM_REG_GUEST_GDTR:
538366f6083SPeter Grehan 		return (TRUE);
539366f6083SPeter Grehan 	default:
540366f6083SPeter Grehan 		return (FALSE);
541366f6083SPeter Grehan 	}
542366f6083SPeter Grehan }
543366f6083SPeter Grehan 
544366f6083SPeter Grehan static boolean_t
545366f6083SPeter Grehan is_segment_register(int reg)
546366f6083SPeter Grehan {
547366f6083SPeter Grehan 
548366f6083SPeter Grehan 	switch (reg) {
549366f6083SPeter Grehan 	case VM_REG_GUEST_ES:
550366f6083SPeter Grehan 	case VM_REG_GUEST_CS:
551366f6083SPeter Grehan 	case VM_REG_GUEST_SS:
552366f6083SPeter Grehan 	case VM_REG_GUEST_DS:
553366f6083SPeter Grehan 	case VM_REG_GUEST_FS:
554366f6083SPeter Grehan 	case VM_REG_GUEST_GS:
555366f6083SPeter Grehan 	case VM_REG_GUEST_TR:
556366f6083SPeter Grehan 	case VM_REG_GUEST_LDTR:
557366f6083SPeter Grehan 		return (TRUE);
558366f6083SPeter Grehan 	default:
559366f6083SPeter Grehan 		return (FALSE);
560366f6083SPeter Grehan 	}
561366f6083SPeter Grehan }
562366f6083SPeter Grehan 
563366f6083SPeter Grehan int
564366f6083SPeter Grehan vm_get_seg_desc(struct vm *vm, int vcpu, int reg,
565366f6083SPeter Grehan 		struct seg_desc *desc)
566366f6083SPeter Grehan {
567366f6083SPeter Grehan 
568366f6083SPeter Grehan 	if (vcpu < 0 || vcpu >= VM_MAXCPU)
569366f6083SPeter Grehan 		return (EINVAL);
570366f6083SPeter Grehan 
571366f6083SPeter Grehan 	if (!is_segment_register(reg) && !is_descriptor_table(reg))
572366f6083SPeter Grehan 		return (EINVAL);
573366f6083SPeter Grehan 
574366f6083SPeter Grehan 	return (VMGETDESC(vm->cookie, vcpu, reg, desc));
575366f6083SPeter Grehan }
576366f6083SPeter Grehan 
577366f6083SPeter Grehan int
578366f6083SPeter Grehan vm_set_seg_desc(struct vm *vm, int vcpu, int reg,
579366f6083SPeter Grehan 		struct seg_desc *desc)
580366f6083SPeter Grehan {
581366f6083SPeter Grehan 	if (vcpu < 0 || vcpu >= VM_MAXCPU)
582366f6083SPeter Grehan 		return (EINVAL);
583366f6083SPeter Grehan 
584366f6083SPeter Grehan 	if (!is_segment_register(reg) && !is_descriptor_table(reg))
585366f6083SPeter Grehan 		return (EINVAL);
586366f6083SPeter Grehan 
587366f6083SPeter Grehan 	return (VMSETDESC(vm->cookie, vcpu, reg, desc));
588366f6083SPeter Grehan }
589366f6083SPeter Grehan 
590366f6083SPeter Grehan int
591366f6083SPeter Grehan vm_get_pinning(struct vm *vm, int vcpuid, int *cpuid)
592366f6083SPeter Grehan {
593366f6083SPeter Grehan 
594366f6083SPeter Grehan 	if (vcpuid < 0 || vcpuid >= VM_MAXCPU)
595366f6083SPeter Grehan 		return (EINVAL);
596366f6083SPeter Grehan 
597366f6083SPeter Grehan 	*cpuid = VCPU_PINCPU(vm, vcpuid);
598366f6083SPeter Grehan 
599366f6083SPeter Grehan 	return (0);
600366f6083SPeter Grehan }
601366f6083SPeter Grehan 
602366f6083SPeter Grehan int
603366f6083SPeter Grehan vm_set_pinning(struct vm *vm, int vcpuid, int host_cpuid)
604366f6083SPeter Grehan {
605366f6083SPeter Grehan 	struct thread *td;
606366f6083SPeter Grehan 
607366f6083SPeter Grehan 	if (vcpuid < 0 || vcpuid >= VM_MAXCPU)
608366f6083SPeter Grehan 		return (EINVAL);
609366f6083SPeter Grehan 
610366f6083SPeter Grehan 	td = curthread;		/* XXXSMP only safe when muxing vcpus */
611366f6083SPeter Grehan 
612366f6083SPeter Grehan 	/* unpin */
613366f6083SPeter Grehan 	if (host_cpuid < 0) {
614366f6083SPeter Grehan 		VCPU_UNPIN(vm, vcpuid);
615366f6083SPeter Grehan 		thread_lock(td);
616366f6083SPeter Grehan 		sched_unbind(td);
617366f6083SPeter Grehan 		thread_unlock(td);
618366f6083SPeter Grehan 		return (0);
619366f6083SPeter Grehan 	}
620366f6083SPeter Grehan 
621366f6083SPeter Grehan 	if (CPU_ABSENT(host_cpuid))
622366f6083SPeter Grehan 		return (EINVAL);
623366f6083SPeter Grehan 
624366f6083SPeter Grehan 	/*
625366f6083SPeter Grehan 	 * XXX we should check that 'host_cpuid' has not already been pinned
626366f6083SPeter Grehan 	 * by another vm.
627366f6083SPeter Grehan 	 */
628366f6083SPeter Grehan 	thread_lock(td);
629366f6083SPeter Grehan 	sched_bind(td, host_cpuid);
630366f6083SPeter Grehan 	thread_unlock(td);
631366f6083SPeter Grehan 	VCPU_PIN(vm, vcpuid, host_cpuid);
632366f6083SPeter Grehan 
633366f6083SPeter Grehan 	return (0);
634366f6083SPeter Grehan }
635366f6083SPeter Grehan 
636366f6083SPeter Grehan static void
637366f6083SPeter Grehan restore_guest_fpustate(struct vcpu *vcpu)
638366f6083SPeter Grehan {
639366f6083SPeter Grehan 
64038f1b189SPeter Grehan 	/* flush host state to the pcb */
64138f1b189SPeter Grehan 	fpuexit(curthread);
642366f6083SPeter Grehan 	fpu_stop_emulating();
64338f1b189SPeter Grehan 	fpurestore(vcpu->guestfpu);
644366f6083SPeter Grehan }
645366f6083SPeter Grehan 
646366f6083SPeter Grehan static void
647366f6083SPeter Grehan save_guest_fpustate(struct vcpu *vcpu)
648366f6083SPeter Grehan {
649366f6083SPeter Grehan 
65038f1b189SPeter Grehan 	fpusave(vcpu->guestfpu);
651366f6083SPeter Grehan 	fpu_start_emulating();
652366f6083SPeter Grehan }
653366f6083SPeter Grehan 
654366f6083SPeter Grehan int
655366f6083SPeter Grehan vm_run(struct vm *vm, struct vm_run *vmrun)
656366f6083SPeter Grehan {
657366f6083SPeter Grehan 	int error, vcpuid;
658366f6083SPeter Grehan 	struct vcpu *vcpu;
659366f6083SPeter Grehan 	struct pcb *pcb;
660366f6083SPeter Grehan 	uint64_t tscval;
661366f6083SPeter Grehan 
662366f6083SPeter Grehan 	vcpuid = vmrun->cpuid;
663366f6083SPeter Grehan 
664366f6083SPeter Grehan 	if (vcpuid < 0 || vcpuid >= VM_MAXCPU)
665366f6083SPeter Grehan 		return (EINVAL);
666366f6083SPeter Grehan 
667366f6083SPeter Grehan 	vcpu = &vm->vcpu[vcpuid];
668366f6083SPeter Grehan 
669366f6083SPeter Grehan 	critical_enter();
670366f6083SPeter Grehan 
671366f6083SPeter Grehan 	tscval = rdtsc();
672366f6083SPeter Grehan 
673366f6083SPeter Grehan 	pcb = PCPU_GET(curpcb);
67434a6b2d6SJohn Baldwin 	set_pcb_flags(pcb, PCB_FULL_IRET);
675366f6083SPeter Grehan 
676366f6083SPeter Grehan 	restore_guest_msrs(vm, vcpuid);
677366f6083SPeter Grehan 	restore_guest_fpustate(vcpu);
67875dd3366SNeel Natu 
67975dd3366SNeel Natu 	vcpu->hostcpu = curcpu;
68098ed632cSNeel Natu 	error = VMRUN(vm->cookie, vcpuid, vmrun->rip);
68175dd3366SNeel Natu 	vcpu->hostcpu = NOCPU;
68275dd3366SNeel Natu 
683366f6083SPeter Grehan 	save_guest_fpustate(vcpu);
684366f6083SPeter Grehan 	restore_host_msrs(vm, vcpuid);
685366f6083SPeter Grehan 
686366f6083SPeter Grehan 	vmm_stat_incr(vm, vcpuid, VCPU_TOTAL_RUNTIME, rdtsc() - tscval);
687366f6083SPeter Grehan 
68898ed632cSNeel Natu 	/* copy the exit information */
68998ed632cSNeel Natu 	bcopy(&vcpu->exitinfo, &vmrun->vm_exit, sizeof(struct vm_exit));
69098ed632cSNeel Natu 
691366f6083SPeter Grehan 	critical_exit();
692366f6083SPeter Grehan 
693366f6083SPeter Grehan 	return (error);
694366f6083SPeter Grehan }
695366f6083SPeter Grehan 
696366f6083SPeter Grehan int
697366f6083SPeter Grehan vm_inject_event(struct vm *vm, int vcpuid, int type,
698366f6083SPeter Grehan 		int vector, uint32_t code, int code_valid)
699366f6083SPeter Grehan {
700366f6083SPeter Grehan 	if (vcpuid < 0 || vcpuid >= VM_MAXCPU)
701366f6083SPeter Grehan 		return (EINVAL);
702366f6083SPeter Grehan 
703366f6083SPeter Grehan 	if ((type > VM_EVENT_NONE && type < VM_EVENT_MAX) == 0)
704366f6083SPeter Grehan 		return (EINVAL);
705366f6083SPeter Grehan 
706366f6083SPeter Grehan 	if (vector < 0 || vector > 255)
707366f6083SPeter Grehan 		return (EINVAL);
708366f6083SPeter Grehan 
709366f6083SPeter Grehan 	return (VMINJECT(vm->cookie, vcpuid, type, vector, code, code_valid));
710366f6083SPeter Grehan }
711366f6083SPeter Grehan 
712*f352ff0cSNeel Natu VMM_STAT_DEFINE(VCPU_NMI_COUNT, "number of NMIs delivered to vcpu");
713366f6083SPeter Grehan 
714*f352ff0cSNeel Natu int
715*f352ff0cSNeel Natu vm_inject_nmi(struct vm *vm, int vcpuid)
716*f352ff0cSNeel Natu {
717*f352ff0cSNeel Natu 	struct vcpu *vcpu;
718*f352ff0cSNeel Natu 
719*f352ff0cSNeel Natu 	if (vcpuid < 0 || vcpuid >= VM_MAXCPU)
720366f6083SPeter Grehan 		return (EINVAL);
721366f6083SPeter Grehan 
722*f352ff0cSNeel Natu 	vcpu = &vm->vcpu[vcpuid];
723*f352ff0cSNeel Natu 
724*f352ff0cSNeel Natu 	vcpu->nmi_pending = 1;
725*f352ff0cSNeel Natu 	vm_interrupt_hostcpu(vm, vcpuid);
726*f352ff0cSNeel Natu 	return (0);
727*f352ff0cSNeel Natu }
728*f352ff0cSNeel Natu 
729*f352ff0cSNeel Natu int
730*f352ff0cSNeel Natu vm_nmi_pending(struct vm *vm, int vcpuid)
731*f352ff0cSNeel Natu {
732*f352ff0cSNeel Natu 	struct vcpu *vcpu;
733*f352ff0cSNeel Natu 
734*f352ff0cSNeel Natu 	if (vcpuid < 0 || vcpuid >= VM_MAXCPU)
735*f352ff0cSNeel Natu 		panic("vm_nmi_pending: invalid vcpuid %d", vcpuid);
736*f352ff0cSNeel Natu 
737*f352ff0cSNeel Natu 	vcpu = &vm->vcpu[vcpuid];
738*f352ff0cSNeel Natu 
739*f352ff0cSNeel Natu 	return (vcpu->nmi_pending);
740*f352ff0cSNeel Natu }
741*f352ff0cSNeel Natu 
742*f352ff0cSNeel Natu void
743*f352ff0cSNeel Natu vm_nmi_clear(struct vm *vm, int vcpuid)
744*f352ff0cSNeel Natu {
745*f352ff0cSNeel Natu 	struct vcpu *vcpu;
746*f352ff0cSNeel Natu 
747*f352ff0cSNeel Natu 	if (vcpuid < 0 || vcpuid >= VM_MAXCPU)
748*f352ff0cSNeel Natu 		panic("vm_nmi_pending: invalid vcpuid %d", vcpuid);
749*f352ff0cSNeel Natu 
750*f352ff0cSNeel Natu 	vcpu = &vm->vcpu[vcpuid];
751*f352ff0cSNeel Natu 
752*f352ff0cSNeel Natu 	if (vcpu->nmi_pending == 0)
753*f352ff0cSNeel Natu 		panic("vm_nmi_clear: inconsistent nmi_pending state");
754*f352ff0cSNeel Natu 
755*f352ff0cSNeel Natu 	vcpu->nmi_pending = 0;
756*f352ff0cSNeel Natu 	vmm_stat_incr(vm, vcpuid, VCPU_NMI_COUNT, 1);
757366f6083SPeter Grehan }
758366f6083SPeter Grehan 
759366f6083SPeter Grehan int
760366f6083SPeter Grehan vm_get_capability(struct vm *vm, int vcpu, int type, int *retval)
761366f6083SPeter Grehan {
762366f6083SPeter Grehan 	if (vcpu < 0 || vcpu >= VM_MAXCPU)
763366f6083SPeter Grehan 		return (EINVAL);
764366f6083SPeter Grehan 
765366f6083SPeter Grehan 	if (type < 0 || type >= VM_CAP_MAX)
766366f6083SPeter Grehan 		return (EINVAL);
767366f6083SPeter Grehan 
768366f6083SPeter Grehan 	return (VMGETCAP(vm->cookie, vcpu, type, retval));
769366f6083SPeter Grehan }
770366f6083SPeter Grehan 
771366f6083SPeter Grehan int
772366f6083SPeter Grehan vm_set_capability(struct vm *vm, int vcpu, int type, int val)
773366f6083SPeter Grehan {
774366f6083SPeter Grehan 	if (vcpu < 0 || vcpu >= VM_MAXCPU)
775366f6083SPeter Grehan 		return (EINVAL);
776366f6083SPeter Grehan 
777366f6083SPeter Grehan 	if (type < 0 || type >= VM_CAP_MAX)
778366f6083SPeter Grehan 		return (EINVAL);
779366f6083SPeter Grehan 
780366f6083SPeter Grehan 	return (VMSETCAP(vm->cookie, vcpu, type, val));
781366f6083SPeter Grehan }
782366f6083SPeter Grehan 
783366f6083SPeter Grehan uint64_t *
784366f6083SPeter Grehan vm_guest_msrs(struct vm *vm, int cpu)
785366f6083SPeter Grehan {
786366f6083SPeter Grehan 	return (vm->vcpu[cpu].guest_msrs);
787366f6083SPeter Grehan }
788366f6083SPeter Grehan 
789366f6083SPeter Grehan struct vlapic *
790366f6083SPeter Grehan vm_lapic(struct vm *vm, int cpu)
791366f6083SPeter Grehan {
792366f6083SPeter Grehan 	return (vm->vcpu[cpu].vlapic);
793366f6083SPeter Grehan }
794366f6083SPeter Grehan 
795366f6083SPeter Grehan boolean_t
796366f6083SPeter Grehan vmm_is_pptdev(int bus, int slot, int func)
797366f6083SPeter Grehan {
798366f6083SPeter Grehan 	int found, b, s, f, n;
799366f6083SPeter Grehan 	char *val, *cp, *cp2;
800366f6083SPeter Grehan 
801366f6083SPeter Grehan 	/*
802366f6083SPeter Grehan 	 * setenv pptdevs "1/2/3 4/5/6 7/8/9 10/11/12"
803366f6083SPeter Grehan 	 */
804366f6083SPeter Grehan 	found = 0;
805366f6083SPeter Grehan 	cp = val = getenv("pptdevs");
806366f6083SPeter Grehan 	while (cp != NULL && *cp != '\0') {
807366f6083SPeter Grehan 		if ((cp2 = strchr(cp, ' ')) != NULL)
808366f6083SPeter Grehan 			*cp2 = '\0';
809366f6083SPeter Grehan 
810366f6083SPeter Grehan 		n = sscanf(cp, "%d/%d/%d", &b, &s, &f);
811366f6083SPeter Grehan 		if (n == 3 && bus == b && slot == s && func == f) {
812366f6083SPeter Grehan 			found = 1;
813366f6083SPeter Grehan 			break;
814366f6083SPeter Grehan 		}
815366f6083SPeter Grehan 
816366f6083SPeter Grehan 		if (cp2 != NULL)
817366f6083SPeter Grehan 			*cp2++ = ' ';
818366f6083SPeter Grehan 
819366f6083SPeter Grehan 		cp = cp2;
820366f6083SPeter Grehan 	}
821366f6083SPeter Grehan 	freeenv(val);
822366f6083SPeter Grehan 	return (found);
823366f6083SPeter Grehan }
824366f6083SPeter Grehan 
825366f6083SPeter Grehan void *
826366f6083SPeter Grehan vm_iommu_domain(struct vm *vm)
827366f6083SPeter Grehan {
828366f6083SPeter Grehan 
829366f6083SPeter Grehan 	return (vm->iommu);
830366f6083SPeter Grehan }
831366f6083SPeter Grehan 
83275dd3366SNeel Natu int
83375dd3366SNeel Natu vcpu_set_state(struct vm *vm, int vcpuid, enum vcpu_state state)
834366f6083SPeter Grehan {
83575dd3366SNeel Natu 	int error;
836366f6083SPeter Grehan 	struct vcpu *vcpu;
837366f6083SPeter Grehan 
838366f6083SPeter Grehan 	if (vcpuid < 0 || vcpuid >= VM_MAXCPU)
839366f6083SPeter Grehan 		panic("vm_set_run_state: invalid vcpuid %d", vcpuid);
840366f6083SPeter Grehan 
841366f6083SPeter Grehan 	vcpu = &vm->vcpu[vcpuid];
842366f6083SPeter Grehan 
84375dd3366SNeel Natu 	vcpu_lock(vcpu);
84475dd3366SNeel Natu 
84575dd3366SNeel Natu 	/*
84675dd3366SNeel Natu 	 * The following state transitions are allowed:
84775dd3366SNeel Natu 	 * IDLE -> RUNNING -> IDLE
84875dd3366SNeel Natu 	 * IDLE -> CANNOT_RUN -> IDLE
84975dd3366SNeel Natu 	 */
85075dd3366SNeel Natu 	if ((vcpu->state == VCPU_IDLE && state != VCPU_IDLE) ||
85175dd3366SNeel Natu 	    (vcpu->state != VCPU_IDLE && state == VCPU_IDLE)) {
85275dd3366SNeel Natu 		error = 0;
85375dd3366SNeel Natu 		vcpu->state = state;
854366f6083SPeter Grehan 	} else {
85575dd3366SNeel Natu 		error = EBUSY;
856366f6083SPeter Grehan 	}
857366f6083SPeter Grehan 
85875dd3366SNeel Natu 	vcpu_unlock(vcpu);
85975dd3366SNeel Natu 
86075dd3366SNeel Natu 	return (error);
86175dd3366SNeel Natu }
86275dd3366SNeel Natu 
86375dd3366SNeel Natu enum vcpu_state
86475dd3366SNeel Natu vcpu_get_state(struct vm *vm, int vcpuid)
865366f6083SPeter Grehan {
866366f6083SPeter Grehan 	struct vcpu *vcpu;
86775dd3366SNeel Natu 	enum vcpu_state state;
868366f6083SPeter Grehan 
869366f6083SPeter Grehan 	if (vcpuid < 0 || vcpuid >= VM_MAXCPU)
870366f6083SPeter Grehan 		panic("vm_get_run_state: invalid vcpuid %d", vcpuid);
871366f6083SPeter Grehan 
872366f6083SPeter Grehan 	vcpu = &vm->vcpu[vcpuid];
873366f6083SPeter Grehan 
87475dd3366SNeel Natu 	vcpu_lock(vcpu);
87575dd3366SNeel Natu 	state = vcpu->state;
87675dd3366SNeel Natu 	vcpu_unlock(vcpu);
877366f6083SPeter Grehan 
87875dd3366SNeel Natu 	return (state);
879366f6083SPeter Grehan }
880366f6083SPeter Grehan 
881366f6083SPeter Grehan void
882366f6083SPeter Grehan vm_activate_cpu(struct vm *vm, int vcpuid)
883366f6083SPeter Grehan {
884366f6083SPeter Grehan 
885366f6083SPeter Grehan 	if (vcpuid >= 0 && vcpuid < VM_MAXCPU)
886a5615c90SPeter Grehan 		CPU_SET(vcpuid, &vm->active_cpus);
887366f6083SPeter Grehan }
888366f6083SPeter Grehan 
889a5615c90SPeter Grehan cpuset_t
890366f6083SPeter Grehan vm_active_cpus(struct vm *vm)
891366f6083SPeter Grehan {
892366f6083SPeter Grehan 
893366f6083SPeter Grehan 	return (vm->active_cpus);
894366f6083SPeter Grehan }
895366f6083SPeter Grehan 
896366f6083SPeter Grehan void *
897366f6083SPeter Grehan vcpu_stats(struct vm *vm, int vcpuid)
898366f6083SPeter Grehan {
899366f6083SPeter Grehan 
900366f6083SPeter Grehan 	return (vm->vcpu[vcpuid].stats);
901366f6083SPeter Grehan }
902e9027382SNeel Natu 
903e9027382SNeel Natu int
904e9027382SNeel Natu vm_get_x2apic_state(struct vm *vm, int vcpuid, enum x2apic_state *state)
905e9027382SNeel Natu {
906e9027382SNeel Natu 	if (vcpuid < 0 || vcpuid >= VM_MAXCPU)
907e9027382SNeel Natu 		return (EINVAL);
908e9027382SNeel Natu 
909e9027382SNeel Natu 	*state = vm->vcpu[vcpuid].x2apic_state;
910e9027382SNeel Natu 
911e9027382SNeel Natu 	return (0);
912e9027382SNeel Natu }
913e9027382SNeel Natu 
914e9027382SNeel Natu int
915e9027382SNeel Natu vm_set_x2apic_state(struct vm *vm, int vcpuid, enum x2apic_state state)
916e9027382SNeel Natu {
917e9027382SNeel Natu 	if (vcpuid < 0 || vcpuid >= VM_MAXCPU)
918e9027382SNeel Natu 		return (EINVAL);
919e9027382SNeel Natu 
920e9027382SNeel Natu 	if (state < 0 || state >= X2APIC_STATE_LAST)
921e9027382SNeel Natu 		return (EINVAL);
922e9027382SNeel Natu 
923e9027382SNeel Natu 	vm->vcpu[vcpuid].x2apic_state = state;
924e9027382SNeel Natu 
92573820fb0SNeel Natu 	vlapic_set_x2apic_state(vm, vcpuid, state);
92673820fb0SNeel Natu 
927e9027382SNeel Natu 	return (0);
928e9027382SNeel Natu }
92975dd3366SNeel Natu 
93075dd3366SNeel Natu void
93175dd3366SNeel Natu vm_interrupt_hostcpu(struct vm *vm, int vcpuid)
93275dd3366SNeel Natu {
93375dd3366SNeel Natu 	int hostcpu;
93475dd3366SNeel Natu 	struct vcpu *vcpu;
93575dd3366SNeel Natu 
93675dd3366SNeel Natu 	vcpu = &vm->vcpu[vcpuid];
93775dd3366SNeel Natu 
93875dd3366SNeel Natu 	/*
93975dd3366SNeel Natu 	 * XXX racy but the worst case is that we'll send an unnecessary IPI
94075dd3366SNeel Natu 	 * to the 'hostcpu'.
94175dd3366SNeel Natu 	 *
94275dd3366SNeel Natu 	 * We cannot use vcpu_is_running() here because it acquires vcpu->mtx
94375dd3366SNeel Natu 	 * which is not allowed inside a critical section.
94475dd3366SNeel Natu 	 */
94575dd3366SNeel Natu 	hostcpu = vcpu->hostcpu;
94675dd3366SNeel Natu 	if (hostcpu == NOCPU || hostcpu == curcpu)
94775dd3366SNeel Natu 		return;
94875dd3366SNeel Natu 
94975dd3366SNeel Natu 	ipi_cpu(hostcpu, vmm_ipinum);
95075dd3366SNeel Natu }
951