Pablo Greco e6a3ae
From 90b8d9d17a9c252427df71fbe5bd76f897529e71 Mon Sep 17 00:00:00 2001
Pablo Greco e6a3ae
From: "plai@redhat.com" <plai@redhat.com>
Pablo Greco e6a3ae
Date: Wed, 3 Apr 2019 15:54:30 +0100
Pablo Greco e6a3ae
Subject: [PATCH 06/10] kvm: Add support to KVM_GET_MSR_FEATURE_INDEX_LIST and
Pablo Greco e6a3ae
 KVM_GET_MSRS system ioctl
Pablo Greco e6a3ae
Pablo Greco e6a3ae
RH-Author: plai@redhat.com
Pablo Greco e6a3ae
Message-id: <1554306874-28796-7-git-send-email-plai@redhat.com>
Pablo Greco e6a3ae
Patchwork-id: 85384
Pablo Greco e6a3ae
O-Subject: [RHEL8.1 qemu-kvm PATCH resend 06/10] kvm: Add support to KVM_GET_MSR_FEATURE_INDEX_LIST and KVM_GET_MSRS system ioctl
Pablo Greco e6a3ae
Bugzilla: 1561761
Pablo Greco e6a3ae
RH-Acked-by: Eduardo Habkost <ehabkost@redhat.com>
Pablo Greco e6a3ae
RH-Acked-by: Igor Mammedov <imammedo@redhat.com>
Pablo Greco e6a3ae
RH-Acked-by: Michael S. Tsirkin <mst@redhat.com>
Pablo Greco e6a3ae
Pablo Greco e6a3ae
From: Robert Hoo <robert.hu@linux.intel.com>
Pablo Greco e6a3ae
Pablo Greco e6a3ae
Add kvm_get_supported_feature_msrs() to get supported MSR feature index list.
Pablo Greco e6a3ae
Add kvm_arch_get_supported_msr_feature() to get each MSR features value.
Pablo Greco e6a3ae
Pablo Greco e6a3ae
Signed-off-by: Robert Hoo <robert.hu@linux.intel.com>
Pablo Greco e6a3ae
Message-Id: <1539578845-37944-2-git-send-email-robert.hu@linux.intel.com>
Pablo Greco e6a3ae
Reviewed-by: Eduardo Habkost <ehabkost@redhat.com>
Pablo Greco e6a3ae
Signed-off-by: Eduardo Habkost <ehabkost@redhat.com>
Pablo Greco e6a3ae
(cherry picked from commit f57bceb6ab5163ddd6c41ff4344ab8cf28a9c63d)
Pablo Greco e6a3ae
Signed-off-by: Paul Lai <plai@redhat.com>
Pablo Greco e6a3ae
Signed-off-by: Danilo C. L. de Paula <ddepaula@redhat.com>
Pablo Greco e6a3ae
---
Pablo Greco e6a3ae
 include/sysemu/kvm.h |  2 ++
Pablo Greco e6a3ae
 target/i386/kvm.c    | 80 ++++++++++++++++++++++++++++++++++++++++++++++++++++
Pablo Greco e6a3ae
 2 files changed, 82 insertions(+)
Pablo Greco e6a3ae
Pablo Greco e6a3ae
diff --git a/include/sysemu/kvm.h b/include/sysemu/kvm.h
Pablo Greco e6a3ae
index 23669c4..3d8f294 100644
Pablo Greco e6a3ae
--- a/include/sysemu/kvm.h
Pablo Greco e6a3ae
+++ b/include/sysemu/kvm.h
Pablo Greco e6a3ae
@@ -464,6 +464,8 @@ int kvm_vm_check_extension(KVMState *s, unsigned int extension);
Pablo Greco e6a3ae
 
Pablo Greco e6a3ae
 uint32_t kvm_arch_get_supported_cpuid(KVMState *env, uint32_t function,
Pablo Greco e6a3ae
                                       uint32_t index, int reg);
Pablo Greco e6a3ae
+uint32_t kvm_arch_get_supported_msr_feature(KVMState *s, uint32_t index);
Pablo Greco e6a3ae
+
Pablo Greco e6a3ae
 
