From: Zhao Liu <zhao1.liu@intel.com>
To: "Paolo Bonzini" <pbonzini@redhat.com>,
"Eric Blake" <eblake@redhat.com>,
"Markus Armbruster" <armbru@redhat.com>,
"Michael Roth" <michael.roth@amd.com>,
"Daniel P . Berrangé" <berrange@redhat.com>,
"Eduardo Habkost" <eduardo@habkost.net>,
"Marcelo Tosatti" <mtosatti@redhat.com>,
"Shaoqin Huang" <shahuang@redhat.com>,
"Eric Auger" <eauger@redhat.com>,
"Peter Maydell" <peter.maydell@linaro.org>,
"Laurent Vivier" <lvivier@redhat.com>,
"Thomas Huth" <thuth@redhat.com>,
"Sebastian Ott" <sebott@redhat.com>,
"Gavin Shan" <gshan@redhat.com>
Cc: qemu-devel@nongnu.org, kvm@vger.kernel.org, qemu-arm@nongnu.org,
Zhenyu Wang <zhenyu.z.wang@intel.com>,
Dapeng Mi <dapeng1.mi@intel.com>, Yuan Yao <yuan.yao@intel.com>,
Xiong Zhang <xiong.y.zhang@intel.com>,
Mingwei Zhang <mizhang@google.com>,
Jim Mattson <jmattson@google.com>, Zhao Liu <zhao1.liu@intel.com>
Subject: [RFC 3/5] i386/kvm: Support event with select&umask format in KVM PMU filter
Date: Wed, 10 Jul 2024 12:51:15 +0800 [thread overview]
Message-ID: <20240710045117.3164577-4-zhao1.liu@intel.com> (raw)
In-Reply-To: <20240710045117.3164577-1-zhao1.liu@intel.com>
The select&umask is the common way for x86 to identify the PMU event,
so support this way as the "x86-default" format in kvm-pmu-filter
object.
Signed-off-by: Zhao Liu <zhao1.liu@intel.com>
---
accel/kvm/kvm-pmu.c | 62 ++++++++++++++++++++++++++++++++++++++++
include/sysemu/kvm-pmu.h | 13 +++++++++
qapi/kvm.json | 46 +++++++++++++++++++++++++++--
target/i386/kvm/kvm.c | 5 ++++
4 files changed, 123 insertions(+), 3 deletions(-)
diff --git a/accel/kvm/kvm-pmu.c b/accel/kvm/kvm-pmu.c
index 483d1bdf4807..51d3fba5a72a 100644
--- a/accel/kvm/kvm-pmu.c
+++ b/accel/kvm/kvm-pmu.c
@@ -17,6 +17,8 @@
#include "qom/object_interfaces.h"
#include "sysemu/kvm-pmu.h"
+#define UINT12_MAX (4095)
+
static void kvm_pmu_filter_get_event(Object *obj, Visitor *v, const char *name,
void *opaque, Error **errp)
{
@@ -38,6 +40,12 @@ static void kvm_pmu_filter_get_event(Object *obj, Visitor *v, const char *name,
str_event->u.raw.code = g_strdup_printf("0x%lx",
event->u.raw.code);
break;
+ case KVM_PMU_EVENT_FMT_X86_DEFAULT:
+ str_event->u.x86_default.select =
+ g_strdup_printf("0x%x", event->u.x86_default.select);
+ str_event->u.x86_default.umask =
+ g_strdup_printf("0x%x", event->u.x86_default.umask);
+ break;
default:
g_assert_not_reached();
}
@@ -83,6 +91,60 @@ static void kvm_pmu_filter_set_event(Object *obj, Visitor *v, const char *name,
goto fail;
}
break;
+ case KVM_PMU_EVENT_FMT_X86_DEFAULT: {
+ uint64_t select, umask;
+
+ ret = qemu_strtou64(str_event->u.x86_default.select, NULL,
+ 0, &select);
+ if (ret < 0) {
+ error_setg(errp,
+ "Invalid %s PMU event (select: %s): %s. "
+ "The select must be a "
+ "12-bit unsigned number string.",
+ KVMPMUEventEncodeFmt_str(str_event->format),
+ str_event->u.x86_default.select,
+ strerror(-ret));
+ g_free(event);
+ goto fail;
+ }
+ if (select > UINT12_MAX) {
+ error_setg(errp,
+ "Invalid %s PMU event (select: %s): "
+ "Numerical result out of range. "
+ "The select must be a "
+ "12-bit unsigned number string.",
+ KVMPMUEventEncodeFmt_str(str_event->format),
+ str_event->u.x86_default.select);
+ g_free(event);
+ goto fail;
+ }
+ event->u.x86_default.select = select;
+
+ ret = qemu_strtou64(str_event->u.x86_default.umask, NULL,
+ 0, &umask);
+ if (ret < 0) {
+ error_setg(errp,
+ "Invalid %s PMU event (umask: %s): %s. "
+ "The umask must be a uint8 string.",
+ KVMPMUEventEncodeFmt_str(str_event->format),
+ str_event->u.x86_default.umask,
+ strerror(-ret));
+ g_free(event);
+ goto fail;
+ }
+ if (umask > UINT8_MAX) {
+ error_setg(errp,
+ "Invalid %s PMU event (umask: %s): "
+ "Numerical result out of range. "
+ "The umask must be a uint8 string.",
+ KVMPMUEventEncodeFmt_str(str_event->format),
+ str_event->u.x86_default.umask);
+ g_free(event);
+ goto fail;
+ }
+ event->u.x86_default.umask = umask;
+ break;
+ }
default:
g_assert_not_reached();
}
diff --git a/include/sysemu/kvm-pmu.h b/include/sysemu/kvm-pmu.h
index 4707759761f1..707f33d604fd 100644
--- a/include/sysemu/kvm-pmu.h
+++ b/include/sysemu/kvm-pmu.h
@@ -26,4 +26,17 @@ struct KVMPMUFilter {
KVMPMUFilterEventList *events;
};
+/*
+ * Stolen from Linux kernel (RAW_EVENT at tools/testing/selftests/kvm/include/
+ * x86_64/pmu.h).
+ *
+ * Encode an eventsel+umask pair into event-select MSR format. Note, this is
+ * technically AMD's format, as Intel's format only supports 8 bits for the
+ * event selector, i.e. doesn't use bits 24:16 for the selector. But, OR-ing
+ * in '0' is a nop and won't clobber the CMASK.
+ */
+#define X86_PMU_RAW_EVENT(eventsel, umask) (((eventsel & 0xf00UL) << 24) | \
+ ((eventsel) & 0xff) | \
+ ((umask) & 0xff) << 8)
+
#endif /* KVM_PMU_H */
diff --git a/qapi/kvm.json b/qapi/kvm.json
index 0619da83c123..0d759884c229 100644
--- a/qapi/kvm.json
+++ b/qapi/kvm.json
@@ -27,11 +27,13 @@
#
# @raw: the encoded event code that KVM can directly consume.
#
+# @x86-default: standard x86 encoding format with select and umask.
+#
# Since 9.1
##
{ 'enum': 'KVMPMUEventEncodeFmt',
'prefix': 'KVM_PMU_EVENT_FMT',
- 'data': ['raw'] }
+ 'data': ['raw', 'x86-default'] }
##
# @KVMPMURawEvent:
@@ -46,6 +48,25 @@
{ 'struct': 'KVMPMURawEvent',
'data': { 'code': 'uint64' } }
+##
+# @KVMPMUX86DefalutEvent:
+#
+# x86 PMU event encoding with select and umask.
+# raw_event = ((select & 0xf00UL) << 24) | \
+# (select) & 0xff) | \
+# ((umask) & 0xff) << 8)
+#
+# @select: x86 PMU event select field, which is a 12-bit unsigned
+# number.
+#
+# @umask: x86 PMU event umask field.
+#
+# Since 9.1
+##
+{ 'struct': 'KVMPMUX86DefalutEvent',
+ 'data': { 'select': 'uint16',
+ 'umask': 'uint8' } }
+
##
# @KVMPMUFilterEvent:
#
@@ -61,7 +82,8 @@
'base': { 'action': 'KVMPMUFilterAction',
'format': 'KVMPMUEventEncodeFmt' },
'discriminator': 'format',
- 'data': { 'raw': 'KVMPMURawEvent' } }
+ 'data': { 'raw': 'KVMPMURawEvent',
+ 'x86-default': 'KVMPMUX86DefalutEvent' } }
##
# @KVMPMUFilterProperty:
@@ -89,6 +111,23 @@
{ 'struct': 'KVMPMURawEventVariant',
'data': { 'code': 'str' } }
+##
+# @KVMPMUX86DefalutEventVariant:
+#
+# The variant of KVMPMUX86DefalutEvent with the string, rather than
+# the numeric value.
+#
+# @select: x86 PMU event select field. This field is a 12-bit
+# unsigned number string.
+#
+# @umask: x86 PMU event umask field. This field is a uint8 string.
+#
+# Since 9.1
+##
+{ 'struct': 'KVMPMUX86DefalutEventVariant',
+ 'data': { 'select': 'str',
+ 'umask': 'str' } }
+
##
# @KVMPMUFilterEventVariant:
#
@@ -104,7 +143,8 @@
'base': { 'action': 'KVMPMUFilterAction',
'format': 'KVMPMUEventEncodeFmt' },
'discriminator': 'format',
- 'data': { 'raw': 'KVMPMURawEventVariant' } }
+ 'data': { 'raw': 'KVMPMURawEventVariant',
+ 'x86-default': 'KVMPMUX86DefalutEventVariant' } }
##
# @KVMPMUFilterPropertyVariant:
diff --git a/target/i386/kvm/kvm.c b/target/i386/kvm/kvm.c
index e9bf79782316..391531c036a6 100644
--- a/target/i386/kvm/kvm.c
+++ b/target/i386/kvm/kvm.c
@@ -5393,6 +5393,10 @@ kvm_config_pmu_event(KVMPMUFilter *filter,
case KVM_PMU_EVENT_FMT_RAW:
code = event->u.raw.code;
break;
+ case KVM_PMU_EVENT_FMT_X86_DEFAULT:
+ code = X86_PMU_RAW_EVENT(event->u.x86_default.select,
+ event->u.x86_default.umask);
+ break;
default:
g_assert_not_reached();
}
@@ -6073,6 +6077,7 @@ static void kvm_arch_check_pmu_filter(const Object *obj, const char *name,
switch (event->format) {
case KVM_PMU_EVENT_FMT_RAW:
+ case KVM_PMU_EVENT_FMT_X86_DEFAULT:
break;
default:
error_setg(errp,
--
2.34.1
next prev parent reply other threads:[~2024-07-10 4:37 UTC|newest]
Thread overview: 15+ messages / expand[flat|nested] mbox.gz Atom feed top
2024-07-10 4:51 [RFC 0/5] accel/kvm: Support KVM PMU filter Zhao Liu
2024-07-10 4:51 ` [RFC 1/5] qapi/qom: Introduce kvm-pmu-filter object Zhao Liu
2024-07-10 4:51 ` [RFC 2/5] i386/kvm: Support initial KVM PMU filter Zhao Liu
2024-07-10 4:51 ` Zhao Liu [this message]
2024-07-18 5:28 ` [RFC 3/5] i386/kvm: Support event with select&umask format in " Mi, Dapeng
2024-07-19 8:40 ` Zhao Liu
2024-07-23 5:36 ` Mi, Dapeng
2024-07-10 4:51 ` [RFC 4/5] i386/kvm: Support event with masked entry " Zhao Liu
2024-07-10 4:51 ` [RFC 5/5] i386/kvm: Support fixed counter " Zhao Liu
2024-07-18 5:27 ` [RFC 0/5] accel/kvm: Support " Mi, Dapeng
2024-07-19 8:05 ` Zhao Liu
2024-08-02 9:01 ` Shaoqin Huang
2024-08-02 9:37 ` Zhao Liu
2024-08-02 9:41 ` Shaoqin Huang
2024-08-02 10:03 ` Zhao Liu
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=20240710045117.3164577-4-zhao1.liu@intel.com \
--to=zhao1.liu@intel.com \
--cc=armbru@redhat.com \
--cc=berrange@redhat.com \
--cc=dapeng1.mi@intel.com \
--cc=eauger@redhat.com \
--cc=eblake@redhat.com \
--cc=eduardo@habkost.net \
--cc=gshan@redhat.com \
--cc=jmattson@google.com \
--cc=kvm@vger.kernel.org \
--cc=lvivier@redhat.com \
--cc=michael.roth@amd.com \
--cc=mizhang@google.com \
--cc=mtosatti@redhat.com \
--cc=pbonzini@redhat.com \
--cc=peter.maydell@linaro.org \
--cc=qemu-arm@nongnu.org \
--cc=qemu-devel@nongnu.org \
--cc=sebott@redhat.com \
--cc=shahuang@redhat.com \
--cc=thuth@redhat.com \
--cc=xiong.y.zhang@intel.com \
--cc=yuan.yao@intel.com \
--cc=zhenyu.z.wang@intel.com \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).