xref: /openbmc/linux/arch/riscv/kvm/vcpu_sbi_v01.c (revision bae0dfd7)
1a046c2d8SAtish Patra // SPDX-License-Identifier: GPL-2.0
2a046c2d8SAtish Patra /*
3a046c2d8SAtish Patra  * Copyright (c) 2021 Western Digital Corporation or its affiliates.
4a046c2d8SAtish Patra  *
5a046c2d8SAtish Patra  * Authors:
6a046c2d8SAtish Patra  *     Atish Patra <atish.patra@wdc.com>
7a046c2d8SAtish Patra  */
8a046c2d8SAtish Patra 
9a046c2d8SAtish Patra #include <linux/errno.h>
10a046c2d8SAtish Patra #include <linux/err.h>
11a046c2d8SAtish Patra #include <linux/kvm_host.h>
12a046c2d8SAtish Patra #include <asm/sbi.h>
13a046c2d8SAtish Patra #include <asm/kvm_vcpu_timer.h>
14a046c2d8SAtish Patra #include <asm/kvm_vcpu_sbi.h>
15a046c2d8SAtish Patra 
kvm_sbi_ext_v01_handler(struct kvm_vcpu * vcpu,struct kvm_run * run,struct kvm_vcpu_sbi_return * retdata)16a046c2d8SAtish Patra static int kvm_sbi_ext_v01_handler(struct kvm_vcpu *vcpu, struct kvm_run *run,
17*bae0dfd7SAtish Patra 				   struct kvm_vcpu_sbi_return *retdata)
18a046c2d8SAtish Patra {
19a046c2d8SAtish Patra 	ulong hmask;
20a046c2d8SAtish Patra 	int i, ret = 0;
21a046c2d8SAtish Patra 	u64 next_cycle;
22a046c2d8SAtish Patra 	struct kvm_vcpu *rvcpu;
23a046c2d8SAtish Patra 	struct kvm *kvm = vcpu->kvm;
24a046c2d8SAtish Patra 	struct kvm_cpu_context *cp = &vcpu->arch.guest_context;
25*bae0dfd7SAtish Patra 	struct kvm_cpu_trap *utrap = retdata->utrap;
26a046c2d8SAtish Patra 
27a046c2d8SAtish Patra 	switch (cp->a7) {
28a046c2d8SAtish Patra 	case SBI_EXT_0_1_CONSOLE_GETCHAR:
29a046c2d8SAtish Patra 	case SBI_EXT_0_1_CONSOLE_PUTCHAR:
30a046c2d8SAtish Patra 		/*
31a046c2d8SAtish Patra 		 * The CONSOLE_GETCHAR/CONSOLE_PUTCHAR SBI calls cannot be
32a046c2d8SAtish Patra 		 * handled in kernel so we forward these to user-space
33a046c2d8SAtish Patra 		 */
34a046c2d8SAtish Patra 		kvm_riscv_vcpu_sbi_forward(vcpu, run);
35*bae0dfd7SAtish Patra 		retdata->uexit = true;
36a046c2d8SAtish Patra 		break;
37a046c2d8SAtish Patra 	case SBI_EXT_0_1_SET_TIMER:
38a046c2d8SAtish Patra #if __riscv_xlen == 32
39a046c2d8SAtish Patra 		next_cycle = ((u64)cp->a1 << 32) | (u64)cp->a0;
40a046c2d8SAtish Patra #else
41a046c2d8SAtish Patra 		next_cycle = (u64)cp->a0;
42a046c2d8SAtish Patra #endif
43a046c2d8SAtish Patra 		ret = kvm_riscv_vcpu_timer_next_event(vcpu, next_cycle);
44a046c2d8SAtish Patra 		break;
45a046c2d8SAtish Patra 	case SBI_EXT_0_1_CLEAR_IPI:
46a046c2d8SAtish Patra 		ret = kvm_riscv_vcpu_unset_interrupt(vcpu, IRQ_VS_SOFT);
47a046c2d8SAtish Patra 		break;
48a046c2d8SAtish Patra 	case SBI_EXT_0_1_SEND_IPI:
49a046c2d8SAtish Patra 		if (cp->a0)
50*bae0dfd7SAtish Patra 			hmask = kvm_riscv_vcpu_unpriv_read(vcpu, false, cp->a0, utrap);
51a046c2d8SAtish Patra 		else
52a046c2d8SAtish Patra 			hmask = (1UL << atomic_read(&kvm->online_vcpus)) - 1;
53a046c2d8SAtish Patra 		if (utrap->scause)
54a046c2d8SAtish Patra 			break;
55a046c2d8SAtish Patra 
56a046c2d8SAtish Patra 		for_each_set_bit(i, &hmask, BITS_PER_LONG) {
57a046c2d8SAtish Patra 			rvcpu = kvm_get_vcpu_by_id(vcpu->kvm, i);
58a046c2d8SAtish Patra 			ret = kvm_riscv_vcpu_set_interrupt(rvcpu, IRQ_VS_SOFT);
59a046c2d8SAtish Patra 			if (ret < 0)
60a046c2d8SAtish Patra 				break;
61a046c2d8SAtish Patra 		}
62a046c2d8SAtish Patra 		break;
63a046c2d8SAtish Patra 	case SBI_EXT_0_1_SHUTDOWN:
644b11d865SAnup Patel 		kvm_riscv_vcpu_sbi_system_reset(vcpu, run,
654b11d865SAnup Patel 						KVM_SYSTEM_EVENT_SHUTDOWN, 0);
66*bae0dfd7SAtish Patra 		retdata->uexit = true;
67a046c2d8SAtish Patra 		break;
68a046c2d8SAtish Patra 	case SBI_EXT_0_1_REMOTE_FENCE_I:
69a046c2d8SAtish Patra 	case SBI_EXT_0_1_REMOTE_SFENCE_VMA:
70a046c2d8SAtish Patra 	case SBI_EXT_0_1_REMOTE_SFENCE_VMA_ASID:
71a046c2d8SAtish Patra 		if (cp->a0)
72*bae0dfd7SAtish Patra 			hmask = kvm_riscv_vcpu_unpriv_read(vcpu, false, cp->a0, utrap);
73a046c2d8SAtish Patra 		else
74a046c2d8SAtish Patra 			hmask = (1UL << atomic_read(&kvm->online_vcpus)) - 1;
75a046c2d8SAtish Patra 		if (utrap->scause)
76a046c2d8SAtish Patra 			break;
77a046c2d8SAtish Patra 
78a046c2d8SAtish Patra 		if (cp->a7 == SBI_EXT_0_1_REMOTE_FENCE_I)
7913acfec2SAnup Patel 			kvm_riscv_fence_i(vcpu->kvm, 0, hmask);
8013acfec2SAnup Patel 		else if (cp->a7 == SBI_EXT_0_1_REMOTE_SFENCE_VMA) {
8113acfec2SAnup Patel 			if (cp->a1 == 0 && cp->a2 == 0)
8213acfec2SAnup Patel 				kvm_riscv_hfence_vvma_all(vcpu->kvm,
8313acfec2SAnup Patel 							  0, hmask);
84a046c2d8SAtish Patra 			else
8513acfec2SAnup Patel 				kvm_riscv_hfence_vvma_gva(vcpu->kvm,
8613acfec2SAnup Patel 							  0, hmask,
8713acfec2SAnup Patel 							  cp->a1, cp->a2,
8813acfec2SAnup Patel 							  PAGE_SHIFT);
8913acfec2SAnup Patel 		} else {
9013acfec2SAnup Patel 			if (cp->a1 == 0 && cp->a2 == 0)
9113acfec2SAnup Patel 				kvm_riscv_hfence_vvma_asid_all(vcpu->kvm,
9213acfec2SAnup Patel 							       0, hmask,
9313acfec2SAnup Patel 							       cp->a3);
9413acfec2SAnup Patel 			else
9513acfec2SAnup Patel 				kvm_riscv_hfence_vvma_asid_gva(vcpu->kvm,
9613acfec2SAnup Patel 							       0, hmask,
9713acfec2SAnup Patel 							       cp->a1, cp->a2,
9813acfec2SAnup Patel 							       PAGE_SHIFT,
9913acfec2SAnup Patel 							       cp->a3);
10013acfec2SAnup Patel 		}
101a046c2d8SAtish Patra 		break;
102a046c2d8SAtish Patra 	default:
103*bae0dfd7SAtish Patra 		retdata->err_val = SBI_ERR_NOT_SUPPORTED;
104a046c2d8SAtish Patra 		break;
1058eb3e1b9SYang Li 	}
106a046c2d8SAtish Patra 
107a046c2d8SAtish Patra 	return ret;
108a046c2d8SAtish Patra }
109a046c2d8SAtish Patra 
110a046c2d8SAtish Patra const struct kvm_vcpu_sbi_extension vcpu_sbi_ext_v01 = {
111a046c2d8SAtish Patra 	.extid_start = SBI_EXT_0_1_SET_TIMER,
112a046c2d8SAtish Patra 	.extid_end = SBI_EXT_0_1_SHUTDOWN,
113a046c2d8SAtish Patra 	.handler = kvm_sbi_ext_v01_handler,
114a046c2d8SAtish Patra };
115