]> www.infradead.org Git - users/hch/misc.git/commitdiff
KVM: selftests: Hoist APIC functions out of individual tests
authorJim Mattson <jmattson@google.com>
Fri, 4 Jun 2021 17:26:08 +0000 (10:26 -0700)
committerPaolo Bonzini <pbonzini@redhat.com>
Thu, 17 Jun 2021 17:09:32 +0000 (13:09 -0400)
Move the APIC functions into the library to encourage code reuse and
to avoid unintended deviations.

Signed-off-by: Jim Mattson <jmattson@google.com>
Reviewed-by: Oliver Upton <oupton@google.com>
Message-Id: <20210604172611.281819-10-jmattson@google.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
tools/testing/selftests/kvm/Makefile
tools/testing/selftests/kvm/include/x86_64/apic.h
tools/testing/selftests/kvm/include/x86_64/processor.h
tools/testing/selftests/kvm/lib/x86_64/apic.c [new file with mode: 0644]
tools/testing/selftests/kvm/x86_64/evmcs_test.c
tools/testing/selftests/kvm/x86_64/set_boot_cpu_id.c
tools/testing/selftests/kvm/x86_64/xapic_ipi_test.c

index 6d241c97a890b8f7832631c3d1c10573d7645515..1c750910c27b150016b9f45a5fde89126cd67c7c 100644 (file)
@@ -34,7 +34,7 @@ ifeq ($(ARCH),s390)
 endif
 
 LIBKVM = lib/assert.c lib/elf.c lib/io.c lib/kvm_util.c lib/rbtree.c lib/sparsebit.c lib/test_util.c lib/guest_modes.c lib/perf_test_util.c
-LIBKVM_x86_64 = lib/x86_64/processor.c lib/x86_64/vmx.c lib/x86_64/svm.c lib/x86_64/ucall.c lib/x86_64/handlers.S
+LIBKVM_x86_64 = lib/x86_64/apic.c lib/x86_64/processor.c lib/x86_64/vmx.c lib/x86_64/svm.c lib/x86_64/ucall.c lib/x86_64/handlers.S
 LIBKVM_aarch64 = lib/aarch64/processor.c lib/aarch64/ucall.c
 LIBKVM_s390x = lib/s390x/processor.c lib/s390x/ucall.c lib/s390x/diag318_test_handler.c
 
index 0d0e35c8866b0a65ca73f8886d861d623225b127..e5a9fe040a6c844952fa6ea012e1c991d3369635 100644 (file)
@@ -8,6 +8,10 @@
 #ifndef SELFTEST_KVM_APIC_H
 #define SELFTEST_KVM_APIC_H
 
+#include <stdint.h>
+
+#include "processor.h"
+
 #define APIC_DEFAULT_GPA               0xfee00000ULL
 
 /* APIC base address MSR and fields */
 #define        APIC_ICR2       0x310
 #define                SET_APIC_DEST_FIELD(x)  ((x) << 24)
 
+void apic_disable(void);
+void xapic_enable(void);
+void x2apic_enable(void);
+
+static inline uint32_t get_bsp_flag(void)
+{
+       return rdmsr(MSR_IA32_APICBASE) & MSR_IA32_APICBASE_BSP;
+}
+
+static inline uint32_t xapic_read_reg(unsigned int reg)
+{
+       return ((volatile uint32_t *)APIC_DEFAULT_GPA)[reg >> 2];
+}
+
+static inline void xapic_write_reg(unsigned int reg, uint32_t val)
+{
+       ((volatile uint32_t *)APIC_DEFAULT_GPA)[reg >> 2] = val;
+}
+
 #endif /* SELFTEST_KVM_APIC_H */
index a4729d9032ced5e72587f5b081066823d8f0d934..9a5b47d2d5d638d242fc679f04b1ade546ee8f2c 100644 (file)
@@ -13,6 +13,8 @@
 
 #include <asm/msr-index.h>
 
+#include "../kvm_util.h"
+
 #define X86_EFLAGS_FIXED        (1u << 1)
 
 #define X86_CR4_VME            (1ul << 0)
