e840f42a49
Russell reported that since 5.13, KVM's probing of the PMU has
started to fail on his HW. As it turns out, there is an implicit
ordering dependency between the architectural PMU probing code and
and KVM's own probing. If, due to probe ordering reasons, KVM probes
before the PMU driver, it will fail to detect the PMU and prevent it
from being advertised to guests as well as the VMM.
Obviously, this is one probing too many, and we should be able to
deal with any ordering.
Add a callback from the PMU code into KVM to advertise the registration
of a host CPU PMU, allowing for any probing order.
Fixes: 5421db1be3
("KVM: arm64: Divorce the perf code from oprofile helpers")
Reported-by: "Russell King (Oracle)" <linux@armlinux.org.uk>
Tested-by: Russell King (Oracle) <rmk+kernel@armlinux.org.uk>
Signed-off-by: Marc Zyngier <maz@kernel.org>
Link: https://lore.kernel.org/r/YUYRKVflRtUytzy5@shell.armlinux.org.uk
Cc: stable@vger.kernel.org
60 lines
1.0 KiB
C
60 lines
1.0 KiB
C
// SPDX-License-Identifier: GPL-2.0-only
|
|
/*
|
|
* Based on the x86 implementation.
|
|
*
|
|
* Copyright (C) 2012 ARM Ltd.
|
|
* Author: Marc Zyngier <marc.zyngier@arm.com>
|
|
*/
|
|
|
|
#include <linux/perf_event.h>
|
|
#include <linux/kvm_host.h>
|
|
|
|
#include <asm/kvm_emulate.h>
|
|
|
|
DEFINE_STATIC_KEY_FALSE(kvm_arm_pmu_available);
|
|
|
|
static int kvm_is_in_guest(void)
|
|
{
|
|
return kvm_get_running_vcpu() != NULL;
|
|
}
|
|
|
|
static int kvm_is_user_mode(void)
|
|
{
|
|
struct kvm_vcpu *vcpu;
|
|
|
|
vcpu = kvm_get_running_vcpu();
|
|
|
|
if (vcpu)
|
|
return !vcpu_mode_priv(vcpu);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static unsigned long kvm_get_guest_ip(void)
|
|
{
|
|
struct kvm_vcpu *vcpu;
|
|
|
|
vcpu = kvm_get_running_vcpu();
|
|
|
|
if (vcpu)
|
|
return *vcpu_pc(vcpu);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static struct perf_guest_info_callbacks kvm_guest_cbs = {
|
|
.is_in_guest = kvm_is_in_guest,
|
|
.is_user_mode = kvm_is_user_mode,
|
|
.get_guest_ip = kvm_get_guest_ip,
|
|
};
|
|
|
|
int kvm_perf_init(void)
|
|
{
|
|
return perf_register_guest_info_callbacks(&kvm_guest_cbs);
|
|
}
|
|
|
|
int kvm_perf_teardown(void)
|
|
{
|
|
return perf_unregister_guest_info_callbacks(&kvm_guest_cbs);
|
|
}
|