xref: /freebsd/sys/dev/vmm/vmm_dev.c (revision c76c2a19ae3763d17aa6a60a5831ed24cbc16e83)
1 b9ef152bSMark Johnston /*-
2 b9ef152bSMark Johnston  * SPDX-License-Identifier: BSD-2-Clause
3 b9ef152bSMark Johnston  *
4 b9ef152bSMark Johnston  * Copyright (c) 2011 NetApp, Inc.
5 b9ef152bSMark Johnston  * Copyright (C) 2015 Mihai Carabas <mihai.carabas@gmail.com>
6 b9ef152bSMark Johnston  * All rights reserved.
7 b9ef152bSMark Johnston  */
8 b9ef152bSMark Johnston 
9 b9ef152bSMark Johnston #include <sys/param.h>
10 b9ef152bSMark Johnston #include <sys/conf.h>
11 a97f683fSMark Johnston #include <sys/fcntl.h>
12 b9ef152bSMark Johnston #include <sys/ioccom.h>
13 b9ef152bSMark Johnston #include <sys/jail.h>
14 b9ef152bSMark Johnston #include <sys/kernel.h>
15 b9ef152bSMark Johnston #include <sys/malloc.h>
16 b9ef152bSMark Johnston #include <sys/mman.h>
17 b9ef152bSMark Johnston #include <sys/proc.h>
18 b9ef152bSMark Johnston #include <sys/queue.h>
19 887c0877SMark Johnston #include <sys/sx.h>
20 b9ef152bSMark Johnston #include <sys/sysctl.h>
21 b9ef152bSMark Johnston #include <sys/ucred.h>
22 b9ef152bSMark Johnston #include <sys/uio.h>
23 b9ef152bSMark Johnston 
24 b9ef152bSMark Johnston #include <machine/vmm.h>
25 b9ef152bSMark Johnston 
26 b9ef152bSMark Johnston #include <vm/vm.h>
27 b9ef152bSMark Johnston #include <vm/vm_object.h>
28 b9ef152bSMark Johnston 
29 b9ef152bSMark Johnston #include <dev/vmm/vmm_dev.h>
30 *c76c2a19SMark Johnston #include <dev/vmm/vmm_mem.h>
31 b9ef152bSMark Johnston #include <dev/vmm/vmm_stat.h>
32 b9ef152bSMark Johnston 
33 e12b6aafSMark Johnston #if defined(__amd64__) && defined(COMPAT_FREEBSD12)
34 a852dc58SMark Johnston struct vm_memseg_12 {
35 e12b6aafSMark Johnston 	int		segid;
36 e12b6aafSMark Johnston 	size_t		len;
37 e12b6aafSMark Johnston 	char		name[64];
38 e12b6aafSMark Johnston };
39 a852dc58SMark Johnston _Static_assert(sizeof(struct vm_memseg_12) == 80, "COMPAT_FREEBSD12 ABI");
40 e12b6aafSMark Johnston 
41 a852dc58SMark Johnston #define	VM_ALLOC_MEMSEG_12	\
42 a852dc58SMark Johnston 	_IOW('v', IOCNUM_ALLOC_MEMSEG, struct vm_memseg_12)
43 a852dc58SMark Johnston #define	VM_GET_MEMSEG_12	\
44 a852dc58SMark Johnston 	_IOWR('v', IOCNUM_GET_MEMSEG, struct vm_memseg_12)
45 e12b6aafSMark Johnston #endif
46 e12b6aafSMark Johnston 
47 b9ef152bSMark Johnston struct devmem_softc {
48 b9ef152bSMark Johnston 	int	segid;
49 b9ef152bSMark Johnston 	char	*name;
50 b9ef152bSMark Johnston 	struct cdev *cdev;
51 b9ef152bSMark Johnston 	struct vmmdev_softc *sc;
52 b9ef152bSMark Johnston 	SLIST_ENTRY(devmem_softc) link;
53 b9ef152bSMark Johnston };
54 b9ef152bSMark Johnston 
55 b9ef152bSMark Johnston struct vmmdev_softc {
56 b9ef152bSMark Johnston 	struct vm	*vm;		/* vm instance cookie */
57 b9ef152bSMark Johnston 	struct cdev	*cdev;
58 b9ef152bSMark Johnston 	struct ucred	*ucred;
59 b9ef152bSMark Johnston 	SLIST_ENTRY(vmmdev_softc) link;
60 b9ef152bSMark Johnston 	SLIST_HEAD(, devmem_softc) devmem;
61 b9ef152bSMark Johnston 	int		flags;
62 b9ef152bSMark Johnston };
63 b9ef152bSMark Johnston 
64 b9ef152bSMark Johnston static SLIST_HEAD(, vmmdev_softc) head;
65 b9ef152bSMark Johnston 
66 b9ef152bSMark Johnston static unsigned pr_allow_flag;
67 887c0877SMark Johnston static struct sx vmmdev_mtx;
68 887c0877SMark Johnston SX_SYSINIT(vmmdev_mtx, &vmmdev_mtx, "vmm device mutex");
69 b9ef152bSMark Johnston 
70 b9ef152bSMark Johnston static MALLOC_DEFINE(M_VMMDEV, "vmmdev", "vmmdev");
71 b9ef152bSMark Johnston 
72 b9ef152bSMark Johnston SYSCTL_DECL(_hw_vmm);
73 b9ef152bSMark Johnston 
74 b9ef152bSMark Johnston static void devmem_destroy(void *arg);
75 f4002135SMark Johnston static int devmem_create_cdev(struct vmmdev_softc *sc, int id, char *devmem);
76 b9ef152bSMark Johnston 
77 b9ef152bSMark Johnston static int
vmm_priv_check(struct ucred * ucred)78 b9ef152bSMark Johnston vmm_priv_check(struct ucred *ucred)
79 b9ef152bSMark Johnston {
80 b9ef152bSMark Johnston 	if (jailed(ucred) &&
81 b9ef152bSMark Johnston 	    !(ucred->cr_prison->pr_allow & pr_allow_flag))
82 b9ef152bSMark Johnston 		return (EPERM);
83 b9ef152bSMark Johnston 
84 b9ef152bSMark Johnston 	return (0);
85 b9ef152bSMark Johnston }
86 b9ef152bSMark Johnston 
87 b9ef152bSMark Johnston static int
vcpu_lock_one(struct vcpu * vcpu)88 b9ef152bSMark Johnston vcpu_lock_one(struct vcpu *vcpu)
89 b9ef152bSMark Johnston {
90 b9ef152bSMark Johnston 	return (vcpu_set_state(vcpu, VCPU_FROZEN, true));
91 b9ef152bSMark Johnston }
92 b9ef152bSMark Johnston 
93 b9ef152bSMark Johnston static void
vcpu_unlock_one(struct vcpu * vcpu)94 b9ef152bSMark Johnston vcpu_unlock_one(struct vcpu *vcpu)
95 b9ef152bSMark Johnston {
96 b9ef152bSMark Johnston 	enum vcpu_state state;
97 b9ef152bSMark Johnston 
98 b9ef152bSMark Johnston 	state = vcpu_get_state(vcpu, NULL);
99 b9ef152bSMark Johnston 	if (state != VCPU_FROZEN) {
100 b9ef152bSMark Johnston 		panic("vcpu %s(%d) has invalid state %d",
101 b9ef152bSMark Johnston 		    vm_name(vcpu_vm(vcpu)), vcpu_vcpuid(vcpu), state);
102 b9ef152bSMark Johnston 	}
103 b9ef152bSMark Johnston 
104 b9ef152bSMark Johnston 	vcpu_set_state(vcpu, VCPU_IDLE, false);
105 b9ef152bSMark Johnston }
106 b9ef152bSMark Johnston 
107 b9ef152bSMark Johnston static int
vcpu_lock_all(struct vmmdev_softc * sc)108 b9ef152bSMark Johnston vcpu_lock_all(struct vmmdev_softc *sc)
109 b9ef152bSMark Johnston {
110 b9ef152bSMark Johnston 	struct vcpu *vcpu;
111 b9ef152bSMark Johnston 	int error;
112 b9ef152bSMark Johnston 	uint16_t i, j, maxcpus;
113 b9ef152bSMark Johnston 
114 b9ef152bSMark Johnston 	error = 0;
115 b9ef152bSMark Johnston 	vm_slock_vcpus(sc->vm);
116 b9ef152bSMark Johnston 	maxcpus = vm_get_maxcpus(sc->vm);
117 b9ef152bSMark Johnston 	for (i = 0; i < maxcpus; i++) {
118 b9ef152bSMark Johnston 		vcpu = vm_vcpu(sc->vm, i);
119 b9ef152bSMark Johnston 		if (vcpu == NULL)
120 b9ef152bSMark Johnston 			continue;
121 b9ef152bSMark Johnston 		error = vcpu_lock_one(vcpu);
122 b9ef152bSMark Johnston 		if (error)
123 b9ef152bSMark Johnston 			break;
124 b9ef152bSMark Johnston 	}
125 b9ef152bSMark Johnston 
126 b9ef152bSMark Johnston 	if (error) {
127 b9ef152bSMark Johnston 		for (j = 0; j < i; j++) {
128 b9ef152bSMark Johnston 			vcpu = vm_vcpu(sc->vm, j);
129 b9ef152bSMark Johnston 			if (vcpu == NULL)
130 b9ef152bSMark Johnston 				continue;
131 b9ef152bSMark Johnston 			vcpu_unlock_one(vcpu);
132 b9ef152bSMark Johnston 		}
133 b9ef152bSMark Johnston 		vm_unlock_vcpus(sc->vm);
134 b9ef152bSMark Johnston 	}
135 b9ef152bSMark Johnston 
136 b9ef152bSMark Johnston 	return (error);
137 b9ef152bSMark Johnston }
138 b9ef152bSMark Johnston 
139 b9ef152bSMark Johnston static void
vcpu_unlock_all(struct vmmdev_softc * sc)140 b9ef152bSMark Johnston vcpu_unlock_all(struct vmmdev_softc *sc)
141 b9ef152bSMark Johnston {
142 b9ef152bSMark Johnston 	struct vcpu *vcpu;
143 b9ef152bSMark Johnston 	uint16_t i, maxcpus;
144 b9ef152bSMark Johnston 
145 b9ef152bSMark Johnston 	maxcpus = vm_get_maxcpus(sc->vm);
146 b9ef152bSMark Johnston 	for (i = 0; i < maxcpus; i++) {
147 b9ef152bSMark Johnston 		vcpu = vm_vcpu(sc->vm, i);
148 b9ef152bSMark Johnston 		if (vcpu == NULL)
149 b9ef152bSMark Johnston 			continue;
150 b9ef152bSMark Johnston 		vcpu_unlock_one(vcpu);
151 b9ef152bSMark Johnston 	}
152 b9ef152bSMark Johnston 	vm_unlock_vcpus(sc->vm);
153 b9ef152bSMark Johnston }
154 b9ef152bSMark Johnston 
155 b9ef152bSMark Johnston static struct vmmdev_softc *
vmmdev_lookup(const char * name,struct ucred * cred)156 c23da668SMark Johnston vmmdev_lookup(const char *name, struct ucred *cred)
157 b9ef152bSMark Johnston {
158 b9ef152bSMark Johnston 	struct vmmdev_softc *sc;
159 b9ef152bSMark Johnston 
160 887c0877SMark Johnston 	sx_assert(&vmmdev_mtx, SA_XLOCKED);
161 b9ef152bSMark Johnston 
162 b9ef152bSMark Johnston 	SLIST_FOREACH(sc, &head, link) {
163 b9ef152bSMark Johnston 		if (strcmp(name, vm_name(sc->vm)) == 0)
164 b9ef152bSMark Johnston 			break;
165 b9ef152bSMark Johnston 	}
166 b9ef152bSMark Johnston 
167 b9ef152bSMark Johnston 	if (sc == NULL)
168 b9ef152bSMark Johnston 		return (NULL);
169 b9ef152bSMark Johnston 
170 c23da668SMark Johnston 	if (cr_cansee(cred, sc->ucred))
171 b9ef152bSMark Johnston 		return (NULL);
172 b9ef152bSMark Johnston 
173 b9ef152bSMark Johnston 	return (sc);
174 b9ef152bSMark Johnston }
175 b9ef152bSMark Johnston 
176 b9ef152bSMark Johnston static struct vmmdev_softc *
vmmdev_lookup2(struct cdev * cdev)177 b9ef152bSMark Johnston vmmdev_lookup2(struct cdev *cdev)
178 b9ef152bSMark Johnston {
179 b9ef152bSMark Johnston 	return (cdev->si_drv1);
180 b9ef152bSMark Johnston }
181 b9ef152bSMark Johnston 
182 b9ef152bSMark Johnston static int
vmmdev_rw(struct cdev * cdev,struct uio * uio,int flags)183 b9ef152bSMark Johnston vmmdev_rw(struct cdev *cdev, struct uio *uio, int flags)
184 b9ef152bSMark Johnston {
185 b9ef152bSMark Johnston 	int error, off, c, prot;
186 b9ef152bSMark Johnston 	vm_paddr_t gpa, maxaddr;
187 b9ef152bSMark Johnston 	void *hpa, *cookie;
188 b9ef152bSMark Johnston 	struct vmmdev_softc *sc;
189 b9ef152bSMark Johnston 
190 b9ef152bSMark Johnston 	sc = vmmdev_lookup2(cdev);
191 b9ef152bSMark Johnston 	if (sc == NULL)
192 b9ef152bSMark Johnston 		return (ENXIO);
193 b9ef152bSMark Johnston 
194 b9ef152bSMark Johnston 	/*
195 b9ef152bSMark Johnston 	 * Get a read lock on the guest memory map.
196 b9ef152bSMark Johnston 	 */
197 b9ef152bSMark Johnston 	vm_slock_memsegs(sc->vm);
198 b9ef152bSMark Johnston 
199 7c89253bSJohn Baldwin 	error = 0;
200 b9ef152bSMark Johnston 	prot = (uio->uio_rw == UIO_WRITE ? VM_PROT_WRITE : VM_PROT_READ);
201 b9ef152bSMark Johnston 	maxaddr = vmm_sysmem_maxaddr(sc->vm);
202 b9ef152bSMark Johnston 	while (uio->uio_resid > 0 && error == 0) {
203 b9ef152bSMark Johnston 		gpa = uio->uio_offset;
204 b9ef152bSMark Johnston 		off = gpa & PAGE_MASK;
205 b9ef152bSMark Johnston 		c = min(uio->uio_resid, PAGE_SIZE - off);
206 b9ef152bSMark Johnston 
207 b9ef152bSMark Johnston 		/*
208 b9ef152bSMark Johnston 		 * The VM has a hole in its physical memory map. If we want to
209 b9ef152bSMark Johnston 		 * use 'dd' to inspect memory beyond the hole we need to
210 b9ef152bSMark Johnston 		 * provide bogus data for memory that lies in the hole.
211 b9ef152bSMark Johnston 		 *
212 b9ef152bSMark Johnston 		 * Since this device does not support lseek(2), dd(1) will
213 b9ef152bSMark Johnston 		 * read(2) blocks of data to simulate the lseek(2).
214 b9ef152bSMark Johnston 		 */
215 b9ef152bSMark Johnston 		hpa = vm_gpa_hold_global(sc->vm, gpa, c, prot, &cookie);
216 b9ef152bSMark Johnston 		if (hpa == NULL) {
217 b9ef152bSMark Johnston 			if (uio->uio_rw == UIO_READ && gpa < maxaddr)
218 b9ef152bSMark Johnston 				error = uiomove(__DECONST(void *, zero_region),
219 b9ef152bSMark Johnston 				    c, uio);
220 b9ef152bSMark Johnston 			else
221 b9ef152bSMark Johnston 				error = EFAULT;
222 b9ef152bSMark Johnston 		} else {
223 b9ef152bSMark Johnston 			error = uiomove(hpa, c, uio);
224 b9ef152bSMark Johnston 			vm_gpa_release(cookie);
225 b9ef152bSMark Johnston 		}
226 b9ef152bSMark Johnston 	}
227 b9ef152bSMark Johnston 	vm_unlock_memsegs(sc->vm);
228 b9ef152bSMark Johnston 	return (error);
229 b9ef152bSMark Johnston }
230 b9ef152bSMark Johnston 
231 b9ef152bSMark Johnston CTASSERT(sizeof(((struct vm_memseg *)0)->name) >= VM_MAX_SUFFIXLEN + 1);
232 b9ef152bSMark Johnston 
233 b9ef152bSMark Johnston static int
get_memseg(struct vmmdev_softc * sc,struct vm_memseg * mseg,size_t len)234 b9ef152bSMark Johnston get_memseg(struct vmmdev_softc *sc, struct vm_memseg *mseg, size_t len)
235 b9ef152bSMark Johnston {
236 b9ef152bSMark Johnston 	struct devmem_softc *dsc;
237 b9ef152bSMark Johnston 	int error;
238 b9ef152bSMark Johnston 	bool sysmem;
239 b9ef152bSMark Johnston 
240 b9ef152bSMark Johnston 	error = vm_get_memseg(sc->vm, mseg->segid, &mseg->len, &sysmem, NULL);
241 b9ef152bSMark Johnston 	if (error || mseg->len == 0)
242 b9ef152bSMark Johnston 		return (error);
243 b9ef152bSMark Johnston 
244 b9ef152bSMark Johnston 	if (!sysmem) {
245 b9ef152bSMark Johnston 		SLIST_FOREACH(dsc, &sc->devmem, link) {
246 b9ef152bSMark Johnston 			if (dsc->segid == mseg->segid)
247 b9ef152bSMark Johnston 				break;
248 b9ef152bSMark Johnston 		}
249 b9ef152bSMark Johnston 		KASSERT(dsc != NULL, ("%s: devmem segment %d not found",
250 b9ef152bSMark Johnston 		    __func__, mseg->segid));
251 b9ef152bSMark Johnston 		error = copystr(dsc->name, mseg->name, len, NULL);
252 b9ef152bSMark Johnston 	} else {
253 b9ef152bSMark Johnston 		bzero(mseg->name, len);
254 b9ef152bSMark Johnston 	}
255 b9ef152bSMark Johnston 
256 b9ef152bSMark Johnston 	return (error);
257 b9ef152bSMark Johnston }
258 b9ef152bSMark Johnston 
259 b9ef152bSMark Johnston static int
alloc_memseg(struct vmmdev_softc * sc,struct vm_memseg * mseg,size_t len)260 b9ef152bSMark Johnston alloc_memseg(struct vmmdev_softc *sc, struct vm_memseg *mseg, size_t len)
261 b9ef152bSMark Johnston {
262 b9ef152bSMark Johnston 	char *name;
263 b9ef152bSMark Johnston 	int error;
264 b9ef152bSMark Johnston 	bool sysmem;
265 b9ef152bSMark Johnston 
266 b9ef152bSMark Johnston 	error = 0;
267 b9ef152bSMark Johnston 	name = NULL;
268 b9ef152bSMark Johnston 	sysmem = true;
269 b9ef152bSMark Johnston 
270 b9ef152bSMark Johnston 	/*
271 b9ef152bSMark Johnston 	 * The allocation is lengthened by 1 to hold a terminating NUL.  It'll
272 b9ef152bSMark Johnston 	 * by stripped off when devfs processes the full string.
273 b9ef152bSMark Johnston 	 */
274 b9ef152bSMark Johnston 	if (VM_MEMSEG_NAME(mseg)) {
275 b9ef152bSMark Johnston 		sysmem = false;
276 b9ef152bSMark Johnston 		name = malloc(len, M_VMMDEV, M_WAITOK);
277 b9ef152bSMark Johnston 		error = copystr(mseg->name, name, len, NULL);
278 b9ef152bSMark Johnston 		if (error)
279 b9ef152bSMark Johnston 			goto done;
280 b9ef152bSMark Johnston 	}
281 b9ef152bSMark Johnston 
282 b9ef152bSMark Johnston 	error = vm_alloc_memseg(sc->vm, mseg->segid, mseg->len, sysmem);
283 b9ef152bSMark Johnston 	if (error)
284 b9ef152bSMark Johnston 		goto done;
285 b9ef152bSMark Johnston 
286 b9ef152bSMark Johnston 	if (VM_MEMSEG_NAME(mseg)) {
287 f4002135SMark Johnston 		error = devmem_create_cdev(sc, mseg->segid, name);
288 b9ef152bSMark Johnston 		if (error)
289 b9ef152bSMark Johnston 			vm_free_memseg(sc->vm, mseg->segid);
290 b9ef152bSMark Johnston 		else
291 b9ef152bSMark Johnston 			name = NULL;	/* freed when 'cdev' is destroyed */
292 b9ef152bSMark Johnston 	}
293 b9ef152bSMark Johnston done:
294 b9ef152bSMark Johnston 	free(name, M_VMMDEV);
295 b9ef152bSMark Johnston 	return (error);
296 b9ef152bSMark Johnston }
297 b9ef152bSMark Johnston 
298 b9ef152bSMark Johnston static int
vm_get_register_set(struct vcpu * vcpu,unsigned int count,int * regnum,uint64_t * regval)299 b9ef152bSMark Johnston vm_get_register_set(struct vcpu *vcpu, unsigned int count, int *regnum,
300 b9ef152bSMark Johnston     uint64_t *regval)
301 b9ef152bSMark Johnston {
302 b9ef152bSMark Johnston 	int error, i;
303 b9ef152bSMark Johnston 
304 b9ef152bSMark Johnston 	error = 0;
305 b9ef152bSMark Johnston 	for (i = 0; i < count; i++) {
306 b9ef152bSMark Johnston 		error = vm_get_register(vcpu, regnum[i], &regval[i]);
307 b9ef152bSMark Johnston 		if (error)
308 b9ef152bSMark Johnston 			break;
309 b9ef152bSMark Johnston 	}
310 b9ef152bSMark Johnston 	return (error);
311 b9ef152bSMark Johnston }
312 b9ef152bSMark Johnston 
313 b9ef152bSMark Johnston static int
vm_set_register_set(struct vcpu * vcpu,unsigned int count,int * regnum,uint64_t * regval)314 b9ef152bSMark Johnston vm_set_register_set(struct vcpu *vcpu, unsigned int count, int *regnum,
315 b9ef152bSMark Johnston     uint64_t *regval)
316 b9ef152bSMark Johnston {
317 b9ef152bSMark Johnston 	int error, i;
318 b9ef152bSMark Johnston 
319 b9ef152bSMark Johnston 	error = 0;
320 b9ef152bSMark Johnston 	for (i = 0; i < count; i++) {
321 b9ef152bSMark Johnston 		error = vm_set_register(vcpu, regnum[i], regval[i]);
322 b9ef152bSMark Johnston 		if (error)
323 b9ef152bSMark Johnston 			break;
324 b9ef152bSMark Johnston 	}
325 b9ef152bSMark Johnston 	return (error);
326 b9ef152bSMark Johnston }
327 b9ef152bSMark Johnston 
328 40087581SMark Johnston static int
vmmdev_open(struct cdev * dev,int flags,int fmt,struct thread * td)329 40087581SMark Johnston vmmdev_open(struct cdev *dev, int flags, int fmt, struct thread *td)
330 40087581SMark Johnston {
331 40087581SMark Johnston 	int error;
332 40087581SMark Johnston 
333 40087581SMark Johnston 	/*
334 40087581SMark Johnston 	 * A jail without vmm access shouldn't be able to access vmm device
335 40087581SMark Johnston 	 * files at all, but check here just to be thorough.
336 40087581SMark Johnston 	 */
337 40087581SMark Johnston 	error = vmm_priv_check(td->td_ucred);
338 40087581SMark Johnston 	if (error != 0)
339 40087581SMark Johnston 		return (error);
340 40087581SMark Johnston 
341 40087581SMark Johnston 	return (0);
342 40087581SMark Johnston }
343 40087581SMark Johnston 
344 b9ef152bSMark Johnston static const struct vmmdev_ioctl vmmdev_ioctls[] = {
345 b9ef152bSMark Johnston 	VMMDEV_IOCTL(VM_GET_REGISTER, VMMDEV_IOCTL_LOCK_ONE_VCPU),
346 b9ef152bSMark Johnston 	VMMDEV_IOCTL(VM_SET_REGISTER, VMMDEV_IOCTL_LOCK_ONE_VCPU),
347 b9ef152bSMark Johnston 	VMMDEV_IOCTL(VM_GET_REGISTER_SET, VMMDEV_IOCTL_LOCK_ONE_VCPU),
348 b9ef152bSMark Johnston 	VMMDEV_IOCTL(VM_SET_REGISTER_SET, VMMDEV_IOCTL_LOCK_ONE_VCPU),
349 b9ef152bSMark Johnston 	VMMDEV_IOCTL(VM_GET_CAPABILITY, VMMDEV_IOCTL_LOCK_ONE_VCPU),
350 b9ef152bSMark Johnston 	VMMDEV_IOCTL(VM_SET_CAPABILITY, VMMDEV_IOCTL_LOCK_ONE_VCPU),
351 b9ef152bSMark Johnston 	VMMDEV_IOCTL(VM_ACTIVATE_CPU, VMMDEV_IOCTL_LOCK_ONE_VCPU),
352 b9ef152bSMark Johnston 	VMMDEV_IOCTL(VM_INJECT_EXCEPTION, VMMDEV_IOCTL_LOCK_ONE_VCPU),
353 b9ef152bSMark Johnston 	VMMDEV_IOCTL(VM_STATS, VMMDEV_IOCTL_LOCK_ONE_VCPU),
354 b9ef152bSMark Johnston 
355 b9ef152bSMark Johnston #if defined(__amd64__) && defined(COMPAT_FREEBSD12)
356 a852dc58SMark Johnston 	VMMDEV_IOCTL(VM_ALLOC_MEMSEG_12,
357 b9ef152bSMark Johnston 	    VMMDEV_IOCTL_XLOCK_MEMSEGS | VMMDEV_IOCTL_LOCK_ALL_VCPUS),
358 b9ef152bSMark Johnston #endif
359 b9ef152bSMark Johnston 	VMMDEV_IOCTL(VM_ALLOC_MEMSEG,
360 b9ef152bSMark Johnston 	    VMMDEV_IOCTL_XLOCK_MEMSEGS | VMMDEV_IOCTL_LOCK_ALL_VCPUS),
361 b9ef152bSMark Johnston 	VMMDEV_IOCTL(VM_MMAP_MEMSEG,
362 b9ef152bSMark Johnston 	    VMMDEV_IOCTL_XLOCK_MEMSEGS | VMMDEV_IOCTL_LOCK_ALL_VCPUS),
363 b9ef152bSMark Johnston 	VMMDEV_IOCTL(VM_MUNMAP_MEMSEG,
364 b9ef152bSMark Johnston 	    VMMDEV_IOCTL_XLOCK_MEMSEGS | VMMDEV_IOCTL_LOCK_ALL_VCPUS),
365 b9ef152bSMark Johnston 	VMMDEV_IOCTL(VM_REINIT,
366 b9ef152bSMark Johnston 	    VMMDEV_IOCTL_XLOCK_MEMSEGS | VMMDEV_IOCTL_LOCK_ALL_VCPUS),
367 b9ef152bSMark Johnston 
368 b9ef152bSMark Johnston #if defined(__amd64__) && defined(COMPAT_FREEBSD12)
369 a852dc58SMark Johnston 	VMMDEV_IOCTL(VM_GET_MEMSEG_12, VMMDEV_IOCTL_SLOCK_MEMSEGS),
370 b9ef152bSMark Johnston #endif
371 b9ef152bSMark Johnston 	VMMDEV_IOCTL(VM_GET_MEMSEG, VMMDEV_IOCTL_SLOCK_MEMSEGS),
372 b9ef152bSMark Johnston 	VMMDEV_IOCTL(VM_MMAP_GETNEXT, VMMDEV_IOCTL_SLOCK_MEMSEGS),
373 b9ef152bSMark Johnston 
374 b9ef152bSMark Johnston 	VMMDEV_IOCTL(VM_SUSPEND_CPU, VMMDEV_IOCTL_MAYBE_ALLOC_VCPU),
375 b9ef152bSMark Johnston 	VMMDEV_IOCTL(VM_RESUME_CPU, VMMDEV_IOCTL_MAYBE_ALLOC_VCPU),
376 b9ef152bSMark Johnston 
377 b9ef152bSMark Johnston 	VMMDEV_IOCTL(VM_SUSPEND, 0),
378 b9ef152bSMark Johnston 	VMMDEV_IOCTL(VM_GET_CPUS, 0),
379 b9ef152bSMark Johnston 	VMMDEV_IOCTL(VM_GET_TOPOLOGY, 0),
380 b9ef152bSMark Johnston 	VMMDEV_IOCTL(VM_SET_TOPOLOGY, 0),
381 b9ef152bSMark Johnston };
382 b9ef152bSMark Johnston 
383 b9ef152bSMark Johnston static int
vmmdev_ioctl(struct cdev * cdev,u_long cmd,caddr_t data,int fflag,struct thread * td)384 b9ef152bSMark Johnston vmmdev_ioctl(struct cdev *cdev, u_long cmd, caddr_t data, int fflag,
385 b9ef152bSMark Johnston     struct thread *td)
386 b9ef152bSMark Johnston {
387 b9ef152bSMark Johnston 	struct vmmdev_softc *sc;
388 b9ef152bSMark Johnston 	struct vcpu *vcpu;
389 b9ef152bSMark Johnston 	const struct vmmdev_ioctl *ioctl;
390 b9ef152bSMark Johnston 	int error, vcpuid;
391 b9ef152bSMark Johnston 
392 b9ef152bSMark Johnston 	sc = vmmdev_lookup2(cdev);
393 b9ef152bSMark Johnston 	if (sc == NULL)
394 b9ef152bSMark Johnston 		return (ENXIO);
395 b9ef152bSMark Johnston 
396 b9ef152bSMark Johnston 	ioctl = NULL;
397 b9ef152bSMark Johnston 	for (size_t i = 0; i < nitems(vmmdev_ioctls); i++) {
398 b9ef152bSMark Johnston 		if (vmmdev_ioctls[i].cmd == cmd) {
399 b9ef152bSMark Johnston 			ioctl = &vmmdev_ioctls[i];
400 b9ef152bSMark Johnston 			break;
401 b9ef152bSMark Johnston 		}
402 b9ef152bSMark Johnston 	}
403 b9ef152bSMark Johnston 	if (ioctl == NULL) {
404 b9ef152bSMark Johnston 		for (size_t i = 0; i < vmmdev_machdep_ioctl_count; i++) {
405 b9ef152bSMark Johnston 			if (vmmdev_machdep_ioctls[i].cmd == cmd) {
406 b9ef152bSMark Johnston 				ioctl = &vmmdev_machdep_ioctls[i];
407 b9ef152bSMark Johnston 				break;
408 b9ef152bSMark Johnston 			}
409 b9ef152bSMark Johnston 		}
410 b9ef152bSMark Johnston 	}
411 b9ef152bSMark Johnston 	if (ioctl == NULL)
412 b9ef152bSMark Johnston 		return (ENOTTY);
413 b9ef152bSMark Johnston 
414 b9ef152bSMark Johnston 	if ((ioctl->flags & VMMDEV_IOCTL_XLOCK_MEMSEGS) != 0)
415 b9ef152bSMark Johnston 		vm_xlock_memsegs(sc->vm);
416 b9ef152bSMark Johnston 	else if ((ioctl->flags & VMMDEV_IOCTL_SLOCK_MEMSEGS) != 0)
417 b9ef152bSMark Johnston 		vm_slock_memsegs(sc->vm);
418 b9ef152bSMark Johnston 
419 b9ef152bSMark Johnston 	vcpu = NULL;
420 b9ef152bSMark Johnston 	vcpuid = -1;
421 b9ef152bSMark Johnston 	if ((ioctl->flags & (VMMDEV_IOCTL_LOCK_ONE_VCPU |
422 b9ef152bSMark Johnston 	    VMMDEV_IOCTL_ALLOC_VCPU | VMMDEV_IOCTL_MAYBE_ALLOC_VCPU)) != 0) {
423 b9ef152bSMark Johnston 		vcpuid = *(int *)data;
424 b9ef152bSMark Johnston 		if (vcpuid == -1) {
425 b9ef152bSMark Johnston 			if ((ioctl->flags &
426 b9ef152bSMark Johnston 			    VMMDEV_IOCTL_MAYBE_ALLOC_VCPU) == 0) {
427 b9ef152bSMark Johnston 				error = EINVAL;
428 b9ef152bSMark Johnston 				goto lockfail;
429 b9ef152bSMark Johnston 			}
430 b9ef152bSMark Johnston 		} else {
431 b9ef152bSMark Johnston 			vcpu = vm_alloc_vcpu(sc->vm, vcpuid);
432 b9ef152bSMark Johnston 			if (vcpu == NULL) {
433 b9ef152bSMark Johnston 				error = EINVAL;
434 b9ef152bSMark Johnston 				goto lockfail;
435 b9ef152bSMark Johnston 			}
436 b9ef152bSMark Johnston 			if ((ioctl->flags & VMMDEV_IOCTL_LOCK_ONE_VCPU) != 0) {
437 b9ef152bSMark Johnston 				error = vcpu_lock_one(vcpu);
438 b9ef152bSMark Johnston 				if (error)
439 b9ef152bSMark Johnston 					goto lockfail;
440 b9ef152bSMark Johnston 			}
441 b9ef152bSMark Johnston 		}
442 b9ef152bSMark Johnston 	}
443 b9ef152bSMark Johnston 	if ((ioctl->flags & VMMDEV_IOCTL_LOCK_ALL_VCPUS) != 0) {
444 b9ef152bSMark Johnston 		error = vcpu_lock_all(sc);
445 b9ef152bSMark Johnston 		if (error)
446 b9ef152bSMark Johnston 			goto lockfail;
447 b9ef152bSMark Johnston 	}
448 b9ef152bSMark Johnston 
449 b9ef152bSMark Johnston 	switch (cmd) {
450 b9ef152bSMark Johnston 	case VM_SUSPEND: {
451 b9ef152bSMark Johnston 		struct vm_suspend *vmsuspend;
452 b9ef152bSMark Johnston 
453 b9ef152bSMark Johnston 		vmsuspend = (struct vm_suspend *)data;
454 b9ef152bSMark Johnston 		error = vm_suspend(sc->vm, vmsuspend->how);
455 b9ef152bSMark Johnston 		break;
456 b9ef152bSMark Johnston 	}
457 b9ef152bSMark Johnston 	case VM_REINIT:
458 b9ef152bSMark Johnston 		error = vm_reinit(sc->vm);
459 b9ef152bSMark Johnston 		break;
460 b9ef152bSMark Johnston 	case VM_STAT_DESC: {
461 b9ef152bSMark Johnston 		struct vm_stat_desc *statdesc;
462 b9ef152bSMark Johnston 
463 b9ef152bSMark Johnston 		statdesc = (struct vm_stat_desc *)data;
464 b9ef152bSMark Johnston 		error = vmm_stat_desc_copy(statdesc->index, statdesc->desc,
465 b9ef152bSMark Johnston 		    sizeof(statdesc->desc));
466 b9ef152bSMark Johnston 		break;
467 b9ef152bSMark Johnston 	}
468 b9ef152bSMark Johnston 	case VM_STATS: {
469 b9ef152bSMark Johnston 		struct vm_stats *vmstats;
470 b9ef152bSMark Johnston 
471 b9ef152bSMark Johnston 		vmstats = (struct vm_stats *)data;
472 b9ef152bSMark Johnston 		getmicrotime(&vmstats->tv);
473 b9ef152bSMark Johnston 		error = vmm_stat_copy(vcpu, vmstats->index,
474 b9ef152bSMark Johnston 		    nitems(vmstats->statbuf), &vmstats->num_entries,
475 b9ef152bSMark Johnston 		    vmstats->statbuf);
476 b9ef152bSMark Johnston 		break;
477 b9ef152bSMark Johnston 	}
478 b9ef152bSMark Johnston 	case VM_MMAP_GETNEXT: {
479 b9ef152bSMark Johnston 		struct vm_memmap *mm;
480 b9ef152bSMark Johnston 
481 b9ef152bSMark Johnston 		mm = (struct vm_memmap *)data;
482 b9ef152bSMark Johnston 		error = vm_mmap_getnext(sc->vm, &mm->gpa, &mm->segid,
483 b9ef152bSMark Johnston 		    &mm->segoff, &mm->len, &mm->prot, &mm->flags);
484 b9ef152bSMark Johnston 		break;
485 b9ef152bSMark Johnston 	}
486 b9ef152bSMark Johnston 	case VM_MMAP_MEMSEG: {
487 b9ef152bSMark Johnston 		struct vm_memmap *mm;
488 b9ef152bSMark Johnston 
489 b9ef152bSMark Johnston 		mm = (struct vm_memmap *)data;
490 b9ef152bSMark Johnston 		error = vm_mmap_memseg(sc->vm, mm->gpa, mm->segid, mm->segoff,
491 b9ef152bSMark Johnston 		    mm->len, mm->prot, mm->flags);
492 b9ef152bSMark Johnston 		break;
493 b9ef152bSMark Johnston 	}
494 b9ef152bSMark Johnston 	case VM_MUNMAP_MEMSEG: {
495 b9ef152bSMark Johnston 		struct vm_munmap *mu;
496 b9ef152bSMark Johnston 
497 b9ef152bSMark Johnston 		mu = (struct vm_munmap *)data;
498 b9ef152bSMark Johnston 		error = vm_munmap_memseg(sc->vm, mu->gpa, mu->len);
499 b9ef152bSMark Johnston 		break;
500 b9ef152bSMark Johnston 	}
501 b9ef152bSMark Johnston #if defined(__amd64__) && defined(COMPAT_FREEBSD12)
502 a852dc58SMark Johnston 	case VM_ALLOC_MEMSEG_12:
503 b9ef152bSMark Johnston 		error = alloc_memseg(sc, (struct vm_memseg *)data,
504 a852dc58SMark Johnston 		    sizeof(((struct vm_memseg_12 *)0)->name));
505 b9ef152bSMark Johnston 		break;
506 a852dc58SMark Johnston 	case VM_GET_MEMSEG_12:
507 b9ef152bSMark Johnston 		error = get_memseg(sc, (struct vm_memseg *)data,
508 a852dc58SMark Johnston 		    sizeof(((struct vm_memseg_12 *)0)->name));
509 b9ef152bSMark Johnston 		break;
510 b9ef152bSMark Johnston #endif
511 b9ef152bSMark Johnston 	case VM_ALLOC_MEMSEG:
512 b9ef152bSMark Johnston 		error = alloc_memseg(sc, (struct vm_memseg *)data,
513 b9ef152bSMark Johnston 		    sizeof(((struct vm_memseg *)0)->name));
514 b9ef152bSMark Johnston 		break;
515 b9ef152bSMark Johnston 	case VM_GET_MEMSEG:
516 b9ef152bSMark Johnston 		error = get_memseg(sc, (struct vm_memseg *)data,
517 b9ef152bSMark Johnston 		    sizeof(((struct vm_memseg *)0)->name));
518 b9ef152bSMark Johnston 		break;
519 b9ef152bSMark Johnston 	case VM_GET_REGISTER: {
520 b9ef152bSMark Johnston 		struct vm_register *vmreg;
521 b9ef152bSMark Johnston 
522 b9ef152bSMark Johnston 		vmreg = (struct vm_register *)data;
523 b9ef152bSMark Johnston 		error = vm_get_register(vcpu, vmreg->regnum, &vmreg->regval);
524 b9ef152bSMark Johnston 		break;
525 b9ef152bSMark Johnston 	}
526 b9ef152bSMark Johnston 	case VM_SET_REGISTER: {
527 b9ef152bSMark Johnston 		struct vm_register *vmreg;
528 b9ef152bSMark Johnston 
529 b9ef152bSMark Johnston 		vmreg = (struct vm_register *)data;
530 b9ef152bSMark Johnston 		error = vm_set_register(vcpu, vmreg->regnum, vmreg->regval);
531 b9ef152bSMark Johnston 		break;
532 b9ef152bSMark Johnston 	}
533 b9ef152bSMark Johnston 	case VM_GET_REGISTER_SET: {
534 b9ef152bSMark Johnston 		struct vm_register_set *vmregset;
535 b9ef152bSMark Johnston 		uint64_t *regvals;
536 b9ef152bSMark Johnston 		int *regnums;
537 b9ef152bSMark Johnston 
538 b9ef152bSMark Johnston 		vmregset = (struct vm_register_set *)data;
539 b9ef152bSMark Johnston 		if (vmregset->count > VM_REG_LAST) {
540 b9ef152bSMark Johnston 			error = EINVAL;
541 b9ef152bSMark Johnston 			break;
542 b9ef152bSMark Johnston 		}
543 b9ef152bSMark Johnston 		regvals = malloc(sizeof(regvals[0]) * vmregset->count, M_VMMDEV,
544 b9ef152bSMark Johnston 		    M_WAITOK);
545 b9ef152bSMark Johnston 		regnums = malloc(sizeof(regnums[0]) * vmregset->count, M_VMMDEV,
546 b9ef152bSMark Johnston 		    M_WAITOK);
547 b9ef152bSMark Johnston 		error = copyin(vmregset->regnums, regnums, sizeof(regnums[0]) *
548 b9ef152bSMark Johnston 		    vmregset->count);
549 b9ef152bSMark Johnston 		if (error == 0)
550 b9ef152bSMark Johnston 			error = vm_get_register_set(vcpu,
551 b9ef152bSMark Johnston 			    vmregset->count, regnums, regvals);
552 b9ef152bSMark Johnston 		if (error == 0)
553 b9ef152bSMark Johnston 			error = copyout(regvals, vmregset->regvals,
554 b9ef152bSMark Johnston 			    sizeof(regvals[0]) * vmregset->count);
555 b9ef152bSMark Johnston 		free(regvals, M_VMMDEV);
556 b9ef152bSMark Johnston 		free(regnums, M_VMMDEV);
557 b9ef152bSMark Johnston 		break;
558 b9ef152bSMark Johnston 	}
559 b9ef152bSMark Johnston 	case VM_SET_REGISTER_SET: {
560 b9ef152bSMark Johnston 		struct vm_register_set *vmregset;
561 b9ef152bSMark Johnston 		uint64_t *regvals;
562 b9ef152bSMark Johnston 		int *regnums;
563 b9ef152bSMark Johnston 
564 b9ef152bSMark Johnston 		vmregset = (struct vm_register_set *)data;
565 b9ef152bSMark Johnston 		if (vmregset->count > VM_REG_LAST) {
566 b9ef152bSMark Johnston 			error = EINVAL;
567 b9ef152bSMark Johnston 			break;
568 b9ef152bSMark Johnston 		}
569 b9ef152bSMark Johnston 		regvals = malloc(sizeof(regvals[0]) * vmregset->count, M_VMMDEV,
570 b9ef152bSMark Johnston 		    M_WAITOK);
571 b9ef152bSMark Johnston 		regnums = malloc(sizeof(regnums[0]) * vmregset->count, M_VMMDEV,
572 b9ef152bSMark Johnston 		    M_WAITOK);
573 b9ef152bSMark Johnston 		error = copyin(vmregset->regnums, regnums, sizeof(regnums[0]) *
574 b9ef152bSMark Johnston 		    vmregset->count);
575 b9ef152bSMark Johnston 		if (error == 0)
576 b9ef152bSMark Johnston 			error = copyin(vmregset->regvals, regvals,
577 b9ef152bSMark Johnston 			    sizeof(regvals[0]) * vmregset->count);
578 b9ef152bSMark Johnston 		if (error == 0)
579 b9ef152bSMark Johnston 			error = vm_set_register_set(vcpu,
580 b9ef152bSMark Johnston 			    vmregset->count, regnums, regvals);
581 b9ef152bSMark Johnston 		free(regvals, M_VMMDEV);
582 b9ef152bSMark Johnston 		free(regnums, M_VMMDEV);
583 b9ef152bSMark Johnston 		break;
584 b9ef152bSMark Johnston 	}
585 b9ef152bSMark Johnston 	case VM_GET_CAPABILITY: {
586 b9ef152bSMark Johnston 		struct vm_capability *vmcap;
587 b9ef152bSMark Johnston 
588 b9ef152bSMark Johnston 		vmcap = (struct vm_capability *)data;
589 b9ef152bSMark Johnston 		error = vm_get_capability(vcpu, vmcap->captype, &vmcap->capval);
590 b9ef152bSMark Johnston 		break;
591 b9ef152bSMark Johnston 	}
592 b9ef152bSMark Johnston 	case VM_SET_CAPABILITY: {
593 b9ef152bSMark Johnston 		struct vm_capability *vmcap;
594 b9ef152bSMark Johnston 
595 b9ef152bSMark Johnston 		vmcap = (struct vm_capability *)data;
596 b9ef152bSMark Johnston 		error = vm_set_capability(vcpu, vmcap->captype, vmcap->capval);
597 b9ef152bSMark Johnston 		break;
598 b9ef152bSMark Johnston 	}
599 b9ef152bSMark Johnston 	case VM_ACTIVATE_CPU:
600 b9ef152bSMark Johnston 		error = vm_activate_cpu(vcpu);
601 b9ef152bSMark Johnston 		break;
602 b9ef152bSMark Johnston 	case VM_GET_CPUS: {
603 b9ef152bSMark Johnston 		struct vm_cpuset *vm_cpuset;
604 b9ef152bSMark Johnston 		cpuset_t *cpuset;
605 b9ef152bSMark Johnston 		int size;
606 b9ef152bSMark Johnston 
607 b9ef152bSMark Johnston 		error = 0;
608 b9ef152bSMark Johnston 		vm_cpuset = (struct vm_cpuset *)data;
609 b9ef152bSMark Johnston 		size = vm_cpuset->cpusetsize;
610 b9ef152bSMark Johnston 		if (size < 1 || size > CPU_MAXSIZE / NBBY) {
611 b9ef152bSMark Johnston 			error = ERANGE;
612 b9ef152bSMark Johnston 			break;
613 b9ef152bSMark Johnston 		}
614 b9ef152bSMark Johnston 		cpuset = malloc(max(size, sizeof(cpuset_t)), M_TEMP,
615 b9ef152bSMark Johnston 		    M_WAITOK | M_ZERO);
616 b9ef152bSMark Johnston 		if (vm_cpuset->which == VM_ACTIVE_CPUS)
617 b9ef152bSMark Johnston 			*cpuset = vm_active_cpus(sc->vm);
618 b9ef152bSMark Johnston 		else if (vm_cpuset->which == VM_SUSPENDED_CPUS)
619 b9ef152bSMark Johnston 			*cpuset = vm_suspended_cpus(sc->vm);
620 b9ef152bSMark Johnston 		else if (vm_cpuset->which == VM_DEBUG_CPUS)
621 b9ef152bSMark Johnston 			*cpuset = vm_debug_cpus(sc->vm);
622 b9ef152bSMark Johnston 		else
623 b9ef152bSMark Johnston 			error = EINVAL;
624 b9ef152bSMark Johnston 		if (error == 0 && size < howmany(CPU_FLS(cpuset), NBBY))
625 b9ef152bSMark Johnston 			error = ERANGE;
626 b9ef152bSMark Johnston 		if (error == 0)
627 b9ef152bSMark Johnston 			error = copyout(cpuset, vm_cpuset->cpus, size);
628 b9ef152bSMark Johnston 		free(cpuset, M_TEMP);
629 b9ef152bSMark Johnston 		break;
630 b9ef152bSMark Johnston 	}
631 b9ef152bSMark Johnston 	case VM_SUSPEND_CPU:
632 b9ef152bSMark Johnston 		error = vm_suspend_cpu(sc->vm, vcpu);
633 b9ef152bSMark Johnston 		break;
634 b9ef152bSMark Johnston 	case VM_RESUME_CPU:
635 b9ef152bSMark Johnston 		error = vm_resume_cpu(sc->vm, vcpu);
636 b9ef152bSMark Johnston 		break;
637 b9ef152bSMark Johnston 	case VM_SET_TOPOLOGY: {
638 b9ef152bSMark Johnston 		struct vm_cpu_topology *topology;
639 b9ef152bSMark Johnston 
640 b9ef152bSMark Johnston 		topology = (struct vm_cpu_topology *)data;
641 b9ef152bSMark Johnston 		error = vm_set_topology(sc->vm, topology->sockets,
642 b9ef152bSMark Johnston 		    topology->cores, topology->threads, topology->maxcpus);
643 b9ef152bSMark Johnston 		break;
644 b9ef152bSMark Johnston 	}
645 b9ef152bSMark Johnston 	case VM_GET_TOPOLOGY: {
646 b9ef152bSMark Johnston 		struct vm_cpu_topology *topology;
647 b9ef152bSMark Johnston 
648 b9ef152bSMark Johnston 		topology = (struct vm_cpu_topology *)data;
649 b9ef152bSMark Johnston 		vm_get_topology(sc->vm, &topology->sockets, &topology->cores,
650 b9ef152bSMark Johnston 		    &topology->threads, &topology->maxcpus);
651 b9ef152bSMark Johnston 		error = 0;
652 b9ef152bSMark Johnston 		break;
653 b9ef152bSMark Johnston 	}
654 b9ef152bSMark Johnston 	default:
655 b9ef152bSMark Johnston 		error = vmmdev_machdep_ioctl(sc->vm, vcpu, cmd, data, fflag,
656 b9ef152bSMark Johnston 		    td);
657 b9ef152bSMark Johnston 		break;
658 b9ef152bSMark Johnston 	}
659 b9ef152bSMark Johnston 
660 b9ef152bSMark Johnston 	if ((ioctl->flags &
661 b9ef152bSMark Johnston 	    (VMMDEV_IOCTL_XLOCK_MEMSEGS | VMMDEV_IOCTL_SLOCK_MEMSEGS)) != 0)
662 b9ef152bSMark Johnston 		vm_unlock_memsegs(sc->vm);
663 b9ef152bSMark Johnston 	if ((ioctl->flags & VMMDEV_IOCTL_LOCK_ALL_VCPUS) != 0)
664 b9ef152bSMark Johnston 		vcpu_unlock_all(sc);
665 b9ef152bSMark Johnston 	else if ((ioctl->flags & VMMDEV_IOCTL_LOCK_ONE_VCPU) != 0)
666 b9ef152bSMark Johnston 		vcpu_unlock_one(vcpu);
667 b9ef152bSMark Johnston 
668 b9ef152bSMark Johnston 	/*
669 b9ef152bSMark Johnston 	 * Make sure that no handler returns a kernel-internal
670 b9ef152bSMark Johnston 	 * error value to userspace.
671 b9ef152bSMark Johnston 	 */
672 b9ef152bSMark Johnston 	KASSERT(error == ERESTART || error >= 0,
673 b9ef152bSMark Johnston 	    ("vmmdev_ioctl: invalid error return %d", error));
674 b9ef152bSMark Johnston 	return (error);
675 b9ef152bSMark Johnston 
676 b9ef152bSMark Johnston lockfail:
677 b9ef152bSMark Johnston 	if ((ioctl->flags &
678 b9ef152bSMark Johnston 	    (VMMDEV_IOCTL_XLOCK_MEMSEGS | VMMDEV_IOCTL_SLOCK_MEMSEGS)) != 0)
679 b9ef152bSMark Johnston 		vm_unlock_memsegs(sc->vm);
680 b9ef152bSMark Johnston 	return (error);
681 b9ef152bSMark Johnston }
682 b9ef152bSMark Johnston 
683 b9ef152bSMark Johnston static int
vmmdev_mmap_single(struct cdev * cdev,vm_ooffset_t * offset,vm_size_t mapsize,struct vm_object ** objp,int nprot)684 b9ef152bSMark Johnston vmmdev_mmap_single(struct cdev *cdev, vm_ooffset_t *offset, vm_size_t mapsize,
685 b9ef152bSMark Johnston     struct vm_object **objp, int nprot)
686 b9ef152bSMark Johnston {
687 b9ef152bSMark Johnston 	struct vmmdev_softc *sc;
688 b9ef152bSMark Johnston 	vm_paddr_t gpa;
689 b9ef152bSMark Johnston 	size_t len;
690 b9ef152bSMark Johnston 	vm_ooffset_t segoff, first, last;
691 b9ef152bSMark Johnston 	int error, found, segid;
692 b9ef152bSMark Johnston 	bool sysmem;
693 b9ef152bSMark Johnston 
694 b9ef152bSMark Johnston 	first = *offset;
695 b9ef152bSMark Johnston 	last = first + mapsize;
696 b9ef152bSMark Johnston 	if ((nprot & PROT_EXEC) || first < 0 || first >= last)
697 b9ef152bSMark Johnston 		return (EINVAL);
698 b9ef152bSMark Johnston 
699 b9ef152bSMark Johnston 	sc = vmmdev_lookup2(cdev);
700 b9ef152bSMark Johnston 	if (sc == NULL) {
701 b9ef152bSMark Johnston 		/* virtual machine is in the process of being created */
702 b9ef152bSMark Johnston 		return (EINVAL);
703 b9ef152bSMark Johnston 	}
704 b9ef152bSMark Johnston 
705 b9ef152bSMark Johnston 	/*
706 b9ef152bSMark Johnston 	 * Get a read lock on the guest memory map.
707 b9ef152bSMark Johnston 	 */
708 b9ef152bSMark Johnston 	vm_slock_memsegs(sc->vm);
709 b9ef152bSMark Johnston 
710 b9ef152bSMark Johnston 	gpa = 0;
711 b9ef152bSMark Johnston 	found = 0;
712 b9ef152bSMark Johnston 	while (!found) {
713 b9ef152bSMark Johnston 		error = vm_mmap_getnext(sc->vm, &gpa, &segid, &segoff, &len,
714 b9ef152bSMark Johnston 		    NULL, NULL);
715 b9ef152bSMark Johnston 		if (error)
716 b9ef152bSMark Johnston 			break;
717 b9ef152bSMark Johnston 
718 b9ef152bSMark Johnston 		if (first >= gpa && last <= gpa + len)
719 b9ef152bSMark Johnston 			found = 1;
720 b9ef152bSMark Johnston 		else
721 b9ef152bSMark Johnston 			gpa += len;
722 b9ef152bSMark Johnston 	}
723 b9ef152bSMark Johnston 
724 b9ef152bSMark Johnston 	if (found) {
725 b9ef152bSMark Johnston 		error = vm_get_memseg(sc->vm, segid, &len, &sysmem, objp);
726 b9ef152bSMark Johnston 		KASSERT(error == 0 && *objp != NULL,
727 b9ef152bSMark Johnston 		    ("%s: invalid memory segment %d", __func__, segid));
728 b9ef152bSMark Johnston 		if (sysmem) {
729 b9ef152bSMark Johnston 			vm_object_reference(*objp);
730 b9ef152bSMark Johnston 			*offset = segoff + (first - gpa);
731 b9ef152bSMark Johnston 		} else {
732 b9ef152bSMark Johnston 			error = EINVAL;
733 b9ef152bSMark Johnston 		}
734 b9ef152bSMark Johnston 	}
735 b9ef152bSMark Johnston 	vm_unlock_memsegs(sc->vm);
736 b9ef152bSMark Johnston 	return (error);
737 b9ef152bSMark Johnston }
738 b9ef152bSMark Johnston 
739 b9ef152bSMark Johnston static void
vmmdev_destroy(struct vmmdev_softc * sc)740 063a8bd9SMark Johnston vmmdev_destroy(struct vmmdev_softc *sc)
741 b9ef152bSMark Johnston {
742 b9ef152bSMark Johnston 	struct devmem_softc *dsc;
743 b9ef152bSMark Johnston 	int error __diagused;
744 b9ef152bSMark Johnston 
745 cef5f43fSMark Johnston 	KASSERT(sc->cdev == NULL, ("%s: cdev not free", __func__));
746 cef5f43fSMark Johnston 
747 063a8bd9SMark Johnston 	/*
748 063a8bd9SMark Johnston 	 * Destroy all cdevs:
749 063a8bd9SMark Johnston 	 *
750 063a8bd9SMark Johnston 	 * - any new operations on the 'cdev' will return an error (ENXIO).
751 063a8bd9SMark Johnston 	 *
752 063a8bd9SMark Johnston 	 * - the 'devmem' cdevs are destroyed before the virtual machine 'cdev'
753 063a8bd9SMark Johnston 	 */
754 063a8bd9SMark Johnston 	SLIST_FOREACH(dsc, &sc->devmem, link) {
755 063a8bd9SMark Johnston 		KASSERT(dsc->cdev != NULL, ("devmem cdev already destroyed"));
756 063a8bd9SMark Johnston 		devmem_destroy(dsc);
757 063a8bd9SMark Johnston 	}
758 063a8bd9SMark Johnston 
759 b9ef152bSMark Johnston 	vm_disable_vcpu_creation(sc->vm);
760 b9ef152bSMark Johnston 	error = vcpu_lock_all(sc);
761 b9ef152bSMark Johnston 	KASSERT(error == 0, ("%s: error %d freezing vcpus", __func__, error));
762 b9ef152bSMark Johnston 	vm_unlock_vcpus(sc->vm);
763 b9ef152bSMark Johnston 
764 b9ef152bSMark Johnston 	while ((dsc = SLIST_FIRST(&sc->devmem)) != NULL) {
765 b9ef152bSMark Johnston 		KASSERT(dsc->cdev == NULL, ("%s: devmem not free", __func__));
766 b9ef152bSMark Johnston 		SLIST_REMOVE_HEAD(&sc->devmem, link);
767 b9ef152bSMark Johnston 		free(dsc->name, M_VMMDEV);
768 b9ef152bSMark Johnston 		free(dsc, M_VMMDEV);
769 b9ef152bSMark Johnston 	}
770 b9ef152bSMark Johnston 
771 b9ef152bSMark Johnston 	if (sc->vm != NULL)
772 b9ef152bSMark Johnston 		vm_destroy(sc->vm);
773 b9ef152bSMark Johnston 
774 b9ef152bSMark Johnston 	if (sc->ucred != NULL)
775 b9ef152bSMark Johnston 		crfree(sc->ucred);
776 b9ef152bSMark Johnston 
777 887c0877SMark Johnston 	sx_xlock(&vmmdev_mtx);
778 b9ef152bSMark Johnston 	SLIST_REMOVE(&head, sc, vmmdev_softc, link);
779 887c0877SMark Johnston 	sx_xunlock(&vmmdev_mtx);
780 b9ef152bSMark Johnston 	free(sc, M_VMMDEV);
781 b9ef152bSMark Johnston }
782 b9ef152bSMark Johnston 
783 b9ef152bSMark Johnston static int
vmmdev_lookup_and_destroy(const char * name,struct ucred * cred)784 063a8bd9SMark Johnston vmmdev_lookup_and_destroy(const char *name, struct ucred *cred)
785 063a8bd9SMark Johnston {
786 063a8bd9SMark Johnston 	struct cdev *cdev;
787 063a8bd9SMark Johnston 	struct vmmdev_softc *sc;
788 063a8bd9SMark Johnston 
789 887c0877SMark Johnston 	sx_xlock(&vmmdev_mtx);
790 c23da668SMark Johnston 	sc = vmmdev_lookup(name, cred);
791 063a8bd9SMark Johnston 	if (sc == NULL || sc->cdev == NULL) {
792 887c0877SMark Johnston 		sx_xunlock(&vmmdev_mtx);
793 063a8bd9SMark Johnston 		return (EINVAL);
794 063a8bd9SMark Johnston 	}
795 063a8bd9SMark Johnston 
796 063a8bd9SMark Johnston 	/*
797 063a8bd9SMark Johnston 	 * Setting 'sc->cdev' to NULL is used to indicate that the VM
798 063a8bd9SMark Johnston 	 * is scheduled for destruction.
799 063a8bd9SMark Johnston 	 */
800 063a8bd9SMark Johnston 	cdev = sc->cdev;
801 063a8bd9SMark Johnston 	sc->cdev = NULL;
802 887c0877SMark Johnston 	sx_xunlock(&vmmdev_mtx);
803 063a8bd9SMark Johnston 
804 063a8bd9SMark Johnston 	destroy_dev(cdev);
805 063a8bd9SMark Johnston 	vmmdev_destroy(sc);
806 063a8bd9SMark Johnston 
807 063a8bd9SMark Johnston 	return (0);
808 063a8bd9SMark Johnston }
809 063a8bd9SMark Johnston 
810 063a8bd9SMark Johnston static int
sysctl_vmm_destroy(SYSCTL_HANDLER_ARGS)811 b9ef152bSMark Johnston sysctl_vmm_destroy(SYSCTL_HANDLER_ARGS)
812 b9ef152bSMark Johnston {
813 b9ef152bSMark Johnston 	char *buf;
814 b9ef152bSMark Johnston 	int error, buflen;
815 b9ef152bSMark Johnston 
816 b9ef152bSMark Johnston 	error = vmm_priv_check(req->td->td_ucred);
817 b9ef152bSMark Johnston 	if (error)
818 b9ef152bSMark Johnston 		return (error);
819 b9ef152bSMark Johnston 
820 b9ef152bSMark Johnston 	buflen = VM_MAX_NAMELEN + 1;
821 b9ef152bSMark Johnston 	buf = malloc(buflen, M_VMMDEV, M_WAITOK | M_ZERO);
822 b9ef152bSMark Johnston 	strlcpy(buf, "beavis", buflen);
823 b9ef152bSMark Johnston 	error = sysctl_handle_string(oidp, buf, buflen, req);
824 063a8bd9SMark Johnston 	if (error == 0 && req->newptr != NULL)
825 063a8bd9SMark Johnston 		error = vmmdev_lookup_and_destroy(buf, req->td->td_ucred);
826 b9ef152bSMark Johnston 	free(buf, M_VMMDEV);
827 b9ef152bSMark Johnston 	return (error);
828 b9ef152bSMark Johnston }
829 b9ef152bSMark Johnston SYSCTL_PROC(_hw_vmm, OID_AUTO, destroy,
830 b9ef152bSMark Johnston     CTLTYPE_STRING | CTLFLAG_RW | CTLFLAG_PRISON | CTLFLAG_MPSAFE,
831 b9ef152bSMark Johnston     NULL, 0, sysctl_vmm_destroy, "A",
832 b9ef152bSMark Johnston     NULL);
833 b9ef152bSMark Johnston 
834 b9ef152bSMark Johnston static struct cdevsw vmmdevsw = {
835 b9ef152bSMark Johnston 	.d_name		= "vmmdev",
836 b9ef152bSMark Johnston 	.d_version	= D_VERSION,
837 40087581SMark Johnston 	.d_open		= vmmdev_open,
838 b9ef152bSMark Johnston 	.d_ioctl	= vmmdev_ioctl,
839 b9ef152bSMark Johnston 	.d_mmap_single	= vmmdev_mmap_single,
840 b9ef152bSMark Johnston 	.d_read		= vmmdev_rw,
841 b9ef152bSMark Johnston 	.d_write	= vmmdev_rw,
842 b9ef152bSMark Johnston };
843 b9ef152bSMark Johnston 
844 d5819709SMark Johnston static struct vmmdev_softc *
vmmdev_alloc(struct vm * vm,struct ucred * cred)845 d5819709SMark Johnston vmmdev_alloc(struct vm *vm, struct ucred *cred)
846 b9ef152bSMark Johnston {
847 d5819709SMark Johnston 	struct vmmdev_softc *sc;
848 b9ef152bSMark Johnston 
849 d5819709SMark Johnston 	sc = malloc(sizeof(*sc), M_VMMDEV, M_WAITOK | M_ZERO);
850 d5819709SMark Johnston 	SLIST_INIT(&sc->devmem);
851 d5819709SMark Johnston 	sc->vm = vm;
852 d5819709SMark Johnston 	sc->ucred = crhold(cred);
853 d5819709SMark Johnston 	return (sc);
854 b9ef152bSMark Johnston }
855 b9ef152bSMark Johnston 
856 d5819709SMark Johnston static int
vmmdev_create(const char * name,struct ucred * cred)857 d5819709SMark Johnston vmmdev_create(const char *name, struct ucred *cred)
858 d5819709SMark Johnston {
859 cef5f43fSMark Johnston 	struct make_dev_args mda;
860 d5819709SMark Johnston 	struct cdev *cdev;
861 cef5f43fSMark Johnston 	struct vmmdev_softc *sc;
862 d5819709SMark Johnston 	struct vm *vm;
863 d5819709SMark Johnston 	int error;
864 b9ef152bSMark Johnston 
865 887c0877SMark Johnston 	sx_xlock(&vmmdev_mtx);
866 c23da668SMark Johnston 	sc = vmmdev_lookup(name, cred);
867 cef5f43fSMark Johnston 	if (sc != NULL) {
868 887c0877SMark Johnston 		sx_xunlock(&vmmdev_mtx);
869 d5819709SMark Johnston 		return (EEXIST);
870 cef5f43fSMark Johnston 	}
871 d5819709SMark Johnston 
872 d5819709SMark Johnston 	error = vm_create(name, &vm);
873 b9ef152bSMark Johnston 	if (error != 0) {
874 cef5f43fSMark Johnston 		sx_xunlock(&vmmdev_mtx);
875 cef5f43fSMark Johnston 		return (error);
876 cef5f43fSMark Johnston 	}
877 cef5f43fSMark Johnston 	sc = vmmdev_alloc(vm, cred);
878 cef5f43fSMark Johnston 	SLIST_INSERT_HEAD(&head, sc, link);
879 cef5f43fSMark Johnston 
880 cef5f43fSMark Johnston 	make_dev_args_init(&mda);
881 cef5f43fSMark Johnston 	mda.mda_devsw = &vmmdevsw;
882 cef5f43fSMark Johnston 	mda.mda_cr = sc->ucred;
883 cef5f43fSMark Johnston 	mda.mda_uid = UID_ROOT;
884 cef5f43fSMark Johnston 	mda.mda_gid = GID_WHEEL;
885 cef5f43fSMark Johnston 	mda.mda_mode = 0600;
886 cef5f43fSMark Johnston 	mda.mda_si_drv1 = sc;
887 cef5f43fSMark Johnston 	mda.mda_flags = MAKEDEV_CHECKNAME | MAKEDEV_WAITOK;
888 cef5f43fSMark Johnston 	error = make_dev_s(&mda, &cdev, "vmm/%s", name);
889 cef5f43fSMark Johnston 	if (error != 0) {
890 cef5f43fSMark Johnston 		sx_xunlock(&vmmdev_mtx);
891 b9ef152bSMark Johnston 		vmmdev_destroy(sc);
892 d5819709SMark Johnston 		return (error);
893 b9ef152bSMark Johnston 	}
894 b9ef152bSMark Johnston 	sc->cdev = cdev;
895 887c0877SMark Johnston 	sx_xunlock(&vmmdev_mtx);
896 d5819709SMark Johnston 	return (0);
897 d5819709SMark Johnston }
898 d5819709SMark Johnston 
899 d5819709SMark Johnston static int
sysctl_vmm_create(SYSCTL_HANDLER_ARGS)900 d5819709SMark Johnston sysctl_vmm_create(SYSCTL_HANDLER_ARGS)
901 d5819709SMark Johnston {
902 d5819709SMark Johnston 	char *buf;
903 d5819709SMark Johnston 	int error, buflen;
904 d5819709SMark Johnston 
905 d5819709SMark Johnston 	error = vmm_priv_check(req->td->td_ucred);
906 d5819709SMark Johnston 	if (error != 0)
907 d5819709SMark Johnston 		return (error);
908 d5819709SMark Johnston 
909 d5819709SMark Johnston 	buflen = VM_MAX_NAMELEN + 1;
910 d5819709SMark Johnston 	buf = malloc(buflen, M_VMMDEV, M_WAITOK | M_ZERO);
911 d5819709SMark Johnston 	strlcpy(buf, "beavis", buflen);
912 d5819709SMark Johnston 	error = sysctl_handle_string(oidp, buf, buflen, req);
913 d5819709SMark Johnston 	if (error == 0 && req->newptr != NULL)
914 d5819709SMark Johnston 		error = vmmdev_create(buf, req->td->td_ucred);
915 b9ef152bSMark Johnston 	free(buf, M_VMMDEV);
916 b9ef152bSMark Johnston 	return (error);
917 b9ef152bSMark Johnston }
918 b9ef152bSMark Johnston SYSCTL_PROC(_hw_vmm, OID_AUTO, create,
919 b9ef152bSMark Johnston     CTLTYPE_STRING | CTLFLAG_RW | CTLFLAG_PRISON | CTLFLAG_MPSAFE,
920 b9ef152bSMark Johnston     NULL, 0, sysctl_vmm_create, "A",
921 b9ef152bSMark Johnston     NULL);
922 b9ef152bSMark Johnston 
923 a97f683fSMark Johnston static int
vmmctl_open(struct cdev * cdev,int flags,int fmt,struct thread * td)924 a97f683fSMark Johnston vmmctl_open(struct cdev *cdev, int flags, int fmt, struct thread *td)
925 a97f683fSMark Johnston {
926 a97f683fSMark Johnston 	int error;
927 a97f683fSMark Johnston 
928 a97f683fSMark Johnston 	error = vmm_priv_check(td->td_ucred);
929 a97f683fSMark Johnston 	if (error != 0)
930 a97f683fSMark Johnston 		return (error);
931 a97f683fSMark Johnston 
932 a97f683fSMark Johnston 	if ((flags & FWRITE) == 0)
933 a97f683fSMark Johnston 		return (EPERM);
934 a97f683fSMark Johnston 
935 a97f683fSMark Johnston 	return (0);
936 a97f683fSMark Johnston }
937 a97f683fSMark Johnston 
938 a97f683fSMark Johnston static int
vmmctl_ioctl(struct cdev * cdev,u_long cmd,caddr_t data,int fflag,struct thread * td)939 a97f683fSMark Johnston vmmctl_ioctl(struct cdev *cdev, u_long cmd, caddr_t data, int fflag,
940 a97f683fSMark Johnston     struct thread *td)
941 a97f683fSMark Johnston {
942 a97f683fSMark Johnston 	int error;
943 a97f683fSMark Johnston 
944 a97f683fSMark Johnston 	switch (cmd) {
945 a97f683fSMark Johnston 	case VMMCTL_VM_CREATE: {
946 a97f683fSMark Johnston 		struct vmmctl_vm_create *vmc;
947 a97f683fSMark Johnston 
948 a97f683fSMark Johnston 		vmc = (struct vmmctl_vm_create *)data;
949 a97f683fSMark Johnston 		vmc->name[VM_MAX_NAMELEN] = '\0';
950 a97f683fSMark Johnston 		for (size_t i = 0; i < nitems(vmc->reserved); i++) {
951 a97f683fSMark Johnston 			if (vmc->reserved[i] != 0) {
952 a97f683fSMark Johnston 				error = EINVAL;
953 a97f683fSMark Johnston 				return (error);
954 a97f683fSMark Johnston 			}
955 a97f683fSMark Johnston 		}
956 a97f683fSMark Johnston 
957 a97f683fSMark Johnston 		error = vmmdev_create(vmc->name, td->td_ucred);
958 a97f683fSMark Johnston 		break;
959 a97f683fSMark Johnston 	}
960 a97f683fSMark Johnston 	case VMMCTL_VM_DESTROY: {
961 a97f683fSMark Johnston 		struct vmmctl_vm_destroy *vmd;
962 a97f683fSMark Johnston 
963 a97f683fSMark Johnston 		vmd = (struct vmmctl_vm_destroy *)data;
964 a97f683fSMark Johnston 		vmd->name[VM_MAX_NAMELEN] = '\0';
965 a97f683fSMark Johnston 		for (size_t i = 0; i < nitems(vmd->reserved); i++) {
966 a97f683fSMark Johnston 			if (vmd->reserved[i] != 0) {
967 a97f683fSMark Johnston 				error = EINVAL;
968 a97f683fSMark Johnston 				return (error);
969 a97f683fSMark Johnston 			}
970 a97f683fSMark Johnston 		}
971 a97f683fSMark Johnston 
972 a97f683fSMark Johnston 		error = vmmdev_lookup_and_destroy(vmd->name, td->td_ucred);
973 a97f683fSMark Johnston 		break;
974 a97f683fSMark Johnston 	}
975 a97f683fSMark Johnston 	default:
976 a97f683fSMark Johnston 		error = ENOTTY;
977 a97f683fSMark Johnston 		break;
978 a97f683fSMark Johnston 	}
979 a97f683fSMark Johnston 
980 a97f683fSMark Johnston 	return (error);
981 a97f683fSMark Johnston }
982 a97f683fSMark Johnston 
983 4a46ece6SMark Johnston static struct cdev *vmmctl_cdev;
984 a97f683fSMark Johnston static struct cdevsw vmmctlsw = {
985 a97f683fSMark Johnston 	.d_name		= "vmmctl",
986 a97f683fSMark Johnston 	.d_version	= D_VERSION,
987 a97f683fSMark Johnston 	.d_open		= vmmctl_open,
988 a97f683fSMark Johnston 	.d_ioctl	= vmmctl_ioctl,
989 a97f683fSMark Johnston };
990 a97f683fSMark Johnston 
991 a97f683fSMark Johnston int
vmmdev_init(void)992 b9ef152bSMark Johnston vmmdev_init(void)
993 b9ef152bSMark Johnston {
994 a97f683fSMark Johnston 	int error;
995 a97f683fSMark Johnston 
996 4a46ece6SMark Johnston 	sx_xlock(&vmmdev_mtx);
997 4a46ece6SMark Johnston 	error = make_dev_p(MAKEDEV_CHECKNAME, &vmmctl_cdev, &vmmctlsw, NULL,
998 a97f683fSMark Johnston 	    UID_ROOT, GID_WHEEL, 0600, "vmmctl");
999 4a46ece6SMark Johnston 	if (error == 0)
1000 b9ef152bSMark Johnston 		pr_allow_flag = prison_add_allow(NULL, "vmm", NULL,
1001 b9ef152bSMark Johnston 		    "Allow use of vmm in a jail.");
1002 4a46ece6SMark Johnston 	sx_xunlock(&vmmdev_mtx);
1003 a97f683fSMark Johnston 
1004 4a46ece6SMark Johnston 	return (error);
1005 b9ef152bSMark Johnston }
1006 b9ef152bSMark Johnston 
1007 b9ef152bSMark Johnston int
vmmdev_cleanup(void)1008 b9ef152bSMark Johnston vmmdev_cleanup(void)
1009 b9ef152bSMark Johnston {
1010 4a46ece6SMark Johnston 	sx_xlock(&vmmdev_mtx);
1011 4a46ece6SMark Johnston 	if (!SLIST_EMPTY(&head)) {
1012 4a46ece6SMark Johnston 		sx_xunlock(&vmmdev_mtx);
1013 4a46ece6SMark Johnston 		return (EBUSY);
1014 4a46ece6SMark Johnston 	}
1015 4a46ece6SMark Johnston 	if (vmmctl_cdev != NULL) {
1016 4a46ece6SMark Johnston 		destroy_dev(vmmctl_cdev);
1017 4a46ece6SMark Johnston 		vmmctl_cdev = NULL;
1018 4a46ece6SMark Johnston 	}
1019 4a46ece6SMark Johnston 	sx_xunlock(&vmmdev_mtx);
1020 b9ef152bSMark Johnston 
1021 4a46ece6SMark Johnston 	return (0);
1022 b9ef152bSMark Johnston }
1023 b9ef152bSMark Johnston 
1024 b9ef152bSMark Johnston static int
devmem_mmap_single(struct cdev * cdev,vm_ooffset_t * offset,vm_size_t len,struct vm_object ** objp,int nprot)1025 b9ef152bSMark Johnston devmem_mmap_single(struct cdev *cdev, vm_ooffset_t *offset, vm_size_t len,
1026 b9ef152bSMark Johnston     struct vm_object **objp, int nprot)
1027 b9ef152bSMark Johnston {
1028 b9ef152bSMark Johnston 	struct devmem_softc *dsc;
1029 b9ef152bSMark Johnston 	vm_ooffset_t first, last;
1030 b9ef152bSMark Johnston 	size_t seglen;
1031 b9ef152bSMark Johnston 	int error;
1032 b9ef152bSMark Johnston 	bool sysmem;
1033 b9ef152bSMark Johnston 
1034 b9ef152bSMark Johnston 	dsc = cdev->si_drv1;
1035 b9ef152bSMark Johnston 	if (dsc == NULL) {
1036 b9ef152bSMark Johnston 		/* 'cdev' has been created but is not ready for use */
1037 b9ef152bSMark Johnston 		return (ENXIO);
1038 b9ef152bSMark Johnston 	}
1039 b9ef152bSMark Johnston 
1040 b9ef152bSMark Johnston 	first = *offset;
1041 b9ef152bSMark Johnston 	last = *offset + len;
1042 b9ef152bSMark Johnston 	if ((nprot & PROT_EXEC) || first < 0 || first >= last)
1043 b9ef152bSMark Johnston 		return (EINVAL);
1044 b9ef152bSMark Johnston 
1045 b9ef152bSMark Johnston 	vm_slock_memsegs(dsc->sc->vm);
1046 b9ef152bSMark Johnston 
1047 b9ef152bSMark Johnston 	error = vm_get_memseg(dsc->sc->vm, dsc->segid, &seglen, &sysmem, objp);
1048 b9ef152bSMark Johnston 	KASSERT(error == 0 && !sysmem && *objp != NULL,
1049 b9ef152bSMark Johnston 	    ("%s: invalid devmem segment %d", __func__, dsc->segid));
1050 b9ef152bSMark Johnston 
1051 b9ef152bSMark Johnston 	if (seglen >= last)
1052 b9ef152bSMark Johnston 		vm_object_reference(*objp);
1053 b9ef152bSMark Johnston 	else
1054 b9ef152bSMark Johnston 		error = EINVAL;
1055 b9ef152bSMark Johnston 
1056 b9ef152bSMark Johnston 	vm_unlock_memsegs(dsc->sc->vm);
1057 b9ef152bSMark Johnston 	return (error);
1058 b9ef152bSMark Johnston }
1059 b9ef152bSMark Johnston 
1060 b9ef152bSMark Johnston static struct cdevsw devmemsw = {
1061 b9ef152bSMark Johnston 	.d_name		= "devmem",
1062 b9ef152bSMark Johnston 	.d_version	= D_VERSION,
1063 b9ef152bSMark Johnston 	.d_mmap_single	= devmem_mmap_single,
1064 b9ef152bSMark Johnston };
1065 b9ef152bSMark Johnston 
1066 b9ef152bSMark Johnston static int
devmem_create_cdev(struct vmmdev_softc * sc,int segid,char * devname)1067 f4002135SMark Johnston devmem_create_cdev(struct vmmdev_softc *sc, int segid, char *devname)
1068 b9ef152bSMark Johnston {
1069 cef5f43fSMark Johnston 	struct make_dev_args mda;
1070 b9ef152bSMark Johnston 	struct devmem_softc *dsc;
1071 b9ef152bSMark Johnston 	int error;
1072 b9ef152bSMark Johnston 
1073 cef5f43fSMark Johnston 	sx_xlock(&vmmdev_mtx);
1074 b9ef152bSMark Johnston 
1075 b9ef152bSMark Johnston 	dsc = malloc(sizeof(struct devmem_softc), M_VMMDEV, M_WAITOK | M_ZERO);
1076 b9ef152bSMark Johnston 	dsc->segid = segid;
1077 b9ef152bSMark Johnston 	dsc->name = devname;
1078 b9ef152bSMark Johnston 	dsc->sc = sc;
1079 b9ef152bSMark Johnston 	SLIST_INSERT_HEAD(&sc->devmem, dsc, link);
1080 cef5f43fSMark Johnston 
1081 cef5f43fSMark Johnston 	make_dev_args_init(&mda);
1082 cef5f43fSMark Johnston 	mda.mda_devsw = &devmemsw;
1083 cef5f43fSMark Johnston 	mda.mda_cr = sc->ucred;
1084 cef5f43fSMark Johnston 	mda.mda_uid = UID_ROOT;
1085 cef5f43fSMark Johnston 	mda.mda_gid = GID_WHEEL;
1086 cef5f43fSMark Johnston 	mda.mda_mode = 0600;
1087 cef5f43fSMark Johnston 	mda.mda_si_drv1 = dsc;
1088 cef5f43fSMark Johnston 	mda.mda_flags = MAKEDEV_CHECKNAME | MAKEDEV_WAITOK;
1089 cef5f43fSMark Johnston 	error = make_dev_s(&mda, &dsc->cdev, "vmm.io/%s.%s", vm_name(sc->vm),
1090 cef5f43fSMark Johnston 	    devname);
1091 cef5f43fSMark Johnston 	if (error != 0) {
1092 cef5f43fSMark Johnston 		SLIST_REMOVE(&sc->devmem, dsc, devmem_softc, link);
1093 cef5f43fSMark Johnston 		free(dsc->name, M_VMMDEV);
1094 cef5f43fSMark Johnston 		free(dsc, M_VMMDEV);
1095 cef5f43fSMark Johnston 	}
1096 cef5f43fSMark Johnston 
1097 887c0877SMark Johnston 	sx_xunlock(&vmmdev_mtx);
1098 b9ef152bSMark Johnston 
1099 cef5f43fSMark Johnston 	return (error);
1100 b9ef152bSMark Johnston }
1101 b9ef152bSMark Johnston 
1102 b9ef152bSMark Johnston static void
devmem_destroy(void * arg)1103 b9ef152bSMark Johnston devmem_destroy(void *arg)
1104 b9ef152bSMark Johnston {
1105 b9ef152bSMark Johnston 	struct devmem_softc *dsc = arg;
1106 b9ef152bSMark Johnston 
1107 cef5f43fSMark Johnston 	destroy_dev(dsc->cdev);
1108 b9ef152bSMark Johnston 	dsc->cdev = NULL;
1109 b9ef152bSMark Johnston 	dsc->sc = NULL;
1110 b9ef152bSMark Johnston }
1111