xref: /OK3568_Linux_fs/kernel/arch/arm64/kvm/pvtime.c (revision 4882a59341e53eb6f0b4789bf948001014eff981)
1*4882a593Smuzhiyun // SPDX-License-Identifier: GPL-2.0
2*4882a593Smuzhiyun // Copyright (C) 2019 Arm Ltd.
3*4882a593Smuzhiyun 
4*4882a593Smuzhiyun #include <linux/arm-smccc.h>
5*4882a593Smuzhiyun #include <linux/kvm_host.h>
6*4882a593Smuzhiyun #include <linux/sched/stat.h>
7*4882a593Smuzhiyun 
8*4882a593Smuzhiyun #include <asm/kvm_mmu.h>
9*4882a593Smuzhiyun #include <asm/pvclock-abi.h>
10*4882a593Smuzhiyun 
11*4882a593Smuzhiyun #include <kvm/arm_hypercalls.h>
12*4882a593Smuzhiyun 
kvm_update_stolen_time(struct kvm_vcpu * vcpu)13*4882a593Smuzhiyun void kvm_update_stolen_time(struct kvm_vcpu *vcpu)
14*4882a593Smuzhiyun {
15*4882a593Smuzhiyun 	struct kvm *kvm = vcpu->kvm;
16*4882a593Smuzhiyun 	u64 base = vcpu->arch.steal.base;
17*4882a593Smuzhiyun 	u64 last_steal = vcpu->arch.steal.last_steal;
18*4882a593Smuzhiyun 	u64 offset = offsetof(struct pvclock_vcpu_stolen_time, stolen_time);
19*4882a593Smuzhiyun 	u64 steal = 0;
20*4882a593Smuzhiyun 	int idx;
21*4882a593Smuzhiyun 
22*4882a593Smuzhiyun 	if (base == GPA_INVALID)
23*4882a593Smuzhiyun 		return;
24*4882a593Smuzhiyun 
25*4882a593Smuzhiyun 	idx = srcu_read_lock(&kvm->srcu);
26*4882a593Smuzhiyun 	if (!kvm_get_guest(kvm, base + offset, steal)) {
27*4882a593Smuzhiyun 		steal = le64_to_cpu(steal);
28*4882a593Smuzhiyun 		vcpu->arch.steal.last_steal = READ_ONCE(current->sched_info.run_delay);
29*4882a593Smuzhiyun 		steal += vcpu->arch.steal.last_steal - last_steal;
30*4882a593Smuzhiyun 		kvm_put_guest(kvm, base + offset, cpu_to_le64(steal));
31*4882a593Smuzhiyun 	}
32*4882a593Smuzhiyun 	srcu_read_unlock(&kvm->srcu, idx);
33*4882a593Smuzhiyun }
34*4882a593Smuzhiyun 
kvm_hypercall_pv_features(struct kvm_vcpu * vcpu)35*4882a593Smuzhiyun long kvm_hypercall_pv_features(struct kvm_vcpu *vcpu)
36*4882a593Smuzhiyun {
37*4882a593Smuzhiyun 	u32 feature = smccc_get_arg1(vcpu);
38*4882a593Smuzhiyun 	long val = SMCCC_RET_NOT_SUPPORTED;
39*4882a593Smuzhiyun 
40*4882a593Smuzhiyun 	switch (feature) {
41*4882a593Smuzhiyun 	case ARM_SMCCC_HV_PV_TIME_FEATURES:
42*4882a593Smuzhiyun 	case ARM_SMCCC_HV_PV_TIME_ST:
43*4882a593Smuzhiyun 		if (vcpu->arch.steal.base != GPA_INVALID)
44*4882a593Smuzhiyun 			val = SMCCC_RET_SUCCESS;
45*4882a593Smuzhiyun 		break;
46*4882a593Smuzhiyun 	}
47*4882a593Smuzhiyun 
48*4882a593Smuzhiyun 	return val;
49*4882a593Smuzhiyun }
50*4882a593Smuzhiyun 
kvm_init_stolen_time(struct kvm_vcpu * vcpu)51*4882a593Smuzhiyun gpa_t kvm_init_stolen_time(struct kvm_vcpu *vcpu)
52*4882a593Smuzhiyun {
53*4882a593Smuzhiyun 	struct pvclock_vcpu_stolen_time init_values = {};
54*4882a593Smuzhiyun 	struct kvm *kvm = vcpu->kvm;
55*4882a593Smuzhiyun 	u64 base = vcpu->arch.steal.base;
56*4882a593Smuzhiyun 
57*4882a593Smuzhiyun 	if (base == GPA_INVALID)
58*4882a593Smuzhiyun 		return base;
59*4882a593Smuzhiyun 
60*4882a593Smuzhiyun 	/*
61*4882a593Smuzhiyun 	 * Start counting stolen time from the time the guest requests
62*4882a593Smuzhiyun 	 * the feature enabled.
63*4882a593Smuzhiyun 	 */
64*4882a593Smuzhiyun 	vcpu->arch.steal.last_steal = current->sched_info.run_delay;
65*4882a593Smuzhiyun 	kvm_write_guest_lock(kvm, base, &init_values, sizeof(init_values));
66*4882a593Smuzhiyun 
67*4882a593Smuzhiyun 	return base;
68*4882a593Smuzhiyun }
69*4882a593Smuzhiyun 
kvm_arm_pvtime_supported(void)70*4882a593Smuzhiyun bool kvm_arm_pvtime_supported(void)
71*4882a593Smuzhiyun {
72*4882a593Smuzhiyun 	return !!sched_info_on();
73*4882a593Smuzhiyun }
74*4882a593Smuzhiyun 
kvm_arm_pvtime_set_attr(struct kvm_vcpu * vcpu,struct kvm_device_attr * attr)75*4882a593Smuzhiyun int kvm_arm_pvtime_set_attr(struct kvm_vcpu *vcpu,
76*4882a593Smuzhiyun 			    struct kvm_device_attr *attr)
77*4882a593Smuzhiyun {
78*4882a593Smuzhiyun 	u64 __user *user = (u64 __user *)attr->addr;
79*4882a593Smuzhiyun 	struct kvm *kvm = vcpu->kvm;
80*4882a593Smuzhiyun 	u64 ipa;
81*4882a593Smuzhiyun 	int ret = 0;
82*4882a593Smuzhiyun 	int idx;
83*4882a593Smuzhiyun 
84*4882a593Smuzhiyun 	if (!kvm_arm_pvtime_supported() ||
85*4882a593Smuzhiyun 	    attr->attr != KVM_ARM_VCPU_PVTIME_IPA)
86*4882a593Smuzhiyun 		return -ENXIO;
87*4882a593Smuzhiyun 
88*4882a593Smuzhiyun 	if (get_user(ipa, user))
89*4882a593Smuzhiyun 		return -EFAULT;
90*4882a593Smuzhiyun 	if (!IS_ALIGNED(ipa, 64))
91*4882a593Smuzhiyun 		return -EINVAL;
92*4882a593Smuzhiyun 	if (vcpu->arch.steal.base != GPA_INVALID)
93*4882a593Smuzhiyun 		return -EEXIST;
94*4882a593Smuzhiyun 
95*4882a593Smuzhiyun 	/* Check the address is in a valid memslot */
96*4882a593Smuzhiyun 	idx = srcu_read_lock(&kvm->srcu);
97*4882a593Smuzhiyun 	if (kvm_is_error_hva(gfn_to_hva(kvm, ipa >> PAGE_SHIFT)))
98*4882a593Smuzhiyun 		ret = -EINVAL;
99*4882a593Smuzhiyun 	srcu_read_unlock(&kvm->srcu, idx);
100*4882a593Smuzhiyun 
101*4882a593Smuzhiyun 	if (!ret)
102*4882a593Smuzhiyun 		vcpu->arch.steal.base = ipa;
103*4882a593Smuzhiyun 
104*4882a593Smuzhiyun 	return ret;
105*4882a593Smuzhiyun }
106*4882a593Smuzhiyun 
kvm_arm_pvtime_get_attr(struct kvm_vcpu * vcpu,struct kvm_device_attr * attr)107*4882a593Smuzhiyun int kvm_arm_pvtime_get_attr(struct kvm_vcpu *vcpu,
108*4882a593Smuzhiyun 			    struct kvm_device_attr *attr)
109*4882a593Smuzhiyun {
110*4882a593Smuzhiyun 	u64 __user *user = (u64 __user *)attr->addr;
111*4882a593Smuzhiyun 	u64 ipa;
112*4882a593Smuzhiyun 
113*4882a593Smuzhiyun 	if (!kvm_arm_pvtime_supported() ||
114*4882a593Smuzhiyun 	    attr->attr != KVM_ARM_VCPU_PVTIME_IPA)
115*4882a593Smuzhiyun 		return -ENXIO;
116*4882a593Smuzhiyun 
117*4882a593Smuzhiyun 	ipa = vcpu->arch.steal.base;
118*4882a593Smuzhiyun 
119*4882a593Smuzhiyun 	if (put_user(ipa, user))
120*4882a593Smuzhiyun 		return -EFAULT;
121*4882a593Smuzhiyun 	return 0;
122*4882a593Smuzhiyun }
123*4882a593Smuzhiyun 
kvm_arm_pvtime_has_attr(struct kvm_vcpu * vcpu,struct kvm_device_attr * attr)124*4882a593Smuzhiyun int kvm_arm_pvtime_has_attr(struct kvm_vcpu *vcpu,
125*4882a593Smuzhiyun 			    struct kvm_device_attr *attr)
126*4882a593Smuzhiyun {
127*4882a593Smuzhiyun 	switch (attr->attr) {
128*4882a593Smuzhiyun 	case KVM_ARM_VCPU_PVTIME_IPA:
129*4882a593Smuzhiyun 		if (kvm_arm_pvtime_supported())
130*4882a593Smuzhiyun 			return 0;
131*4882a593Smuzhiyun 	}
132*4882a593Smuzhiyun 	return -ENXIO;
133*4882a593Smuzhiyun }
134