diff --git a/tools/testing/selftests/kvm/lib/x86_64/apic.c b/tools/testing/selftests/kvm/lib/x86_64/apic.c
new file mode 100644 (file)
index 0000000..31f318a
--- /dev/null
@@ -0,0 +1,46 @@
+// SPDX-License-Identifier: GPL-2.0-only
+/*
+ * tools/testing/selftests/kvm/lib/x86_64/processor.c
+ *
+ * Copyright (C) 2021, Google LLC.
+ */
+
+#include "apic.h"
+
+void apic_disable(void)
+{
+       wrmsr(MSR_IA32_APICBASE,
+             rdmsr(MSR_IA32_APICBASE) &
+               ~(MSR_IA32_APICBASE_ENABLE | MSR_IA32_APICBASE_EXTD));
+}
+
+void xapic_enable(void)
+{
+       uint64_t val = rdmsr(MSR_IA32_APICBASE);
+
+       /* Per SDM: to enable xAPIC when in x2APIC must first disable APIC */
+       if (val & MSR_IA32_APICBASE_EXTD) {
+               apic_disable();
+               wrmsr(MSR_IA32_APICBASE,
+                     rdmsr(MSR_IA32_APICBASE) | MSR_IA32_APICBASE_ENABLE);
+       } else if (!(val & MSR_IA32_APICBASE_ENABLE)) {
+               wrmsr(MSR_IA32_APICBASE, val | MSR_IA32_APICBASE_ENABLE);
+       }
+
+       /*
+        * Per SDM: reset value of spurious interrupt vector register has the
+        * APIC software enabled bit=0. It must be enabled in addition to the
+        * enable bit in the MSR.
+        */
+       val = xapic_read_reg(APIC_SPIV) | APIC_SPIV_APIC_ENABLED;
+       xapic_write_reg(APIC_SPIV, val);
+}
+
+void x2apic_enable(void)
+{
+       uint32_t spiv_reg = APIC_BASE_MSR + (APIC_SPIV >> 4);
+
+       wrmsr(MSR_IA32_APICBASE, rdmsr(MSR_IA32_APICBASE) |
+             MSR_IA32_APICBASE_ENABLE | MSR_IA32_APICBASE_EXTD);
+       wrmsr(spiv_reg, rdmsr(spiv_reg) | APIC_SPIV_APIC_ENABLED);
+}
index 63096cea26c61beb0cca78149e110758d60c0d98..d058d9e428c639d9fa8e6268cf8b9113cae2e73a 100644 (file)
 
 static int ud_count;
 
