diff mbox

[v11,16/21] KVM: ARM64: Add PMU overflow interrupt routing

Message ID 1454656456-11640-17-git-send-email-zhaoshenglong@huawei.com
State Superseded
Headers show

Commit Message

Shannon Zhao Feb. 5, 2016, 7:14 a.m. UTC
From: Shannon Zhao <shannon.zhao@linaro.org>


When calling perf_event_create_kernel_counter to create perf_event,
assign a overflow handler. Then when the perf event overflows, set the
corresponding bit of guest PMOVSSET register. If this counter is enabled
and its interrupt is enabled as well, kick the vcpu to sync the
interrupt.

On VM entry, if there is counter overflowed, inject the interrupt with
the level set to 1. Otherwise, inject the interrupt with the level set
to 0.

Signed-off-by: Shannon Zhao <shannon.zhao@linaro.org>

Reviewed-by: Marc Zyngier <marc.zyngier@arm.com>

Reviewed-by: Andrew Jones <drjones@redhat.com>

---
 arch/arm/kvm/arm.c    |  2 ++
 include/kvm/arm_pmu.h |  2 ++
 virt/kvm/arm/pmu.c    | 50 +++++++++++++++++++++++++++++++++++++++++++++++++-
 3 files changed, 53 insertions(+), 1 deletion(-)

-- 
2.0.4



_______________________________________________
linux-arm-kernel mailing list
linux-arm-kernel@lists.infradead.org
http://lists.infradead.org/mailman/listinfo/linux-arm-kernel

Comments

Christoffer Dall Feb. 8, 2016, 12:26 p.m. UTC | #1
On Fri, Feb 05, 2016 at 03:14:11PM +0800, Shannon Zhao wrote:
> From: Shannon Zhao <shannon.zhao@linaro.org>

> 

> When calling perf_event_create_kernel_counter to create perf_event,

> assign a overflow handler. Then when the perf event overflows, set the

> corresponding bit of guest PMOVSSET register. If this counter is enabled

> and its interrupt is enabled as well, kick the vcpu to sync the

> interrupt.

> 

> On VM entry, if there is counter overflowed, inject the interrupt with

> the level set to 1. Otherwise, inject the interrupt with the level set

> to 0.

> 

> Signed-off-by: Shannon Zhao <shannon.zhao@linaro.org>

> Reviewed-by: Marc Zyngier <marc.zyngier@arm.com>

> Reviewed-by: Andrew Jones <drjones@redhat.com>

> ---

>  arch/arm/kvm/arm.c    |  2 ++

>  include/kvm/arm_pmu.h |  2 ++

>  virt/kvm/arm/pmu.c    | 50 +++++++++++++++++++++++++++++++++++++++++++++++++-

>  3 files changed, 53 insertions(+), 1 deletion(-)

> 

> diff --git a/arch/arm/kvm/arm.c b/arch/arm/kvm/arm.c

> index dda1959..f54264c 100644

> --- a/arch/arm/kvm/arm.c

> +++ b/arch/arm/kvm/arm.c

> @@ -28,6 +28,7 @@

>  #include <linux/sched.h>

>  #include <linux/kvm.h>

>  #include <trace/events/kvm.h>

> +#include <kvm/arm_pmu.h>

>  

>  #define CREATE_TRACE_POINTS

>  #include "trace.h"

> @@ -577,6 +578,7 @@ int kvm_arch_vcpu_ioctl_run(struct kvm_vcpu *vcpu, struct kvm_run *run)

>  		 * non-preemptible context.

>  		 */

>  		preempt_disable();

> +		kvm_pmu_flush_hwstate(vcpu);

>  		kvm_timer_flush_hwstate(vcpu);

>  		kvm_vgic_flush_hwstate(vcpu);

>  

> diff --git a/include/kvm/arm_pmu.h b/include/kvm/arm_pmu.h

> index 1f4bfa2..44a3c75 100644

> --- a/include/kvm/arm_pmu.h

> +++ b/include/kvm/arm_pmu.h

> @@ -44,6 +44,7 @@ u64 kvm_pmu_valid_counter_mask(struct kvm_vcpu *vcpu);

