xref: /freebsd/sys/riscv/vmm/vmm_riscv.c (revision f1ddb6fb8c4d051a205dae3a848776c9d56f86ff)
1 /*-
2  * SPDX-License-Identifier: BSD-2-Clause
3  *
4  * Copyright (c) 2024 Ruslan Bukin <br@bsdpad.com>
5  *
6  * This software was developed by the University of Cambridge Computer
7  * Laboratory (Department of Computer Science and Technology) under Innovate
8  * UK project 105694, "Digital Security by Design (DSbD) Technology Platform
9  * Prototype".
10  *
11  * Redistribution and use in source and binary forms, with or without
12  * modification, are permitted provided that the following conditions
13  * are met:
14  * 1. Redistributions of source code must retain the above copyright
15  *    notice, this list of conditions and the following disclaimer.
16  * 2. Redistributions in binary form must reproduce the above copyright
17  *    notice, this list of conditions and the following disclaimer in the
18  *    documentation and/or other materials provided with the distribution.
19  *
20  * THIS SOFTWARE IS PROVIDED BY AUTHOR AND CONTRIBUTORS ``AS IS'' AND
21  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
22  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
23  * ARE DISCLAIMED.  IN NO EVENT SHALL AUTHOR OR CONTRIBUTORS BE LIABLE
24  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
25  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
26  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
27  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
28  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
29  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
30  * SUCH DAMAGE.
31  */
32 
33 #include <sys/param.h>
34 #include <sys/systm.h>
35 #include <sys/smp.h>
36 #include <sys/kernel.h>
37 #include <sys/malloc.h>
38 #include <sys/mman.h>
39 #include <sys/pcpu.h>
40 #include <sys/proc.h>
41 #include <sys/rman.h>
42 #include <sys/sysctl.h>
43 #include <sys/lock.h>
44 #include <sys/mutex.h>
45 #include <sys/vmem.h>
46 #include <sys/bus.h>
47 
48 #include <vm/vm.h>
49 #include <vm/pmap.h>
50 #include <vm/vm_extern.h>
51 #include <vm/vm_map.h>
52 #include <vm/vm_page.h>
53 #include <vm/vm_param.h>
54 
55 #include <machine/md_var.h>
56 #include <machine/riscvreg.h>
57 #include <machine/vm.h>
58 #include <machine/cpufunc.h>
59 #include <machine/cpu.h>
60 #include <machine/machdep.h>
61 #include <machine/vmm.h>
62 #include <machine/vmm_dev.h>
63 #include <machine/atomic.h>
64 #include <machine/pmap.h>
65 #include <machine/intr.h>
66 #include <machine/encoding.h>
67 #include <machine/db_machdep.h>
68 
69 #include "riscv.h"
70 #include "vmm_aplic.h"
71 #include "vmm_stat.h"
72 
73 MALLOC_DEFINE(M_HYP, "RISC-V VMM HYP", "RISC-V VMM HYP");
74 
75 DPCPU_DEFINE_STATIC(struct hypctx *, vcpu);
76 
77 static int
78 m_op(uint32_t insn, int match, int mask)
79 {
80 
81 	if (((insn ^ match) & mask) == 0)
82 		return (1);
83 
84 	return (0);
85 }
86 
87 static inline void
88 riscv_set_active_vcpu(struct hypctx *hypctx)
89 {
90 
91 	DPCPU_SET(vcpu, hypctx);
92 }
93 
94 struct hypctx *
95 riscv_get_active_vcpu(void)
96 {
97 
98 	return (DPCPU_GET(vcpu));
99 }
100 
101 int
102 vmmops_modinit(void)
103 {
104 
105 	if (!has_hyp) {
106 		printf("vmm: riscv hart doesn't support H-extension.\n");
107 		return (ENXIO);
108 	}
109 
110 	if (!has_sstc) {
111 		printf("vmm: riscv hart doesn't support SSTC extension.\n");
112 		return (ENXIO);
113 	}
114 
115 	return (0);
116 }
117 
118 int
119 vmmops_modcleanup(void)
120 {
121 
122 	return (0);
123 }
124 
125 void *
126 vmmops_init(struct vm *vm, pmap_t pmap)
127 {
128 	struct hyp *hyp;
129 	vm_size_t size;
130 
131 	size = round_page(sizeof(struct hyp) +
132 	    sizeof(struct hypctx *) * vm_get_maxcpus(vm));
133 	hyp = malloc_aligned(size, PAGE_SIZE, M_HYP, M_WAITOK | M_ZERO);
134 	hyp->vm = vm;
135 	hyp->aplic_attached = false;
136 
137 	aplic_vminit(hyp);
138 
139 	return (hyp);
140 }
141 
142 static void
143 vmmops_delegate(void)
144 {
145 	uint64_t hedeleg;
146 	uint64_t hideleg;
147 
148 	hedeleg  = (1UL << SCAUSE_INST_MISALIGNED);
149 	hedeleg |= (1UL << SCAUSE_ILLEGAL_INSTRUCTION);
150 	hedeleg |= (1UL << SCAUSE_BREAKPOINT);
151 	hedeleg |= (1UL << SCAUSE_ECALL_USER);
152 	hedeleg |= (1UL << SCAUSE_INST_PAGE_FAULT);
153 	hedeleg |= (1UL << SCAUSE_LOAD_PAGE_FAULT);
154 	hedeleg |= (1UL << SCAUSE_STORE_PAGE_FAULT);
155 	csr_write(hedeleg, hedeleg);
156 
157 	hideleg  = (1UL << IRQ_SOFTWARE_HYPERVISOR);
158 	hideleg |= (1UL << IRQ_TIMER_HYPERVISOR);
159 	hideleg |= (1UL << IRQ_EXTERNAL_HYPERVISOR);
160 	csr_write(hideleg, hideleg);
161 }
162 
163 static void
164 vmmops_vcpu_restore_csrs(struct hypctx *hypctx)
165 {
166 	struct hypcsr *csrs;
167 
168 	csrs = &hypctx->guest_csrs;
169 
170 	csr_write(vsstatus, csrs->vsstatus);
171 	csr_write(vsie, csrs->vsie);
172 	csr_write(vstvec, csrs->vstvec);
173 	csr_write(vsscratch, csrs->vsscratch);
174 	csr_write(vsepc, csrs->vsepc);
175 	csr_write(vscause, csrs->vscause);
176 	csr_write(vstval, csrs->vstval);
177 	csr_write(hvip, csrs->hvip);
178 	csr_write(vsatp, csrs->vsatp);
179 }
180 
181 static void
182 vmmops_vcpu_save_csrs(struct hypctx *hypctx)
183 {
184 	struct hypcsr *csrs;
185 
186 	csrs = &hypctx->guest_csrs;
187 
188 	csrs->vsstatus = csr_read(vsstatus);
189 	csrs->vsie = csr_read(vsie);
190 	csrs->vstvec = csr_read(vstvec);
191 	csrs->vsscratch = csr_read(vsscratch);
192 	csrs->vsepc = csr_read(vsepc);
193 	csrs->vscause = csr_read(vscause);
194 	csrs->vstval = csr_read(vstval);
195 	csrs->hvip = csr_read(hvip);
196 	csrs->vsatp = csr_read(vsatp);
197 }
198 
199 void *
200 vmmops_vcpu_init(void *vmi, struct vcpu *vcpu1, int vcpuid)
201 {
202 	struct hypctx *hypctx;
203 	struct hyp *hyp;
204 	vm_size_t size;
205 
206 	hyp = vmi;
207 
208 	dprintf("%s: hyp %p\n", __func__, hyp);
209 
210 	KASSERT(vcpuid >= 0 && vcpuid < vm_get_maxcpus(hyp->vm),
211 	    ("%s: Invalid vcpuid %d", __func__, vcpuid));
212 
213 	size = round_page(sizeof(struct hypctx));
214 
215 	hypctx = malloc_aligned(size, PAGE_SIZE, M_HYP, M_WAITOK | M_ZERO);
216 	hypctx->hyp = hyp;
217 	hypctx->vcpu = vcpu1;
218 	hypctx->guest_scounteren = HCOUNTEREN_CY | HCOUNTEREN_TM;
219 
220 	/* sstatus */
221 	hypctx->guest_regs.hyp_sstatus = SSTATUS_SPP | SSTATUS_SPIE;
222 	hypctx->guest_regs.hyp_sstatus |= SSTATUS_FS_INITIAL;
223 
224 	/* hstatus */
225 	hypctx->guest_regs.hyp_hstatus = HSTATUS_SPV | HSTATUS_VTW;
226 	hypctx->guest_regs.hyp_hstatus |= HSTATUS_SPVP;
227 
228 	hypctx->cpu_id = vcpuid;
229 	hyp->ctx[vcpuid] = hypctx;
230 
231 	aplic_cpuinit(hypctx);
232 
233 	return (hypctx);
234 }
235 
236 static int
237 riscv_vmm_pinit(pmap_t pmap)
238 {
239 
240 	dprintf("%s: pmap %p\n", __func__, pmap);
241 
242 	pmap_pinit_stage(pmap, PM_STAGE2);
243 
244 	return (1);
245 }
246 
247 struct vmspace *
248 vmmops_vmspace_alloc(vm_offset_t min, vm_offset_t max)
249 {
250 
251 	return (vmspace_alloc(min, max, riscv_vmm_pinit));
252 }
253 
254 void
255 vmmops_vmspace_free(struct vmspace *vmspace)
256 {
257 
258 	pmap_remove_pages(vmspace_pmap(vmspace));
259 	vmspace_free(vmspace);
260 }
261 
262 static void
263 riscv_unpriv_read(struct hypctx *hypctx, uintptr_t guest_addr, uint64_t *data,
264     struct hyptrap *trap)
265 {
266 	register struct hyptrap * htrap asm("a0");
267 	uintptr_t old_hstatus;
268 	uintptr_t old_stvec;
269 	uintptr_t entry;
270 	uint64_t val;
271 	uint64_t tmp;
272 	int intr;
273 
274 	entry = (uintptr_t)&vmm_unpriv_trap;
275 	htrap = trap;
276 
277 	intr = intr_disable();
278 
279 	old_hstatus = csr_swap(hstatus, hypctx->guest_regs.hyp_hstatus);
280 	/*
281 	 * Setup a temporary exception vector, so that if hlvx.hu raises
282 	 * an exception we catch it in the vmm_unpriv_trap().
283 	 */
284 	old_stvec = csr_swap(stvec, entry);
285 
286 	/*
287 	 * Read first two bytes of instruction assuming it could be a
288 	 * compressed one.
289 	 */
290 	__asm __volatile(".option push\n"
291 			 ".option norvc\n"
292 			"hlvx.hu %[val], (%[addr])\n"
293 			".option pop\n"
294 	    : [val] "=r" (val)
295 	    : [addr] "r" (guest_addr), "r" (htrap)
296 	    : "a1", "memory");
297 
298 	/*
299 	 * Check if previous hlvx.hu did not raise an exception, and then
300 	 * read the rest of instruction if it is a full-length one.
301 	 */
302 	if (trap->scause == -1 && (val & 0x3) == 0x3) {
303 		guest_addr += 2;
304 		__asm __volatile(".option push\n"
305 				 ".option norvc\n"
306 				"hlvx.hu %[tmp], (%[addr])\n"
307 				".option pop\n"
308 		    : [tmp] "=r" (tmp)
309 		    : [addr] "r" (guest_addr), "r" (htrap)
310 		    : "a1", "memory");
311 		val |= (tmp << 16);
312 	}
313 
314 	csr_write(hstatus, old_hstatus);
315 	csr_write(stvec, old_stvec);
316 
317 	intr_restore(intr);
318 
319 	*data = val;
320 }
321 
322 static int
323 riscv_gen_inst_emul_data(struct hypctx *hypctx, struct vm_exit *vme_ret,
324     struct hyptrap *trap)
325 {
326 	uintptr_t guest_addr;
327 	struct vie *vie;
328 	uint64_t insn;
329 	int reg_num;
330 	int rs2, rd;
331 	int direction;
332 	int sign_extend;
333 	int access_size;
334 
335 	guest_addr = vme_ret->sepc;
336 
337 	KASSERT(vme_ret->scause == SCAUSE_FETCH_GUEST_PAGE_FAULT ||
338 	    vme_ret->scause == SCAUSE_LOAD_GUEST_PAGE_FAULT ||
339 	    vme_ret->scause == SCAUSE_STORE_GUEST_PAGE_FAULT,
340 	    ("Invalid scause"));
341 
342 	direction = vme_ret->scause == SCAUSE_STORE_GUEST_PAGE_FAULT ?
343 	    VM_DIR_WRITE : VM_DIR_READ;
344 
345 	sign_extend = 1;
346 
347 	bzero(trap, sizeof(struct hyptrap));
348 	trap->scause = -1;
349 	riscv_unpriv_read(hypctx, guest_addr, &insn, trap);
350 	if (trap->scause != -1)
351 		return (-1);
352 
353 	if ((insn & 0x3) == 0x3) {
354 		rs2 = (insn & RS2_MASK) >> RS2_SHIFT;
355 		rd = (insn & RD_MASK) >> RD_SHIFT;
356 
357 		if (direction == VM_DIR_WRITE) {
358 			if (m_op(insn, MATCH_SB, MASK_SB))
359 				access_size = 1;
360 			else if (m_op(insn, MATCH_SH, MASK_SH))
361 				access_size = 2;
362 			else if (m_op(insn, MATCH_SW, MASK_SW))
363 				access_size = 4;
364 			else if (m_op(insn, MATCH_SD, MASK_SD))
365 				access_size = 8;
366 			else {
367 				printf("unknown store instr at %lx",
368 				    guest_addr);
369 				return (-2);
370 			}
371 			reg_num = rs2;
372 		} else {
373 			if (m_op(insn, MATCH_LB, MASK_LB))
374 				access_size = 1;
375 			else if (m_op(insn, MATCH_LH, MASK_LH))
376 				access_size = 2;
377 			else if (m_op(insn, MATCH_LW, MASK_LW))
378 				access_size = 4;
379 			else if (m_op(insn, MATCH_LD, MASK_LD))
380 				access_size = 8;
381 			else if (m_op(insn, MATCH_LBU, MASK_LBU)) {
382 				access_size = 1;
383 				sign_extend = 0;
384 			} else if (m_op(insn, MATCH_LHU, MASK_LHU)) {
385 				access_size = 2;
386 				sign_extend = 0;
387 			} else if (m_op(insn, MATCH_LWU, MASK_LWU)) {
388 				access_size = 4;
389 				sign_extend = 0;
390 			} else {
391 				printf("unknown load instr at %lx",
392 				    guest_addr);
393 				return (-3);
394 			}
395 			reg_num = rd;
396 		}
397 		vme_ret->inst_length = 4;
398 	} else {
399 		rs2 = (insn >> 7) & 0x7;
400 		rs2 += 0x8;
401 		rd = (insn >> 2) & 0x7;
402 		rd += 0x8;
403 
404 		if (direction == VM_DIR_WRITE) {
405 			if (m_op(insn, MATCH_C_SW, MASK_C_SW))
406 				access_size = 4;
407 			else if (m_op(insn, MATCH_C_SD, MASK_C_SD))
408 				access_size = 8;
409 			else {
410 				printf("unknown compressed store instr at %lx",
411 				    guest_addr);
412 				return (-4);
413 			}
414 		} else  {
415 			if (m_op(insn, MATCH_C_LW, MASK_C_LW))
416 				access_size = 4;
417 			else if (m_op(insn, MATCH_C_LD, MASK_C_LD))
418 				access_size = 8;
419 			else {
420 				printf("unknown load instr at %lx", guest_addr);
421 				return (-5);
422 			}
423 		}
424 		reg_num = rd;
425 		vme_ret->inst_length = 2;
426 	}
427 
428 	vme_ret->u.inst_emul.gpa = (vme_ret->htval << 2) |
429 	    (vme_ret->stval & 0x3);
430 
431 	dprintf("guest_addr %lx insn %lx, reg %d, gpa %lx\n", guest_addr, insn,
432 	    reg_num, vme_ret->u.inst_emul.gpa);
433 
434 	vie = &vme_ret->u.inst_emul.vie;
435 	vie->dir = direction;
436 	vie->reg = reg_num;
437 	vie->sign_extend = sign_extend;
438 	vie->access_size = access_size;
439 
440 	return (0);
441 }
442 
443 static bool
444 riscv_handle_world_switch(struct hypctx *hypctx, struct vm_exit *vme,
445     pmap_t pmap)
446 {
447 	struct hyptrap trap;
448 	uint64_t insn;
449 	uint64_t gpa;
450 	bool handled;
451 	bool retu;
452 	int ret;
453 	int i;
454 
455 	handled = false;
456 
457 	if (vme->scause & SCAUSE_INTR) {
458 		/*
459 		 * Host interrupt? Leave critical section to handle.
460 		 */
461 		vmm_stat_incr(hypctx->vcpu, VMEXIT_IRQ, 1);
462 		vme->exitcode = VM_EXITCODE_BOGUS;
463 		vme->inst_length = 0;
464 		return (handled);
465 	}
466 
467 	switch (vme->scause) {
468 	case SCAUSE_FETCH_GUEST_PAGE_FAULT:
469 	case SCAUSE_LOAD_GUEST_PAGE_FAULT:
470 	case SCAUSE_STORE_GUEST_PAGE_FAULT:
471 		gpa = (vme->htval << 2) | (vme->stval & 0x3);
472 		if (vm_mem_allocated(hypctx->vcpu, gpa)) {
473 			vme->exitcode = VM_EXITCODE_PAGING;
474 			vme->inst_length = 0;
475 			vme->u.paging.gpa = gpa;
476 		} else {
477 			ret = riscv_gen_inst_emul_data(hypctx, vme, &trap);
478 			if (ret != 0) {
479 				vme->exitcode = VM_EXITCODE_HYP;
480 				vme->u.hyp.scause = trap.scause;
481 				break;
482 			}
483 			vme->exitcode = VM_EXITCODE_INST_EMUL;
484 		}
485 		break;
486 	case SCAUSE_ILLEGAL_INSTRUCTION:
487 		/*
488 		 * TODO: handle illegal instruction properly.
489 		 */
490 		printf("%s: Illegal instruction at %lx stval 0x%lx htval "
491 		    "0x%lx\n", __func__, vme->sepc, vme->stval, vme->htval);
492 		vmm_stat_incr(hypctx->vcpu, VMEXIT_UNHANDLED, 1);
493 		vme->exitcode = VM_EXITCODE_BOGUS;
494 		handled = false;
495 		break;
496 	case SCAUSE_VIRTUAL_SUPERVISOR_ECALL:
497 		retu = false;
498 		vmm_sbi_ecall(hypctx->vcpu, &retu);
499 		if (retu == false) {
500 			handled = true;
501 			break;
502 		}
503 		for (i = 0; i < nitems(vme->u.ecall.args); i++)
504 			vme->u.ecall.args[i] = hypctx->guest_regs.hyp_a[i];
505 		vme->exitcode = VM_EXITCODE_ECALL;
506 		handled = false;
507 		break;
508 	case SCAUSE_VIRTUAL_INSTRUCTION:
509 		insn = vme->stval;
510 		if (m_op(insn, MATCH_WFI, MASK_WFI))
511 			vme->exitcode = VM_EXITCODE_WFI;
512 		else
513 			vme->exitcode = VM_EXITCODE_BOGUS;
514 		handled = false;
515 		break;
516 	default:
517 		printf("unknown scause %lx\n", vme->scause);
518 		vmm_stat_incr(hypctx->vcpu, VMEXIT_UNHANDLED, 1);
519 		vme->exitcode = VM_EXITCODE_BOGUS;
520 		handled = false;
521 		break;
522 	}
523 
524 	return (handled);
525 }
526 
527 int
528 vmmops_gla2gpa(void *vcpui, struct vm_guest_paging *paging, uint64_t gla,
529     int prot, uint64_t *gpa, int *is_fault)
530 {
531 
532 	/* Implement me. */
533 
534 	return (ENOSYS);
535 }
536 
537 void
538 riscv_send_ipi(struct hypctx *hypctx, int hart_id)
539 {
540 	struct hyp *hyp;
541 	struct vm *vm;
542 
543 	hyp = hypctx->hyp;
544 	vm = hyp->vm;
545 
546 	atomic_set_32(&hypctx->ipi_pending, 1);
547 
548 	vcpu_notify_event(vm_vcpu(vm, hart_id));
549 }
550 
551 int
552 riscv_check_ipi(struct hypctx *hypctx, bool clear)
553 {
554 	int val;
555 
556 	if (clear)
557 		val = atomic_swap_32(&hypctx->ipi_pending, 0);
558 	else
559 		val = hypctx->ipi_pending;
560 
561 	return (val);
562 }
563 
564 static void
565 riscv_sync_interrupts(struct hypctx *hypctx)
566 {
567 	int pending;
568 
569 	pending = aplic_check_pending(hypctx);
570 
571 	if (pending)
572 		hypctx->guest_csrs.hvip |= HVIP_VSEIP;
573 	else
574 		hypctx->guest_csrs.hvip &= ~HVIP_VSEIP;
575 
576 	csr_write(hvip, hypctx->guest_csrs.hvip);
577 }
578 
579 static void
580 riscv_sync_ipi(struct hypctx *hypctx)
581 {
582 
583 	/* Guest clears VSSIP bit manually. */
584 	if (riscv_check_ipi(hypctx, true))
585 		hypctx->guest_csrs.hvip |= HVIP_VSSIP;
586 
587 	csr_write(hvip, hypctx->guest_csrs.hvip);
588 }
589 
590 int
591 vmmops_run(void *vcpui, register_t pc, pmap_t pmap, struct vm_eventinfo *evinfo)
592 {
593 	struct hypctx *hypctx;
594 	struct vm_exit *vme;
595 	struct vcpu *vcpu;
596 	register_t val;
597 	bool handled;
598 
599 	hypctx = (struct hypctx *)vcpui;
600 	vcpu = hypctx->vcpu;
601 	vme = vm_exitinfo(vcpu);
602 
603 	hypctx->guest_regs.hyp_sepc = (uint64_t)pc;
604 
605 	vmmops_delegate();
606 
607 	/*
608 	 * From The RISC-V Instruction Set Manual
609 	 * Volume II: RISC-V Privileged Architectures
610 	 *
611 	 * If the new virtual machine's guest physical page tables
612 	 * have been modified, it may be necessary to execute an HFENCE.GVMA
613 	 * instruction (see Section 5.3.2) before or after writing hgatp.
614 	 */
615 	__asm __volatile("hfence.gvma" ::: "memory");
616 
617 	csr_write(hgatp, pmap->pm_satp);
618 	csr_write(henvcfg, HENVCFG_STCE);
619 	csr_write(hie, HIE_VSEIE | HIE_VSSIE | HIE_SGEIE);
620 	/* TODO: should we trap rdcycle / rdtime? */
621 	csr_write(hcounteren, HCOUNTEREN_CY | HCOUNTEREN_TM);
622 
623 	vmmops_vcpu_restore_csrs(hypctx);
624 
625 	for (;;) {
626 		dprintf("%s: pc %lx\n", __func__, pc);
627 
628 		if (hypctx->has_exception) {
629 			hypctx->has_exception = false;
630 			/*
631 			 * TODO: implement exception injection.
632 			 */
633 		}
634 
635 		val = intr_disable();
636 
637 		/* Check if the vcpu is suspended */
638 		if (vcpu_suspended(evinfo)) {
639 			intr_restore(val);
640 			vm_exit_suspended(vcpu, pc);
641 			break;
642 		}
643 
644 		if (vcpu_debugged(vcpu)) {
645 			intr_restore(val);
646 			vm_exit_debug(vcpu, pc);
647 			break;
648 		}
649 
650 		/*
651 		 * TODO: What happens if a timer interrupt is asserted exactly
652 		 * here, but for the previous VM?
653 		 */
654 		riscv_set_active_vcpu(hypctx);
655 		aplic_flush_hwstate(hypctx);
656 
657 		riscv_sync_interrupts(hypctx);
658 		riscv_sync_ipi(hypctx);
659 
660 		dprintf("%s: Entering guest VM, vsatp %lx, ss %lx hs %lx\n",
661 		    __func__, csr_read(vsatp), hypctx->guest_regs.hyp_sstatus,
662 		    hypctx->guest_regs.hyp_hstatus);
663 
664 		vmm_switch(hypctx);
665 
666 		dprintf("%s: Leaving guest VM, hstatus %lx\n", __func__,
667 		    hypctx->guest_regs.hyp_hstatus);
668 
669 		aplic_sync_hwstate(hypctx);
670 		riscv_sync_interrupts(hypctx);
671 
672 		/*
673 		 * TODO: deactivate stage 2 pmap here if needed.
674 		 */
675 
676 		vme->scause = csr_read(scause);
677 		vme->sepc = csr_read(sepc);
678 		vme->stval = csr_read(stval);
679 		vme->htval = csr_read(htval);
680 		vme->htinst = csr_read(htinst);
681 
682 		intr_restore(val);
683 
684 		vmm_stat_incr(vcpu, VMEXIT_COUNT, 1);
685 		vme->pc = hypctx->guest_regs.hyp_sepc;
686 		vme->inst_length = INSN_SIZE;
687 
688 		handled = riscv_handle_world_switch(hypctx, vme, pmap);
689 		if (handled == false)
690 			/* Exit loop to emulate instruction. */
691 			break;
692 		else {
693 			/* Resume guest execution from the next instruction. */
694 			hypctx->guest_regs.hyp_sepc += vme->inst_length;
695 		}
696 	}
697 
698 	vmmops_vcpu_save_csrs(hypctx);
699 
700 	return (0);
701 }
702 
703 static void
704 riscv_pcpu_vmcleanup(void *arg)
705 {
706 	struct hyp *hyp;
707 	int i, maxcpus;
708 
709 	hyp = arg;
710 	maxcpus = vm_get_maxcpus(hyp->vm);
711 	for (i = 0; i < maxcpus; i++) {
712 		if (riscv_get_active_vcpu() == hyp->ctx[i]) {
713 			riscv_set_active_vcpu(NULL);
714 			break;
715 		}
716 	}
717 }
718 
719 void
720 vmmops_vcpu_cleanup(void *vcpui)
721 {
722 	struct hypctx *hypctx;
723 
724 	hypctx = vcpui;
725 
726 	dprintf("%s\n", __func__);
727 
728 	aplic_cpucleanup(hypctx);
729 
730 	free(hypctx, M_HYP);
731 }
732 
733 void
734 vmmops_cleanup(void *vmi)
735 {
736 	struct hyp *hyp;
737 
738 	hyp = vmi;
739 
740 	dprintf("%s\n", __func__);
741 
742 	aplic_vmcleanup(hyp);
743 
744 	smp_rendezvous(NULL, riscv_pcpu_vmcleanup, NULL, hyp);
745 
746 	free(hyp, M_HYP);
747 }
748 
749 /*
750  * Return register value. Registers have different sizes and an explicit cast
751  * must be made to ensure proper conversion.
752  */
753 static uint64_t *
754 hypctx_regptr(struct hypctx *hypctx, int reg)
755 {
756 
757 	switch (reg) {
758 	case VM_REG_GUEST_RA:
759 		return (&hypctx->guest_regs.hyp_ra);
760 	case VM_REG_GUEST_SP:
761 		return (&hypctx->guest_regs.hyp_sp);
762 	case VM_REG_GUEST_GP:
763 		return (&hypctx->guest_regs.hyp_gp);
764 	case VM_REG_GUEST_TP:
765 		return (&hypctx->guest_regs.hyp_tp);
766 	case VM_REG_GUEST_T0:
767 		return (&hypctx->guest_regs.hyp_t[0]);
768 	case VM_REG_GUEST_T1:
769 		return (&hypctx->guest_regs.hyp_t[1]);
770 	case VM_REG_GUEST_T2:
771 		return (&hypctx->guest_regs.hyp_t[2]);
772 	case VM_REG_GUEST_S0:
773 		return (&hypctx->guest_regs.hyp_s[0]);
774 	case VM_REG_GUEST_S1:
775 		return (&hypctx->guest_regs.hyp_s[1]);
776 	case VM_REG_GUEST_A0:
777 		return (&hypctx->guest_regs.hyp_a[0]);
778 	case VM_REG_GUEST_A1:
779 		return (&hypctx->guest_regs.hyp_a[1]);
780 	case VM_REG_GUEST_A2:
781 		return (&hypctx->guest_regs.hyp_a[2]);
782 	case VM_REG_GUEST_A3:
783 		return (&hypctx->guest_regs.hyp_a[3]);
784 	case VM_REG_GUEST_A4:
785 		return (&hypctx->guest_regs.hyp_a[4]);
786 	case VM_REG_GUEST_A5:
787 		return (&hypctx->guest_regs.hyp_a[5]);
788 	case VM_REG_GUEST_A6:
789 		return (&hypctx->guest_regs.hyp_a[6]);
790 	case VM_REG_GUEST_A7:
791 		return (&hypctx->guest_regs.hyp_a[7]);
792 	case VM_REG_GUEST_S2:
793 		return (&hypctx->guest_regs.hyp_s[2]);
794 	case VM_REG_GUEST_S3:
795 		return (&hypctx->guest_regs.hyp_s[3]);
796 	case VM_REG_GUEST_S4:
797 		return (&hypctx->guest_regs.hyp_s[4]);
798 	case VM_REG_GUEST_S5:
799 		return (&hypctx->guest_regs.hyp_s[5]);
800 	case VM_REG_GUEST_S6:
801 		return (&hypctx->guest_regs.hyp_s[6]);
802 	case VM_REG_GUEST_S7:
803 		return (&hypctx->guest_regs.hyp_s[7]);
804 	case VM_REG_GUEST_S8:
805 		return (&hypctx->guest_regs.hyp_s[8]);
806 	case VM_REG_GUEST_S9:
807 		return (&hypctx->guest_regs.hyp_s[9]);
808 	case VM_REG_GUEST_S10:
809 		return (&hypctx->guest_regs.hyp_s[10]);
810 	case VM_REG_GUEST_S11:
811 		return (&hypctx->guest_regs.hyp_s[11]);
812 	case VM_REG_GUEST_T3:
813 		return (&hypctx->guest_regs.hyp_t[3]);
814 	case VM_REG_GUEST_T4:
815 		return (&hypctx->guest_regs.hyp_t[4]);
816 	case VM_REG_GUEST_T5:
817 		return (&hypctx->guest_regs.hyp_t[5]);
818 	case VM_REG_GUEST_T6:
819 		return (&hypctx->guest_regs.hyp_t[6]);
820 	case VM_REG_GUEST_SEPC:
821 		return (&hypctx->guest_regs.hyp_sepc);
822 	default:
823 		break;
824 	}
825 
826 	return (NULL);
827 }
828 
829 int
830 vmmops_getreg(void *vcpui, int reg, uint64_t *retval)
831 {
832 	uint64_t *regp;
833 	int running, hostcpu;
834 	struct hypctx *hypctx;
835 
836 	hypctx = vcpui;
837 
838 	running = vcpu_is_running(hypctx->vcpu, &hostcpu);
839 	if (running && hostcpu != curcpu)
840 		panic("%s: %s%d is running", __func__, vm_name(hypctx->hyp->vm),
841 		    vcpu_vcpuid(hypctx->vcpu));
842 
843 	if (reg == VM_REG_GUEST_ZERO) {
844 		*retval = 0;
845 		return (0);
846 	}
847 
848 	regp = hypctx_regptr(hypctx, reg);
849 	if (regp == NULL)
850 		return (EINVAL);
851 
852 	*retval = *regp;
853 
854 	return (0);
855 }
856 
857 int
858 vmmops_setreg(void *vcpui, int reg, uint64_t val)
859 {
860 	struct hypctx *hypctx;
861 	int running, hostcpu;
862 	uint64_t *regp;
863 
864 	hypctx = vcpui;
865 
866 	running = vcpu_is_running(hypctx->vcpu, &hostcpu);
867 	if (running && hostcpu != curcpu)
868 		panic("%s: %s%d is running", __func__, vm_name(hypctx->hyp->vm),
869 		    vcpu_vcpuid(hypctx->vcpu));
870 
871 	regp = hypctx_regptr(hypctx, reg);
872 	if (regp == NULL)
873 		return (EINVAL);
874 
875 	*regp = val;
876 
877 	return (0);
878 }
879 
880 int
881 vmmops_exception(void *vcpui, uint64_t scause)
882 {
883 	struct hypctx *hypctx;
884 	int running, hostcpu;
885 
886 	hypctx = vcpui;
887 
888 	running = vcpu_is_running(hypctx->vcpu, &hostcpu);
889 	if (running && hostcpu != curcpu)
890 		panic("%s: %s%d is running", __func__, vm_name(hypctx->hyp->vm),
891 		    vcpu_vcpuid(hypctx->vcpu));
892 
893 	/* TODO: implement me. */
894 
895 	return (ENOSYS);
896 }
897 
898 int
899 vmmops_getcap(void *vcpui, int num, int *retval)
900 {
901 	int ret;
902 
903 	ret = ENOENT;
904 
905 	switch (num) {
906 	case VM_CAP_SSTC:
907 		*retval = has_sstc;
908 		ret = 0;
909 		break;
910 	case VM_CAP_UNRESTRICTED_GUEST:
911 		*retval = 1;
912 		ret = 0;
913 		break;
914 	default:
915 		break;
916 	}
917 
918 	return (ret);
919 }
920 
921 int
922 vmmops_setcap(void *vcpui, int num, int val)
923 {
924 
925 	return (ENOENT);
926 }
927