mirror of
https://github.com/projectacrn/acrn-hypervisor.git
synced 2025-09-24 10:17:28 +00:00
HV: Added Initial support for SEP/SOCWATCH profiling
This patch adds support to sep/socwatch profiling Adds 2 new files include/arch/x86/profiling.h and arch/x86/profiling.c which contains most of the implementation for profiling,most of the functions in profiling.c have dummy implementation and will be implemented in next patches a. cpu.c, Initial profiling setup is done as part of bsp_boot_post and cpu_secondary_post flow b. vmcall.c, New ioctl is added for performing profiling related operations in vmcall_vmexit_handler ioctl - HC_PROFILING_OPS function - hcall_profiling_ops() c. common/hypercall.c, hcall_profiling_ops() implementation. d. hv_main.c, In vcpu_thread calling profiling related functions to save vm context e. acrn_hv_defs.h, list all the profiling command types Tracked-On: projectacrn#1409 Acked-by: Eddie Dong <eddie.dong@intel.com> Signed-off-by: Chinthapally, Manisha <manisha.chinthapally@intel.com>
This commit is contained in:
committed by
wenlingz
parent
3010718d4a
commit
8ba333d275
216
hypervisor/debug/profiling.c
Normal file
216
hypervisor/debug/profiling.c
Normal file
@@ -0,0 +1,216 @@
|
||||
/*
|
||||
* Copyright (C) 2018 Intel Corporation. All rights reserved.
|
||||
*
|
||||
* SPDX-License-Identifier: BSD-3-Clause
|
||||
*/
|
||||
|
||||
#ifdef PROFILING_ON
|
||||
|
||||
#include <hypervisor.h>
|
||||
|
||||
#define ACRN_DBG_PROFILING 5U
|
||||
|
||||
#define LVT_PERFCTR_BIT_MASK 0x10000U
|
||||
|
||||
static uint32_t profiling_pmi_irq = IRQ_INVALID;
|
||||
|
||||
static void profiling_initialize_vmsw(void)
|
||||
{
|
||||
/* to be implemented */
|
||||
}
|
||||
|
||||
/*
|
||||
* Configure the PMU's for sep/socwatch profiling.
|
||||
*/
|
||||
static void profiling_initialize_pmi(void)
|
||||
{
|
||||
/* to be implemented */
|
||||
}
|
||||
|
||||
/*
|
||||
* Enable all the Performance Monitoring Control registers.
|
||||
*/
|
||||
static void profiling_enable_pmu(void)
|
||||
{
|
||||
/* to be implemented */
|
||||
}
|
||||
|
||||
/*
|
||||
* Disable all Performance Monitoring Control registers
|
||||
*/
|
||||
static void profiling_disable_pmu(void)
|
||||
{
|
||||
/* to be implemented */
|
||||
}
|
||||
|
||||
/*
|
||||
* Performs MSR operations - read, write and clear
|
||||
*/
|
||||
static void profiling_handle_msrops(void)
|
||||
{
|
||||
/* to be implemented */
|
||||
}
|
||||
|
||||
/*
|
||||
* Interrupt handler for performance monitoring interrupts
|
||||
*/
|
||||
static void profiling_pmi_handler(__unused unsigned int irq, __unused void *data)
|
||||
{
|
||||
/* to be implemented */
|
||||
}
|
||||
|
||||
/*
|
||||
* Performs MSR operations on all the CPU's
|
||||
*/
|
||||
int32_t profiling_msr_ops_all_cpus(__unused struct vm *vm, __unused uint64_t addr)
|
||||
{
|
||||
/* to be implemented
|
||||
* call to smp_call_function profiling_ipi_handler
|
||||
*/
|
||||
return 0;
|
||||
}
|
||||
|
||||
/*
|
||||
* Generate VM info list
|
||||
*/
|
||||
int32_t profiling_vm_list_info(__unused struct vm *vm, __unused uint64_t addr)
|
||||
{
|
||||
/* to be implemented */
|
||||
return 0;
|
||||
}
|
||||
|
||||
/*
|
||||
* Sep/socwatch profiling version
|
||||
*/
|
||||
int32_t profiling_get_version_info(__unused struct vm *vm, __unused uint64_t addr)
|
||||
{
|
||||
/* to be implemented */
|
||||
return 0;
|
||||
}
|
||||
|
||||
/*
|
||||
* Gets type of profiling - sep/socwatch
|
||||
*/
|
||||
int32_t profiling_get_control(__unused struct vm *vm, __unused uint64_t addr)
|
||||
{
|
||||
/* to be implemented */
|
||||
return 0;
|
||||
}
|
||||
|
||||
/*
|
||||
* Update the profiling type based on control switch
|
||||
*/
|
||||
int32_t profiling_set_control(__unused struct vm *vm, __unused uint64_t addr)
|
||||
{
|
||||
/* to be implemented */
|
||||
return 0;
|
||||
}
|
||||
|
||||
/*
|
||||
* Configure PMI on all cpus
|
||||
*/
|
||||
int32_t profiling_configure_pmi(__unused struct vm *vm, __unused uint64_t addr)
|
||||
{
|
||||
/* to be implemented
|
||||
* call to smp_call_function profiling_ipi_handler
|
||||
*/
|
||||
return 0;
|
||||
}
|
||||
|
||||
/*
|
||||
* Configure for VM-switch data on all cpus
|
||||
*/
|
||||
int32_t profiling_configure_vmsw(__unused struct vm *vm, __unused uint64_t addr)
|
||||
{
|
||||
/* to be implemented
|
||||
* call to smp_call_function profiling_ipi_handler
|
||||
*/
|
||||
return 0;
|
||||
}
|
||||
|
||||
/*
|
||||
* Get the physical cpu id
|
||||
*/
|
||||
int32_t profiling_get_pcpu_id(__unused struct vm *vm, __unused uint64_t addr)
|
||||
{
|
||||
/* to be implemented */
|
||||
return 0;
|
||||
}
|
||||
|
||||
/*
|
||||
* IPI interrupt handler function
|
||||
*/
|
||||
void profiling_ipi_handler(__unused void *data)
|
||||
{
|
||||
switch (get_cpu_var(profiling_info.ipi_cmd)) {
|
||||
case IPI_PMU_START:
|
||||
profiling_enable_pmu();
|
||||
break;
|
||||
case IPI_PMU_STOP:
|
||||
profiling_disable_pmu();
|
||||
break;
|
||||
case IPI_MSR_OP:
|
||||
profiling_handle_msrops();
|
||||
break;
|
||||
case IPI_PMU_CONFIG:
|
||||
profiling_initialize_pmi();
|
||||
break;
|
||||
case IPI_VMSW_CONFIG:
|
||||
profiling_initialize_vmsw();
|
||||
break;
|
||||
default:
|
||||
pr_err("%s: unknown IPI command %d on cpu %d",
|
||||
__func__, get_cpu_var(profiling_info.ipi_cmd), get_cpu_id());
|
||||
break;
|
||||
}
|
||||
get_cpu_var(profiling_info.ipi_cmd) = IPI_UNKNOWN;
|
||||
}
|
||||
|
||||
/*
|
||||
* Save the VCPU info on vmenter
|
||||
*/
|
||||
void profiling_vmenter_handler(__unused struct vcpu *vcpu)
|
||||
{
|
||||
/* to be implemented */
|
||||
}
|
||||
|
||||
/*
|
||||
* Save the VCPU info on vmexit
|
||||
*/
|
||||
void profiling_vmexit_handler(__unused struct vcpu *vcpu, __unused uint64_t exit_reason)
|
||||
{
|
||||
if (exit_reason == VMX_EXIT_REASON_EXTERNAL_INTERRUPT) {
|
||||
/* to be implemented */
|
||||
} else {
|
||||
/* to be implemented */
|
||||
}
|
||||
}
|
||||
|
||||
/*
|
||||
* Setup PMI irq vector
|
||||
*/
|
||||
void profiling_setup(void)
|
||||
{
|
||||
uint16_t cpu;
|
||||
int32_t retval;
|
||||
dev_dbg(ACRN_DBG_PROFILING, "%s: entering", __func__);
|
||||
cpu = get_cpu_id();
|
||||
/* support PMI notification, VM0 will register all CPU */
|
||||
if ((cpu == BOOT_CPU_ID) && (profiling_pmi_irq == IRQ_INVALID)) {
|
||||
pr_info("%s: calling request_irq", __func__);
|
||||
retval = request_irq(PMI_IRQ,
|
||||
profiling_pmi_handler, NULL, IRQF_NONE);
|
||||
if (retval < 0) {
|
||||
pr_err("Failed to add PMI isr");
|
||||
return;
|
||||
}
|
||||
profiling_pmi_irq = (uint32_t)retval;
|
||||
}
|
||||
|
||||
msr_write(MSR_IA32_EXT_APIC_LVT_PMI,
|
||||
VECTOR_PMI | LVT_PERFCTR_BIT_MASK);
|
||||
|
||||
dev_dbg(ACRN_DBG_PROFILING, "%s: exiting", __func__);
|
||||
}
|
||||
|
||||
#endif
|
Reference in New Issue
Block a user