Pablo Greco e6a3ae
 void kvm_set_sigmask_len(KVMState *s, unsigned int sigmask_len);
Pablo Greco e6a3ae
 
Pablo Greco e6a3ae
diff --git a/target/i386/kvm.c b/target/i386/kvm.c
Pablo Greco e6a3ae
index 702e3bf..096ed24 100644
Pablo Greco e6a3ae
--- a/target/i386/kvm.c
Pablo Greco e6a3ae
+++ b/target/i386/kvm.c
Pablo Greco e6a3ae
@@ -106,6 +106,7 @@ static int has_pit_state2;
Pablo Greco e6a3ae
 static bool has_msr_mcg_ext_ctl;
Pablo Greco e6a3ae
 
Pablo Greco e6a3ae
 static struct kvm_cpuid2 *cpuid_cache;
Pablo Greco e6a3ae
+static struct kvm_msr_list *kvm_feature_msrs;
Pablo Greco e6a3ae
 
Pablo Greco e6a3ae
 int kvm_has_pit_state2(void)
Pablo Greco e6a3ae
 {
Pablo Greco e6a3ae
@@ -405,6 +406,42 @@ uint32_t kvm_arch_get_supported_cpuid(KVMState *s, uint32_t function,
Pablo Greco e6a3ae
     return ret;
Pablo Greco e6a3ae
 }
Pablo Greco e6a3ae
 
Pablo Greco e6a3ae
+uint32_t kvm_arch_get_supported_msr_feature(KVMState *s, uint32_t index)
Pablo Greco e6a3ae
+{
Pablo Greco e6a3ae
+    struct {
Pablo Greco e6a3ae
+        struct kvm_msrs info;
Pablo Greco e6a3ae
+        struct kvm_msr_entry entries[1];
Pablo Greco e6a3ae
+    } msr_data;
Pablo Greco e6a3ae
+    uint32_t ret;
Pablo Greco e6a3ae
+
Pablo Greco e6a3ae
+    if (kvm_feature_msrs == NULL) { /* Host doesn't support feature MSRs */
Pablo Greco e6a3ae
+        return 0;
Pablo Greco e6a3ae
+    }
Pablo Greco e6a3ae
+
Pablo Greco e6a3ae
+    /* Check if requested MSR is supported feature MSR */
Pablo Greco e6a3ae
+    int i;
Pablo Greco e6a3ae
+    for (i = 0; i < kvm_feature_msrs->nmsrs; i++)
Pablo Greco e6a3ae
+        if (kvm_feature_msrs->indices[i] == index) {
Pablo Greco e6a3ae
+            break;
Pablo Greco e6a3ae
+        }
Pablo Greco e6a3ae
+    if (i == kvm_feature_msrs->nmsrs) {
Pablo Greco e6a3ae
+        return 0; /* if the feature MSR is not supported, simply return 0 */
Pablo Greco e6a3ae
+    }
Pablo Greco e6a3ae
+
Pablo Greco e6a3ae
+    msr_data.info.nmsrs = 1;
Pablo Greco e6a3ae
+    msr_data.entries[0].index = index;
Pablo Greco e6a3ae
+
Pablo Greco e6a3ae
+    ret = kvm_ioctl(s, KVM_GET_MSRS, &msr_data);
Pablo Greco e6a3ae
+    if (ret != 1) {
Pablo Greco e6a3ae
+        error_report("KVM get MSR (index=0x%x) feature failed, %s",
Pablo Greco e6a3ae
+            index, strerror(-ret));
Pablo Greco e6a3ae
+        exit(1);
Pablo Greco e6a3ae
+    }
Pablo Greco e6a3ae
+
Pablo Greco e6a3ae
+    return msr_data.entries[0].data;
Pablo Greco e6a3ae
+}
Pablo Greco e6a3ae
+
Pablo Greco e6a3ae
+
Pablo Greco e6a3ae
 typedef struct HWPoisonPage {
Pablo Greco e6a3ae
     ram_addr_t ram_addr;
Pablo Greco e6a3ae
     QLIST_ENTRY(HWPoisonPage) list;
Pablo Greco e6a3ae
@@ -1164,6 +1201,47 @@ void kvm_arch_do_init_vcpu(X86CPU *cpu)
Pablo Greco e6a3ae
     }
Pablo Greco e6a3ae
 }
