blob: 36c9a8fa08c8cfcb708a485ea56d22971be016b4 [file] [log] [blame]
#include "x86/msr.h"
#include "x86/processor.h"
#include "x86/pmu.h"
#include "x86/desc.h"
#define N 1000000
volatile int count;
u32 lbr_from, lbr_to;
static noinline int compute_flag(int i)
{
if (i % 10 < 4)
return i + 1;
return 0;
}
static noinline int lbr_test(void)
{
int i;
int flag;
volatile double x = 1212121212, y = 121212;
for (i = 0; i < 200000000; i++) {
flag = compute_flag(i);
count++;
if (flag)
x += x / y + y / x;
}
return 0;
}
static void init_lbr(void *index)
{
wrmsr(lbr_from + *(int *) index, 0);
wrmsr(lbr_to + *(int *)index, 0);
}
static bool test_init_lbr_from_exception(u64 index)
{
return test_for_exception(GP_VECTOR, init_lbr, &index);
}
int main(int ac, char **av)
{
int max, i;
setup_vm();
if (!is_intel()) {
report_skip("PMU_LBR test is for intel CPU's only");
return report_summary();
}
if (!this_cpu_has_pmu()) {
report_skip("No pmu is detected!");
return report_summary();
}
if (!this_cpu_has(X86_FEATURE_PDCM)) {
report_skip("Perfmon/Debug Capabilities MSR isn't supported.");
return report_summary();
}
if (!pmu_lbr_version()) {
report_skip("(Architectural) LBR is not supported.");
return report_summary();
}
printf("PMU version: %d\n", pmu.version);
printf("LBR version: %ld\n", pmu_lbr_version());
/* Look for LBR from and to MSRs */
lbr_from = MSR_LBR_CORE_FROM;
lbr_to = MSR_LBR_CORE_TO;
if (test_init_lbr_from_exception(0)) {
lbr_from = MSR_LBR_NHM_FROM;
lbr_to = MSR_LBR_NHM_TO;
}
if (test_init_lbr_from_exception(0)) {
report_skip("LBR on this platform is not supported!");
return report_summary();
}
wrmsr(MSR_LBR_SELECT, 0);
wrmsr(MSR_LBR_TOS, 0);
for (max = 0; max < MAX_NUM_LBR_ENTRY; max++) {
if (test_init_lbr_from_exception(max))
break;
}
report(max > 0, "The number of guest LBR entries is good.");
/* Do some branch instructions. */
wrmsr(MSR_IA32_DEBUGCTLMSR, DEBUGCTLMSR_LBR);
lbr_test();
wrmsr(MSR_IA32_DEBUGCTLMSR, 0);
report(rdmsr(MSR_LBR_TOS) != 0, "The guest LBR MSR_LBR_TOS value is good.");
for (i = 0; i < max; ++i) {
if (!rdmsr(lbr_to + i) || !rdmsr(lbr_from + i))
break;
}
report(i == max, "The guest LBR FROM_IP/TO_IP values are good.");
return report_summary();
}