mirror of
				https://github.com/kata-containers/kata-containers.git
				synced 2025-10-25 14:23:11 +00:00 
			
		
		
		
	kernel: PTP_KVM support for arm/arm64 in Kata
This work patched the 4.19, 5.4 and 5.10 kernels, and now ptp_kvm can work correctly when the host and guest use different kernel versions.. Fixes: #2123 Signed-off-by: Damon Kwok <damon-kwok@outlook.com>
This commit is contained in:
		| @@ -1,457 +0,0 @@ | ||||
| From bee1ae5587a7427dbb9e9e313f6d0a43a9e0ec2e Mon Sep 17 00:00:00 2001 | ||||
| From: Jianyong Wu <jianyong.wu@arm.com> | ||||
| Date: Mon, 30 Sep 2019 09:26:22 +0800 | ||||
| Subject: [PATCH] 4.19: enable ptp_kvm for arm64 in kata | ||||
|  | ||||
| --- | ||||
|  drivers/clocksource/arm_arch_timer.c        | 25 ++++++ | ||||
|  drivers/ptp/Kconfig                         |  2 +- | ||||
|  drivers/ptp/Makefile                        |  1 + | ||||
|  drivers/ptp/ptp_kvm_arm64.c                 | 59 ++++++++++++++ | ||||
|  drivers/ptp/{ptp_kvm.c => ptp_kvm_common.c} | 89 +++++---------------- | ||||
|  drivers/ptp/ptp_kvm_x86.c                   | 87 ++++++++++++++++++++ | ||||
|  include/asm-generic/ptp_kvm.h               | 12 +++ | ||||
|  include/linux/arm-smccc.h                   |  5 ++ | ||||
|  virt/kvm/arm/psci.c                         | 12 +++ | ||||
|  9 files changed, 221 insertions(+), 71 deletions(-) | ||||
|  create mode 100644 drivers/ptp/ptp_kvm_arm64.c | ||||
|  rename drivers/ptp/{ptp_kvm.c => ptp_kvm_common.c} (56%) | ||||
|  create mode 100644 drivers/ptp/ptp_kvm_x86.c | ||||
|  create mode 100644 include/asm-generic/ptp_kvm.h | ||||
|  | ||||
| diff --git a/drivers/clocksource/arm_arch_timer.c b/drivers/clocksource/arm_arch_timer.c | ||||
| index d8c7f5750cdb..84ba8f9e57be 100644 | ||||
| --- a/drivers/clocksource/arm_arch_timer.c | ||||
| +++ b/drivers/clocksource/arm_arch_timer.c | ||||
| @@ -1571,3 +1571,28 @@ static int __init arch_timer_acpi_init(struct acpi_table_header *table) | ||||
|  } | ||||
|  TIMER_ACPI_DECLARE(arch_timer, ACPI_SIG_GTDT, arch_timer_acpi_init); | ||||
|  #endif | ||||
| + | ||||
| +#if IS_ENABLED(CONFIG_PTP_1588_CLOCK_KVM) | ||||
| +#include <linux/arm-smccc.h> | ||||
| +int kvm_arch_ptp_get_clock_fn(long *cycle, struct timespec64 *ts, | ||||
| +			      struct clocksource **cs) | ||||
| +{ | ||||
| +	struct arm_smccc_res hvc_res; | ||||
| +	ktime_t ktime_overall; | ||||
| +	struct arm_smccc_quirk hvc_quirk; | ||||
| + | ||||
| +	 __arm_smccc_hvc(ARM_SMCCC_VENDOR_HYP_KVM_PTP_FUNC_ID, 0, 0, 0, 0, 0, 0, 0, &hvc_res, &hvc_quirk); | ||||
| + | ||||
| +       if ((long)(hvc_res.a0) < 0) | ||||
| +               return -EOPNOTSUPP; | ||||
| + | ||||
| +       ts->tv_sec = hvc_res.a0; | ||||
| +       ts->tv_nsec = hvc_res.a1; | ||||
| +       *cycle = hvc_res.a2 << 32 | hvc_res.a3; | ||||
| +       *cs = &clocksource_counter; | ||||
| + | ||||
| +      return 0; | ||||
| +} | ||||
| +EXPORT_SYMBOL_GPL(kvm_arch_ptp_get_clock_fn); | ||||
| +#endif | ||||
| + | ||||
| diff --git a/drivers/ptp/Kconfig b/drivers/ptp/Kconfig | ||||
| index d137c480db46..318b3f5df1ea 100644 | ||||
| --- a/drivers/ptp/Kconfig | ||||
| +++ b/drivers/ptp/Kconfig | ||||
| @@ -109,7 +109,7 @@ config PTP_1588_CLOCK_PCH | ||||
|  config PTP_1588_CLOCK_KVM | ||||
|  	tristate "KVM virtual PTP clock" | ||||
|  	depends on PTP_1588_CLOCK | ||||
| -	depends on KVM_GUEST && X86 | ||||
| +	depends on KVM_GUEST && X86 || ARM64 | ||||
|  	default y | ||||
|  	help | ||||
|  	  This driver adds support for using kvm infrastructure as a PTP | ||||
| diff --git a/drivers/ptp/Makefile b/drivers/ptp/Makefile | ||||
| index 19efa9cfa950..1bf4940a88a6 100644 | ||||
| --- a/drivers/ptp/Makefile | ||||
| +++ b/drivers/ptp/Makefile | ||||
| @@ -4,6 +4,7 @@ | ||||
|  # | ||||
|   | ||||
|  ptp-y					:= ptp_clock.o ptp_chardev.o ptp_sysfs.o | ||||
| +ptp_kvm-y				:= ptp_kvm_common.o ptp_kvm_$(ARCH).o | ||||
|  obj-$(CONFIG_PTP_1588_CLOCK)		+= ptp.o | ||||
|  obj-$(CONFIG_PTP_1588_CLOCK_DTE)	+= ptp_dte.o | ||||
|  obj-$(CONFIG_PTP_1588_CLOCK_IXP46X)	+= ptp_ixp46x.o | ||||
| diff --git a/drivers/ptp/ptp_kvm_arm64.c b/drivers/ptp/ptp_kvm_arm64.c | ||||
| new file mode 100644 | ||||
| index 000000000000..fcd83324c7e1 | ||||
| --- /dev/null | ||||
| +++ b/drivers/ptp/ptp_kvm_arm64.c | ||||
| @@ -0,0 +1,59 @@ | ||||
| +// SPDX-License-Identifier: GPL-2.0-only | ||||
| +/* | ||||
| + *  Virtual PTP 1588 clock for use with KVM guests | ||||
| + *  Copyright (C) 2019 ARM Ltd. | ||||
| + *  All Rights Reserved | ||||
| + */ | ||||
| + | ||||
| +#include <linux/kernel.h> | ||||
| +#include <linux/err.h> | ||||
| +#include <asm/hypervisor.h> | ||||
| +#include <linux/module.h> | ||||
| +#include <linux/psci.h> | ||||
| +#include <linux/arm-smccc.h> | ||||
| +#include <linux/timecounter.h> | ||||
| +#include <linux/sched/clock.h> | ||||
| +#include <asm/arch_timer.h> | ||||
| + | ||||
| + | ||||
| +void arm_smccc_1_1_invoke(u32 id, struct arm_smccc_res *res) | ||||
| +{ | ||||
| +	struct arm_smccc_quirk hvc_quirk; | ||||
| + | ||||
| +	 __arm_smccc_hvc(id, 0, 0, 0, 0, 0, 0, 0, res, &hvc_quirk); | ||||
| +} | ||||
| + | ||||
| +int kvm_arch_ptp_init(void) | ||||
| +{ | ||||
| +	struct arm_smccc_res hvc_res; | ||||
| + | ||||
| +	arm_smccc_1_1_invoke(ARM_SMCCC_VENDOR_HYP_KVM_PTP_FUNC_ID, | ||||
| +					&hvc_res); | ||||
| +	if ((long)(hvc_res.a0) < 0) | ||||
| +		return -EOPNOTSUPP; | ||||
| + | ||||
| +	return 0; | ||||
| +} | ||||
| + | ||||
| +int kvm_arch_ptp_get_clock_generic(struct timespec64 *ts, | ||||
| +				   struct arm_smccc_res *hvc_res) | ||||
| +{ | ||||
| +	arm_smccc_1_1_invoke(ARM_SMCCC_VENDOR_HYP_KVM_PTP_FUNC_ID, | ||||
| +					hvc_res); | ||||
| +	if ((long)(hvc_res->a0) < 0) | ||||
| +		return -EOPNOTSUPP; | ||||
| + | ||||
| +	ts->tv_sec = hvc_res->a0; | ||||
| +	ts->tv_nsec = hvc_res->a1; | ||||
| + | ||||
| +	return 0; | ||||
| +} | ||||
| + | ||||
| +int kvm_arch_ptp_get_clock(struct timespec64 *ts) | ||||
| +{ | ||||
| +	struct arm_smccc_res hvc_res; | ||||
| + | ||||
| +	kvm_arch_ptp_get_clock_generic(ts, &hvc_res); | ||||
| + | ||||
| +	return 0; | ||||
| +} | ||||
| diff --git a/drivers/ptp/ptp_kvm.c b/drivers/ptp/ptp_kvm_common.c | ||||
| similarity index 56% | ||||
| rename from drivers/ptp/ptp_kvm.c | ||||
| rename to drivers/ptp/ptp_kvm_common.c | ||||
| index c67dd11e08b1..c0b445fa6144 100644 | ||||
| --- a/drivers/ptp/ptp_kvm.c | ||||
| +++ b/drivers/ptp/ptp_kvm_common.c | ||||
| @@ -1,29 +1,19 @@ | ||||
| +// SPDX-License-Identifier: GPL-2.0-or-later | ||||
|  /* | ||||
|   * Virtual PTP 1588 clock for use with KVM guests | ||||
|   * | ||||
|   * Copyright (C) 2017 Red Hat Inc. | ||||
| - * | ||||
| - *  This program is free software; you can redistribute it and/or modify | ||||
| - *  it under the terms of the GNU General Public License as published by | ||||
| - *  the Free Software Foundation; either version 2 of the License, or | ||||
| - *  (at your option) any later version. | ||||
| - * | ||||
| - *  This program is distributed in the hope that it will be useful, | ||||
| - *  but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
| - *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
| - *  GNU General Public License for more details. | ||||
| - * | ||||
|   */ | ||||
|  #include <linux/device.h> | ||||
|  #include <linux/err.h> | ||||
|  #include <linux/init.h> | ||||
|  #include <linux/kernel.h> | ||||
| +#include <linux/slab.h> | ||||
|  #include <linux/module.h> | ||||
|  #include <uapi/linux/kvm_para.h> | ||||
|  #include <asm/kvm_para.h> | ||||
| -#include <asm/pvclock.h> | ||||
| -#include <asm/kvmclock.h> | ||||
|  #include <uapi/asm/kvm_para.h> | ||||
| +#include <asm-generic/ptp_kvm.h> | ||||
|   | ||||
|  #include <linux/ptp_clock_kernel.h> | ||||
|   | ||||
| @@ -34,56 +24,29 @@ struct kvm_ptp_clock { | ||||
|   | ||||
|  DEFINE_SPINLOCK(kvm_ptp_lock); | ||||
|   | ||||
| -static struct pvclock_vsyscall_time_info *hv_clock; | ||||
| - | ||||
| -static struct kvm_clock_pairing clock_pair; | ||||
| -static phys_addr_t clock_pair_gpa; | ||||
| - | ||||
|  static int ptp_kvm_get_time_fn(ktime_t *device_time, | ||||
|  			       struct system_counterval_t *system_counter, | ||||
|  			       void *ctx) | ||||
|  { | ||||
| -	unsigned long ret; | ||||
| +	unsigned long ret, cycle; | ||||
|  	struct timespec64 tspec; | ||||
| -	unsigned version; | ||||
| -	int cpu; | ||||
| -	struct pvclock_vcpu_time_info *src; | ||||
| +	struct clocksource *cs; | ||||
|   | ||||
|  	spin_lock(&kvm_ptp_lock); | ||||
|   | ||||
|  	preempt_disable_notrace(); | ||||
| -	cpu = smp_processor_id(); | ||||
| -	src = &hv_clock[cpu].pvti; | ||||
| - | ||||
| -	do { | ||||
| -		/* | ||||
| -		 * We are using a TSC value read in the hosts | ||||
| -		 * kvm_hc_clock_pairing handling. | ||||
| -		 * So any changes to tsc_to_system_mul | ||||
| -		 * and tsc_shift or any other pvclock | ||||
| -		 * data invalidate that measurement. | ||||
| -		 */ | ||||
| -		version = pvclock_read_begin(src); | ||||
| - | ||||
| -		ret = kvm_hypercall2(KVM_HC_CLOCK_PAIRING, | ||||
| -				     clock_pair_gpa, | ||||
| -				     KVM_CLOCK_PAIRING_WALLCLOCK); | ||||
| -		if (ret != 0) { | ||||
| -			pr_err_ratelimited("clock pairing hypercall ret %lu\n", ret); | ||||
| -			spin_unlock(&kvm_ptp_lock); | ||||
| -			preempt_enable_notrace(); | ||||
| -			return -EOPNOTSUPP; | ||||
| -		} | ||||
| - | ||||
| -		tspec.tv_sec = clock_pair.sec; | ||||
| -		tspec.tv_nsec = clock_pair.nsec; | ||||
| -		ret = __pvclock_read_cycles(src, clock_pair.tsc); | ||||
| -	} while (pvclock_read_retry(src, version)); | ||||
| +	ret = kvm_arch_ptp_get_clock_fn(&cycle, &tspec, &cs); | ||||
| +	if (ret != 0) { | ||||
| +		pr_err_ratelimited("clock pairing hypercall ret %lu\n", ret); | ||||
| +		spin_unlock(&kvm_ptp_lock); | ||||
| +		preempt_enable_notrace(); | ||||
| +		return -EOPNOTSUPP; | ||||
| +	} | ||||
|   | ||||
|  	preempt_enable_notrace(); | ||||
|   | ||||
| -	system_counter->cycles = ret; | ||||
| -	system_counter->cs = &kvm_clock; | ||||
| +	system_counter->cycles = cycle; | ||||
| +	system_counter->cs = cs; | ||||
|   | ||||
|  	*device_time = timespec64_to_ktime(tspec); | ||||
|   | ||||
| @@ -126,17 +89,13 @@ static int ptp_kvm_gettime(struct ptp_clock_info *ptp, struct timespec64 *ts) | ||||
|   | ||||
|  	spin_lock(&kvm_ptp_lock); | ||||
|   | ||||
| -	ret = kvm_hypercall2(KVM_HC_CLOCK_PAIRING, | ||||
| -			     clock_pair_gpa, | ||||
| -			     KVM_CLOCK_PAIRING_WALLCLOCK); | ||||
| +	ret = kvm_arch_ptp_get_clock(&tspec); | ||||
|  	if (ret != 0) { | ||||
|  		pr_err_ratelimited("clock offset hypercall ret %lu\n", ret); | ||||
|  		spin_unlock(&kvm_ptp_lock); | ||||
|  		return -EOPNOTSUPP; | ||||
|  	} | ||||
|   | ||||
| -	tspec.tv_sec = clock_pair.sec; | ||||
| -	tspec.tv_nsec = clock_pair.nsec; | ||||
|  	spin_unlock(&kvm_ptp_lock); | ||||
|   | ||||
|  	memcpy(ts, &tspec, sizeof(struct timespec64)); | ||||
| @@ -176,21 +135,11 @@ static void __exit ptp_kvm_exit(void) | ||||
|   | ||||
|  static int __init ptp_kvm_init(void) | ||||
|  { | ||||
| -	long ret; | ||||
| - | ||||
| -	if (!kvm_para_available()) | ||||
| -		return -ENODEV; | ||||
| +	int ret; | ||||
|   | ||||
| -	clock_pair_gpa = slow_virt_to_phys(&clock_pair); | ||||
| -	hv_clock = pvclock_get_pvti_cpu0_va(); | ||||
| - | ||||
| -	if (!hv_clock) | ||||
| -		return -ENODEV; | ||||
| - | ||||
| -	ret = kvm_hypercall2(KVM_HC_CLOCK_PAIRING, clock_pair_gpa, | ||||
| -			KVM_CLOCK_PAIRING_WALLCLOCK); | ||||
| -	if (ret == -KVM_ENOSYS || ret == -KVM_EOPNOTSUPP) | ||||
| -		return -ENODEV; | ||||
| +	ret = kvm_arch_ptp_init(); | ||||
| +	if (ret) | ||||
| +		return -EOPNOTSUPP; | ||||
|   | ||||
|  	kvm_ptp_clock.caps = ptp_kvm_caps; | ||||
|   | ||||
| diff --git a/drivers/ptp/ptp_kvm_x86.c b/drivers/ptp/ptp_kvm_x86.c | ||||
| new file mode 100644 | ||||
| index 000000000000..a52cf1c2990c | ||||
| --- /dev/null | ||||
| +++ b/drivers/ptp/ptp_kvm_x86.c | ||||
| @@ -0,0 +1,87 @@ | ||||
| +// SPDX-License-Identifier: GPL-2.0-or-later | ||||
| +/* | ||||
| + * Virtual PTP 1588 clock for use with KVM guests | ||||
| + * | ||||
| + * Copyright (C) 2017 Red Hat Inc. | ||||
| + */ | ||||
| + | ||||
| +#include <asm/pvclock.h> | ||||
| +#include <asm/kvmclock.h> | ||||
| +#include <linux/module.h> | ||||
| +#include <uapi/asm/kvm_para.h> | ||||
| +#include <uapi/linux/kvm_para.h> | ||||
| +#include <linux/ptp_clock_kernel.h> | ||||
| + | ||||
| +phys_addr_t clock_pair_gpa; | ||||
| +struct kvm_clock_pairing clock_pair; | ||||
| +struct pvclock_vsyscall_time_info *hv_clock; | ||||
| + | ||||
| +int kvm_arch_ptp_init(void) | ||||
| +{ | ||||
| +	int ret; | ||||
| + | ||||
| +	if (!kvm_para_available()) | ||||
| +		return -ENODEV; | ||||
| + | ||||
| +	clock_pair_gpa = slow_virt_to_phys(&clock_pair); | ||||
| +	hv_clock = pvclock_get_pvti_cpu0_va(); | ||||
| +	if (!hv_clock) | ||||
| +		return -ENODEV; | ||||
| + | ||||
| +	ret = kvm_hypercall2(KVM_HC_CLOCK_PAIRING, clock_pair_gpa, | ||||
| +			     KVM_CLOCK_PAIRING_WALLCLOCK); | ||||
| +	if (ret == -KVM_ENOSYS || ret == -KVM_EOPNOTSUPP) | ||||
| +		return -ENODEV; | ||||
| + | ||||
| +	return 0; | ||||
| +} | ||||
| + | ||||
| +int kvm_arch_ptp_get_clock(struct timespec64 *ts) | ||||
| +{ | ||||
| +	long ret; | ||||
| + | ||||
| +	ret = kvm_hypercall2(KVM_HC_CLOCK_PAIRING, | ||||
| +			     clock_pair_gpa, | ||||
| +			     KVM_CLOCK_PAIRING_WALLCLOCK); | ||||
| +	if (ret != 0) | ||||
| +		return -EOPNOTSUPP; | ||||
| + | ||||
| +	ts->tv_sec = clock_pair.sec; | ||||
| +	ts->tv_nsec = clock_pair.nsec; | ||||
| + | ||||
| +	return 0; | ||||
| +} | ||||
| + | ||||
| +int kvm_arch_ptp_get_clock_fn(unsigned long *cycle, struct timespec64 *tspec, | ||||
| +			      struct clocksource **cs) | ||||
| +{ | ||||
| +	unsigned long ret; | ||||
| +	unsigned int version; | ||||
| +	int cpu; | ||||
| +	struct pvclock_vcpu_time_info *src; | ||||
| + | ||||
| +	cpu = smp_processor_id(); | ||||
| +	src = &hv_clock[cpu].pvti; | ||||
| + | ||||
| +	do { | ||||
| +		/* | ||||
| +		 * We are using a TSC value read in the hosts | ||||
| +		 * kvm_hc_clock_pairing handling. | ||||
| +		 * So any changes to tsc_to_system_mul | ||||
| +		 * and tsc_shift or any other pvclock | ||||
| +		 * data invalidate that measurement. | ||||
| +		 */ | ||||
| +		version = pvclock_read_begin(src); | ||||
| + | ||||
| +		ret = kvm_hypercall2(KVM_HC_CLOCK_PAIRING, | ||||
| +				     clock_pair_gpa, | ||||
| +				     KVM_CLOCK_PAIRING_WALLCLOCK); | ||||
| +		tspec->tv_sec = clock_pair.sec; | ||||
| +		tspec->tv_nsec = clock_pair.nsec; | ||||
| +		*cycle = __pvclock_read_cycles(src, clock_pair.tsc); | ||||
| +	} while (pvclock_read_retry(src, version)); | ||||
| + | ||||
| +	*cs = &kvm_clock; | ||||
| + | ||||
| +	return 0; | ||||
| +} | ||||
| diff --git a/include/asm-generic/ptp_kvm.h b/include/asm-generic/ptp_kvm.h | ||||
| new file mode 100644 | ||||
| index 000000000000..883eea494a80 | ||||
| --- /dev/null | ||||
| +++ b/include/asm-generic/ptp_kvm.h | ||||
| @@ -0,0 +1,12 @@ | ||||
| +// SPDX-License-Identifier: GPL-2.0-only | ||||
| +/* | ||||
| + *  linux/drivers/clocksource/arm_arch_timer.c | ||||
| + * | ||||
| + *  Copyright (C) 2019 ARM Ltd. | ||||
| + *  All Rights Reserved | ||||
| + */ | ||||
| + | ||||
| +int kvm_arch_ptp_init(void); | ||||
| +int kvm_arch_ptp_get_clock(struct timespec64 *ts); | ||||
| +int kvm_arch_ptp_get_clock_fn(unsigned long *cycle, | ||||
| +                struct timespec64 *tspec, void *cs); | ||||
| diff --git a/include/linux/arm-smccc.h b/include/linux/arm-smccc.h | ||||
| index 18863d56273c..10e99c82d098 100644 | ||||
| --- a/include/linux/arm-smccc.h | ||||
| +++ b/include/linux/arm-smccc.h | ||||
| @@ -75,6 +75,11 @@ | ||||
|  			   ARM_SMCCC_SMC_32,				\ | ||||
|  			   0, 1) | ||||
|   | ||||
| +#define ARM_SMCCC_VENDOR_HYP_KVM_PTP_FUNC_ID				\ | ||||
| +	ARM_SMCCC_CALL_VAL(ARM_SMCCC_FAST_CALL,				\ | ||||
| +			   ARM_SMCCC_SMC_32,				\ | ||||
| +			   0, 2) | ||||
| + | ||||
|  #define ARM_SMCCC_ARCH_WORKAROUND_1					\ | ||||
|  	ARM_SMCCC_CALL_VAL(ARM_SMCCC_FAST_CALL,				\ | ||||
|  			   ARM_SMCCC_SMC_32,				\ | ||||
| diff --git a/virt/kvm/arm/psci.c b/virt/kvm/arm/psci.c | ||||
| index 9b73d3ad918a..9b9999bdeab7 100644 | ||||
| --- a/virt/kvm/arm/psci.c | ||||
| +++ b/virt/kvm/arm/psci.c | ||||
| @@ -407,6 +407,9 @@ int kvm_hvc_call_handler(struct kvm_vcpu *vcpu) | ||||
|  	u32 func_id = smccc_get_function(vcpu); | ||||
|  	u32 val = SMCCC_RET_NOT_SUPPORTED; | ||||
|  	u32 feature; | ||||
| +	struct timespec64 ts; | ||||
| +	u64 cycles, cycle_high, cycle_low; | ||||
| +	struct system_time_snapshot systime_snapshot; | ||||
|   | ||||
|  	switch (func_id) { | ||||
|  	case ARM_SMCCC_VERSION_FUNC_ID: | ||||
| @@ -435,6 +438,15 @@ int kvm_hvc_call_handler(struct kvm_vcpu *vcpu) | ||||
|  			break; | ||||
|  		} | ||||
|  		break; | ||||
| +	case ARM_SMCCC_VENDOR_HYP_KVM_PTP_FUNC_ID: | ||||
| +		ktime_get_real_ts64(&ts); | ||||
| +		ktime_get_snapshot(&systime_snapshot); | ||||
| +		cycles = systime_snapshot.cycles - vcpu_vtimer(vcpu)->cntvoff; | ||||
| +		cycle_high = cycles >> 32; | ||||
| +		cycle_low = cycles << 32 >> 32; | ||||
| + | ||||
| +		smccc_set_retval(vcpu, ts.tv_sec, ts.tv_nsec, cycle_high, cycle_low); | ||||
| +		return 1; | ||||
|  	default: | ||||
|  		return kvm_psci_call(vcpu); | ||||
|  	} | ||||
| --  | ||||
| 2.17.1 | ||||
|  | ||||
| @@ -0,0 +1,977 @@ | ||||
| From f96a27de35d241a4f5909493ca99847512ec748a Mon Sep 17 00:00:00 2001 | ||||
| From: Damon Kwok <damon-kwok@outlook.com> | ||||
| Date: Thu, 24 Jun 2021 14:25:01 +0800 | ||||
| Subject: [PATCH] 4.19.x: PTP_KVM support for arm/arm64 | ||||
|  | ||||
| --- | ||||
|  arch/arm/kernel/setup.c                     |   2 + | ||||
|  arch/arm64/include/asm/kvm_host.h           |   2 + | ||||
|  arch/arm64/kernel/setup.c                   |   3 + | ||||
|  drivers/clocksource/arm_arch_timer.c        |  33 ++++++ | ||||
|  drivers/firmware/psci.c                     |  53 +++++++++ | ||||
|  drivers/ptp/Kconfig                         |   2 +- | ||||
|  drivers/ptp/Makefile                        |   2 + | ||||
|  drivers/ptp/ptp_kvm_arm.c                   |  28 +++++ | ||||
|  drivers/ptp/{ptp_kvm.c => ptp_kvm_common.c} |  99 ++++------------ | ||||
|  drivers/ptp/ptp_kvm_x86.c                   |  87 ++++++++++++++ | ||||
|  include/linux/arm-smccc.h                   | 125 ++++++++++++++++++++ | ||||
|  include/linux/clocksource.h                 |   6 + | ||||
|  include/linux/clocksource_ids.h             |  13 ++ | ||||
|  include/linux/ptp_kvm.h                     |  19 +++ | ||||
|  include/linux/timekeeping.h                 |   2 + | ||||
|  include/uapi/linux/kvm.h                    |   1 + | ||||
|  kernel/time/clocksource.c                   |   3 + | ||||
|  kernel/time/timekeeping.c                   |   1 + | ||||
|  virt/kvm/arm/arm.c                          |   1 + | ||||
|  virt/kvm/arm/psci.c                         |  80 ++++++++++++- | ||||
|  20 files changed, 481 insertions(+), 81 deletions(-) | ||||
|  create mode 100644 drivers/ptp/ptp_kvm_arm.c | ||||
|  rename drivers/ptp/{ptp_kvm.c => ptp_kvm_common.c} (53%) | ||||
|  create mode 100644 drivers/ptp/ptp_kvm_x86.c | ||||
|  create mode 100644 include/linux/clocksource_ids.h | ||||
|  create mode 100644 include/linux/ptp_kvm.h | ||||
|  | ||||
| diff --git a/arch/arm/kernel/setup.c b/arch/arm/kernel/setup.c | ||||
| index 7bbaa293a..137320130 100644 | ||||
| --- a/arch/arm/kernel/setup.c | ||||
| +++ b/arch/arm/kernel/setup.c | ||||
| @@ -7,6 +7,7 @@ | ||||
|   * it under the terms of the GNU General Public License version 2 as | ||||
|   * published by the Free Software Foundation. | ||||
|   */ | ||||
| + #include <linux/arm-smccc.h> | ||||
|  #include <linux/efi.h> | ||||
|  #include <linux/export.h> | ||||
|  #include <linux/kernel.h> | ||||
| @@ -1137,6 +1138,7 @@ void __init setup_arch(char **cmdline_p) | ||||
|   | ||||
|  	arm_dt_init_cpu_maps(); | ||||
|  	psci_dt_init(); | ||||
| +	kvm_init_hyp_services(); | ||||
|  #ifdef CONFIG_SMP | ||||
|  	if (is_smp()) { | ||||
|  		if (!mdesc->smp_init || !mdesc->smp_init()) { | ||||
| diff --git a/arch/arm64/include/asm/kvm_host.h b/arch/arm64/include/asm/kvm_host.h | ||||
| index 5e720742d..ffc729a43 100644 | ||||
| --- a/arch/arm64/include/asm/kvm_host.h | ||||
| +++ b/arch/arm64/include/asm/kvm_host.h | ||||
| @@ -145,6 +145,8 @@ enum vcpu_sysreg { | ||||
|  	PMSWINC_EL0,	/* Software Increment Register */ | ||||
|  	PMUSERENR_EL0,	/* User Enable Register */ | ||||
|   | ||||
| +	CNTVOFF_EL2, | ||||
| +	 | ||||
|  	/* 32bit specific registers. Keep them at the end of the range */ | ||||
|  	DACR32_EL2,	/* Domain Access Control Register */ | ||||
|  	IFSR32_EL2,	/* Instruction Fault Status Register */ | ||||
| diff --git a/arch/arm64/kernel/setup.c b/arch/arm64/kernel/setup.c | ||||
| index b3354ff94..071f26ef4 100644 | ||||
| --- a/arch/arm64/kernel/setup.c | ||||
| +++ b/arch/arm64/kernel/setup.c | ||||
| @@ -18,6 +18,7 @@ | ||||
|   */ | ||||
|   | ||||
|  #include <linux/acpi.h> | ||||
| +#include <linux/arm-smccc.h> | ||||
|  #include <linux/export.h> | ||||
|  #include <linux/kernel.h> | ||||
|  #include <linux/stddef.h> | ||||
| @@ -334,6 +335,8 @@ void __init setup_arch(char **cmdline_p) | ||||
|  		psci_dt_init(); | ||||
|  	else | ||||
|  		psci_acpi_init(); | ||||
| +		 | ||||
| +	kvm_init_hyp_services(); | ||||
|   | ||||
|  	cpu_read_bootcpu_ops(); | ||||
|  	smp_init_cpus(); | ||||
| diff --git a/drivers/clocksource/arm_arch_timer.c b/drivers/clocksource/arm_arch_timer.c | ||||
| index 0445ad7e5..b4e8a0a0b 100644 | ||||
| --- a/drivers/clocksource/arm_arch_timer.c | ||||
| +++ b/drivers/clocksource/arm_arch_timer.c | ||||
| @@ -19,6 +19,7 @@ | ||||
|  #include <linux/cpu_pm.h> | ||||
|  #include <linux/clockchips.h> | ||||
|  #include <linux/clocksource.h> | ||||
| +#include <linux/clocksource_ids.h> | ||||
|  #include <linux/interrupt.h> | ||||
|  #include <linux/of_irq.h> | ||||
|  #include <linux/of_address.h> | ||||
| @@ -27,6 +28,8 @@ | ||||
|  #include <linux/sched/clock.h> | ||||
|  #include <linux/sched_clock.h> | ||||
|  #include <linux/acpi.h> | ||||
| +#include <linux/arm-smccc.h> | ||||
| +#include <linux/ptp_kvm.h> | ||||
|   | ||||
|  #include <asm/arch_timer.h> | ||||
|  #include <asm/virt.h> | ||||
| @@ -173,6 +176,7 @@ static u64 arch_counter_read_cc(const struct cyclecounter *cc) | ||||
|   | ||||
|  static struct clocksource clocksource_counter = { | ||||
|  	.name	= "arch_sys_counter", | ||||
| +	.id	= CSID_ARM_ARCH_COUNTER, | ||||
|  	.rating	= 400, | ||||
|  	.read	= arch_counter_read, | ||||
|  	.mask	= CLOCKSOURCE_MASK(56), | ||||
| @@ -1626,3 +1630,32 @@ static int __init arch_timer_acpi_init(struct acpi_table_header *table) | ||||
|  } | ||||
|  TIMER_ACPI_DECLARE(arch_timer, ACPI_SIG_GTDT, arch_timer_acpi_init); | ||||
|  #endif | ||||
| + | ||||
| +int kvm_arch_ptp_get_crosststamp(u64 *cycle, struct timespec64 *ts, | ||||
| +			      struct clocksource **cs) | ||||
| +{ | ||||
| +	struct arm_smccc_res hvc_res; | ||||
| +	ktime_t ktime; | ||||
| +	u32 ptp_counter; | ||||
| + | ||||
| +	if (arch_timer_uses_ppi == ARCH_TIMER_VIRT_PPI) | ||||
| +		ptp_counter = ARM_PTP_VIRT_COUNTER; | ||||
| +	else | ||||
| +		ptp_counter = ARM_PTP_PHY_COUNTER; | ||||
| + | ||||
| +	arm_smccc_1_1_invoke(ARM_SMCCC_VENDOR_HYP_KVM_PTP_FUNC_ID, | ||||
| +			     ptp_counter, &hvc_res); | ||||
| + | ||||
| +	if ((int)(hvc_res.a0) < 0) | ||||
| +		return -EOPNOTSUPP; | ||||
| + | ||||
| +	ktime = (u64)hvc_res.a0 << 32 | hvc_res.a1; | ||||
| +	*ts = ktime_to_timespec64(ktime); | ||||
| +	if (cycle) | ||||
| +		*cycle = (u64)hvc_res.a2 << 32 | hvc_res.a3; | ||||
| +	if (cs) | ||||
| +		*cs = &clocksource_counter; | ||||
| + | ||||
| +	return 0; | ||||
| +} | ||||
| +EXPORT_SYMBOL_GPL(kvm_arch_ptp_get_crosststamp); | ||||
| diff --git a/drivers/firmware/psci.c b/drivers/firmware/psci.c | ||||
| index c80ec1d03..dc958d0b2 100644 | ||||
| --- a/drivers/firmware/psci.c | ||||
| +++ b/drivers/firmware/psci.c | ||||
| @@ -15,6 +15,8 @@ | ||||
|   | ||||
|  #include <linux/acpi.h> | ||||
|  #include <linux/arm-smccc.h> | ||||
| +#include <linux/kernel.h> | ||||
| +#include <linux/string.h> | ||||
|  #include <linux/cpuidle.h> | ||||
|  #include <linux/errno.h> | ||||
|  #include <linux/linkage.h> | ||||
| @@ -64,6 +66,57 @@ struct psci_operations psci_ops = { | ||||
|  	.smccc_version = SMCCC_VERSION_1_0, | ||||
|  }; | ||||
|   | ||||
| +DECLARE_BITMAP(__kvm_arm_hyp_services, ARM_SMCCC_KVM_NUM_FUNCS) = { }; | ||||
| +EXPORT_SYMBOL_GPL(__kvm_arm_hyp_services); | ||||
| + | ||||
| +void __init kvm_init_hyp_services(void) | ||||
| +{ | ||||
| +	int i; | ||||
| +	struct arm_smccc_res res; | ||||
| + | ||||
| +	if (psci_ops.smccc_version == ARM_SMCCC_VERSION_1_0) | ||||
| +		return; | ||||
| + | ||||
| +	arm_smccc_1_1_invoke(ARM_SMCCC_VENDOR_HYP_CALL_UID_FUNC_ID, &res); | ||||
| +	if (res.a0 != ARM_SMCCC_VENDOR_HYP_UID_KVM_REG_0 || | ||||
| +	    res.a1 != ARM_SMCCC_VENDOR_HYP_UID_KVM_REG_1 || | ||||
| +	    res.a2 != ARM_SMCCC_VENDOR_HYP_UID_KVM_REG_2 || | ||||
| +	    res.a3 != ARM_SMCCC_VENDOR_HYP_UID_KVM_REG_3) | ||||
| +		return; | ||||
| + | ||||
| +	memset(&res, 0, sizeof(res)); | ||||
| +	arm_smccc_1_1_invoke(ARM_SMCCC_VENDOR_HYP_KVM_FEATURES_FUNC_ID, &res); | ||||
| +	for (i = 0; i < 32; ++i) { | ||||
| +		if (res.a0 & (i)) | ||||
| +			set_bit(i + (32 * 0), __kvm_arm_hyp_services); | ||||
| +		if (res.a1 & (i)) | ||||
| +			set_bit(i + (32 * 1), __kvm_arm_hyp_services); | ||||
| +		if (res.a2 & (i)) | ||||
| +			set_bit(i + (32 * 2), __kvm_arm_hyp_services); | ||||
| +		if (res.a3 & (i)) | ||||
| +			set_bit(i + (32 * 3), __kvm_arm_hyp_services); | ||||
| +	} | ||||
| + | ||||
| +	pr_info("KVM hypervisor services detected (0x%08lx 0x%08lx 0x%08lx 0x%08lx)\n", | ||||
| +		 res.a3, res.a2, res.a1, res.a0); | ||||
| +} | ||||
| + | ||||
| +enum arm_smccc_conduit arm_smccc_1_1_get_conduit(void) | ||||
| +{ | ||||
| +	if (psci_ops.smccc_version < SMCCC_VERSION_1_1) | ||||
| +		return SMCCC_CONDUIT_NONE; | ||||
| + | ||||
| +	switch (psci_ops.conduit) { | ||||
| +	case PSCI_CONDUIT_SMC: | ||||
| +		return SMCCC_CONDUIT_SMC; | ||||
| +	case PSCI_CONDUIT_HVC: | ||||
| +		return SMCCC_CONDUIT_HVC; | ||||
| +	default: | ||||
| +		return SMCCC_CONDUIT_NONE; | ||||
| +	} | ||||
| +} | ||||
| +EXPORT_SYMBOL(arm_smccc_1_1_get_conduit); | ||||
| + | ||||
|  typedef unsigned long (psci_fn)(unsigned long, unsigned long, | ||||
|  				unsigned long, unsigned long); | ||||
|  static psci_fn *invoke_psci_fn; | ||||
| diff --git a/drivers/ptp/Kconfig b/drivers/ptp/Kconfig | ||||
| index d137c480d..7860ea3cb 100644 | ||||
| --- a/drivers/ptp/Kconfig | ||||
| +++ b/drivers/ptp/Kconfig | ||||
| @@ -109,7 +109,7 @@ config PTP_1588_CLOCK_PCH | ||||
|  config PTP_1588_CLOCK_KVM | ||||
|  	tristate "KVM virtual PTP clock" | ||||
|  	depends on PTP_1588_CLOCK | ||||
| -	depends on KVM_GUEST && X86 | ||||
| +	depends on X86 || ((ARM || ARM64) && ARM_ARCH_TIMER && ARM_PSCI_FW) | ||||
|  	default y | ||||
|  	help | ||||
|  	  This driver adds support for using kvm infrastructure as a PTP | ||||
| diff --git a/drivers/ptp/Makefile b/drivers/ptp/Makefile | ||||
| index 19efa9cfa..50a6fbd30 100644 | ||||
| --- a/drivers/ptp/Makefile | ||||
| +++ b/drivers/ptp/Makefile | ||||
| @@ -4,6 +4,8 @@ | ||||
|  # | ||||
|   | ||||
|  ptp-y					:= ptp_clock.o ptp_chardev.o ptp_sysfs.o | ||||
| +ptp_kvm-$(CONFIG_X86)			:= ptp_kvm_x86.o ptp_kvm_common.o | ||||
| +ptp_kvm-$(CONFIG_HAVE_ARM_SMCCC)	:= ptp_kvm_arm.o ptp_kvm_common.o | ||||
|  obj-$(CONFIG_PTP_1588_CLOCK)		+= ptp.o | ||||
|  obj-$(CONFIG_PTP_1588_CLOCK_DTE)	+= ptp_dte.o | ||||
|  obj-$(CONFIG_PTP_1588_CLOCK_IXP46X)	+= ptp_ixp46x.o | ||||
| diff --git a/drivers/ptp/ptp_kvm_arm.c b/drivers/ptp/ptp_kvm_arm.c | ||||
| new file mode 100644 | ||||
| index 000000000..b7d28c8df | ||||
| --- /dev/null | ||||
| +++ b/drivers/ptp/ptp_kvm_arm.c | ||||
| @@ -0,0 +1,28 @@ | ||||
| +// SPDX-License-Identifier: GPL-2.0-only | ||||
| +/* | ||||
| + *  Virtual PTP 1588 clock for use with KVM guests | ||||
| + *  Copyright (C) 2019 ARM Ltd. | ||||
| + *  All Rights Reserved | ||||
| + */ | ||||
| + | ||||
| +#include <linux/arm-smccc.h> | ||||
| +#include <linux/ptp_kvm.h> | ||||
| + | ||||
| +#include <asm/arch_timer.h> | ||||
| +#include <asm/hypervisor.h> | ||||
| + | ||||
| +int kvm_arch_ptp_init(void) | ||||
| +{ | ||||
| +	int ret; | ||||
| + | ||||
| +	ret = kvm_arm_hyp_service_available(ARM_SMCCC_KVM_FUNC_PTP); | ||||
| +	if (ret <= 0) | ||||
| +		return -EOPNOTSUPP; | ||||
| + | ||||
| +	return 0; | ||||
| +} | ||||
| + | ||||
| +int kvm_arch_ptp_get_clock(struct timespec64 *ts) | ||||
| +{ | ||||
| +	return kvm_arch_ptp_get_crosststamp(NULL, ts, NULL); | ||||
| +} | ||||
| diff --git a/drivers/ptp/ptp_kvm.c b/drivers/ptp/ptp_kvm_common.c | ||||
| similarity index 53% | ||||
| rename from drivers/ptp/ptp_kvm.c | ||||
| rename to drivers/ptp/ptp_kvm_common.c | ||||
| index c67dd11e0..5c36e2fa0 100644 | ||||
| --- a/drivers/ptp/ptp_kvm.c | ||||
| +++ b/drivers/ptp/ptp_kvm_common.c | ||||
| @@ -1,28 +1,18 @@ | ||||
| +// SPDX-License-Identifier: GPL-2.0-or-later | ||||
|  /* | ||||
|   * Virtual PTP 1588 clock for use with KVM guests | ||||
|   * | ||||
|   * Copyright (C) 2017 Red Hat Inc. | ||||
| - * | ||||
| - *  This program is free software; you can redistribute it and/or modify | ||||
| - *  it under the terms of the GNU General Public License as published by | ||||
| - *  the Free Software Foundation; either version 2 of the License, or | ||||
| - *  (at your option) any later version. | ||||
| - * | ||||
| - *  This program is distributed in the hope that it will be useful, | ||||
| - *  but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
| - *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
| - *  GNU General Public License for more details. | ||||
| - * | ||||
|   */ | ||||
|  #include <linux/device.h> | ||||
|  #include <linux/err.h> | ||||
|  #include <linux/init.h> | ||||
|  #include <linux/kernel.h> | ||||
| +#include <linux/slab.h> | ||||
|  #include <linux/module.h> | ||||
| +#include <linux/ptp_kvm.h> | ||||
|  #include <uapi/linux/kvm_para.h> | ||||
|  #include <asm/kvm_para.h> | ||||
| -#include <asm/pvclock.h> | ||||
| -#include <asm/kvmclock.h> | ||||
|  #include <uapi/asm/kvm_para.h> | ||||
|   | ||||
|  #include <linux/ptp_clock_kernel.h> | ||||
| @@ -32,58 +22,31 @@ struct kvm_ptp_clock { | ||||
|  	struct ptp_clock_info caps; | ||||
|  }; | ||||
|   | ||||
| -DEFINE_SPINLOCK(kvm_ptp_lock); | ||||
| - | ||||
| -static struct pvclock_vsyscall_time_info *hv_clock; | ||||
| - | ||||
| -static struct kvm_clock_pairing clock_pair; | ||||
| -static phys_addr_t clock_pair_gpa; | ||||
| +static DEFINE_SPINLOCK(kvm_ptp_lock); | ||||
|   | ||||
|  static int ptp_kvm_get_time_fn(ktime_t *device_time, | ||||
|  			       struct system_counterval_t *system_counter, | ||||
|  			       void *ctx) | ||||
|  { | ||||
| -	unsigned long ret; | ||||
| +	long ret; | ||||
| +	u64 cycle; | ||||
|  	struct timespec64 tspec; | ||||
| -	unsigned version; | ||||
| -	int cpu; | ||||
| -	struct pvclock_vcpu_time_info *src; | ||||
| +	struct clocksource *cs; | ||||
|   | ||||
|  	spin_lock(&kvm_ptp_lock); | ||||
|   | ||||
|  	preempt_disable_notrace(); | ||||
| -	cpu = smp_processor_id(); | ||||
| -	src = &hv_clock[cpu].pvti; | ||||
| - | ||||
| -	do { | ||||
| -		/* | ||||
| -		 * We are using a TSC value read in the hosts | ||||
| -		 * kvm_hc_clock_pairing handling. | ||||
| -		 * So any changes to tsc_to_system_mul | ||||
| -		 * and tsc_shift or any other pvclock | ||||
| -		 * data invalidate that measurement. | ||||
| -		 */ | ||||
| -		version = pvclock_read_begin(src); | ||||
| - | ||||
| -		ret = kvm_hypercall2(KVM_HC_CLOCK_PAIRING, | ||||
| -				     clock_pair_gpa, | ||||
| -				     KVM_CLOCK_PAIRING_WALLCLOCK); | ||||
| -		if (ret != 0) { | ||||
| -			pr_err_ratelimited("clock pairing hypercall ret %lu\n", ret); | ||||
| -			spin_unlock(&kvm_ptp_lock); | ||||
| -			preempt_enable_notrace(); | ||||
| -			return -EOPNOTSUPP; | ||||
| -		} | ||||
| - | ||||
| -		tspec.tv_sec = clock_pair.sec; | ||||
| -		tspec.tv_nsec = clock_pair.nsec; | ||||
| -		ret = __pvclock_read_cycles(src, clock_pair.tsc); | ||||
| -	} while (pvclock_read_retry(src, version)); | ||||
| +	ret = kvm_arch_ptp_get_crosststamp(&cycle, &tspec, &cs); | ||||
| +	if (ret!=0) { | ||||
| +		spin_unlock(&kvm_ptp_lock); | ||||
| +		preempt_enable_notrace(); | ||||
| +		return ret; | ||||
| +	} | ||||
|   | ||||
|  	preempt_enable_notrace(); | ||||
|   | ||||
| -	system_counter->cycles = ret; | ||||
| -	system_counter->cs = &kvm_clock; | ||||
| +	system_counter->cycles = cycle; | ||||
| +	system_counter->cs = cs; | ||||
|   | ||||
|  	*device_time = timespec64_to_ktime(tspec); | ||||
|   | ||||
| @@ -121,22 +84,17 @@ static int ptp_kvm_settime(struct ptp_clock_info *ptp, | ||||
|   | ||||
|  static int ptp_kvm_gettime(struct ptp_clock_info *ptp, struct timespec64 *ts) | ||||
|  { | ||||
| -	unsigned long ret; | ||||
| +	long ret; | ||||
|  	struct timespec64 tspec; | ||||
|   | ||||
|  	spin_lock(&kvm_ptp_lock); | ||||
|   | ||||
| -	ret = kvm_hypercall2(KVM_HC_CLOCK_PAIRING, | ||||
| -			     clock_pair_gpa, | ||||
| -			     KVM_CLOCK_PAIRING_WALLCLOCK); | ||||
| -	if (ret != 0) { | ||||
| -		pr_err_ratelimited("clock offset hypercall ret %lu\n", ret); | ||||
| +	ret = kvm_arch_ptp_get_clock(&tspec); | ||||
| +	if (ret) { | ||||
|  		spin_unlock(&kvm_ptp_lock); | ||||
| -		return -EOPNOTSUPP; | ||||
| +		return ret; | ||||
|  	} | ||||
|   | ||||
| -	tspec.tv_sec = clock_pair.sec; | ||||
| -	tspec.tv_nsec = clock_pair.nsec; | ||||
|  	spin_unlock(&kvm_ptp_lock); | ||||
|   | ||||
|  	memcpy(ts, &tspec, sizeof(struct timespec64)); | ||||
| @@ -178,19 +136,12 @@ static int __init ptp_kvm_init(void) | ||||
|  { | ||||
|  	long ret; | ||||
|   | ||||
| -	if (!kvm_para_available()) | ||||
| -		return -ENODEV; | ||||
| - | ||||
| -	clock_pair_gpa = slow_virt_to_phys(&clock_pair); | ||||
| -	hv_clock = pvclock_get_pvti_cpu0_va(); | ||||
| - | ||||
| -	if (!hv_clock) | ||||
| -		return -ENODEV; | ||||
| - | ||||
| -	ret = kvm_hypercall2(KVM_HC_CLOCK_PAIRING, clock_pair_gpa, | ||||
| -			KVM_CLOCK_PAIRING_WALLCLOCK); | ||||
| -	if (ret == -KVM_ENOSYS || ret == -KVM_EOPNOTSUPP) | ||||
| -		return -ENODEV; | ||||
| +	ret = kvm_arch_ptp_init(); | ||||
| +	if (ret) { | ||||
| +		if (ret != -EOPNOTSUPP) | ||||
| +			pr_err("fail to initialize ptp_kvm"); | ||||
| +		return ret; | ||||
| +	} | ||||
|   | ||||
|  	kvm_ptp_clock.caps = ptp_kvm_caps; | ||||
|   | ||||
| diff --git a/drivers/ptp/ptp_kvm_x86.c b/drivers/ptp/ptp_kvm_x86.c | ||||
| new file mode 100644 | ||||
| index 000000000..4add42ae3 | ||||
| --- /dev/null | ||||
| +++ b/drivers/ptp/ptp_kvm_x86.c | ||||
| @@ -0,0 +1,87 @@ | ||||
| +// SPDX-License-Identifier: GPL-2.0-or-later | ||||
| +/* | ||||
| + * Virtual PTP 1588 clock for use with KVM guests | ||||
| + * | ||||
| + * Copyright (C) 2017 Red Hat Inc. | ||||
| + */ | ||||
| + | ||||
| +#include <asm/pvclock.h> | ||||
| +#include <asm/kvmclock.h> | ||||
| +#include <linux/module.h> | ||||
| +#include <uapi/asm/kvm_para.h> | ||||
| +#include <uapi/linux/kvm_para.h> | ||||
| +#include <linux/ptp_clock_kernel.h> | ||||
| + | ||||
| +phys_addr_t clock_pair_gpa; | ||||
| +struct kvm_clock_pairing clock_pair; | ||||
| +struct pvclock_vsyscall_time_info *hv_clock; | ||||
| + | ||||
| +int kvm_arch_ptp_init(void) | ||||
| +{ | ||||
| +	int ret; | ||||
| + | ||||
| +	if (!kvm_para_available()) | ||||
| +		return -ENODEV; | ||||
| + | ||||
| +	clock_pair_gpa = slow_virt_to_phys(&clock_pair); | ||||
| +	hv_clock = pvclock_get_pvti_cpu0_va(); | ||||
| +	if (!hv_clock) | ||||
| +		return -ENODEV; | ||||
| + | ||||
| +	ret = kvm_hypercall2(KVM_HC_CLOCK_PAIRING, clock_pair_gpa, | ||||
| +			     KVM_CLOCK_PAIRING_WALLCLOCK); | ||||
| +	if (ret == -KVM_ENOSYS || ret == -KVM_EOPNOTSUPP) | ||||
| +		return -ENODEV; | ||||
| + | ||||
| +	return 0; | ||||
| +} | ||||
| + | ||||
| +int kvm_arch_ptp_get_clock(struct timespec64 *ts) | ||||
| +{ | ||||
| +	long ret; | ||||
| + | ||||
| +	ret = kvm_hypercall2(KVM_HC_CLOCK_PAIRING, | ||||
| +			     clock_pair_gpa, | ||||
| +			     KVM_CLOCK_PAIRING_WALLCLOCK); | ||||
| +	if (ret != 0) | ||||
| +		return -EOPNOTSUPP; | ||||
| + | ||||
| +	ts->tv_sec = clock_pair.sec; | ||||
| +	ts->tv_nsec = clock_pair.nsec; | ||||
| + | ||||
| +	return 0; | ||||
| +} | ||||
| + | ||||
| +int kvm_arch_ptp_get_crosststamp(u64 *cycle, struct timespec64 *tspec, | ||||
| +			      struct clocksource **cs) | ||||
| +{ | ||||
| +	unsigned long ret; | ||||
| +	unsigned int version; | ||||
| +	int cpu; | ||||
| +	struct pvclock_vcpu_time_info *src; | ||||
| + | ||||
| +	cpu = smp_processor_id(); | ||||
| +	src = &hv_clock[cpu].pvti; | ||||
| + | ||||
| +	do { | ||||
| +		/* | ||||
| +		 * We are using a TSC value read in the hosts | ||||
| +		 * kvm_hc_clock_pairing handling. | ||||
| +		 * So any changes to tsc_to_system_mul | ||||
| +		 * and tsc_shift or any other pvclock | ||||
| +		 * data invalidate that measurement. | ||||
| +		 */ | ||||
| +		version = pvclock_read_begin(src); | ||||
| + | ||||
| +		ret = kvm_hypercall2(KVM_HC_CLOCK_PAIRING, | ||||
| +				     clock_pair_gpa, | ||||
| +				     KVM_CLOCK_PAIRING_WALLCLOCK); | ||||
| +		tspec->tv_sec = clock_pair.sec; | ||||
| +		tspec->tv_nsec = clock_pair.nsec; | ||||
| +		*cycle = __pvclock_read_cycles(src, clock_pair.tsc); | ||||
| +	} while (pvclock_read_retry(src, version)); | ||||
| + | ||||
| +	*cs = &kvm_clock; | ||||
| + | ||||
| +	return 0; | ||||
| +} | ||||
| diff --git a/include/linux/arm-smccc.h b/include/linux/arm-smccc.h | ||||
| index 18863d562..af78262d4 100644 | ||||
| --- a/include/linux/arm-smccc.h | ||||
| +++ b/include/linux/arm-smccc.h | ||||
| @@ -1,3 +1,4 @@ | ||||
| +/* SPDX-License-Identifier: GPL-2.0-only */ | ||||
|  /* | ||||
|   * Copyright (c) 2015, Linaro Limited | ||||
|   * | ||||
| @@ -14,6 +15,7 @@ | ||||
|  #ifndef __LINUX_ARM_SMCCC_H | ||||
|  #define __LINUX_ARM_SMCCC_H | ||||
|   | ||||
| +#include <linux/init.h> | ||||
|  #include <uapi/linux/const.h> | ||||
|   | ||||
|  /* | ||||
| @@ -54,11 +56,15 @@ | ||||
|  #define ARM_SMCCC_OWNER_SIP		2 | ||||
|  #define ARM_SMCCC_OWNER_OEM		3 | ||||
|  #define ARM_SMCCC_OWNER_STANDARD	4 | ||||
| +#define ARM_SMCCC_OWNER_STANDARD_HYP	5 | ||||
| +#define ARM_SMCCC_OWNER_VENDOR_HYP	6 | ||||
|  #define ARM_SMCCC_OWNER_TRUSTED_APP	48 | ||||
|  #define ARM_SMCCC_OWNER_TRUSTED_APP_END	49 | ||||
|  #define ARM_SMCCC_OWNER_TRUSTED_OS	50 | ||||
|  #define ARM_SMCCC_OWNER_TRUSTED_OS_END	63 | ||||
|   | ||||
| +#define ARM_SMCCC_FUNC_QUERY_CALL_UID  0xff01 | ||||
| + | ||||
|  #define ARM_SMCCC_QUIRK_NONE		0 | ||||
|  #define ARM_SMCCC_QUIRK_QCOM_A6		1 /* Save/restore register a6 */ | ||||
|   | ||||
| @@ -84,11 +90,67 @@ | ||||
|  	ARM_SMCCC_CALL_VAL(ARM_SMCCC_FAST_CALL,				\ | ||||
|  			   ARM_SMCCC_SMC_32,				\ | ||||
|  			   0, 0x7fff) | ||||
| +#define ARM_SMCCC_VENDOR_HYP_CALL_UID_FUNC_ID				\ | ||||
| +	ARM_SMCCC_CALL_VAL(ARM_SMCCC_FAST_CALL,				\ | ||||
| +			   ARM_SMCCC_SMC_32,				\ | ||||
| +			   ARM_SMCCC_OWNER_VENDOR_HYP,			\ | ||||
| +			   ARM_SMCCC_FUNC_QUERY_CALL_UID) | ||||
| + | ||||
| +/* KVM UID value: 28b46fb6-2ec5-11e9-a9ca-4b564d003a74 */ | ||||
| +#define ARM_SMCCC_VENDOR_HYP_UID_KVM_REG_0	0xb66fb428U | ||||
| +#define ARM_SMCCC_VENDOR_HYP_UID_KVM_REG_1	0xe911c52eU | ||||
| +#define ARM_SMCCC_VENDOR_HYP_UID_KVM_REG_2	0x564bcaa9U | ||||
| +#define ARM_SMCCC_VENDOR_HYP_UID_KVM_REG_3	0x743a004dU | ||||
| + | ||||
| +/* KVM "vendor specific" services */ | ||||
| +#define ARM_SMCCC_KVM_FUNC_FEATURES		0 | ||||
| +#define ARM_SMCCC_KVM_FUNC_PTP			1 | ||||
| +#define ARM_SMCCC_KVM_FUNC_FEATURES_2		127 | ||||
| +#define ARM_SMCCC_KVM_NUM_FUNCS			128 | ||||
| + | ||||
| +#define ARM_SMCCC_VENDOR_HYP_KVM_FEATURES_FUNC_ID			\ | ||||
| +	ARM_SMCCC_CALL_VAL(ARM_SMCCC_FAST_CALL,				\ | ||||
| +			   ARM_SMCCC_SMC_32,				\ | ||||
| +			   ARM_SMCCC_OWNER_VENDOR_HYP,			\ | ||||
| +			   ARM_SMCCC_KVM_FUNC_FEATURES) | ||||
| + | ||||
| +#define SMCCC_ARCH_WORKAROUND_RET_UNAFFECTED	1 | ||||
| + | ||||
| +/* | ||||
| + * ptp_kvm is a feature used for time sync between vm and host. | ||||
| + * ptp_kvm module in guest kernel will get service from host using | ||||
| + * this hypercall ID. | ||||
| + */ | ||||
| +#define ARM_SMCCC_VENDOR_HYP_KVM_PTP_FUNC_ID				\ | ||||
| +	ARM_SMCCC_CALL_VAL(ARM_SMCCC_FAST_CALL,				\ | ||||
| +			   ARM_SMCCC_SMC_32,				\ | ||||
| +			   ARM_SMCCC_OWNER_VENDOR_HYP,			\ | ||||
| +			   ARM_SMCCC_KVM_FUNC_PTP) | ||||
| + | ||||
| +/* ptp_kvm counter type ID */ | ||||
| +#define ARM_PTP_VIRT_COUNTER			0 | ||||
| +#define ARM_PTP_PHY_COUNTER			1 | ||||
|   | ||||
|  #ifndef __ASSEMBLY__ | ||||
|   | ||||
|  #include <linux/linkage.h> | ||||
|  #include <linux/types.h> | ||||
| + | ||||
| +enum arm_smccc_conduit { | ||||
| +	SMCCC_CONDUIT_NONE, | ||||
| +	SMCCC_CONDUIT_SMC, | ||||
| +	SMCCC_CONDUIT_HVC, | ||||
| +}; | ||||
| + | ||||
| +/** | ||||
| + * arm_smccc_1_1_get_conduit() | ||||
| + * | ||||
| + * Returns the conduit to be used for SMCCCv1.1 or later. | ||||
| + * | ||||
| + * When SMCCCv1.1 is not present, returns SMCCC_CONDUIT_NONE. | ||||
| + */ | ||||
| +enum arm_smccc_conduit arm_smccc_1_1_get_conduit(void); | ||||
| + | ||||
|  /** | ||||
|   * struct arm_smccc_res - Result from SMC/HVC call | ||||
|   * @a0-a3 result values from registers 0 to 3 | ||||
| @@ -311,5 +373,68 @@ asmlinkage void __arm_smccc_hvc(unsigned long a0, unsigned long a1, | ||||
|  #define SMCCC_RET_NOT_SUPPORTED			-1 | ||||
|  #define SMCCC_RET_NOT_REQUIRED			-2 | ||||
|   | ||||
| +/* | ||||
| + * Like arm_smccc_1_1* but always returns SMCCC_RET_NOT_SUPPORTED. | ||||
| + * Used when the SMCCC conduit is not defined. The empty asm statement | ||||
| + * avoids compiler warnings about unused variables. | ||||
| + */ | ||||
| +#define __fail_smccc_1_1(...)                                          \ | ||||
| +	do {                                                            \ | ||||
| +		__declare_args(__count_args(__VA_ARGS__), __VA_ARGS__); \ | ||||
| +		asm ("" __constraints(__count_args(__VA_ARGS__)));      \ | ||||
| +		if (___res)                                             \ | ||||
| +			___res->a0 = SMCCC_RET_NOT_SUPPORTED;           \ | ||||
| +	} while (0) | ||||
| + | ||||
| +/* | ||||
| + * arm_smccc_1_1_invoke() - make an SMCCC v1.1 compliant call | ||||
| + * | ||||
| + * This is a variadic macro taking one to eight source arguments, and | ||||
| + * an optional return structure. | ||||
| + * | ||||
| + * @a0-a7: arguments passed in registers 0 to 7 | ||||
| + * @res: result values from registers 0 to 3 | ||||
| + * | ||||
| + * This macro will make either an HVC call or an SMC call depending on the | ||||
| + * current SMCCC conduit. If no valid conduit is available then -1 | ||||
| + * (SMCCC_RET_NOT_SUPPORTED) is returned in @res.a0 (if supplied). | ||||
| + * | ||||
| + * The return value also provides the conduit that was used. | ||||
| + */ | ||||
| +#define arm_smccc_1_1_invoke(...) ({					\ | ||||
| +		int method = arm_smccc_1_1_get_conduit();		\ | ||||
| +		switch (method) {					\ | ||||
| +		case SMCCC_CONDUIT_HVC:					\ | ||||
| +			arm_smccc_1_1_hvc(__VA_ARGS__);			\ | ||||
| +			break;						\ | ||||
| +		case SMCCC_CONDUIT_SMC:					\ | ||||
| +			arm_smccc_1_1_smc(__VA_ARGS__);			\ | ||||
| +			break;						\ | ||||
| +		default:						\ | ||||
| +			__fail_smccc_1_1(__VA_ARGS__);			\ | ||||
| +			method = SMCCC_CONDUIT_NONE;			\ | ||||
| +			break;						\ | ||||
| +		}							\ | ||||
| +		method;							\ | ||||
| +	}) | ||||
| + | ||||
| +void __init kvm_init_hyp_services(void); | ||||
| +	 | ||||
| +/* | ||||
| + * This helper will be called in guest. We put it here then both arm and arm64 | ||||
| + * guest can touch it. | ||||
| + */ | ||||
| +#include <linux/kernel.h> | ||||
| +#include <linux/err.h> | ||||
| +static inline bool kvm_arm_hyp_service_available(u32 func_id) | ||||
| +{ | ||||
| +	extern DECLARE_BITMAP(__kvm_arm_hyp_services, ARM_SMCCC_KVM_NUM_FUNCS); | ||||
| + | ||||
| +	if (func_id >= ARM_SMCCC_KVM_NUM_FUNCS) | ||||
| +		return false; | ||||
| + | ||||
| +	return test_bit(func_id, __kvm_arm_hyp_services); | ||||
| +} | ||||
| + | ||||
|  #endif /*__ASSEMBLY__*/ | ||||
|  #endif /*__LINUX_ARM_SMCCC_H*/ | ||||
| diff --git a/include/linux/clocksource.h b/include/linux/clocksource.h | ||||
| index 308918928..043967d3f 100644 | ||||
| --- a/include/linux/clocksource.h | ||||
| +++ b/include/linux/clocksource.h | ||||
| @@ -17,6 +17,7 @@ | ||||
|  #include <linux/timer.h> | ||||
|  #include <linux/init.h> | ||||
|  #include <linux/of.h> | ||||
| +#include <linux/clocksource_ids.h> | ||||
|  #include <asm/div64.h> | ||||
|  #include <asm/io.h> | ||||
|   | ||||
| @@ -49,6 +50,10 @@ struct module; | ||||
|   *			400-499: Perfect | ||||
|   *				The ideal clocksource. A must-use where | ||||
|   *				available. | ||||
| + * @id:			Defaults to CSID_GENERIC. The id value is captured | ||||
| + *			in certain snapshot functions to allow callers to | ||||
| + *			validate the clocksource from which the snapshot was | ||||
| + *			taken. | ||||
|   * @read:		returns a cycle value, passes clocksource as argument | ||||
|   * @enable:		optional function to enable the clocksource | ||||
|   * @disable:		optional function to disable the clocksource | ||||
| @@ -91,6 +96,7 @@ struct clocksource { | ||||
|  	const char *name; | ||||
|  	struct list_head list; | ||||
|  	int rating; | ||||
| +	enum clocksource_ids id; | ||||
|  	int (*enable)(struct clocksource *cs); | ||||
|  	void (*disable)(struct clocksource *cs); | ||||
|  	unsigned long flags; | ||||
| diff --git a/include/linux/clocksource_ids.h b/include/linux/clocksource_ids.h | ||||
| new file mode 100644 | ||||
| index 000000000..93bec8426 | ||||
| --- /dev/null | ||||
| +++ b/include/linux/clocksource_ids.h | ||||
| @@ -0,0 +1,13 @@ | ||||
| +/* SPDX-License-Identifier: GPL-2.0 */ | ||||
| +#ifndef _LINUX_CLOCKSOURCE_IDS_H | ||||
| +#define _LINUX_CLOCKSOURCE_IDS_H | ||||
| + | ||||
| +/* Enum to give clocksources a unique identifier */ | ||||
| +enum clocksource_ids { | ||||
| +	CSID_GENERIC		= 0, | ||||
| +	CSID_ARM_ARCH_COUNTER, | ||||
| +	CSID_MAX, | ||||
| +}; | ||||
| + | ||||
| +#endif | ||||
| + | ||||
| diff --git a/include/linux/ptp_kvm.h b/include/linux/ptp_kvm.h | ||||
| new file mode 100644 | ||||
| index 000000000..f960a719f | ||||
| --- /dev/null | ||||
| +++ b/include/linux/ptp_kvm.h | ||||
| @@ -0,0 +1,19 @@ | ||||
| +/* SPDX-License-Identifier: GPL-2.0-or-later */ | ||||
| +/* | ||||
| + * Virtual PTP 1588 clock for use with KVM guests | ||||
| + * | ||||
| + * Copyright (C) 2017 Red Hat Inc. | ||||
| + */ | ||||
| + | ||||
| +#ifndef _PTP_KVM_H_ | ||||
| +#define _PTP_KVM_H_ | ||||
| + | ||||
| +struct timespec64; | ||||
| +struct clocksource; | ||||
| + | ||||
| +int kvm_arch_ptp_init(void); | ||||
| +int kvm_arch_ptp_get_clock(struct timespec64 *ts); | ||||
| +int kvm_arch_ptp_get_crosststamp(u64 *cycle, | ||||
| +		struct timespec64 *tspec, struct clocksource **cs); | ||||
| + | ||||
| +#endif /* _PTP_KVM_H_ */ | ||||
| diff --git a/include/linux/timekeeping.h b/include/linux/timekeeping.h | ||||
| index a5a3cfc3c..24808479f 100644 | ||||
| --- a/include/linux/timekeeping.h | ||||
| +++ b/include/linux/timekeeping.h | ||||
| @@ -2,6 +2,7 @@ | ||||
|  #ifndef _LINUX_TIMEKEEPING_H | ||||
|  #define _LINUX_TIMEKEEPING_H | ||||
|   | ||||
| +#include <linux/clocksource_ids.h> | ||||
|  #include <linux/errno.h> | ||||
|   | ||||
|  /* Included from linux/ktime.h */ | ||||
| @@ -207,6 +208,7 @@ struct system_time_snapshot { | ||||
|  	u64		cycles; | ||||
|  	ktime_t		real; | ||||
|  	ktime_t		raw; | ||||
| +	enum clocksource_ids	cs_id; | ||||
|  	unsigned int	clock_was_set_seq; | ||||
|  	u8		cs_was_changed_seq; | ||||
|  }; | ||||
| diff --git a/include/uapi/linux/kvm.h b/include/uapi/linux/kvm.h | ||||
| index c297abc4e..e5962ad27 100644 | ||||
| --- a/include/uapi/linux/kvm.h | ||||
| +++ b/include/uapi/linux/kvm.h | ||||
| @@ -956,6 +956,7 @@ struct kvm_ppc_resize_hpt { | ||||
|  #define KVM_CAP_NESTED_STATE 157 | ||||
|  #define KVM_CAP_ARM_INJECT_SERROR_ESR 158 | ||||
|  #define KVM_CAP_MSR_PLATFORM_INFO 159 | ||||
| +#define KVM_CAP_ARM_KVM_PTP 160 | ||||
|   | ||||
|  #ifdef KVM_CAP_IRQ_ROUTING | ||||
|   | ||||
| diff --git a/kernel/time/clocksource.c b/kernel/time/clocksource.c | ||||
| index f80bb104c..6b3a072a1 100644 | ||||
| --- a/kernel/time/clocksource.c | ||||
| +++ b/kernel/time/clocksource.c | ||||
| @@ -944,6 +944,9 @@ int __clocksource_register_scale(struct clocksource *cs, u32 scale, u32 freq) | ||||
|  { | ||||
|  	unsigned long flags; | ||||
|   | ||||
| +	if (WARN_ON_ONCE((unsigned int)cs->id >= CSID_MAX)) | ||||
| +		cs->id = CSID_GENERIC; | ||||
| + | ||||
|  	/* Initialize mult/shift and max_idle_ns */ | ||||
|  	__clocksource_update_freq_scale(cs, scale, freq); | ||||
|   | ||||
| diff --git a/kernel/time/timekeeping.c b/kernel/time/timekeeping.c | ||||
| index c66fd11d9..348ea7de1 100644 | ||||
| --- a/kernel/time/timekeeping.c | ||||
| +++ b/kernel/time/timekeeping.c | ||||
| @@ -978,6 +978,7 @@ void ktime_get_snapshot(struct system_time_snapshot *systime_snapshot) | ||||
|  	do { | ||||
|  		seq = read_seqcount_begin(&tk_core.seq); | ||||
|  		now = tk_clock_read(&tk->tkr_mono); | ||||
| +		systime_snapshot->cs_id = tk->tkr_mono.clock->id; | ||||
|  		systime_snapshot->cs_was_changed_seq = tk->cs_was_changed_seq; | ||||
|  		systime_snapshot->clock_was_set_seq = tk->clock_was_set_seq; | ||||
|  		base_real = ktime_add(tk->tkr_mono.base, | ||||
| diff --git a/virt/kvm/arm/arm.c b/virt/kvm/arm/arm.c | ||||
| index d982650de..716b9c7e8 100644 | ||||
| --- a/virt/kvm/arm/arm.c | ||||
| +++ b/virt/kvm/arm/arm.c | ||||
| @@ -212,6 +212,7 @@ int kvm_vm_ioctl_check_extension(struct kvm *kvm, long ext) | ||||
|  	case KVM_CAP_READONLY_MEM: | ||||
|  	case KVM_CAP_MP_STATE: | ||||
|  	case KVM_CAP_IMMEDIATE_EXIT: | ||||
| +	case KVM_CAP_ARM_KVM_PTP: | ||||
|  		r = 1; | ||||
|  		break; | ||||
|  	case KVM_CAP_ARM_SET_DEVICE_ADDR: | ||||
| diff --git a/virt/kvm/arm/psci.c b/virt/kvm/arm/psci.c | ||||
| index 34d08ee63..0f35819ba 100644 | ||||
| --- a/virt/kvm/arm/psci.c | ||||
| +++ b/virt/kvm/arm/psci.c | ||||
| @@ -398,22 +398,65 @@ static int kvm_psci_call(struct kvm_vcpu *vcpu) | ||||
|  	}; | ||||
|  } | ||||
|   | ||||
| +static void kvm_ptp_get_time(struct kvm_vcpu *vcpu, u64 *val) | ||||
| +{ | ||||
| +	struct system_time_snapshot systime_snapshot; | ||||
| +	u64 cycles = ~0UL; | ||||
| +	u32 feature; | ||||
| + | ||||
| +	/* | ||||
| +	 * system time and counter value must captured in the same | ||||
| +	 * time to keep consistency and precision. | ||||
| +	 */ | ||||
| +	ktime_get_snapshot(&systime_snapshot); | ||||
| + | ||||
| +	// binding ptp_kvm clocksource to arm_arch_counter | ||||
| +	if (systime_snapshot.cs_id != CSID_ARM_ARCH_COUNTER) | ||||
| +		return; | ||||
| + | ||||
| +	val[0] = upper_32_bits(systime_snapshot.real); | ||||
| +	val[1] = lower_32_bits(systime_snapshot.real); | ||||
| + | ||||
| +	/* | ||||
| +	 * which of virtual counter or physical counter being | ||||
| +	 * asked for is decided by the r1 value of SMCCC | ||||
| +	 * call. If no invalid r1 value offered, default cycle | ||||
| +	 * value(-1) will be returned. | ||||
| +	 * Note: keep in mind that feature is u32 and smccc_get_arg1 | ||||
| +	 * will return u64, so need auto cast here. | ||||
| +	 */ | ||||
| +	feature = smccc_get_arg1(vcpu); | ||||
| +	switch (feature) { | ||||
| +	case ARM_PTP_VIRT_COUNTER: | ||||
| +		cycles = systime_snapshot.cycles - vcpu_read_sys_reg(vcpu, CNTVOFF_EL2); | ||||
| +		break; | ||||
| +	case ARM_PTP_PHY_COUNTER: | ||||
| +		cycles = systime_snapshot.cycles; | ||||
| +		break; | ||||
| +	default: | ||||
| +		val[0] = SMCCC_RET_NOT_SUPPORTED; | ||||
| +		break; | ||||
| +	} | ||||
| +	val[2] = upper_32_bits(cycles); | ||||
| +	val[3] = lower_32_bits(cycles); | ||||
| +} | ||||
| + | ||||
|  int kvm_hvc_call_handler(struct kvm_vcpu *vcpu) | ||||
|  { | ||||
|  	u32 func_id = smccc_get_function(vcpu); | ||||
| -	u32 val = SMCCC_RET_NOT_SUPPORTED; | ||||
| +	u64 val[4] = {SMCCC_RET_NOT_SUPPORTED}; | ||||
|  	u32 feature; | ||||
|   | ||||
|  	switch (func_id) { | ||||
|  	case ARM_SMCCC_VERSION_FUNC_ID: | ||||
| -		val = ARM_SMCCC_VERSION_1_1; | ||||
| +		val[0] = ARM_SMCCC_VERSION_1_1; | ||||
|  		break; | ||||
|  	case ARM_SMCCC_ARCH_FEATURES_FUNC_ID: | ||||
|  		feature = smccc_get_arg1(vcpu); | ||||
|  		switch(feature) { | ||||
|  		case ARM_SMCCC_ARCH_WORKAROUND_1: | ||||
|  			if (kvm_arm_harden_branch_predictor()) | ||||
| -				val = SMCCC_RET_SUCCESS; | ||||
| +				val[0] = SMCCC_RET_SUCCESS; | ||||
|  			break; | ||||
|  		case ARM_SMCCC_ARCH_WORKAROUND_2: | ||||
|  			switch (kvm_arm_have_ssbd()) { | ||||
| @@ -421,21 +464,46 @@ int kvm_hvc_call_handler(struct kvm_vcpu *vcpu) | ||||
|  			case KVM_SSBD_UNKNOWN: | ||||
|  				break; | ||||
|  			case KVM_SSBD_KERNEL: | ||||
| -				val = SMCCC_RET_SUCCESS; | ||||
| +				val[0] = SMCCC_RET_SUCCESS; | ||||
|  				break; | ||||
|  			case KVM_SSBD_FORCE_ENABLE: | ||||
|  			case KVM_SSBD_MITIGATED: | ||||
| -				val = SMCCC_RET_NOT_REQUIRED; | ||||
| +				val[0] = SMCCC_RET_NOT_REQUIRED; | ||||
|  				break; | ||||
|  			} | ||||
|  			break; | ||||
|  		} | ||||
|  		break; | ||||
| +	case ARM_SMCCC_VENDOR_HYP_CALL_UID_FUNC_ID: | ||||
| +		val[0] = ARM_SMCCC_VENDOR_HYP_UID_KVM_REG_0; | ||||
| +		val[1] = ARM_SMCCC_VENDOR_HYP_UID_KVM_REG_1; | ||||
| +		val[2] = ARM_SMCCC_VENDOR_HYP_UID_KVM_REG_2; | ||||
| +		val[3] = ARM_SMCCC_VENDOR_HYP_UID_KVM_REG_3; | ||||
| +		break; | ||||
| +	case ARM_SMCCC_VENDOR_HYP_KVM_FEATURES_FUNC_ID: | ||||
| +		val[0] = BIT(ARM_SMCCC_KVM_FUNC_FEATURES); | ||||
| +		val[0] |= BIT(ARM_SMCCC_KVM_FUNC_PTP); | ||||
| +		break; | ||||
| +	/* | ||||
| +	 * This serves virtual kvm_ptp. | ||||
| +	 * Four values will be passed back. | ||||
| +	 * reg0 stores high 32-bits of host ktime; | ||||
| +	 * reg1 stores low 32-bits of host ktime; | ||||
| +	 * For ARM_PTP_VIRT_COUNTER: | ||||
| +	 * reg2 stores high 32-bits of difference of host cycles and cntvoff; | ||||
| +	 * reg3 stores low 32-bits of difference of host cycles and cntvoff. | ||||
| +	 * For ARM_PTP_PHY_COUNTER: | ||||
| +	 * reg2 stores the high 32-bits of host cycles; | ||||
| +	 * reg3 stores the low 32-bits of host cycles. | ||||
| +	 */ | ||||
| +	case ARM_SMCCC_VENDOR_HYP_KVM_PTP_FUNC_ID: | ||||
| +		kvm_ptp_get_time(vcpu, val); | ||||
| +		break; | ||||
|  	default: | ||||
|  		return kvm_psci_call(vcpu); | ||||
|  	} | ||||
|   | ||||
| -	smccc_set_retval(vcpu, val, 0, 0, 0); | ||||
| +	smccc_set_retval(vcpu, val[0], val[1], val[2], val[3]); | ||||
|  	return 1; | ||||
|  } | ||||
|   | ||||
| --  | ||||
| 2.32.0 | ||||
|  | ||||
| @@ -0,0 +1,974 @@ | ||||
| From 772e627bb2aa29a6d006f08bdf7bc1492671d1fe Mon Sep 17 00:00:00 2001 | ||||
| From: Damon Kwok <damon-kwok@outlook.com> | ||||
| Date: Thu, 24 Jun 2021 15:04:19 +0800 | ||||
| Subject: [PATCH] 5.10.x: PTP_KVM support for arm/arm64 | ||||
|  | ||||
| --- | ||||
|  Documentation/virt/kvm/api.rst              |  9 ++ | ||||
|  Documentation/virt/kvm/arm/index.rst        |  1 + | ||||
|  Documentation/virt/kvm/arm/ptp_kvm.rst      | 31 +++++++ | ||||
|  Documentation/virt/kvm/timekeeping.rst      | 35 ++++++++ | ||||
|  arch/arm/kernel/setup.c                     |  5 ++ | ||||
|  arch/arm64/kernel/setup.c                   |  1 + | ||||
|  arch/arm64/kvm/hypercalls.c                 | 86 ++++++++++++++++-- | ||||
|  arch/arm64/kvm/reset.c                      |  1 + | ||||
|  drivers/clocksource/arm_arch_timer.c        | 33 +++++++ | ||||
|  drivers/firmware/smccc/smccc.c              | 37 ++++++++ | ||||
|  drivers/ptp/Kconfig                         |  2 +- | ||||
|  drivers/ptp/Makefile                        |  2 + | ||||
|  drivers/ptp/ptp_kvm_arm.c                   | 28 ++++++ | ||||
|  drivers/ptp/{ptp_kvm.c => ptp_kvm_common.c} | 85 +++++------------- | ||||
|  drivers/ptp/ptp_kvm_x86.c                   | 96 +++++++++++++++++++++ | ||||
|  include/linux/arm-smccc.h                   | 59 +++++++++++++ | ||||
|  include/linux/clocksource.h                 |  6 ++ | ||||
|  include/linux/clocksource_ids.h             | 12 +++ | ||||
|  include/linux/ptp_kvm.h                     | 19 ++++ | ||||
|  include/linux/timekeeping.h                 | 12 +-- | ||||
|  include/uapi/linux/kvm.h                    |  1 + | ||||
|  kernel/time/clocksource.c                   |  2 + | ||||
|  kernel/time/timekeeping.c                   |  1 + | ||||
|  23 files changed, 487 insertions(+), 77 deletions(-) | ||||
|  create mode 100644 Documentation/virt/kvm/arm/ptp_kvm.rst | ||||
|  create mode 100644 drivers/ptp/ptp_kvm_arm.c | ||||
|  rename drivers/ptp/{ptp_kvm.c => ptp_kvm_common.c} (60%) | ||||
|  create mode 100644 drivers/ptp/ptp_kvm_x86.c | ||||
|  create mode 100644 include/linux/clocksource_ids.h | ||||
|  create mode 100644 include/linux/ptp_kvm.h | ||||
|  | ||||
| diff --git a/Documentation/virt/kvm/api.rst b/Documentation/virt/kvm/api.rst | ||||
| index cd8a585..4b97c22 100644 | ||||
| --- a/Documentation/virt/kvm/api.rst | ||||
| +++ b/Documentation/virt/kvm/api.rst | ||||
| @@ -6398,3 +6398,12 @@ When enabled, KVM will disable paravirtual features provided to the | ||||
|  guest according to the bits in the KVM_CPUID_FEATURES CPUID leaf | ||||
|  (0x40000001). Otherwise, a guest may use the paravirtual features | ||||
|  regardless of what has actually been exposed through the CPUID leaf. | ||||
| + | ||||
| +8.27 KVM_CAP_PTP_KVM | ||||
| +-------------------- | ||||
| + | ||||
| +:Architectures: arm64 | ||||
| + | ||||
| +This capability indicates that KVM virtual PTP service is supported in host. | ||||
| +It must company with the implementation of KVM virtual PTP service in host | ||||
| +so VMM can probe if there is the service in host by checking this capability. | ||||
| diff --git a/Documentation/virt/kvm/arm/index.rst b/Documentation/virt/kvm/arm/index.rst | ||||
| index 3e2b2ab..78a9b67 100644 | ||||
| --- a/Documentation/virt/kvm/arm/index.rst | ||||
| +++ b/Documentation/virt/kvm/arm/index.rst | ||||
| @@ -10,3 +10,4 @@ ARM | ||||
|     hyp-abi | ||||
|     psci | ||||
|     pvtime | ||||
| +   ptp_kvm | ||||
| diff --git a/Documentation/virt/kvm/arm/ptp_kvm.rst b/Documentation/virt/kvm/arm/ptp_kvm.rst | ||||
| new file mode 100644 | ||||
| index 0000000..7be7e65 | ||||
| --- /dev/null | ||||
| +++ b/Documentation/virt/kvm/arm/ptp_kvm.rst | ||||
| @@ -0,0 +1,31 @@ | ||||
| +.. SPDX-License-Identifier: GPL-2.0 | ||||
| + | ||||
| +PTP_KVM support for arm/arm64 | ||||
| +============================= | ||||
| + | ||||
| +PTP_KVM is used for time sync between guest and host in a high precision. | ||||
| +It needs to get the wall time and counter value from the host and transfer these | ||||
| +to guest via hypercall service. So one more hypercall service has been added. | ||||
| + | ||||
| +This new SMCCC hypercall is defined as: | ||||
| + | ||||
| +* ARM_SMCCC_VENDOR_HYP_KVM_PTP_FUNC_ID: 0x86000001 | ||||
| + | ||||
| +As both 32 and 64-bits ptp_kvm client should be supported, we choose SMC32/HVC32 | ||||
| +calling convention. | ||||
| + | ||||
| +ARM_SMCCC_VENDOR_HYP_KVM_PTP_FUNC_ID: | ||||
| + | ||||
| +    =============    ==========    ========== | ||||
| +    Function ID:     (uint32)      0x86000001 | ||||
| +    Arguments:       (uint32)      ARM_PTP_PHY_COUNTER(1) or ARM_PTP_VIRT_COUNTER(0) | ||||
| +                                   which indicate acquiring physical counter or | ||||
| +                                   virtual counter respectively. | ||||
| +    Return Value:    val0(uint32)  NOT_SUPPORTED(-1) or upper 32 bits of wall clock time(64-bits). | ||||
| +                     val1(uint32)  Lower 32 bits of wall clock time. | ||||
| +                     val2(uint32)  Upper 32 bits of counter cycle(64-bits). | ||||
| +                     val3(uint32)  Lower 32 bits of counter cycle. | ||||
| +    Endianness:                    No Restrictions. | ||||
| +    =============    ==========    ========== | ||||
| + | ||||
| +More info see section 5 in Documentation/virt/kvm/timekeeping.rst. | ||||
| diff --git a/Documentation/virt/kvm/timekeeping.rst b/Documentation/virt/kvm/timekeeping.rst | ||||
| index 21ae7ef..c81383e 100644 | ||||
| --- a/Documentation/virt/kvm/timekeeping.rst | ||||
| +++ b/Documentation/virt/kvm/timekeeping.rst | ||||
| @@ -13,6 +13,7 @@ Timekeeping Virtualization for X86-Based Architectures | ||||
|     2) Timing Devices | ||||
|     3) TSC Hardware | ||||
|     4) Virtualization Problems | ||||
| +   5) KVM virtual PTP clock | ||||
|   | ||||
|  1. Overview | ||||
|  =========== | ||||
| @@ -643,3 +644,37 @@ by using CPU utilization itself as a signalling channel.  Preventing such | ||||
|  problems would require completely isolated virtual time which may not track | ||||
|  real time any longer.  This may be useful in certain security or QA contexts, | ||||
|  but in general isn't recommended for real-world deployment scenarios. | ||||
| + | ||||
| +5. KVM virtual PTP clock | ||||
| +======================== | ||||
| + | ||||
| +NTP (Network Time Protocol) is often used to sync time in a VM. Unfortunately, | ||||
| +the precision of NTP is limited due to unknown delays in the network. | ||||
| + | ||||
| +KVM virtual PTP clock (PTP_KVM) offers another way to sync time in VM; use the | ||||
| +host's clock rather than one from a remote machine. Having a synchronization | ||||
| +mechanism for the virtualization environment allows us to keep all the guests | ||||
| +running on the same host in sync. | ||||
| +In general, the delay of communication between host and guest is quite | ||||
| +small, so ptp_kvm can offer time sync precision up to in order of nanoseconds. | ||||
| +Please keep in mind that ptp_kvm just limits itself to be a channel which | ||||
| +transmits the remote clock from host to guest. An application, eg. chrony, is | ||||
| +needed in usersapce of VM in order to set the guest time. | ||||
| + | ||||
| +After ptp_kvm is initialized, there will be a new device node under /dev called | ||||
| +ptp%d. A guest userspace service, like chrony, can use this device to get host | ||||
| +walltime, sometimes also counter cycle, which depends on the service it calls. | ||||
| +Then this guest userspace service can use those data to do the time sync for | ||||
| +the guest. | ||||
| +The following is the work flow of ptp_kvm: | ||||
| + | ||||
| +a) time sync service in guest userspace call ioctl on ptp device /dev/ptp%d. | ||||
| +b) ptp_kvm module in guest receives this request then invokes hypercall to | ||||
| +   route into host kernel to request host's walltime/counter cycle. | ||||
| +c) ptp_kvm hypercall service on the host responds to the request and sends data | ||||
| +   back. | ||||
| +d) ptp in guest copies the data to userspace. | ||||
| + | ||||
| +ptp_kvm consists of components running on the guest and host. Step 2 consists of | ||||
| +a guest driver making a hypercall whilst step 3 involves the hypervisor responding | ||||
| +with information. | ||||
| diff --git a/arch/arm/kernel/setup.c b/arch/arm/kernel/setup.c | ||||
| index f90479d..472daea 100644 | ||||
| --- a/arch/arm/kernel/setup.c | ||||
| +++ b/arch/arm/kernel/setup.c | ||||
| @@ -1154,6 +1154,11 @@ void __init setup_arch(char **cmdline_p) | ||||
|   | ||||
|  	arm_dt_init_cpu_maps(); | ||||
|  	psci_dt_init(); | ||||
| + | ||||
| +#ifdef CONFIG_HAVE_ARM_SMCCC_DISCOVERY | ||||
| +	kvm_init_hyp_services(); | ||||
| +#endif | ||||
| + | ||||
|  #ifdef CONFIG_SMP | ||||
|  	if (is_smp()) { | ||||
|  		if (!mdesc->smp_init || !mdesc->smp_init()) { | ||||
| diff --git a/arch/arm64/kernel/setup.c b/arch/arm64/kernel/setup.c | ||||
| index 133257f..d1dbe41 100644 | ||||
| --- a/arch/arm64/kernel/setup.c | ||||
| +++ b/arch/arm64/kernel/setup.c | ||||
| @@ -353,6 +353,7 @@ void __init __no_sanitize_address setup_arch(char **cmdline_p) | ||||
|  	else | ||||
|  		psci_acpi_init(); | ||||
|   | ||||
| +	kvm_init_hyp_services(); | ||||
|  	init_bootcpu_ops(); | ||||
|  	smp_init_cpus(); | ||||
|  	smp_build_mpidr_hash(); | ||||
| diff --git a/arch/arm64/kvm/hypercalls.c b/arch/arm64/kvm/hypercalls.c | ||||
| index 25ea4ec..9a48345 100644 | ||||
| --- a/arch/arm64/kvm/hypercalls.c | ||||
| +++ b/arch/arm64/kvm/hypercalls.c | ||||
| @@ -9,16 +9,59 @@ | ||||
|  #include <kvm/arm_hypercalls.h> | ||||
|  #include <kvm/arm_psci.h> | ||||
|   | ||||
| +static void kvm_ptp_get_time(struct kvm_vcpu *vcpu, u64 *val) | ||||
| +{ | ||||
| +	struct system_time_snapshot systime_snapshot; | ||||
| +	u64 cycles = ~0UL; | ||||
| +	u32 feature; | ||||
| + | ||||
| +	/* | ||||
| +	 * system time and counter value must captured in the same | ||||
| +	 * time to keep consistency and precision. | ||||
| +	 */ | ||||
| +	ktime_get_snapshot(&systime_snapshot); | ||||
| + | ||||
| +	// binding ptp_kvm clocksource to arm_arch_counter | ||||
| +	if (systime_snapshot.cs_id != CSID_ARM_ARCH_COUNTER) | ||||
| +		return; | ||||
| + | ||||
| +	val[0] = upper_32_bits(systime_snapshot.real); | ||||
| +	val[1] = lower_32_bits(systime_snapshot.real); | ||||
| + | ||||
| +	/* | ||||
| +	 * which of virtual counter or physical counter being | ||||
| +	 * asked for is decided by the r1 value of SMCCC | ||||
| +	 * call. If no invalid r1 value offered, default cycle | ||||
| +	 * value(-1) will be returned. | ||||
| +	 * Note: keep in mind that feature is u32 and smccc_get_arg1 | ||||
| +	 * will return u64, so need auto cast here. | ||||
| +	 */ | ||||
| +	feature = smccc_get_arg1(vcpu); | ||||
| +	switch (feature) { | ||||
| +	case ARM_PTP_VIRT_COUNTER: | ||||
| +		cycles = systime_snapshot.cycles - vcpu_read_sys_reg(vcpu, CNTVOFF_EL2); | ||||
| +		break; | ||||
| +	case ARM_PTP_PHY_COUNTER: | ||||
| +		cycles = systime_snapshot.cycles; | ||||
| +		break; | ||||
| +	default: | ||||
| +		val[0] = SMCCC_RET_NOT_SUPPORTED; | ||||
| +		break; | ||||
| +	} | ||||
| +	val[2] = upper_32_bits(cycles); | ||||
| +	val[3] = lower_32_bits(cycles); | ||||
| +} | ||||
| + | ||||
|  int kvm_hvc_call_handler(struct kvm_vcpu *vcpu) | ||||
|  { | ||||
|  	u32 func_id = smccc_get_function(vcpu); | ||||
| -	long val = SMCCC_RET_NOT_SUPPORTED; | ||||
| +	u64 val[4] = {SMCCC_RET_NOT_SUPPORTED}; | ||||
|  	u32 feature; | ||||
|  	gpa_t gpa; | ||||
|   | ||||
|  	switch (func_id) { | ||||
|  	case ARM_SMCCC_VERSION_FUNC_ID: | ||||
| -		val = ARM_SMCCC_VERSION_1_1; | ||||
| +		val[0] = ARM_SMCCC_VERSION_1_1; | ||||
|  		break; | ||||
|  	case ARM_SMCCC_ARCH_FEATURES_FUNC_ID: | ||||
|  		feature = smccc_get_arg1(vcpu); | ||||
| @@ -28,10 +71,10 @@ int kvm_hvc_call_handler(struct kvm_vcpu *vcpu) | ||||
|  			case SPECTRE_VULNERABLE: | ||||
|  				break; | ||||
|  			case SPECTRE_MITIGATED: | ||||
| -				val = SMCCC_RET_SUCCESS; | ||||
| +				val[0] = SMCCC_RET_SUCCESS; | ||||
|  				break; | ||||
|  			case SPECTRE_UNAFFECTED: | ||||
| -				val = SMCCC_ARCH_WORKAROUND_RET_UNAFFECTED; | ||||
| +				val[0] = SMCCC_ARCH_WORKAROUND_RET_UNAFFECTED; | ||||
|  				break; | ||||
|  			} | ||||
|  			break; | ||||
| @@ -54,27 +97,52 @@ int kvm_hvc_call_handler(struct kvm_vcpu *vcpu) | ||||
|  					break; | ||||
|  				fallthrough; | ||||
|  			case SPECTRE_UNAFFECTED: | ||||
| -				val = SMCCC_RET_NOT_REQUIRED; | ||||
| +				val[0] = SMCCC_RET_NOT_REQUIRED; | ||||
|  				break; | ||||
|  			} | ||||
|  			break; | ||||
|  		case ARM_SMCCC_HV_PV_TIME_FEATURES: | ||||
| -			val = SMCCC_RET_SUCCESS; | ||||
| +			val[0] = SMCCC_RET_SUCCESS; | ||||
|  			break; | ||||
|  		} | ||||
|  		break; | ||||
|  	case ARM_SMCCC_HV_PV_TIME_FEATURES: | ||||
| -		val = kvm_hypercall_pv_features(vcpu); | ||||
| +		val[0] = kvm_hypercall_pv_features(vcpu); | ||||
|  		break; | ||||
|  	case ARM_SMCCC_HV_PV_TIME_ST: | ||||
|  		gpa = kvm_init_stolen_time(vcpu); | ||||
|  		if (gpa != GPA_INVALID) | ||||
| -			val = gpa; | ||||
| +			val[0] = gpa; | ||||
| +		break; | ||||
| +	case ARM_SMCCC_VENDOR_HYP_CALL_UID_FUNC_ID: | ||||
| +		val[0] = ARM_SMCCC_VENDOR_HYP_UID_KVM_REG_0; | ||||
| +		val[1] = ARM_SMCCC_VENDOR_HYP_UID_KVM_REG_1; | ||||
| +		val[2] = ARM_SMCCC_VENDOR_HYP_UID_KVM_REG_2; | ||||
| +		val[3] = ARM_SMCCC_VENDOR_HYP_UID_KVM_REG_3; | ||||
| +		break; | ||||
| +	case ARM_SMCCC_VENDOR_HYP_KVM_FEATURES_FUNC_ID: | ||||
| +		val[0] = BIT(ARM_SMCCC_KVM_FUNC_FEATURES); | ||||
| +		val[0] |= BIT(ARM_SMCCC_KVM_FUNC_PTP); | ||||
| +		break; | ||||
| +	/* | ||||
| +	 * This serves virtual kvm_ptp. | ||||
| +	 * Four values will be passed back. | ||||
| +	 * reg0 stores high 32-bits of host ktime; | ||||
| +	 * reg1 stores low 32-bits of host ktime; | ||||
| +	 * For ARM_PTP_VIRT_COUNTER: | ||||
| +	 * reg2 stores high 32-bits of difference of host cycles and cntvoff; | ||||
| +	 * reg3 stores low 32-bits of difference of host cycles and cntvoff. | ||||
| +	 * For ARM_PTP_PHY_COUNTER: | ||||
| +	 * reg2 stores the high 32-bits of host cycles; | ||||
| +	 * reg3 stores the low 32-bits of host cycles. | ||||
| +	 */ | ||||
| +	case ARM_SMCCC_VENDOR_HYP_KVM_PTP_FUNC_ID: | ||||
| +		kvm_ptp_get_time(vcpu, val); | ||||
|  		break; | ||||
|  	default: | ||||
|  		return kvm_psci_call(vcpu); | ||||
|  	} | ||||
|   | ||||
| -	smccc_set_retval(vcpu, val, 0, 0, 0); | ||||
| +	smccc_set_retval(vcpu, val[0], val[1], val[2], val[3]); | ||||
|  	return 1; | ||||
|  } | ||||
| diff --git a/arch/arm64/kvm/reset.c b/arch/arm64/kvm/reset.c | ||||
| index b969c21..77fae02 100644 | ||||
| --- a/arch/arm64/kvm/reset.c | ||||
| +++ b/arch/arm64/kvm/reset.c | ||||
| @@ -75,6 +75,7 @@ int kvm_arch_vm_ioctl_check_extension(struct kvm *kvm, long ext) | ||||
|  		break; | ||||
|  	case KVM_CAP_SET_GUEST_DEBUG: | ||||
|  	case KVM_CAP_VCPU_ATTRIBUTES: | ||||
| +	case KVM_CAP_PTP_KVM: | ||||
|  		r = 1; | ||||
|  		break; | ||||
|  	case KVM_CAP_ARM_VM_IPA_SIZE: | ||||
| diff --git a/drivers/clocksource/arm_arch_timer.c b/drivers/clocksource/arm_arch_timer.c | ||||
| index d017782..7cbb9fa 100644 | ||||
| --- a/drivers/clocksource/arm_arch_timer.c | ||||
| +++ b/drivers/clocksource/arm_arch_timer.c | ||||
| @@ -16,6 +16,7 @@ | ||||
|  #include <linux/cpu_pm.h> | ||||
|  #include <linux/clockchips.h> | ||||
|  #include <linux/clocksource.h> | ||||
| +#include <linux/clocksource_ids.h> | ||||
|  #include <linux/interrupt.h> | ||||
|  #include <linux/of_irq.h> | ||||
|  #include <linux/of_address.h> | ||||
| @@ -24,6 +25,8 @@ | ||||
|  #include <linux/sched/clock.h> | ||||
|  #include <linux/sched_clock.h> | ||||
|  #include <linux/acpi.h> | ||||
| +#include <linux/arm-smccc.h> | ||||
| +#include <linux/ptp_kvm.h> | ||||
|   | ||||
|  #include <asm/arch_timer.h> | ||||
|  #include <asm/virt.h> | ||||
| @@ -191,6 +194,7 @@ static u64 arch_counter_read_cc(const struct cyclecounter *cc) | ||||
|   | ||||
|  static struct clocksource clocksource_counter = { | ||||
|  	.name	= "arch_sys_counter", | ||||
| +	.id	= CSID_ARM_ARCH_COUNTER, | ||||
|  	.rating	= 400, | ||||
|  	.read	= arch_counter_read, | ||||
|  	.mask	= CLOCKSOURCE_MASK(56), | ||||
| @@ -1657,3 +1661,32 @@ static int __init arch_timer_acpi_init(struct acpi_table_header *table) | ||||
|  } | ||||
|  TIMER_ACPI_DECLARE(arch_timer, ACPI_SIG_GTDT, arch_timer_acpi_init); | ||||
|  #endif | ||||
| + | ||||
| +int kvm_arch_ptp_get_crosststamp(u64 *cycle, struct timespec64 *ts, | ||||
| +			      struct clocksource **cs) | ||||
| +{ | ||||
| +	struct arm_smccc_res hvc_res; | ||||
| +	ktime_t ktime; | ||||
| +	u32 ptp_counter; | ||||
| + | ||||
| +	if (arch_timer_uses_ppi == ARCH_TIMER_VIRT_PPI) | ||||
| +		ptp_counter = ARM_PTP_VIRT_COUNTER; | ||||
| +	else | ||||
| +		ptp_counter = ARM_PTP_PHY_COUNTER; | ||||
| + | ||||
| +	arm_smccc_1_1_invoke(ARM_SMCCC_VENDOR_HYP_KVM_PTP_FUNC_ID, | ||||
| +			     ptp_counter, &hvc_res); | ||||
| + | ||||
| +	if ((int)(hvc_res.a0) < 0) | ||||
| +		return -EOPNOTSUPP; | ||||
| + | ||||
| +	ktime = (u64)hvc_res.a0 << 32 | hvc_res.a1; | ||||
| +	*ts = ktime_to_timespec64(ktime); | ||||
| +	if (cycle) | ||||
| +		*cycle = (u64)hvc_res.a2 << 32 | hvc_res.a3; | ||||
| +	if (cs) | ||||
| +		*cs = &clocksource_counter; | ||||
| + | ||||
| +	return 0; | ||||
| +} | ||||
| +EXPORT_SYMBOL_GPL(kvm_arch_ptp_get_crosststamp); | ||||
| diff --git a/drivers/firmware/smccc/smccc.c b/drivers/firmware/smccc/smccc.c | ||||
| index 00c88b8..e153c71 100644 | ||||
| --- a/drivers/firmware/smccc/smccc.c | ||||
| +++ b/drivers/firmware/smccc/smccc.c | ||||
| @@ -7,10 +7,47 @@ | ||||
|   | ||||
|  #include <linux/init.h> | ||||
|  #include <linux/arm-smccc.h> | ||||
| +#include <linux/kernel.h> | ||||
| +#include <linux/string.h> | ||||
|   | ||||
|  static u32 smccc_version = ARM_SMCCC_VERSION_1_0; | ||||
|  static enum arm_smccc_conduit smccc_conduit = SMCCC_CONDUIT_NONE; | ||||
|   | ||||
| +DECLARE_BITMAP(__kvm_arm_hyp_services, ARM_SMCCC_KVM_NUM_FUNCS) = { }; | ||||
| +EXPORT_SYMBOL_GPL(__kvm_arm_hyp_services); | ||||
| + | ||||
| +void __init kvm_init_hyp_services(void) | ||||
| +{ | ||||
| +	int i; | ||||
| +	struct arm_smccc_res res; | ||||
| + | ||||
| +	if (arm_smccc_get_version() == ARM_SMCCC_VERSION_1_0) | ||||
| +		return; | ||||
| + | ||||
| +	arm_smccc_1_1_invoke(ARM_SMCCC_VENDOR_HYP_CALL_UID_FUNC_ID, &res); | ||||
| +	if (res.a0 != ARM_SMCCC_VENDOR_HYP_UID_KVM_REG_0 || | ||||
| +	    res.a1 != ARM_SMCCC_VENDOR_HYP_UID_KVM_REG_1 || | ||||
| +	    res.a2 != ARM_SMCCC_VENDOR_HYP_UID_KVM_REG_2 || | ||||
| +	    res.a3 != ARM_SMCCC_VENDOR_HYP_UID_KVM_REG_3) | ||||
| +		return; | ||||
| + | ||||
| +	memset(&res, 0, sizeof(res)); | ||||
| +	arm_smccc_1_1_invoke(ARM_SMCCC_VENDOR_HYP_KVM_FEATURES_FUNC_ID, &res); | ||||
| +	for (i = 0; i < 32; ++i) { | ||||
| +		if (res.a0 & (i)) | ||||
| +			set_bit(i + (32 * 0), __kvm_arm_hyp_services); | ||||
| +		if (res.a1 & (i)) | ||||
| +			set_bit(i + (32 * 1), __kvm_arm_hyp_services); | ||||
| +		if (res.a2 & (i)) | ||||
| +			set_bit(i + (32 * 2), __kvm_arm_hyp_services); | ||||
| +		if (res.a3 & (i)) | ||||
| +			set_bit(i + (32 * 3), __kvm_arm_hyp_services); | ||||
| +	} | ||||
| + | ||||
| +	pr_info("KVM hypervisor services detected (0x%08lx 0x%08lx 0x%08lx 0x%08lx)\n", | ||||
| +		 res.a3, res.a2, res.a1, res.a0); | ||||
| +} | ||||
| + | ||||
|  void __init arm_smccc_version_init(u32 version, enum arm_smccc_conduit conduit) | ||||
|  { | ||||
|  	smccc_version = version; | ||||
| diff --git a/drivers/ptp/Kconfig b/drivers/ptp/Kconfig | ||||
| index deb429a..5523f96 100644 | ||||
| --- a/drivers/ptp/Kconfig | ||||
| +++ b/drivers/ptp/Kconfig | ||||
| @@ -108,7 +108,7 @@ config PTP_1588_CLOCK_PCH | ||||
|  config PTP_1588_CLOCK_KVM | ||||
|  	tristate "KVM virtual PTP clock" | ||||
|  	depends on PTP_1588_CLOCK | ||||
| -	depends on KVM_GUEST && X86 | ||||
| +	depends on X86 || (HAVE_ARM_SMCCC_DISCOVERY && ARM_ARCH_TIMER) | ||||
|  	default y | ||||
|  	help | ||||
|  	  This driver adds support for using kvm infrastructure as a PTP | ||||
| diff --git a/drivers/ptp/Makefile b/drivers/ptp/Makefile | ||||
| index 7aff75f..9fa5ede 100644 | ||||
| --- a/drivers/ptp/Makefile | ||||
| +++ b/drivers/ptp/Makefile | ||||
| @@ -4,6 +4,8 @@ | ||||
|  # | ||||
|   | ||||
|  ptp-y					:= ptp_clock.o ptp_chardev.o ptp_sysfs.o | ||||
| +ptp_kvm-$(CONFIG_X86)			:= ptp_kvm_x86.o ptp_kvm_common.o | ||||
| +ptp_kvm-$(CONFIG_HAVE_ARM_SMCCC)	:= ptp_kvm_arm.o ptp_kvm_common.o | ||||
|  obj-$(CONFIG_PTP_1588_CLOCK)		+= ptp.o | ||||
|  obj-$(CONFIG_PTP_1588_CLOCK_DTE)	+= ptp_dte.o | ||||
|  obj-$(CONFIG_PTP_1588_CLOCK_INES)	+= ptp_ines.o | ||||
| diff --git a/drivers/ptp/ptp_kvm_arm.c b/drivers/ptp/ptp_kvm_arm.c | ||||
| new file mode 100644 | ||||
| index 0000000..b7d28c8 | ||||
| --- /dev/null | ||||
| +++ b/drivers/ptp/ptp_kvm_arm.c | ||||
| @@ -0,0 +1,28 @@ | ||||
| +// SPDX-License-Identifier: GPL-2.0-only | ||||
| +/* | ||||
| + *  Virtual PTP 1588 clock for use with KVM guests | ||||
| + *  Copyright (C) 2019 ARM Ltd. | ||||
| + *  All Rights Reserved | ||||
| + */ | ||||
| + | ||||
| +#include <linux/arm-smccc.h> | ||||
| +#include <linux/ptp_kvm.h> | ||||
| + | ||||
| +#include <asm/arch_timer.h> | ||||
| +#include <asm/hypervisor.h> | ||||
| + | ||||
| +int kvm_arch_ptp_init(void) | ||||
| +{ | ||||
| +	int ret; | ||||
| + | ||||
| +	ret = kvm_arm_hyp_service_available(ARM_SMCCC_KVM_FUNC_PTP); | ||||
| +	if (ret <= 0) | ||||
| +		return -EOPNOTSUPP; | ||||
| + | ||||
| +	return 0; | ||||
| +} | ||||
| + | ||||
| +int kvm_arch_ptp_get_clock(struct timespec64 *ts) | ||||
| +{ | ||||
| +	return kvm_arch_ptp_get_crosststamp(NULL, ts, NULL); | ||||
| +} | ||||
| diff --git a/drivers/ptp/ptp_kvm.c b/drivers/ptp/ptp_kvm_common.c | ||||
| similarity index 60% | ||||
| rename from drivers/ptp/ptp_kvm.c | ||||
| rename to drivers/ptp/ptp_kvm_common.c | ||||
| index 658d33f..5c36e2f 100644 | ||||
| --- a/drivers/ptp/ptp_kvm.c | ||||
| +++ b/drivers/ptp/ptp_kvm_common.c | ||||
| @@ -8,11 +8,11 @@ | ||||
|  #include <linux/err.h> | ||||
|  #include <linux/init.h> | ||||
|  #include <linux/kernel.h> | ||||
| +#include <linux/slab.h> | ||||
|  #include <linux/module.h> | ||||
| +#include <linux/ptp_kvm.h> | ||||
|  #include <uapi/linux/kvm_para.h> | ||||
|  #include <asm/kvm_para.h> | ||||
| -#include <asm/pvclock.h> | ||||
| -#include <asm/kvmclock.h> | ||||
|  #include <uapi/asm/kvm_para.h> | ||||
|   | ||||
|  #include <linux/ptp_clock_kernel.h> | ||||
| @@ -24,56 +24,29 @@ struct kvm_ptp_clock { | ||||
|   | ||||
|  static DEFINE_SPINLOCK(kvm_ptp_lock); | ||||
|   | ||||
| -static struct pvclock_vsyscall_time_info *hv_clock; | ||||
| - | ||||
| -static struct kvm_clock_pairing clock_pair; | ||||
| -static phys_addr_t clock_pair_gpa; | ||||
| - | ||||
|  static int ptp_kvm_get_time_fn(ktime_t *device_time, | ||||
|  			       struct system_counterval_t *system_counter, | ||||
|  			       void *ctx) | ||||
|  { | ||||
| -	unsigned long ret; | ||||
| +	long ret; | ||||
| +	u64 cycle; | ||||
|  	struct timespec64 tspec; | ||||
| -	unsigned version; | ||||
| -	int cpu; | ||||
| -	struct pvclock_vcpu_time_info *src; | ||||
| +	struct clocksource *cs; | ||||
|   | ||||
|  	spin_lock(&kvm_ptp_lock); | ||||
|   | ||||
|  	preempt_disable_notrace(); | ||||
| -	cpu = smp_processor_id(); | ||||
| -	src = &hv_clock[cpu].pvti; | ||||
| - | ||||
| -	do { | ||||
| -		/* | ||||
| -		 * We are using a TSC value read in the hosts | ||||
| -		 * kvm_hc_clock_pairing handling. | ||||
| -		 * So any changes to tsc_to_system_mul | ||||
| -		 * and tsc_shift or any other pvclock | ||||
| -		 * data invalidate that measurement. | ||||
| -		 */ | ||||
| -		version = pvclock_read_begin(src); | ||||
| - | ||||
| -		ret = kvm_hypercall2(KVM_HC_CLOCK_PAIRING, | ||||
| -				     clock_pair_gpa, | ||||
| -				     KVM_CLOCK_PAIRING_WALLCLOCK); | ||||
| -		if (ret != 0) { | ||||
| -			pr_err_ratelimited("clock pairing hypercall ret %lu\n", ret); | ||||
| -			spin_unlock(&kvm_ptp_lock); | ||||
| -			preempt_enable_notrace(); | ||||
| -			return -EOPNOTSUPP; | ||||
| -		} | ||||
| - | ||||
| -		tspec.tv_sec = clock_pair.sec; | ||||
| -		tspec.tv_nsec = clock_pair.nsec; | ||||
| -		ret = __pvclock_read_cycles(src, clock_pair.tsc); | ||||
| -	} while (pvclock_read_retry(src, version)); | ||||
| +	ret = kvm_arch_ptp_get_crosststamp(&cycle, &tspec, &cs); | ||||
| +	if (ret!=0) { | ||||
| +		spin_unlock(&kvm_ptp_lock); | ||||
| +		preempt_enable_notrace(); | ||||
| +		return ret; | ||||
| +	} | ||||
|   | ||||
|  	preempt_enable_notrace(); | ||||
|   | ||||
| -	system_counter->cycles = ret; | ||||
| -	system_counter->cs = &kvm_clock; | ||||
| +	system_counter->cycles = cycle; | ||||
| +	system_counter->cs = cs; | ||||
|   | ||||
|  	*device_time = timespec64_to_ktime(tspec); | ||||
|   | ||||
| @@ -111,22 +84,17 @@ static int ptp_kvm_settime(struct ptp_clock_info *ptp, | ||||
|   | ||||
|  static int ptp_kvm_gettime(struct ptp_clock_info *ptp, struct timespec64 *ts) | ||||
|  { | ||||
| -	unsigned long ret; | ||||
| +	long ret; | ||||
|  	struct timespec64 tspec; | ||||
|   | ||||
|  	spin_lock(&kvm_ptp_lock); | ||||
|   | ||||
| -	ret = kvm_hypercall2(KVM_HC_CLOCK_PAIRING, | ||||
| -			     clock_pair_gpa, | ||||
| -			     KVM_CLOCK_PAIRING_WALLCLOCK); | ||||
| -	if (ret != 0) { | ||||
| -		pr_err_ratelimited("clock offset hypercall ret %lu\n", ret); | ||||
| +	ret = kvm_arch_ptp_get_clock(&tspec); | ||||
| +	if (ret) { | ||||
|  		spin_unlock(&kvm_ptp_lock); | ||||
| -		return -EOPNOTSUPP; | ||||
| +		return ret; | ||||
|  	} | ||||
|   | ||||
| -	tspec.tv_sec = clock_pair.sec; | ||||
| -	tspec.tv_nsec = clock_pair.nsec; | ||||
|  	spin_unlock(&kvm_ptp_lock); | ||||
|   | ||||
|  	memcpy(ts, &tspec, sizeof(struct timespec64)); | ||||
| @@ -168,19 +136,12 @@ static int __init ptp_kvm_init(void) | ||||
|  { | ||||
|  	long ret; | ||||
|   | ||||
| -	if (!kvm_para_available()) | ||||
| -		return -ENODEV; | ||||
| - | ||||
| -	clock_pair_gpa = slow_virt_to_phys(&clock_pair); | ||||
| -	hv_clock = pvclock_get_pvti_cpu0_va(); | ||||
| - | ||||
| -	if (!hv_clock) | ||||
| -		return -ENODEV; | ||||
| - | ||||
| -	ret = kvm_hypercall2(KVM_HC_CLOCK_PAIRING, clock_pair_gpa, | ||||
| -			KVM_CLOCK_PAIRING_WALLCLOCK); | ||||
| -	if (ret == -KVM_ENOSYS || ret == -KVM_EOPNOTSUPP) | ||||
| -		return -ENODEV; | ||||
| +	ret = kvm_arch_ptp_init(); | ||||
| +	if (ret) { | ||||
| +		if (ret != -EOPNOTSUPP) | ||||
| +			pr_err("fail to initialize ptp_kvm"); | ||||
| +		return ret; | ||||
| +	} | ||||
|   | ||||
|  	kvm_ptp_clock.caps = ptp_kvm_caps; | ||||
|   | ||||
| diff --git a/drivers/ptp/ptp_kvm_x86.c b/drivers/ptp/ptp_kvm_x86.c | ||||
| new file mode 100644 | ||||
| index 0000000..e011d69 | ||||
| --- /dev/null | ||||
| +++ b/drivers/ptp/ptp_kvm_x86.c | ||||
| @@ -0,0 +1,96 @@ | ||||
| +// SPDX-License-Identifier: GPL-2.0-or-later | ||||
| +/* | ||||
| + * Virtual PTP 1588 clock for use with KVM guests | ||||
| + * | ||||
| + * Copyright (C) 2017 Red Hat Inc. | ||||
| + */ | ||||
| + | ||||
| +#include <linux/device.h> | ||||
| +#include <linux/kernel.h> | ||||
| +#include <asm/pvclock.h> | ||||
| +#include <asm/kvmclock.h> | ||||
| +#include <linux/module.h> | ||||
| +#include <uapi/asm/kvm_para.h> | ||||
| +#include <uapi/linux/kvm_para.h> | ||||
| +#include <linux/ptp_clock_kernel.h> | ||||
| +#include <linux/ptp_kvm.h> | ||||
| + | ||||
| +phys_addr_t clock_pair_gpa; | ||||
| +struct kvm_clock_pairing clock_pair; | ||||
| +struct pvclock_vsyscall_time_info *hv_clock; | ||||
| + | ||||
| +int kvm_arch_ptp_init(void) | ||||
| +{ | ||||
| +	long ret; | ||||
| + | ||||
| +	if (!kvm_para_available()) | ||||
| +		return -ENODEV; | ||||
| + | ||||
| +	clock_pair_gpa = slow_virt_to_phys(&clock_pair); | ||||
| +	hv_clock = pvclock_get_pvti_cpu0_va(); | ||||
| +	if (!hv_clock) | ||||
| +		return -ENODEV; | ||||
| + | ||||
| +	ret = kvm_hypercall2(KVM_HC_CLOCK_PAIRING, clock_pair_gpa, | ||||
| +			     KVM_CLOCK_PAIRING_WALLCLOCK); | ||||
| +	if (ret == -KVM_ENOSYS || ret == -KVM_EOPNOTSUPP) | ||||
| +		return -ENODEV; | ||||
| + | ||||
| +	return 0; | ||||
| +} | ||||
| + | ||||
| +int kvm_arch_ptp_get_clock(struct timespec64 *ts) | ||||
| +{ | ||||
| +	unsigned long ret; | ||||
| + | ||||
| +	ret = kvm_hypercall2(KVM_HC_CLOCK_PAIRING, | ||||
| +			     clock_pair_gpa, | ||||
| +			     KVM_CLOCK_PAIRING_WALLCLOCK); | ||||
| +	if (ret != 0) { | ||||
| +		pr_err_ratelimited("clock offset hypercall ret %lu\n", ret); | ||||
| +		return -EOPNOTSUPP; | ||||
| +	} | ||||
| + | ||||
| +	ts->tv_sec = clock_pair.sec; | ||||
| +	ts->tv_nsec = clock_pair.nsec; | ||||
| + | ||||
| +	return 0; | ||||
| +} | ||||
| + | ||||
| +int kvm_arch_ptp_get_crosststamp(u64 *cycle, struct timespec64 *tspec, | ||||
| +			      struct clocksource **cs) | ||||
| +{ | ||||
| +	unsigned long ret; | ||||
| +	unsigned int version; | ||||
| +	int cpu; | ||||
| +	struct pvclock_vcpu_time_info *src; | ||||
| + | ||||
| +	cpu = smp_processor_id(); | ||||
| +	src = &hv_clock[cpu].pvti; | ||||
| + | ||||
| +	do { | ||||
| +		/* | ||||
| +		 * We are using a TSC value read in the hosts | ||||
| +		 * kvm_hc_clock_pairing handling. | ||||
| +		 * So any changes to tsc_to_system_mul | ||||
| +		 * and tsc_shift or any other pvclock | ||||
| +		 * data invalidate that measurement. | ||||
| +		 */ | ||||
| +		version = pvclock_read_begin(src); | ||||
| + | ||||
| +		ret = kvm_hypercall2(KVM_HC_CLOCK_PAIRING, | ||||
| +				     clock_pair_gpa, | ||||
| +				     KVM_CLOCK_PAIRING_WALLCLOCK); | ||||
| +		if (ret != 0) { | ||||
| +			pr_err_ratelimited("clock pairing hypercall ret %lu\n", ret); | ||||
| +			return -EOPNOTSUPP; | ||||
| +		} | ||||
| +		tspec->tv_sec = clock_pair.sec; | ||||
| +		tspec->tv_nsec = clock_pair.nsec; | ||||
| +		*cycle = __pvclock_read_cycles(src, clock_pair.tsc); | ||||
| +	} while (pvclock_read_retry(src, version)); | ||||
| + | ||||
| +	*cs = &kvm_clock; | ||||
| + | ||||
| +	return 0; | ||||
| +} | ||||
| diff --git a/include/linux/arm-smccc.h b/include/linux/arm-smccc.h | ||||
| index f860645..92b46cb 100644 | ||||
| --- a/include/linux/arm-smccc.h | ||||
| +++ b/include/linux/arm-smccc.h | ||||
| @@ -55,6 +55,8 @@ | ||||
|  #define ARM_SMCCC_OWNER_TRUSTED_OS	50 | ||||
|  #define ARM_SMCCC_OWNER_TRUSTED_OS_END	63 | ||||
|   | ||||
| +#define ARM_SMCCC_FUNC_QUERY_CALL_UID  0xff01 | ||||
| + | ||||
|  #define ARM_SMCCC_QUIRK_NONE		0 | ||||
|  #define ARM_SMCCC_QUIRK_QCOM_A6		1 /* Save/restore register a6 */ | ||||
|   | ||||
| @@ -87,8 +89,47 @@ | ||||
|  			   ARM_SMCCC_SMC_32,				\ | ||||
|  			   0, 0x7fff) | ||||
|   | ||||
| +#define ARM_SMCCC_VENDOR_HYP_CALL_UID_FUNC_ID				\ | ||||
| +	ARM_SMCCC_CALL_VAL(ARM_SMCCC_FAST_CALL,				\ | ||||
| +			   ARM_SMCCC_SMC_32,				\ | ||||
| +			   ARM_SMCCC_OWNER_VENDOR_HYP,			\ | ||||
| +			   ARM_SMCCC_FUNC_QUERY_CALL_UID) | ||||
| + | ||||
| +/* KVM UID value: 28b46fb6-2ec5-11e9-a9ca-4b564d003a74 */ | ||||
| +#define ARM_SMCCC_VENDOR_HYP_UID_KVM_REG_0	0xb66fb428U | ||||
| +#define ARM_SMCCC_VENDOR_HYP_UID_KVM_REG_1	0xe911c52eU | ||||
| +#define ARM_SMCCC_VENDOR_HYP_UID_KVM_REG_2	0x564bcaa9U | ||||
| +#define ARM_SMCCC_VENDOR_HYP_UID_KVM_REG_3	0x743a004dU | ||||
| + | ||||
| +/* KVM "vendor specific" services */ | ||||
| +#define ARM_SMCCC_KVM_FUNC_FEATURES		0 | ||||
| +#define ARM_SMCCC_KVM_FUNC_PTP			1 | ||||
| +#define ARM_SMCCC_KVM_FUNC_FEATURES_2		127 | ||||
| +#define ARM_SMCCC_KVM_NUM_FUNCS			128 | ||||
| + | ||||
| +#define ARM_SMCCC_VENDOR_HYP_KVM_FEATURES_FUNC_ID			\ | ||||
| +	ARM_SMCCC_CALL_VAL(ARM_SMCCC_FAST_CALL,				\ | ||||
| +			   ARM_SMCCC_SMC_32,				\ | ||||
| +			   ARM_SMCCC_OWNER_VENDOR_HYP,			\ | ||||
| +			   ARM_SMCCC_KVM_FUNC_FEATURES) | ||||
| + | ||||
|  #define SMCCC_ARCH_WORKAROUND_RET_UNAFFECTED	1 | ||||
|   | ||||
| +/* | ||||
| + * ptp_kvm is a feature used for time sync between vm and host. | ||||
| + * ptp_kvm module in guest kernel will get service from host using | ||||
| + * this hypercall ID. | ||||
| + */ | ||||
| +#define ARM_SMCCC_VENDOR_HYP_KVM_PTP_FUNC_ID				\ | ||||
| +	ARM_SMCCC_CALL_VAL(ARM_SMCCC_FAST_CALL,				\ | ||||
| +			   ARM_SMCCC_SMC_32,				\ | ||||
| +			   ARM_SMCCC_OWNER_VENDOR_HYP,			\ | ||||
| +			   ARM_SMCCC_KVM_FUNC_PTP) | ||||
| + | ||||
| +/* ptp_kvm counter type ID */ | ||||
| +#define ARM_PTP_VIRT_COUNTER			0 | ||||
| +#define ARM_PTP_PHY_COUNTER			1 | ||||
| + | ||||
|  /* Paravirtualised time calls (defined by ARM DEN0057A) */ | ||||
|  #define ARM_SMCCC_HV_PV_TIME_FEATURES				\ | ||||
|  	ARM_SMCCC_CALL_VAL(ARM_SMCCC_FAST_CALL,			\ | ||||
| @@ -391,5 +432,23 @@ asmlinkage void __arm_smccc_hvc(unsigned long a0, unsigned long a1, | ||||
|  		method;							\ | ||||
|  	}) | ||||
|   | ||||
| +void __init kvm_init_hyp_services(void); | ||||
| + | ||||
| +/* | ||||
| + * This helper will be called in guest. We put it here then both arm and arm64 | ||||
| + * guest can touch it. | ||||
| + */ | ||||
| +#include <linux/kernel.h> | ||||
| +#include <linux/err.h> | ||||
| +static inline bool kvm_arm_hyp_service_available(u32 func_id) | ||||
| +{ | ||||
| +	extern DECLARE_BITMAP(__kvm_arm_hyp_services, ARM_SMCCC_KVM_NUM_FUNCS); | ||||
| + | ||||
| +	if (func_id >= ARM_SMCCC_KVM_NUM_FUNCS) | ||||
| +		return false; | ||||
| + | ||||
| +	return test_bit(func_id, __kvm_arm_hyp_services); | ||||
| +} | ||||
| + | ||||
|  #endif /*__ASSEMBLY__*/ | ||||
|  #endif /*__LINUX_ARM_SMCCC_H*/ | ||||
| diff --git a/include/linux/clocksource.h b/include/linux/clocksource.h | ||||
| index 86d143d..1290d0d 100644 | ||||
| --- a/include/linux/clocksource.h | ||||
| +++ b/include/linux/clocksource.h | ||||
| @@ -17,6 +17,7 @@ | ||||
|  #include <linux/timer.h> | ||||
|  #include <linux/init.h> | ||||
|  #include <linux/of.h> | ||||
| +#include <linux/clocksource_ids.h> | ||||
|  #include <asm/div64.h> | ||||
|  #include <asm/io.h> | ||||
|   | ||||
| @@ -62,6 +63,10 @@ struct module; | ||||
|   *			400-499: Perfect | ||||
|   *				The ideal clocksource. A must-use where | ||||
|   *				available. | ||||
| + * @id:			Defaults to CSID_GENERIC. The id value is captured | ||||
| + *			in certain snapshot functions to allow callers to | ||||
| + *			validate the clocksource from which the snapshot was | ||||
| + *			taken. | ||||
|   * @flags:		Flags describing special properties | ||||
|   * @enable:		Optional function to enable the clocksource | ||||
|   * @disable:		Optional function to disable the clocksource | ||||
| @@ -100,6 +105,7 @@ struct clocksource { | ||||
|  	const char		*name; | ||||
|  	struct list_head	list; | ||||
|  	int			rating; | ||||
| +	enum clocksource_ids	id; | ||||
|  	enum vdso_clock_mode	vdso_clock_mode; | ||||
|  	unsigned long		flags; | ||||
|   | ||||
| diff --git a/include/linux/clocksource_ids.h b/include/linux/clocksource_ids.h | ||||
| new file mode 100644 | ||||
| index 0000000..16775d7 | ||||
| --- /dev/null | ||||
| +++ b/include/linux/clocksource_ids.h | ||||
| @@ -0,0 +1,12 @@ | ||||
| +/* SPDX-License-Identifier: GPL-2.0 */ | ||||
| +#ifndef _LINUX_CLOCKSOURCE_IDS_H | ||||
| +#define _LINUX_CLOCKSOURCE_IDS_H | ||||
| + | ||||
| +/* Enum to give clocksources a unique identifier */ | ||||
| +enum clocksource_ids { | ||||
| +	CSID_GENERIC		= 0, | ||||
| +	CSID_ARM_ARCH_COUNTER, | ||||
| +	CSID_MAX, | ||||
| +}; | ||||
| + | ||||
| +#endif | ||||
| diff --git a/include/linux/ptp_kvm.h b/include/linux/ptp_kvm.h | ||||
| new file mode 100644 | ||||
| index 0000000..f960a71 | ||||
| --- /dev/null | ||||
| +++ b/include/linux/ptp_kvm.h | ||||
| @@ -0,0 +1,19 @@ | ||||
| +/* SPDX-License-Identifier: GPL-2.0-or-later */ | ||||
| +/* | ||||
| + * Virtual PTP 1588 clock for use with KVM guests | ||||
| + * | ||||
| + * Copyright (C) 2017 Red Hat Inc. | ||||
| + */ | ||||
| + | ||||
| +#ifndef _PTP_KVM_H_ | ||||
| +#define _PTP_KVM_H_ | ||||
| + | ||||
| +struct timespec64; | ||||
| +struct clocksource; | ||||
| + | ||||
| +int kvm_arch_ptp_init(void); | ||||
| +int kvm_arch_ptp_get_clock(struct timespec64 *ts); | ||||
| +int kvm_arch_ptp_get_crosststamp(u64 *cycle, | ||||
| +		struct timespec64 *tspec, struct clocksource **cs); | ||||
| + | ||||
| +#endif /* _PTP_KVM_H_ */ | ||||
| diff --git a/include/linux/timekeeping.h b/include/linux/timekeeping.h | ||||
| index 7f7e4a3..2ee0535 100644 | ||||
| --- a/include/linux/timekeeping.h | ||||
| +++ b/include/linux/timekeeping.h | ||||
| @@ -3,6 +3,7 @@ | ||||
|  #define _LINUX_TIMEKEEPING_H | ||||
|   | ||||
|  #include <linux/errno.h> | ||||
| +#include <linux/clocksource_ids.h> | ||||
|   | ||||
|  /* Included from linux/ktime.h */ | ||||
|   | ||||
| @@ -244,11 +245,12 @@ struct ktime_timestamps { | ||||
|   * @cs_was_changed_seq:	The sequence number of clocksource change events | ||||
|   */ | ||||
|  struct system_time_snapshot { | ||||
| -	u64		cycles; | ||||
| -	ktime_t		real; | ||||
| -	ktime_t		raw; | ||||
| -	unsigned int	clock_was_set_seq; | ||||
| -	u8		cs_was_changed_seq; | ||||
| +	u64			cycles; | ||||
| +	ktime_t			real; | ||||
| +	ktime_t			raw; | ||||
| +	enum clocksource_ids	cs_id; | ||||
| +	unsigned int		clock_was_set_seq; | ||||
| +	u8			cs_was_changed_seq; | ||||
|  }; | ||||
|   | ||||
|  /** | ||||
| diff --git a/include/uapi/linux/kvm.h b/include/uapi/linux/kvm.h | ||||
| index ca41220..797c40b 100644 | ||||
| --- a/include/uapi/linux/kvm.h | ||||
| +++ b/include/uapi/linux/kvm.h | ||||
| @@ -1053,6 +1053,7 @@ struct kvm_ppc_resize_hpt { | ||||
|  #define KVM_CAP_X86_USER_SPACE_MSR 188 | ||||
|  #define KVM_CAP_X86_MSR_FILTER 189 | ||||
|  #define KVM_CAP_ENFORCE_PV_FEATURE_CPUID 190 | ||||
| +#define KVM_CAP_PTP_KVM 191 | ||||
|   | ||||
|  #ifdef KVM_CAP_IRQ_ROUTING | ||||
|   | ||||
| diff --git a/kernel/time/clocksource.c b/kernel/time/clocksource.c | ||||
| index 02441ea..6b38d49 100644 | ||||
| --- a/kernel/time/clocksource.c | ||||
| +++ b/kernel/time/clocksource.c | ||||
| @@ -928,6 +928,8 @@ int __clocksource_register_scale(struct clocksource *cs, u32 scale, u32 freq) | ||||
|   | ||||
|  	clocksource_arch_init(cs); | ||||
|   | ||||
| +	if (WARN_ON_ONCE((unsigned int)cs->id >= CSID_MAX)) | ||||
| +		cs->id = CSID_GENERIC; | ||||
|  	if (cs->vdso_clock_mode < 0 || | ||||
|  	    cs->vdso_clock_mode >= VDSO_CLOCKMODE_MAX) { | ||||
|  		pr_warn("clocksource %s registered with invalid VDSO mode %d. Disabling VDSO support.\n", | ||||
| diff --git a/kernel/time/timekeeping.c b/kernel/time/timekeeping.c | ||||
| index 6858a31..eb04a2d 100644 | ||||
| --- a/kernel/time/timekeeping.c | ||||
| +++ b/kernel/time/timekeeping.c | ||||
| @@ -1053,6 +1053,7 @@ void ktime_get_snapshot(struct system_time_snapshot *systime_snapshot) | ||||
|  	do { | ||||
|  		seq = read_seqcount_begin(&tk_core.seq); | ||||
|  		now = tk_clock_read(&tk->tkr_mono); | ||||
| +		systime_snapshot->cs_id = tk->tkr_mono.clock->id; | ||||
|  		systime_snapshot->cs_was_changed_seq = tk->cs_was_changed_seq; | ||||
|  		systime_snapshot->clock_was_set_seq = tk->clock_was_set_seq; | ||||
|  		base_real = ktime_add(tk->tkr_mono.base, | ||||
| --  | ||||
| 2.32.0 | ||||
|  | ||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -1,81 +0,0 @@ | ||||
| From 3d1d7f8922ed2f080f6d8e08df0d51e22f9590ec Mon Sep 17 00:00:00 2001 | ||||
| From: Jianyong Wu <jianyong.wu@arm.com> | ||||
| Date: Wed, 1 Apr 2020 15:19:29 +0800 | ||||
| Subject: [PATCH 1/9] arm/arm64: Provide a wrapper for SMCCC 1.1 calls | ||||
|  | ||||
| From: Steven Price <steven.price@arm.com> | ||||
|  | ||||
| SMCCC 1.1 calls may use either HVC or SMC depending on the PSCI | ||||
| conduit. Rather than coding this in every call site, provide a macro | ||||
| which uses the correct instruction. The macro also handles the case | ||||
| where no conduit is configured/available returning a not supported error | ||||
| in res, along with returning the conduit used for the call. | ||||
|  | ||||
| This allow us to remove some duplicated code and will be useful later | ||||
| when adding paravirtualized time hypervisor calls. | ||||
|  | ||||
| Signed-off-by: Steven Price <steven.price@arm.com> | ||||
| Acked-by: Will Deacon <will@kernel.org> | ||||
| Signed-off-by: Marc Zyngier <maz@kernel.org> | ||||
| --- | ||||
|  include/linux/arm-smccc.h | 45 +++++++++++++++++++++++++++++++++++++++ | ||||
|  1 file changed, 45 insertions(+) | ||||
|  | ||||
| diff --git a/include/linux/arm-smccc.h b/include/linux/arm-smccc.h | ||||
| index 080012a6f025..131edde5d37e 100644 | ||||
| --- a/include/linux/arm-smccc.h | ||||
| +++ b/include/linux/arm-smccc.h | ||||
| @@ -302,5 +302,50 @@ asmlinkage void __arm_smccc_hvc(unsigned long a0, unsigned long a1, | ||||
|  #define SMCCC_RET_NOT_SUPPORTED			-1 | ||||
|  #define SMCCC_RET_NOT_REQUIRED			-2 | ||||
|   | ||||
| +/* | ||||
| + * Like arm_smccc_1_1* but always returns SMCCC_RET_NOT_SUPPORTED. | ||||
| + * Used when the SMCCC conduit is not defined. The empty asm statement | ||||
| + * avoids compiler warnings about unused variables. | ||||
| + */ | ||||
| +#define __fail_smccc_1_1(...)                                          \ | ||||
| +	do {                                                            \ | ||||
| +		__declare_args(__count_args(__VA_ARGS__), __VA_ARGS__); \ | ||||
| +		asm ("" __constraints(__count_args(__VA_ARGS__)));      \ | ||||
| +		if (___res)                                             \ | ||||
| +			___res->a0 = SMCCC_RET_NOT_SUPPORTED;           \ | ||||
| +	} while (0) | ||||
| + | ||||
| +/* | ||||
| + * arm_smccc_1_1_invoke() - make an SMCCC v1.1 compliant call | ||||
| + * | ||||
| + * This is a variadic macro taking one to eight source arguments, and | ||||
| + * an optional return structure. | ||||
| + * | ||||
| + * @a0-a7: arguments passed in registers 0 to 7 | ||||
| + * @res: result values from registers 0 to 3 | ||||
| + * | ||||
| + * This macro will make either an HVC call or an SMC call depending on the | ||||
| + * current SMCCC conduit. If no valid conduit is available then -1 | ||||
| + * (SMCCC_RET_NOT_SUPPORTED) is returned in @res.a0 (if supplied). | ||||
| + * | ||||
| + * The return value also provides the conduit that was used. | ||||
| + */ | ||||
| +#define arm_smccc_1_1_invoke(...) ({					\ | ||||
| +		int method = arm_smccc_1_1_get_conduit();		\ | ||||
| +		switch (method) {					\ | ||||
| +		case SMCCC_CONDUIT_HVC:					\ | ||||
| +			arm_smccc_1_1_hvc(__VA_ARGS__);			\ | ||||
| +			break;						\ | ||||
| +		case SMCCC_CONDUIT_SMC:					\ | ||||
| +			arm_smccc_1_1_smc(__VA_ARGS__);			\ | ||||
| +			break;						\ | ||||
| +		default:						\ | ||||
| +			__fail_smccc_1_1(__VA_ARGS__);			\ | ||||
| +			method = SMCCC_CONDUIT_NONE;			\ | ||||
| +			break;						\ | ||||
| +		}							\ | ||||
| +		method;							\ | ||||
| +	}) | ||||
| + | ||||
|  #endif /*__ASSEMBLY__*/ | ||||
|  #endif /*__LINUX_ARM_SMCCC_H*/ | ||||
| --  | ||||
| 2.17.1 | ||||
|  | ||||
| @@ -1,81 +0,0 @@ | ||||
| From b830806f5cd02119be9b25812b3ea56d97cd08f3 Mon Sep 17 00:00:00 2001 | ||||
| From: Mark Rutland <mark.rutland@arm.com> | ||||
| Date: Fri, 9 Aug 2019 14:22:40 +0100 | ||||
| Subject: [PATCH 2/9] arm/arm64: smccc/psci: add arm_smccc_1_1_get_conduit() | ||||
|  | ||||
| SMCCC callers are currently amassing a collection of enums for the SMCCC | ||||
| conduit, and are having to dig into the PSCI driver's internals in order | ||||
| to figure out what to do. | ||||
|  | ||||
| Let's clean this up, with common SMCCC_CONDUIT_* definitions, and an | ||||
| arm_smccc_1_1_get_conduit() helper that abstracts the PSCI driver's | ||||
| internal state. | ||||
|  | ||||
| We can kill off the PSCI_CONDUIT_* definitions once we've migrated users | ||||
| over to the new interface. | ||||
|  | ||||
| Signed-off-by: Mark Rutland <mark.rutland@arm.com> | ||||
| Acked-by: Lorenzo Pieralisi <lorenzo.pieralisi@arm.com> | ||||
| Acked-by: Will Deacon <will.deacon@arm.com> | ||||
| Signed-off-by: Catalin Marinas <catalin.marinas@arm.com> | ||||
| --- | ||||
|  drivers/firmware/psci/psci.c | 15 +++++++++++++++ | ||||
|  include/linux/arm-smccc.h    | 16 ++++++++++++++++ | ||||
|  2 files changed, 31 insertions(+) | ||||
|  | ||||
| diff --git a/drivers/firmware/psci/psci.c b/drivers/firmware/psci/psci.c | ||||
| index 84f4ff351c62..eb797081d159 100644 | ||||
| --- a/drivers/firmware/psci/psci.c | ||||
| +++ b/drivers/firmware/psci/psci.c | ||||
| @@ -57,6 +57,21 @@ struct psci_operations psci_ops = { | ||||
|  	.smccc_version = SMCCC_VERSION_1_0, | ||||
|  }; | ||||
|   | ||||
| +enum arm_smccc_conduit arm_smccc_1_1_get_conduit(void) | ||||
| +{ | ||||
| +	if (psci_ops.smccc_version < SMCCC_VERSION_1_1) | ||||
| +		return SMCCC_CONDUIT_NONE; | ||||
| + | ||||
| +	switch (psci_ops.conduit) { | ||||
| +	case PSCI_CONDUIT_SMC: | ||||
| +		return SMCCC_CONDUIT_SMC; | ||||
| +	case PSCI_CONDUIT_HVC: | ||||
| +		return SMCCC_CONDUIT_HVC; | ||||
| +	default: | ||||
| +		return SMCCC_CONDUIT_NONE; | ||||
| +	} | ||||
| +} | ||||
| + | ||||
|  typedef unsigned long (psci_fn)(unsigned long, unsigned long, | ||||
|  				unsigned long, unsigned long); | ||||
|  static psci_fn *invoke_psci_fn; | ||||
| diff --git a/include/linux/arm-smccc.h b/include/linux/arm-smccc.h | ||||
| index 131edde5d37e..e6d4cb4f61f1 100644 | ||||
| --- a/include/linux/arm-smccc.h | ||||
| +++ b/include/linux/arm-smccc.h | ||||
| @@ -80,6 +80,22 @@ | ||||
|   | ||||
|  #include <linux/linkage.h> | ||||
|  #include <linux/types.h> | ||||
| + | ||||
| +enum arm_smccc_conduit { | ||||
| +	SMCCC_CONDUIT_NONE, | ||||
| +	SMCCC_CONDUIT_SMC, | ||||
| +	SMCCC_CONDUIT_HVC, | ||||
| +}; | ||||
| + | ||||
| +/** | ||||
| + * arm_smccc_1_1_get_conduit() | ||||
| + * | ||||
| + * Returns the conduit to be used for SMCCCv1.1 or later. | ||||
| + * | ||||
| + * When SMCCCv1.1 is not present, returns SMCCC_CONDUIT_NONE. | ||||
| + */ | ||||
| +enum arm_smccc_conduit arm_smccc_1_1_get_conduit(void); | ||||
| + | ||||
|  /** | ||||
|   * struct arm_smccc_res - Result from SMC/HVC call | ||||
|   * @a0-a3 result values from registers 0 to 3 | ||||
| --  | ||||
| 2.17.1 | ||||
|  | ||||
		Reference in New Issue
	
	Block a user