]> git.ipfire.org Git - thirdparty/linux.git/commitdiff
KVM: arm64: Add clock support to nVHE/pKVM hyp
authorVincent Donnefort <vdonnefort@google.com>
Mon, 9 Mar 2026 16:25:06 +0000 (16:25 +0000)
committerMarc Zyngier <maz@kernel.org>
Wed, 11 Mar 2026 08:51:16 +0000 (08:51 +0000)
In preparation for supporting tracing from the nVHE hyp, add support to
generate timestamps with a clock fed by the CNTCVT counter. The clock
can be kept in sync with the kernel's by updating the slope values. This
will be done later.

As current we do only create a trace clock, make the whole support
dependent on the upcoming CONFIG_NVHE_EL2_TRACING.

Signed-off-by: Vincent Donnefort <vdonnefort@google.com>
Link: https://patch.msgid.link/20260309162516.2623589-21-vdonnefort@google.com
Signed-off-by: Marc Zyngier <maz@kernel.org>
arch/arm64/kvm/hyp/include/nvhe/clock.h [new file with mode: 0644]
arch/arm64/kvm/hyp/nvhe/Makefile
arch/arm64/kvm/hyp/nvhe/clock.c [new file with mode: 0644]

diff --git a/arch/arm64/kvm/hyp/include/nvhe/clock.h b/arch/arm64/kvm/hyp/include/nvhe/clock.h
new file mode 100644 (file)
index 0000000..9f429f5
--- /dev/null
@@ -0,0 +1,16 @@
+/* SPDX-License-Identifier: GPL-2.0 */
+#ifndef __ARM64_KVM_HYP_NVHE_CLOCK_H
+#define __ARM64_KVM_HYP_NVHE_CLOCK_H
+#include <linux/types.h>
+
+#include <asm/kvm_hyp.h>
+
+#ifdef CONFIG_NVHE_EL2_TRACING
+void trace_clock_update(u32 mult, u32 shift, u64 epoch_ns, u64 epoch_cyc);
+u64 trace_clock(void);
+#else
+static inline void
+trace_clock_update(u32 mult, u32 shift, u64 epoch_ns, u64 epoch_cyc) { }
+static inline u64 trace_clock(void) { return 0; }
+#endif
+#endif
index a244ec25f8c5bd0a744f7791102265323ecc681c..8dc95257c2910f547388f9df5735730036ed007c 100644 (file)
@@ -17,7 +17,7 @@ ccflags-y += -fno-stack-protector     \
 hostprogs := gen-hyprel
 HOST_EXTRACFLAGS += -I$(objtree)/include
 
-lib-objs := clear_page.o copy_page.o memcpy.o memset.o
+lib-objs := clear_page.o copy_page.o memcpy.o memset.o tishift.o
 lib-objs := $(addprefix ../../../lib/, $(lib-objs))
 
 CFLAGS_switch.nvhe.o += -Wno-override-init
@@ -29,6 +29,7 @@ hyp-obj-y += ../vgic-v3-sr.o ../aarch32.o ../vgic-v2-cpuif-proxy.o ../entry.o \
         ../fpsimd.o ../hyp-entry.o ../exception.o ../pgtable.o
 hyp-obj-y += ../../../kernel/smccc-call.o
 hyp-obj-$(CONFIG_LIST_HARDENED) += list_debug.o
+hyp-obj-$(CONFIG_NVHE_EL2_TRACING) += clock.o
 hyp-obj-y += $(lib-objs)
 
 ##
diff --git a/arch/arm64/kvm/hyp/nvhe/clock.c b/arch/arm64/kvm/hyp/nvhe/clock.c
new file mode 100644 (file)
index 0000000..32fc431
--- /dev/null
@@ -0,0 +1,65 @@
+// SPDX-License-Identifier: GPL-2.0
+/*
+ * Copyright (C) 2025 Google LLC
+ * Author: Vincent Donnefort <vdonnefort@google.com>
+ */
+
+#include <nvhe/clock.h>
+
+#include <asm/arch_timer.h>
+#include <asm/div64.h>
+
+static struct clock_data {
+       struct {
+               u32 mult;
+               u32 shift;
+               u64 epoch_ns;
+               u64 epoch_cyc;
+               u64 cyc_overflow64;
+       } data[2];
+       u64 cur;
+} trace_clock_data;
+
+static u64 __clock_mult_uint128(u64 cyc, u32 mult, u32 shift)
+{
+       __uint128_t ns = (__uint128_t)cyc * mult;
+
+       ns >>= shift;
+
+       return (u64)ns;
+}
+
+/* Does not guarantee no reader on the modified bank. */
+void trace_clock_update(u32 mult, u32 shift, u64 epoch_ns, u64 epoch_cyc)
+{
+       struct clock_data *clock = &trace_clock_data;
+       u64 bank = clock->cur ^ 1;
+
+       clock->data[bank].mult                  = mult;
+       clock->data[bank].shift                 = shift;
+       clock->data[bank].epoch_ns              = epoch_ns;
+       clock->data[bank].epoch_cyc             = epoch_cyc;
+       clock->data[bank].cyc_overflow64        = ULONG_MAX / mult;
+
+       smp_store_release(&clock->cur, bank);
+}
+
+/* Use untrusted host data */
+u64 trace_clock(void)
+{
+       struct clock_data *clock = &trace_clock_data;
+       u64 bank = smp_load_acquire(&clock->cur);
+       u64 cyc, ns;
+
+       cyc = __arch_counter_get_cntvct() - clock->data[bank].epoch_cyc;
+
+       if (likely(cyc < clock->data[bank].cyc_overflow64)) {
+               ns = cyc * clock->data[bank].mult;
+               ns >>= clock->data[bank].shift;
+       } else {
+               ns = __clock_mult_uint128(cyc, clock->data[bank].mult,
+                                         clock->data[bank].shift);
+       }
+
+       return (u64)ns + clock->data[bank].epoch_ns;
+}