Pablo Greco e6a3ae
 
Pablo Greco e6a3ae
+static int kvm_get_supported_feature_msrs(KVMState *s)
Pablo Greco e6a3ae
+{
Pablo Greco e6a3ae
+    int ret = 0;
Pablo Greco e6a3ae
+
Pablo Greco e6a3ae
+    if (kvm_feature_msrs != NULL) {
Pablo Greco e6a3ae
+        return 0;
Pablo Greco e6a3ae
+    }
Pablo Greco e6a3ae
+
Pablo Greco e6a3ae
+    if (!kvm_check_extension(s, KVM_CAP_GET_MSR_FEATURES)) {
Pablo Greco e6a3ae
+        return 0;
Pablo Greco e6a3ae
+    }
Pablo Greco e6a3ae
+
Pablo Greco e6a3ae
+    struct kvm_msr_list msr_list;
Pablo Greco e6a3ae
+
Pablo Greco e6a3ae
+    msr_list.nmsrs = 0;
Pablo Greco e6a3ae
+    ret = kvm_ioctl(s, KVM_GET_MSR_FEATURE_INDEX_LIST, &msr_list);
Pablo Greco e6a3ae
+    if (ret < 0 && ret != -E2BIG) {
Pablo Greco e6a3ae
+        error_report("Fetch KVM feature MSR list failed: %s",
Pablo Greco e6a3ae
+            strerror(-ret));
Pablo Greco e6a3ae
+        return ret;
Pablo Greco e6a3ae
+    }
Pablo Greco e6a3ae
+
Pablo Greco e6a3ae
+    assert(msr_list.nmsrs > 0);
Pablo Greco e6a3ae
+    kvm_feature_msrs = (struct kvm_msr_list *) \
Pablo Greco e6a3ae
+        g_malloc0(sizeof(msr_list) +
Pablo Greco e6a3ae
+                 msr_list.nmsrs * sizeof(msr_list.indices[0]));
Pablo Greco e6a3ae
+
Pablo Greco e6a3ae
+    kvm_feature_msrs->nmsrs = msr_list.nmsrs;
Pablo Greco e6a3ae
+    ret = kvm_ioctl(s, KVM_GET_MSR_FEATURE_INDEX_LIST, kvm_feature_msrs);
Pablo Greco e6a3ae
+
Pablo Greco e6a3ae
+    if (ret < 0) {
Pablo Greco e6a3ae
+        error_report("Fetch KVM feature MSR list failed: %s",
Pablo Greco e6a3ae
+            strerror(-ret));
Pablo Greco e6a3ae
+        g_free(kvm_feature_msrs);
Pablo Greco e6a3ae
+        kvm_feature_msrs = NULL;
Pablo Greco e6a3ae
+        return ret;
Pablo Greco e6a3ae
+    }
Pablo Greco e6a3ae
+
Pablo Greco e6a3ae
+    return 0;
Pablo Greco e6a3ae
+}
Pablo Greco e6a3ae
+
Pablo Greco e6a3ae
 static int kvm_get_supported_msrs(KVMState *s)
Pablo Greco e6a3ae
 {
Pablo Greco e6a3ae
     static int kvm_supported_msrs;
Pablo Greco e6a3ae
@@ -1320,6 +1398,8 @@ int kvm_arch_init(MachineState *ms, KVMState *s)
Pablo Greco e6a3ae
         return ret;
Pablo Greco e6a3ae
     }
Pablo Greco e6a3ae
 
Pablo Greco e6a3ae
+    kvm_get_supported_feature_msrs(s);
Pablo Greco e6a3ae
+
Pablo Greco e6a3ae
     uname(&utsname);
Pablo Greco e6a3ae
     lm_capable_kernel = strcmp(utsname.machine, "x86_64") == 0;
Pablo Greco e6a3ae
 
Pablo Greco e6a3ae
-- 
Pablo Greco e6a3ae
1.8.3.1
Pablo Greco e6a3ae