>  void kvm_pmu_disable_counter(struct kvm_vcpu *vcpu, u64 val);

>  void kvm_pmu_enable_counter(struct kvm_vcpu *vcpu, u64 val);

>  void kvm_pmu_overflow_set(struct kvm_vcpu *vcpu, u64 val);

> +void kvm_pmu_flush_hwstate(struct kvm_vcpu *vcpu);

>  void kvm_pmu_software_increment(struct kvm_vcpu *vcpu, u64 val);

>  void kvm_pmu_handle_pmcr(struct kvm_vcpu *vcpu, u64 val);

>  void kvm_pmu_set_counter_event_type(struct kvm_vcpu *vcpu, u64 data,

> @@ -67,6 +68,7 @@ static inline u64 kvm_pmu_valid_counter_mask(struct kvm_vcpu *vcpu)

>  static inline void kvm_pmu_disable_counter(struct kvm_vcpu *vcpu, u64 val) {}

>  static inline void kvm_pmu_enable_counter(struct kvm_vcpu *vcpu, u64 val) {}

>  static inline void kvm_pmu_overflow_set(struct kvm_vcpu *vcpu, u64 val) {}

> +static inline void kvm_pmu_flush_hwstate(struct kvm_vcpu *vcpu) {}

>  static inline void kvm_pmu_software_increment(struct kvm_vcpu *vcpu, u64 val) {}

>  static inline void kvm_pmu_handle_pmcr(struct kvm_vcpu *vcpu, u64 val) {}

>  static inline void kvm_pmu_set_counter_event_type(struct kvm_vcpu *vcpu,

> diff --git a/virt/kvm/arm/pmu.c b/virt/kvm/arm/pmu.c

> index c8ea825..5f983cb 100644

> --- a/virt/kvm/arm/pmu.c

> +++ b/virt/kvm/arm/pmu.c

> @@ -21,6 +21,7 @@

>  #include <linux/perf_event.h>

>  #include <asm/kvm_emulate.h>

>  #include <kvm/arm_pmu.h>

> +#include <kvm/arm_vgic.h>

>  

>  /**

>   * kvm_pmu_get_counter_value - get PMU counter value

> @@ -180,6 +181,52 @@ void kvm_pmu_overflow_set(struct kvm_vcpu *vcpu, u64 val)

>  }

>  

>  /**

> + * kvm_pmu_flush_hwstate - flush pmu state to cpu

> + * @vcpu: The vcpu pointer

> + *

> + * Inject virtual PMU IRQ if IRQ is pending for this cpu.

> + */

> +void kvm_pmu_flush_hwstate(struct kvm_vcpu *vcpu)

> +{

> +	struct kvm_pmu *pmu = &vcpu->arch.pmu;

> +	u64 overflow;

> +

> +	if (!kvm_arm_pmu_v3_ready(vcpu))

> +		return;

> +

> +	if (!(vcpu_sys_reg(vcpu, PMCR_EL0) & ARMV8_PMCR_E))

> +		return;


are we modeling the PMU interrupt as level-triggered?

In that case, shouldn't we lower the interrupt line on flush when
PMCR_EL0.E == 0 ?


Thanks,
-Christoffer

_______________________________________________
linux-arm-kernel mailing list
linux-arm-kernel@lists.infradead.org
http://lists.infradead.org/mailman/listinfo/linux-arm-kernel
Shannon Zhao Feb. 20, 2016, 1:32 p.m. UTC | #2
On 2016/2/8 20:26, Christoffer Dall wrote:
> On Fri, Feb 05, 2016 at 03:14:11PM +0800, Shannon Zhao wrote:

>> >From: Shannon Zhao<shannon.zhao@linaro.org>

>> >

>> >When calling perf_event_create_kernel_counter to create perf_event,

>> >assign a overflow handler. Then when the perf event overflows, set the

>> >corresponding bit of guest PMOVSSET register. If this counter is enabled

>> >and its interrupt is enabled as well, kick the vcpu to sync the

>> >interrupt.

>> >

>> >On VM entry, if there is counter overflowed, inject the interrupt with

>> >the level set to 1. Otherwise, inject the interrupt with the level set

>> >to 0.

>> >

>> >Signed-off-by: Shannon Zhao<shannon.zhao@linaro.org>

>> >Reviewed-by: Marc Zyngier<marc.zyngier@arm.com>

>> >Reviewed-by: Andrew Jones<drjones@redhat.com>

>> >---

>> >  arch/arm/kvm/arm.c    |  2 ++

>> >  include/kvm/arm_pmu.h |  2 ++

>> >  virt/kvm/arm/pmu.c    | 50 +++++++++++++++++++++++++++++++++++++++++++++++++-

>> >  3 files changed, 53 insertions(+), 1 deletion(-)

>> >

>> >diff --git a/arch/arm/kvm/arm.c b/arch/arm/kvm/arm.c

>> >index dda1959..f54264c 100644

>> >--- a/arch/arm/kvm/arm.c

>> >+++ b/arch/arm/kvm/arm.c

>> >@@ -28,6 +28,7 @@

>> >  #include <linux/sched.h>

>> >  #include <linux/kvm.h>

>> >  #include <trace/events/kvm.h>

>> >+#include <kvm/arm_pmu.h>

>> >

>> >  #define CREATE_TRACE_POINTS

>> >  #include "trace.h"

>> >@@ -577,6 +578,7 @@ int kvm_arch_vcpu_ioctl_run(struct kvm_vcpu *vcpu, struct kvm_run *run)

>> >  		 * non-preemptible context.

>> >  		 */

>> >  		preempt_disable();

>> >+		kvm_pmu_flush_hwstate(vcpu);

>> >  		kvm_timer_flush_hwstate(vcpu);

>> >  		kvm_vgic_flush_hwstate(vcpu);

>> >

>> >diff --git a/include/kvm/arm_pmu.h b/include/kvm/arm_pmu.h

>> >index 1f4bfa2..44a3c75 100644

>> >--- a/include/kvm/arm_pmu.h

>> >+++ b/include/kvm/arm_pmu.h

>> >@@ -44,6 +44,7 @@ u64 kvm_pmu_valid_counter_mask(struct kvm_vcpu *vcpu);

>> >  void kvm_pmu_disable_counter(struct kvm_vcpu *vcpu, u64 val);

>> >  void kvm_pmu_enable_counter(struct kvm_vcpu *vcpu, u64 val);

>> >  void kvm_pmu_overflow_set(struct kvm_vcpu *vcpu, u64 val);

>> >+void kvm_pmu_flush_hwstate(struct kvm_vcpu *vcpu);

>> >  void kvm_pmu_software_increment(struct kvm_vcpu *vcpu, u64 val);

>> >  void kvm_pmu_handle_pmcr(struct kvm_vcpu *vcpu, u64 val);

>> >  void kvm_pmu_set_counter_event_type(struct kvm_vcpu *vcpu, u64 data,

>> >@@ -67,6 +68,7 @@ static inline u64 kvm_pmu_valid_counter_mask(struct kvm_vcpu *vcpu)

>> >  static inline void kvm_pmu_disable_counter(struct kvm_vcpu *vcpu, u64 val) {}

>> >  static inline void kvm_pmu_enable_counter(struct kvm_vcpu *vcpu, u64 val) {}

>> >  static inline void kvm_pmu_overflow_set(struct kvm_vcpu *vcpu, u64 val) {}

>> >+static inline void kvm_pmu_flush_hwstate(struct kvm_vcpu *vcpu) {}

>> >  static inline void kvm_pmu_software_increment(struct kvm_vcpu *vcpu, u64 val) {}

>> >  static inline void kvm_pmu_handle_pmcr(struct kvm_vcpu *vcpu, u64 val) {}

>> >  static inline void kvm_pmu_set_counter_event_type(struct kvm_vcpu *vcpu,

>> >diff --git a/virt/kvm/arm/pmu.c b/virt/kvm/arm/pmu.c

>> >index c8ea825..5f983cb 100644

>> >--- a/virt/kvm/arm/pmu.c

>> >+++ b/virt/kvm/arm/pmu.c

>> >@@ -21,6 +21,7 @@

>> >  #include <linux/perf_event.h>

>> >  #include <asm/kvm_emulate.h>

>> >  #include <kvm/arm_pmu.h>

>> >+#include <kvm/arm_vgic.h>

>> >

>> >  /**

>> >   * kvm_pmu_get_counter_value - get PMU counter value

>> >@@ -180,6 +181,52 @@ void kvm_pmu_overflow_set(struct kvm_vcpu *vcpu, u64 val)

>> >  }

>> >

>> >  /**

>> >+ * kvm_pmu_flush_hwstate - flush pmu state to cpu

>> >+ * @vcpu: The vcpu pointer

>> >+ *

>> >+ * Inject virtual PMU IRQ if IRQ is pending for this cpu.

>> >+ */

>> >+void kvm_pmu_flush_hwstate(struct kvm_vcpu *vcpu)

>> >+{

>> >+	struct kvm_pmu *pmu = &vcpu->arch.pmu;

>> >+	u64 overflow;

>> >+

>> >+	if (!kvm_arm_pmu_v3_ready(vcpu))

>> >+		return;

>> >+

>> >+	if (!(vcpu_sys_reg(vcpu, PMCR_EL0) & ARMV8_PMCR_E))

>> >+		return;

> are we modeling the PMU interrupt as level-triggered?

>

Yes.
> In that case, shouldn't we lower the interrupt line on flush when

> PMCR_EL0.E == 0 ?

We inject the interrupt only based on the value of PMOVSSET. There might 
be a case that PMCR_EL0.E is 0 and PMOVSSET is not zero. In this case it 
should pull up the interrupt line.

-- 
Shannon

_______________________________________________
linux-arm-kernel mailing list
linux-arm-kernel@lists.infradead.org
http://lists.infradead.org/mailman/listinfo/linux-arm-kernel
diff mbox

Patch

diff --git a/arch/arm/kvm/arm.c b/arch/arm/kvm/arm.c
index dda1959..f54264c 100644
--- a/arch/arm/kvm/arm.c
+++ b/arch/arm/kvm/arm.c
@@ -28,6 +28,7 @@ 
 #include <linux/sched.h>
 #include <linux/kvm.h>
 #include <trace/events/kvm.h>
+#include <kvm/arm_pmu.h>
 
 #define CREATE_TRACE_POINTS
 #include "trace.h"
@@ -577,6 +578,7 @@  int kvm_arch_vcpu_ioctl_run(struct kvm_vcpu *vcpu, struct kvm_run *run)
 		 * non-preemptible context.
 		 */
 		preempt_disable();
+		kvm_pmu_flush_hwstate(vcpu);
 		kvm_timer_flush_hwstate(vcpu);
 		kvm_vgic_flush_hwstate(vcpu);
 
diff --git a/include/kvm/arm_pmu.h b/include/kvm/arm_pmu.h
index 1f4bfa2..44a3c75 100644
--- a/include/kvm/arm_pmu.h
+++ b/include/kvm/arm_pmu.h
@@ -44,6 +44,7 @@  u64 kvm_pmu_valid_counter_mask(struct kvm_vcpu *vcpu);
 void kvm_pmu_disable_counter(struct kvm_vcpu *vcpu, u64 val);
 void kvm_pmu_enable_counter(struct kvm_vcpu *vcpu, u64 val);
 void kvm_pmu_overflow_set(struct kvm_vcpu *vcpu, u64 val);
+void kvm_pmu_flush_hwstate(struct kvm_vcpu *vcpu);
 void kvm_pmu_software_increment(struct kvm_vcpu *vcpu, u64 val);
 void kvm_pmu_handle_pmcr(struct kvm_vcpu *vcpu, u64 val);
 void kvm_pmu_set_counter_event_type(struct kvm_vcpu *vcpu, u64 data,
@@ -67,6 +68,7 @@  static inline u64 kvm_pmu_valid_counter_mask(struct kvm_vcpu *vcpu)
 static inline void kvm_pmu_disable_counter(struct kvm_vcpu *vcpu, u64 val) {}
 static inline void kvm_pmu_enable_counter(struct kvm_vcpu *vcpu, u64 val) {}
 static inline void kvm_pmu_overflow_set(struct kvm_vcpu *vcpu, u64 val) {}
+static inline void kvm_pmu_flush_hwstate(struct kvm_vcpu *vcpu) {}
 static inline void kvm_pmu_software_increment(struct kvm_vcpu *vcpu, u64 val) {}
 static inline void kvm_pmu_handle_pmcr(struct kvm_vcpu *vcpu, u64 val) {}
 static inline void kvm_pmu_set_counter_event_type(struct kvm_vcpu *vcpu,
diff --git a/virt/kvm/arm/pmu.c b/virt/kvm/arm/pmu.c
index c8ea825..5f983cb 100644
--- a/virt/kvm/arm/pmu.c
+++ b/virt/kvm/arm/pmu.c
@@ -21,6 +21,7 @@ 
 #include <linux/perf_event.h>
 #include <asm/kvm_emulate.h>
 #include <kvm/arm_pmu.h>
+#include <kvm/arm_vgic.h>
 
 /**
  * kvm_pmu_get_counter_value - get PMU counter value
@@ -180,6 +181,52 @@  void kvm_pmu_overflow_set(struct kvm_vcpu *vcpu, u64 val)
 }
 
 /**
+ * kvm_pmu_flush_hwstate - flush pmu state to cpu
+ * @vcpu: The vcpu pointer
+ *
+ * Inject virtual PMU IRQ if IRQ is pending for this cpu.
+ */
+void kvm_pmu_flush_hwstate(struct kvm_vcpu *vcpu)
+{
+	struct kvm_pmu *pmu = &vcpu->arch.pmu;
+	u64 overflow;
+
+	if (!kvm_arm_pmu_v3_ready(vcpu))
+		return;
+
+	if (!(vcpu_sys_reg(vcpu, PMCR_EL0) & ARMV8_PMCR_E))
+		return;
+
+	overflow = kvm_pmu_overflow_status(vcpu);
+	kvm_vgic_inject_irq(vcpu->kvm, vcpu->vcpu_id, pmu->irq_num, !!overflow);
+}
+
+static inline struct kvm_vcpu *kvm_pmc_to_vcpu(struct kvm_pmc *pmc)
+{
+	struct kvm_pmu *pmu;
+	struct kvm_vcpu_arch *vcpu_arch;
+
+	pmc -= pmc->idx;
+	pmu = container_of(pmc, struct kvm_pmu, pmc[0]);
+	vcpu_arch = container_of(pmu, struct kvm_vcpu_arch, pmu);
+	return container_of(vcpu_arch, struct kvm_vcpu, arch);
+}
+
+/**
+ * When perf event overflows, call kvm_pmu_overflow_set to set overflow status.
+ */
+static void kvm_pmu_perf_overflow(struct perf_event *perf_event,
+				  struct perf_sample_data *data,
+				  struct pt_regs *regs)
+{
+	struct kvm_pmc *pmc = perf_event->overflow_handler_context;
+	struct kvm_vcpu *vcpu = kvm_pmc_to_vcpu(pmc);
+	int idx = pmc->idx;
+
+	kvm_pmu_overflow_set(vcpu, BIT(idx));
+}
+
+/**
  * kvm_pmu_software_increment - do software increment
  * @vcpu: The vcpu pointer
  * @val: the value guest writes to PMSWINC register
@@ -289,7 +336,8 @@  void kvm_pmu_set_counter_event_type(struct kvm_vcpu *vcpu, u64 data,
 	/* The initial sample period (overflow count) of an event. */
 	attr.sample_period = (-counter) & pmc->bitmask;
 
-	event = perf_event_create_kernel_counter(&attr, -1, current, NULL, pmc);
+	event = perf_event_create_kernel_counter(&attr, -1, current,
+						 kvm_pmu_perf_overflow, pmc);
 	if (IS_ERR(event)) {
 		pr_err_once("kvm: pmu event creation failed %ld\n",
 			    PTR_ERR(event));