-void enable_x2apic(void)
-{
-       uint32_t spiv_reg = APIC_BASE_MSR + (APIC_SPIV >> 4);
-
-       wrmsr(MSR_IA32_APICBASE, rdmsr(MSR_IA32_APICBASE) |
-             MSR_IA32_APICBASE_ENABLE | MSR_IA32_APICBASE_EXTD);
-       wrmsr(spiv_reg, rdmsr(spiv_reg) | APIC_SPIV_APIC_ENABLED);
-}
-
 static void guest_ud_handler(struct ex_regs *regs)
 {
        ud_count++;
@@ -59,7 +50,7 @@ void guest_code(struct vmx_pages *vmx_pages)
 #define L2_GUEST_STACK_SIZE 64
        unsigned long l2_guest_stack[L2_GUEST_STACK_SIZE];
 
-       enable_x2apic();
+       x2apic_enable();
 
        GUEST_SYNC(1);
        GUEST_SYNC(2);
index 12c558fc8074a8ba6320f34c01a3ca66b78bb5c7..5f8dd74d415f83680d36c737bafa4e98c5f85979 100644 (file)
 #include "test_util.h"
 #include "kvm_util.h"
 #include "processor.h"
+#include "apic.h"
 
 #define N_VCPU 2
 #define VCPU_ID0 0
 #define VCPU_ID1 1
 
-static uint32_t get_bsp_flag(void)
-{
-       return rdmsr(MSR_IA32_APICBASE) & MSR_IA32_APICBASE_BSP;
-}
-
 static void guest_bsp_vcpu(void *arg)
 {
        GUEST_SYNC(1);
index 2f964cdc273c94207d99a8c4182b7a8ac0c80a1c..21b22718a9dbd5b7c8f18acf7fe9628f34c228f0 100644 (file)
@@ -42,8 +42,6 @@
 #define HALTER_VCPU_ID 0
 #define SENDER_VCPU_ID 1
 
-volatile uint32_t *apic_base = (volatile uint32_t *)APIC_DEFAULT_GPA;
-
 /*
  * Vector for IPI from sender vCPU to halting vCPU.
  * Value is arbitrary and was chosen for the alternating bit pattern. Any
@@ -86,45 +84,6 @@ struct thread_params {
        uint64_t *pipis_rcvd; /* host address of ipis_rcvd global */
 };
 
-uint32_t read_apic_reg(uint reg)
-{
-       return apic_base[reg >> 2];
-}
-
-void write_apic_reg(uint reg, uint32_t val)
-{
-       apic_base[reg >> 2] = val;
-}
-
-void disable_apic(void)
-{
-       wrmsr(MSR_IA32_APICBASE,
-             rdmsr(MSR_IA32_APICBASE) &
-               ~(MSR_IA32_APICBASE_ENABLE | MSR_IA32_APICBASE_EXTD));
-}
-
-void enable_xapic(void)
-{
-       uint64_t val = rdmsr(MSR_IA32_APICBASE);
-
-       /* Per SDM: to enable xAPIC when in x2APIC must first disable APIC */
-       if (val & MSR_IA32_APICBASE_EXTD) {
-               disable_apic();
-               wrmsr(MSR_IA32_APICBASE,
-                     rdmsr(MSR_IA32_APICBASE) | MSR_IA32_APICBASE_ENABLE);
-       } else if (!(val & MSR_IA32_APICBASE_ENABLE)) {
-               wrmsr(MSR_IA32_APICBASE, val | MSR_IA32_APICBASE_ENABLE);
-       }
-
-       /*
-        * Per SDM: reset value of spurious interrupt vector register has the
-        * APIC software enabled bit=0. It must be enabled in addition to the
-        * enable bit in the MSR.
-        */
-       val = read_apic_reg(APIC_SPIV) | APIC_SPIV_APIC_ENABLED;
-       write_apic_reg(APIC_SPIV, val);
-}
-
 void verify_apic_base_addr(void)
 {
        uint64_t msr = rdmsr(MSR_IA32_APICBASE);
@@ -136,10 +95,10 @@ void verify_apic_base_addr(void)
 static void halter_guest_code(struct test_data_page *data)
 {
        verify_apic_base_addr();
-       enable_xapic();
+       xapic_enable();
 
-       data->halter_apic_id = GET_APIC_ID_FIELD(read_apic_reg(APIC_ID));
-       data->halter_lvr = read_apic_reg(APIC_LVR);
+       data->halter_apic_id = GET_APIC_ID_FIELD(xapic_read_reg(APIC_ID));
+       data->halter_lvr = xapic_read_reg(APIC_LVR);
 
        /*
         * Loop forever HLTing and recording halts & wakes. Disable interrupts
@@ -150,8 +109,8 @@ static void halter_guest_code(struct test_data_page *data)
         * TPR and PPR for diagnostic purposes in case the test fails.
         */
        for (;;) {
-               data->halter_tpr = read_apic_reg(APIC_TASKPRI);
-               data->halter_ppr = read_apic_reg(APIC_PROCPRI);
+               data->halter_tpr = xapic_read_reg(APIC_TASKPRI);
+               data->halter_ppr = xapic_read_reg(APIC_PROCPRI);
                data->hlt_count++;
                asm volatile("sti; hlt; cli");
                data->wake_count++;
@@ -166,7 +125,7 @@ static void halter_guest_code(struct test_data_page *data)
 static void guest_ipi_handler(struct ex_regs *regs)
 {
        ipis_rcvd++;
-       write_apic_reg(APIC_EOI, 77);
+       xapic_write_reg(APIC_EOI, 77);
 }
 
 static void sender_guest_code(struct test_data_page *data)
@@ -179,7 +138,7 @@ static void sender_guest_code(struct test_data_page *data)
        uint64_t tsc_start;
 
        verify_apic_base_addr();
-       enable_xapic();
+       xapic_enable();
 
        /*
         * Init interrupt command register for sending IPIs
@@ -206,8 +165,8 @@ static void sender_guest_code(struct test_data_page *data)
                 * First IPI can be sent unconditionally because halter vCPU
                 * starts earlier.
                 */
-               write_apic_reg(APIC_ICR2, icr2_val);
-               write_apic_reg(APIC_ICR, icr_val);
+               xapic_write_reg(APIC_ICR2, icr2_val);
+               xapic_write_reg(APIC_ICR, icr_val);
                data->ipis_sent++;
 
                /*