18c2ecf20Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-only 28c2ecf20Sopenharmony_ci/* 38c2ecf20Sopenharmony_ci * Copyright (C) 2015 Linaro Ltd. 48c2ecf20Sopenharmony_ci * Author: Shannon Zhao <shannon.zhao@linaro.org> 58c2ecf20Sopenharmony_ci */ 68c2ecf20Sopenharmony_ci 78c2ecf20Sopenharmony_ci#include <linux/cpu.h> 88c2ecf20Sopenharmony_ci#include <linux/kvm.h> 98c2ecf20Sopenharmony_ci#include <linux/kvm_host.h> 108c2ecf20Sopenharmony_ci#include <linux/perf_event.h> 118c2ecf20Sopenharmony_ci#include <linux/perf/arm_pmu.h> 128c2ecf20Sopenharmony_ci#include <linux/uaccess.h> 138c2ecf20Sopenharmony_ci#include <asm/kvm_emulate.h> 148c2ecf20Sopenharmony_ci#include <kvm/arm_pmu.h> 158c2ecf20Sopenharmony_ci#include <kvm/arm_vgic.h> 168c2ecf20Sopenharmony_ci 178c2ecf20Sopenharmony_cistatic void kvm_pmu_create_perf_event(struct kvm_vcpu *vcpu, u64 select_idx); 188c2ecf20Sopenharmony_cistatic void kvm_pmu_update_pmc_chained(struct kvm_vcpu *vcpu, u64 select_idx); 198c2ecf20Sopenharmony_cistatic void kvm_pmu_stop_counter(struct kvm_vcpu *vcpu, struct kvm_pmc *pmc); 208c2ecf20Sopenharmony_ci 218c2ecf20Sopenharmony_ci#define PERF_ATTR_CFG1_KVM_PMU_CHAINED 0x1 228c2ecf20Sopenharmony_ci 238c2ecf20Sopenharmony_cistatic u32 kvm_pmu_event_mask(struct kvm *kvm) 248c2ecf20Sopenharmony_ci{ 258c2ecf20Sopenharmony_ci switch (kvm->arch.pmuver) { 268c2ecf20Sopenharmony_ci case 1: /* ARMv8.0 */ 278c2ecf20Sopenharmony_ci return GENMASK(9, 0); 288c2ecf20Sopenharmony_ci case 4: /* ARMv8.1 */ 298c2ecf20Sopenharmony_ci case 5: /* ARMv8.4 */ 308c2ecf20Sopenharmony_ci case 6: /* ARMv8.5 */ 318c2ecf20Sopenharmony_ci return GENMASK(15, 0); 328c2ecf20Sopenharmony_ci default: /* Shouldn't be here, just for sanity */ 338c2ecf20Sopenharmony_ci WARN_ONCE(1, "Unknown PMU version %d\n", kvm->arch.pmuver); 348c2ecf20Sopenharmony_ci return 0; 358c2ecf20Sopenharmony_ci } 368c2ecf20Sopenharmony_ci} 378c2ecf20Sopenharmony_ci 388c2ecf20Sopenharmony_ci/** 398c2ecf20Sopenharmony_ci * kvm_pmu_idx_is_64bit - determine if select_idx is a 64bit counter 408c2ecf20Sopenharmony_ci * @vcpu: The vcpu pointer 418c2ecf20Sopenharmony_ci * @select_idx: The counter index 428c2ecf20Sopenharmony_ci */ 438c2ecf20Sopenharmony_cistatic bool kvm_pmu_idx_is_64bit(struct kvm_vcpu *vcpu, u64 select_idx) 448c2ecf20Sopenharmony_ci{ 458c2ecf20Sopenharmony_ci return (select_idx == ARMV8_PMU_CYCLE_IDX && 468c2ecf20Sopenharmony_ci __vcpu_sys_reg(vcpu, PMCR_EL0) & ARMV8_PMU_PMCR_LC); 478c2ecf20Sopenharmony_ci} 488c2ecf20Sopenharmony_ci 498c2ecf20Sopenharmony_cistatic struct kvm_vcpu *kvm_pmc_to_vcpu(struct kvm_pmc *pmc) 508c2ecf20Sopenharmony_ci{ 518c2ecf20Sopenharmony_ci struct kvm_pmu *pmu; 528c2ecf20Sopenharmony_ci struct kvm_vcpu_arch *vcpu_arch; 538c2ecf20Sopenharmony_ci 548c2ecf20Sopenharmony_ci pmc -= pmc->idx; 558c2ecf20Sopenharmony_ci pmu = container_of(pmc, struct kvm_pmu, pmc[0]); 568c2ecf20Sopenharmony_ci vcpu_arch = container_of(pmu, struct kvm_vcpu_arch, pmu); 578c2ecf20Sopenharmony_ci return container_of(vcpu_arch, struct kvm_vcpu, arch); 588c2ecf20Sopenharmony_ci} 598c2ecf20Sopenharmony_ci 608c2ecf20Sopenharmony_ci/** 618c2ecf20Sopenharmony_ci * kvm_pmu_pmc_is_chained - determine if the pmc is chained 628c2ecf20Sopenharmony_ci * @pmc: The PMU counter pointer 638c2ecf20Sopenharmony_ci */ 648c2ecf20Sopenharmony_cistatic bool kvm_pmu_pmc_is_chained(struct kvm_pmc *pmc) 658c2ecf20Sopenharmony_ci{ 668c2ecf20Sopenharmony_ci struct kvm_vcpu *vcpu = kvm_pmc_to_vcpu(pmc); 678c2ecf20Sopenharmony_ci 688c2ecf20Sopenharmony_ci return test_bit(pmc->idx >> 1, vcpu->arch.pmu.chained); 698c2ecf20Sopenharmony_ci} 708c2ecf20Sopenharmony_ci 718c2ecf20Sopenharmony_ci/** 728c2ecf20Sopenharmony_ci * kvm_pmu_idx_is_high_counter - determine if select_idx is a high/low counter 738c2ecf20Sopenharmony_ci * @select_idx: The counter index 748c2ecf20Sopenharmony_ci */ 758c2ecf20Sopenharmony_cistatic bool kvm_pmu_idx_is_high_counter(u64 select_idx) 768c2ecf20Sopenharmony_ci{ 778c2ecf20Sopenharmony_ci return select_idx & 0x1; 788c2ecf20Sopenharmony_ci} 798c2ecf20Sopenharmony_ci 808c2ecf20Sopenharmony_ci/** 818c2ecf20Sopenharmony_ci * kvm_pmu_get_canonical_pmc - obtain the canonical pmc 828c2ecf20Sopenharmony_ci * @pmc: The PMU counter pointer 838c2ecf20Sopenharmony_ci * 848c2ecf20Sopenharmony_ci * When a pair of PMCs are chained together we use the low counter (canonical) 858c2ecf20Sopenharmony_ci * to hold the underlying perf event. 868c2ecf20Sopenharmony_ci */ 878c2ecf20Sopenharmony_cistatic struct kvm_pmc *kvm_pmu_get_canonical_pmc(struct kvm_pmc *pmc) 888c2ecf20Sopenharmony_ci{ 898c2ecf20Sopenharmony_ci if (kvm_pmu_pmc_is_chained(pmc) && 908c2ecf20Sopenharmony_ci kvm_pmu_idx_is_high_counter(pmc->idx)) 918c2ecf20Sopenharmony_ci return pmc - 1; 928c2ecf20Sopenharmony_ci 938c2ecf20Sopenharmony_ci return pmc; 948c2ecf20Sopenharmony_ci} 958c2ecf20Sopenharmony_cistatic struct kvm_pmc *kvm_pmu_get_alternate_pmc(struct kvm_pmc *pmc) 968c2ecf20Sopenharmony_ci{ 978c2ecf20Sopenharmony_ci if (kvm_pmu_idx_is_high_counter(pmc->idx)) 988c2ecf20Sopenharmony_ci return pmc - 1; 998c2ecf20Sopenharmony_ci else 1008c2ecf20Sopenharmony_ci return pmc + 1; 1018c2ecf20Sopenharmony_ci} 1028c2ecf20Sopenharmony_ci 1038c2ecf20Sopenharmony_ci/** 1048c2ecf20Sopenharmony_ci * kvm_pmu_idx_has_chain_evtype - determine if the event type is chain 1058c2ecf20Sopenharmony_ci * @vcpu: The vcpu pointer 1068c2ecf20Sopenharmony_ci * @select_idx: The counter index 1078c2ecf20Sopenharmony_ci */ 1088c2ecf20Sopenharmony_cistatic bool kvm_pmu_idx_has_chain_evtype(struct kvm_vcpu *vcpu, u64 select_idx) 1098c2ecf20Sopenharmony_ci{ 1108c2ecf20Sopenharmony_ci u64 eventsel, reg; 1118c2ecf20Sopenharmony_ci 1128c2ecf20Sopenharmony_ci select_idx |= 0x1; 1138c2ecf20Sopenharmony_ci 1148c2ecf20Sopenharmony_ci if (select_idx == ARMV8_PMU_CYCLE_IDX) 1158c2ecf20Sopenharmony_ci return false; 1168c2ecf20Sopenharmony_ci 1178c2ecf20Sopenharmony_ci reg = PMEVTYPER0_EL0 + select_idx; 1188c2ecf20Sopenharmony_ci eventsel = __vcpu_sys_reg(vcpu, reg) & kvm_pmu_event_mask(vcpu->kvm); 1198c2ecf20Sopenharmony_ci 1208c2ecf20Sopenharmony_ci return eventsel == ARMV8_PMUV3_PERFCTR_CHAIN; 1218c2ecf20Sopenharmony_ci} 1228c2ecf20Sopenharmony_ci 1238c2ecf20Sopenharmony_ci/** 1248c2ecf20Sopenharmony_ci * kvm_pmu_get_pair_counter_value - get PMU counter value 1258c2ecf20Sopenharmony_ci * @vcpu: The vcpu pointer 1268c2ecf20Sopenharmony_ci * @pmc: The PMU counter pointer 1278c2ecf20Sopenharmony_ci */ 1288c2ecf20Sopenharmony_cistatic u64 kvm_pmu_get_pair_counter_value(struct kvm_vcpu *vcpu, 1298c2ecf20Sopenharmony_ci struct kvm_pmc *pmc) 1308c2ecf20Sopenharmony_ci{ 1318c2ecf20Sopenharmony_ci u64 counter, counter_high, reg, enabled, running; 1328c2ecf20Sopenharmony_ci 1338c2ecf20Sopenharmony_ci if (kvm_pmu_pmc_is_chained(pmc)) { 1348c2ecf20Sopenharmony_ci pmc = kvm_pmu_get_canonical_pmc(pmc); 1358c2ecf20Sopenharmony_ci reg = PMEVCNTR0_EL0 + pmc->idx; 1368c2ecf20Sopenharmony_ci 1378c2ecf20Sopenharmony_ci counter = __vcpu_sys_reg(vcpu, reg); 1388c2ecf20Sopenharmony_ci counter_high = __vcpu_sys_reg(vcpu, reg + 1); 1398c2ecf20Sopenharmony_ci 1408c2ecf20Sopenharmony_ci counter = lower_32_bits(counter) | (counter_high << 32); 1418c2ecf20Sopenharmony_ci } else { 1428c2ecf20Sopenharmony_ci reg = (pmc->idx == ARMV8_PMU_CYCLE_IDX) 1438c2ecf20Sopenharmony_ci ? PMCCNTR_EL0 : PMEVCNTR0_EL0 + pmc->idx; 1448c2ecf20Sopenharmony_ci counter = __vcpu_sys_reg(vcpu, reg); 1458c2ecf20Sopenharmony_ci } 1468c2ecf20Sopenharmony_ci 1478c2ecf20Sopenharmony_ci /* 1488c2ecf20Sopenharmony_ci * The real counter value is equal to the value of counter register plus 1498c2ecf20Sopenharmony_ci * the value perf event counts. 1508c2ecf20Sopenharmony_ci */ 1518c2ecf20Sopenharmony_ci if (pmc->perf_event) 1528c2ecf20Sopenharmony_ci counter += perf_event_read_value(pmc->perf_event, &enabled, 1538c2ecf20Sopenharmony_ci &running); 1548c2ecf20Sopenharmony_ci 1558c2ecf20Sopenharmony_ci return counter; 1568c2ecf20Sopenharmony_ci} 1578c2ecf20Sopenharmony_ci 1588c2ecf20Sopenharmony_ci/** 1598c2ecf20Sopenharmony_ci * kvm_pmu_get_counter_value - get PMU counter value 1608c2ecf20Sopenharmony_ci * @vcpu: The vcpu pointer 1618c2ecf20Sopenharmony_ci * @select_idx: The counter index 1628c2ecf20Sopenharmony_ci */ 1638c2ecf20Sopenharmony_ciu64 kvm_pmu_get_counter_value(struct kvm_vcpu *vcpu, u64 select_idx) 1648c2ecf20Sopenharmony_ci{ 1658c2ecf20Sopenharmony_ci u64 counter; 1668c2ecf20Sopenharmony_ci struct kvm_pmu *pmu = &vcpu->arch.pmu; 1678c2ecf20Sopenharmony_ci struct kvm_pmc *pmc = &pmu->pmc[select_idx]; 1688c2ecf20Sopenharmony_ci 1698c2ecf20Sopenharmony_ci counter = kvm_pmu_get_pair_counter_value(vcpu, pmc); 1708c2ecf20Sopenharmony_ci 1718c2ecf20Sopenharmony_ci if (kvm_pmu_pmc_is_chained(pmc) && 1728c2ecf20Sopenharmony_ci kvm_pmu_idx_is_high_counter(select_idx)) 1738c2ecf20Sopenharmony_ci counter = upper_32_bits(counter); 1748c2ecf20Sopenharmony_ci else if (select_idx != ARMV8_PMU_CYCLE_IDX) 1758c2ecf20Sopenharmony_ci counter = lower_32_bits(counter); 1768c2ecf20Sopenharmony_ci 1778c2ecf20Sopenharmony_ci return counter; 1788c2ecf20Sopenharmony_ci} 1798c2ecf20Sopenharmony_ci 1808c2ecf20Sopenharmony_ci/** 1818c2ecf20Sopenharmony_ci * kvm_pmu_set_counter_value - set PMU counter value 1828c2ecf20Sopenharmony_ci * @vcpu: The vcpu pointer 1838c2ecf20Sopenharmony_ci * @select_idx: The counter index 1848c2ecf20Sopenharmony_ci * @val: The counter value 1858c2ecf20Sopenharmony_ci */ 1868c2ecf20Sopenharmony_civoid kvm_pmu_set_counter_value(struct kvm_vcpu *vcpu, u64 select_idx, u64 val) 1878c2ecf20Sopenharmony_ci{ 1888c2ecf20Sopenharmony_ci u64 reg; 1898c2ecf20Sopenharmony_ci 1908c2ecf20Sopenharmony_ci reg = (select_idx == ARMV8_PMU_CYCLE_IDX) 1918c2ecf20Sopenharmony_ci ? PMCCNTR_EL0 : PMEVCNTR0_EL0 + select_idx; 1928c2ecf20Sopenharmony_ci __vcpu_sys_reg(vcpu, reg) += (s64)val - kvm_pmu_get_counter_value(vcpu, select_idx); 1938c2ecf20Sopenharmony_ci 1948c2ecf20Sopenharmony_ci /* Recreate the perf event to reflect the updated sample_period */ 1958c2ecf20Sopenharmony_ci kvm_pmu_create_perf_event(vcpu, select_idx); 1968c2ecf20Sopenharmony_ci} 1978c2ecf20Sopenharmony_ci 1988c2ecf20Sopenharmony_ci/** 1998c2ecf20Sopenharmony_ci * kvm_pmu_release_perf_event - remove the perf event 2008c2ecf20Sopenharmony_ci * @pmc: The PMU counter pointer 2018c2ecf20Sopenharmony_ci */ 2028c2ecf20Sopenharmony_cistatic void kvm_pmu_release_perf_event(struct kvm_pmc *pmc) 2038c2ecf20Sopenharmony_ci{ 2048c2ecf20Sopenharmony_ci pmc = kvm_pmu_get_canonical_pmc(pmc); 2058c2ecf20Sopenharmony_ci if (pmc->perf_event) { 2068c2ecf20Sopenharmony_ci perf_event_disable(pmc->perf_event); 2078c2ecf20Sopenharmony_ci perf_event_release_kernel(pmc->perf_event); 2088c2ecf20Sopenharmony_ci pmc->perf_event = NULL; 2098c2ecf20Sopenharmony_ci } 2108c2ecf20Sopenharmony_ci} 2118c2ecf20Sopenharmony_ci 2128c2ecf20Sopenharmony_ci/** 2138c2ecf20Sopenharmony_ci * kvm_pmu_stop_counter - stop PMU counter 2148c2ecf20Sopenharmony_ci * @pmc: The PMU counter pointer 2158c2ecf20Sopenharmony_ci * 2168c2ecf20Sopenharmony_ci * If this counter has been configured to monitor some event, release it here. 2178c2ecf20Sopenharmony_ci */ 2188c2ecf20Sopenharmony_cistatic void kvm_pmu_stop_counter(struct kvm_vcpu *vcpu, struct kvm_pmc *pmc) 2198c2ecf20Sopenharmony_ci{ 2208c2ecf20Sopenharmony_ci u64 counter, reg, val; 2218c2ecf20Sopenharmony_ci 2228c2ecf20Sopenharmony_ci pmc = kvm_pmu_get_canonical_pmc(pmc); 2238c2ecf20Sopenharmony_ci if (!pmc->perf_event) 2248c2ecf20Sopenharmony_ci return; 2258c2ecf20Sopenharmony_ci 2268c2ecf20Sopenharmony_ci counter = kvm_pmu_get_pair_counter_value(vcpu, pmc); 2278c2ecf20Sopenharmony_ci 2288c2ecf20Sopenharmony_ci if (pmc->idx == ARMV8_PMU_CYCLE_IDX) { 2298c2ecf20Sopenharmony_ci reg = PMCCNTR_EL0; 2308c2ecf20Sopenharmony_ci val = counter; 2318c2ecf20Sopenharmony_ci } else { 2328c2ecf20Sopenharmony_ci reg = PMEVCNTR0_EL0 + pmc->idx; 2338c2ecf20Sopenharmony_ci val = lower_32_bits(counter); 2348c2ecf20Sopenharmony_ci } 2358c2ecf20Sopenharmony_ci 2368c2ecf20Sopenharmony_ci __vcpu_sys_reg(vcpu, reg) = val; 2378c2ecf20Sopenharmony_ci 2388c2ecf20Sopenharmony_ci if (kvm_pmu_pmc_is_chained(pmc)) 2398c2ecf20Sopenharmony_ci __vcpu_sys_reg(vcpu, reg + 1) = upper_32_bits(counter); 2408c2ecf20Sopenharmony_ci 2418c2ecf20Sopenharmony_ci kvm_pmu_release_perf_event(pmc); 2428c2ecf20Sopenharmony_ci} 2438c2ecf20Sopenharmony_ci 2448c2ecf20Sopenharmony_ci/** 2458c2ecf20Sopenharmony_ci * kvm_pmu_vcpu_init - assign pmu counter idx for cpu 2468c2ecf20Sopenharmony_ci * @vcpu: The vcpu pointer 2478c2ecf20Sopenharmony_ci * 2488c2ecf20Sopenharmony_ci */ 2498c2ecf20Sopenharmony_civoid kvm_pmu_vcpu_init(struct kvm_vcpu *vcpu) 2508c2ecf20Sopenharmony_ci{ 2518c2ecf20Sopenharmony_ci int i; 2528c2ecf20Sopenharmony_ci struct kvm_pmu *pmu = &vcpu->arch.pmu; 2538c2ecf20Sopenharmony_ci 2548c2ecf20Sopenharmony_ci for (i = 0; i < ARMV8_PMU_MAX_COUNTERS; i++) 2558c2ecf20Sopenharmony_ci pmu->pmc[i].idx = i; 2568c2ecf20Sopenharmony_ci} 2578c2ecf20Sopenharmony_ci 2588c2ecf20Sopenharmony_ci/** 2598c2ecf20Sopenharmony_ci * kvm_pmu_vcpu_reset - reset pmu state for cpu 2608c2ecf20Sopenharmony_ci * @vcpu: The vcpu pointer 2618c2ecf20Sopenharmony_ci * 2628c2ecf20Sopenharmony_ci */ 2638c2ecf20Sopenharmony_civoid kvm_pmu_vcpu_reset(struct kvm_vcpu *vcpu) 2648c2ecf20Sopenharmony_ci{ 2658c2ecf20Sopenharmony_ci unsigned long mask = kvm_pmu_valid_counter_mask(vcpu); 2668c2ecf20Sopenharmony_ci struct kvm_pmu *pmu = &vcpu->arch.pmu; 2678c2ecf20Sopenharmony_ci int i; 2688c2ecf20Sopenharmony_ci 2698c2ecf20Sopenharmony_ci for_each_set_bit(i, &mask, 32) 2708c2ecf20Sopenharmony_ci kvm_pmu_stop_counter(vcpu, &pmu->pmc[i]); 2718c2ecf20Sopenharmony_ci 2728c2ecf20Sopenharmony_ci bitmap_zero(vcpu->arch.pmu.chained, ARMV8_PMU_MAX_COUNTER_PAIRS); 2738c2ecf20Sopenharmony_ci} 2748c2ecf20Sopenharmony_ci 2758c2ecf20Sopenharmony_ci/** 2768c2ecf20Sopenharmony_ci * kvm_pmu_vcpu_destroy - free perf event of PMU for cpu 2778c2ecf20Sopenharmony_ci * @vcpu: The vcpu pointer 2788c2ecf20Sopenharmony_ci * 2798c2ecf20Sopenharmony_ci */ 2808c2ecf20Sopenharmony_civoid kvm_pmu_vcpu_destroy(struct kvm_vcpu *vcpu) 2818c2ecf20Sopenharmony_ci{ 2828c2ecf20Sopenharmony_ci int i; 2838c2ecf20Sopenharmony_ci struct kvm_pmu *pmu = &vcpu->arch.pmu; 2848c2ecf20Sopenharmony_ci 2858c2ecf20Sopenharmony_ci for (i = 0; i < ARMV8_PMU_MAX_COUNTERS; i++) 2868c2ecf20Sopenharmony_ci kvm_pmu_release_perf_event(&pmu->pmc[i]); 2878c2ecf20Sopenharmony_ci irq_work_sync(&vcpu->arch.pmu.overflow_work); 2888c2ecf20Sopenharmony_ci} 2898c2ecf20Sopenharmony_ci 2908c2ecf20Sopenharmony_ciu64 kvm_pmu_valid_counter_mask(struct kvm_vcpu *vcpu) 2918c2ecf20Sopenharmony_ci{ 2928c2ecf20Sopenharmony_ci u64 val = __vcpu_sys_reg(vcpu, PMCR_EL0) >> ARMV8_PMU_PMCR_N_SHIFT; 2938c2ecf20Sopenharmony_ci 2948c2ecf20Sopenharmony_ci val &= ARMV8_PMU_PMCR_N_MASK; 2958c2ecf20Sopenharmony_ci if (val == 0) 2968c2ecf20Sopenharmony_ci return BIT(ARMV8_PMU_CYCLE_IDX); 2978c2ecf20Sopenharmony_ci else 2988c2ecf20Sopenharmony_ci return GENMASK(val - 1, 0) | BIT(ARMV8_PMU_CYCLE_IDX); 2998c2ecf20Sopenharmony_ci} 3008c2ecf20Sopenharmony_ci 3018c2ecf20Sopenharmony_ci/** 3028c2ecf20Sopenharmony_ci * kvm_pmu_enable_counter_mask - enable selected PMU counters 3038c2ecf20Sopenharmony_ci * @vcpu: The vcpu pointer 3048c2ecf20Sopenharmony_ci * @val: the value guest writes to PMCNTENSET register 3058c2ecf20Sopenharmony_ci * 3068c2ecf20Sopenharmony_ci * Call perf_event_enable to start counting the perf event 3078c2ecf20Sopenharmony_ci */ 3088c2ecf20Sopenharmony_civoid kvm_pmu_enable_counter_mask(struct kvm_vcpu *vcpu, u64 val) 3098c2ecf20Sopenharmony_ci{ 3108c2ecf20Sopenharmony_ci int i; 3118c2ecf20Sopenharmony_ci struct kvm_pmu *pmu = &vcpu->arch.pmu; 3128c2ecf20Sopenharmony_ci struct kvm_pmc *pmc; 3138c2ecf20Sopenharmony_ci 3148c2ecf20Sopenharmony_ci if (!(__vcpu_sys_reg(vcpu, PMCR_EL0) & ARMV8_PMU_PMCR_E) || !val) 3158c2ecf20Sopenharmony_ci return; 3168c2ecf20Sopenharmony_ci 3178c2ecf20Sopenharmony_ci for (i = 0; i < ARMV8_PMU_MAX_COUNTERS; i++) { 3188c2ecf20Sopenharmony_ci if (!(val & BIT(i))) 3198c2ecf20Sopenharmony_ci continue; 3208c2ecf20Sopenharmony_ci 3218c2ecf20Sopenharmony_ci pmc = &pmu->pmc[i]; 3228c2ecf20Sopenharmony_ci 3238c2ecf20Sopenharmony_ci /* A change in the enable state may affect the chain state */ 3248c2ecf20Sopenharmony_ci kvm_pmu_update_pmc_chained(vcpu, i); 3258c2ecf20Sopenharmony_ci kvm_pmu_create_perf_event(vcpu, i); 3268c2ecf20Sopenharmony_ci 3278c2ecf20Sopenharmony_ci /* At this point, pmc must be the canonical */ 3288c2ecf20Sopenharmony_ci if (pmc->perf_event) { 3298c2ecf20Sopenharmony_ci perf_event_enable(pmc->perf_event); 3308c2ecf20Sopenharmony_ci if (pmc->perf_event->state != PERF_EVENT_STATE_ACTIVE) 3318c2ecf20Sopenharmony_ci kvm_debug("fail to enable perf event\n"); 3328c2ecf20Sopenharmony_ci } 3338c2ecf20Sopenharmony_ci } 3348c2ecf20Sopenharmony_ci} 3358c2ecf20Sopenharmony_ci 3368c2ecf20Sopenharmony_ci/** 3378c2ecf20Sopenharmony_ci * kvm_pmu_disable_counter_mask - disable selected PMU counters 3388c2ecf20Sopenharmony_ci * @vcpu: The vcpu pointer 3398c2ecf20Sopenharmony_ci * @val: the value guest writes to PMCNTENCLR register 3408c2ecf20Sopenharmony_ci * 3418c2ecf20Sopenharmony_ci * Call perf_event_disable to stop counting the perf event 3428c2ecf20Sopenharmony_ci */ 3438c2ecf20Sopenharmony_civoid kvm_pmu_disable_counter_mask(struct kvm_vcpu *vcpu, u64 val) 3448c2ecf20Sopenharmony_ci{ 3458c2ecf20Sopenharmony_ci int i; 3468c2ecf20Sopenharmony_ci struct kvm_pmu *pmu = &vcpu->arch.pmu; 3478c2ecf20Sopenharmony_ci struct kvm_pmc *pmc; 3488c2ecf20Sopenharmony_ci 3498c2ecf20Sopenharmony_ci if (!val) 3508c2ecf20Sopenharmony_ci return; 3518c2ecf20Sopenharmony_ci 3528c2ecf20Sopenharmony_ci for (i = 0; i < ARMV8_PMU_MAX_COUNTERS; i++) { 3538c2ecf20Sopenharmony_ci if (!(val & BIT(i))) 3548c2ecf20Sopenharmony_ci continue; 3558c2ecf20Sopenharmony_ci 3568c2ecf20Sopenharmony_ci pmc = &pmu->pmc[i]; 3578c2ecf20Sopenharmony_ci 3588c2ecf20Sopenharmony_ci /* A change in the enable state may affect the chain state */ 3598c2ecf20Sopenharmony_ci kvm_pmu_update_pmc_chained(vcpu, i); 3608c2ecf20Sopenharmony_ci kvm_pmu_create_perf_event(vcpu, i); 3618c2ecf20Sopenharmony_ci 3628c2ecf20Sopenharmony_ci /* At this point, pmc must be the canonical */ 3638c2ecf20Sopenharmony_ci if (pmc->perf_event) 3648c2ecf20Sopenharmony_ci perf_event_disable(pmc->perf_event); 3658c2ecf20Sopenharmony_ci } 3668c2ecf20Sopenharmony_ci} 3678c2ecf20Sopenharmony_ci 3688c2ecf20Sopenharmony_cistatic u64 kvm_pmu_overflow_status(struct kvm_vcpu *vcpu) 3698c2ecf20Sopenharmony_ci{ 3708c2ecf20Sopenharmony_ci u64 reg = 0; 3718c2ecf20Sopenharmony_ci 3728c2ecf20Sopenharmony_ci if ((__vcpu_sys_reg(vcpu, PMCR_EL0) & ARMV8_PMU_PMCR_E)) { 3738c2ecf20Sopenharmony_ci reg = __vcpu_sys_reg(vcpu, PMOVSSET_EL0); 3748c2ecf20Sopenharmony_ci reg &= __vcpu_sys_reg(vcpu, PMCNTENSET_EL0); 3758c2ecf20Sopenharmony_ci reg &= __vcpu_sys_reg(vcpu, PMINTENSET_EL1); 3768c2ecf20Sopenharmony_ci reg &= kvm_pmu_valid_counter_mask(vcpu); 3778c2ecf20Sopenharmony_ci } 3788c2ecf20Sopenharmony_ci 3798c2ecf20Sopenharmony_ci return reg; 3808c2ecf20Sopenharmony_ci} 3818c2ecf20Sopenharmony_ci 3828c2ecf20Sopenharmony_cistatic void kvm_pmu_update_state(struct kvm_vcpu *vcpu) 3838c2ecf20Sopenharmony_ci{ 3848c2ecf20Sopenharmony_ci struct kvm_pmu *pmu = &vcpu->arch.pmu; 3858c2ecf20Sopenharmony_ci bool overflow; 3868c2ecf20Sopenharmony_ci 3878c2ecf20Sopenharmony_ci if (!kvm_arm_pmu_v3_ready(vcpu)) 3888c2ecf20Sopenharmony_ci return; 3898c2ecf20Sopenharmony_ci 3908c2ecf20Sopenharmony_ci overflow = !!kvm_pmu_overflow_status(vcpu); 3918c2ecf20Sopenharmony_ci if (pmu->irq_level == overflow) 3928c2ecf20Sopenharmony_ci return; 3938c2ecf20Sopenharmony_ci 3948c2ecf20Sopenharmony_ci pmu->irq_level = overflow; 3958c2ecf20Sopenharmony_ci 3968c2ecf20Sopenharmony_ci if (likely(irqchip_in_kernel(vcpu->kvm))) { 3978c2ecf20Sopenharmony_ci int ret = kvm_vgic_inject_irq(vcpu->kvm, vcpu->vcpu_id, 3988c2ecf20Sopenharmony_ci pmu->irq_num, overflow, pmu); 3998c2ecf20Sopenharmony_ci WARN_ON(ret); 4008c2ecf20Sopenharmony_ci } 4018c2ecf20Sopenharmony_ci} 4028c2ecf20Sopenharmony_ci 4038c2ecf20Sopenharmony_cibool kvm_pmu_should_notify_user(struct kvm_vcpu *vcpu) 4048c2ecf20Sopenharmony_ci{ 4058c2ecf20Sopenharmony_ci struct kvm_pmu *pmu = &vcpu->arch.pmu; 4068c2ecf20Sopenharmony_ci struct kvm_sync_regs *sregs = &vcpu->run->s.regs; 4078c2ecf20Sopenharmony_ci bool run_level = sregs->device_irq_level & KVM_ARM_DEV_PMU; 4088c2ecf20Sopenharmony_ci 4098c2ecf20Sopenharmony_ci if (likely(irqchip_in_kernel(vcpu->kvm))) 4108c2ecf20Sopenharmony_ci return false; 4118c2ecf20Sopenharmony_ci 4128c2ecf20Sopenharmony_ci return pmu->irq_level != run_level; 4138c2ecf20Sopenharmony_ci} 4148c2ecf20Sopenharmony_ci 4158c2ecf20Sopenharmony_ci/* 4168c2ecf20Sopenharmony_ci * Reflect the PMU overflow interrupt output level into the kvm_run structure 4178c2ecf20Sopenharmony_ci */ 4188c2ecf20Sopenharmony_civoid kvm_pmu_update_run(struct kvm_vcpu *vcpu) 4198c2ecf20Sopenharmony_ci{ 4208c2ecf20Sopenharmony_ci struct kvm_sync_regs *regs = &vcpu->run->s.regs; 4218c2ecf20Sopenharmony_ci 4228c2ecf20Sopenharmony_ci /* Populate the timer bitmap for user space */ 4238c2ecf20Sopenharmony_ci regs->device_irq_level &= ~KVM_ARM_DEV_PMU; 4248c2ecf20Sopenharmony_ci if (vcpu->arch.pmu.irq_level) 4258c2ecf20Sopenharmony_ci regs->device_irq_level |= KVM_ARM_DEV_PMU; 4268c2ecf20Sopenharmony_ci} 4278c2ecf20Sopenharmony_ci 4288c2ecf20Sopenharmony_ci/** 4298c2ecf20Sopenharmony_ci * kvm_pmu_flush_hwstate - flush pmu state to cpu 4308c2ecf20Sopenharmony_ci * @vcpu: The vcpu pointer 4318c2ecf20Sopenharmony_ci * 4328c2ecf20Sopenharmony_ci * Check if the PMU has overflowed while we were running in the host, and inject 4338c2ecf20Sopenharmony_ci * an interrupt if that was the case. 4348c2ecf20Sopenharmony_ci */ 4358c2ecf20Sopenharmony_civoid kvm_pmu_flush_hwstate(struct kvm_vcpu *vcpu) 4368c2ecf20Sopenharmony_ci{ 4378c2ecf20Sopenharmony_ci kvm_pmu_update_state(vcpu); 4388c2ecf20Sopenharmony_ci} 4398c2ecf20Sopenharmony_ci 4408c2ecf20Sopenharmony_ci/** 4418c2ecf20Sopenharmony_ci * kvm_pmu_sync_hwstate - sync pmu state from cpu 4428c2ecf20Sopenharmony_ci * @vcpu: The vcpu pointer 4438c2ecf20Sopenharmony_ci * 4448c2ecf20Sopenharmony_ci * Check if the PMU has overflowed while we were running in the guest, and 4458c2ecf20Sopenharmony_ci * inject an interrupt if that was the case. 4468c2ecf20Sopenharmony_ci */ 4478c2ecf20Sopenharmony_civoid kvm_pmu_sync_hwstate(struct kvm_vcpu *vcpu) 4488c2ecf20Sopenharmony_ci{ 4498c2ecf20Sopenharmony_ci kvm_pmu_update_state(vcpu); 4508c2ecf20Sopenharmony_ci} 4518c2ecf20Sopenharmony_ci 4528c2ecf20Sopenharmony_ci/** 4538c2ecf20Sopenharmony_ci * When perf interrupt is an NMI, we cannot safely notify the vcpu corresponding 4548c2ecf20Sopenharmony_ci * to the event. 4558c2ecf20Sopenharmony_ci * This is why we need a callback to do it once outside of the NMI context. 4568c2ecf20Sopenharmony_ci */ 4578c2ecf20Sopenharmony_cistatic void kvm_pmu_perf_overflow_notify_vcpu(struct irq_work *work) 4588c2ecf20Sopenharmony_ci{ 4598c2ecf20Sopenharmony_ci struct kvm_vcpu *vcpu; 4608c2ecf20Sopenharmony_ci struct kvm_pmu *pmu; 4618c2ecf20Sopenharmony_ci 4628c2ecf20Sopenharmony_ci pmu = container_of(work, struct kvm_pmu, overflow_work); 4638c2ecf20Sopenharmony_ci vcpu = kvm_pmc_to_vcpu(pmu->pmc); 4648c2ecf20Sopenharmony_ci 4658c2ecf20Sopenharmony_ci kvm_vcpu_kick(vcpu); 4668c2ecf20Sopenharmony_ci} 4678c2ecf20Sopenharmony_ci 4688c2ecf20Sopenharmony_ci/** 4698c2ecf20Sopenharmony_ci * When the perf event overflows, set the overflow status and inform the vcpu. 4708c2ecf20Sopenharmony_ci */ 4718c2ecf20Sopenharmony_cistatic void kvm_pmu_perf_overflow(struct perf_event *perf_event, 4728c2ecf20Sopenharmony_ci struct perf_sample_data *data, 4738c2ecf20Sopenharmony_ci struct pt_regs *regs) 4748c2ecf20Sopenharmony_ci{ 4758c2ecf20Sopenharmony_ci struct kvm_pmc *pmc = perf_event->overflow_handler_context; 4768c2ecf20Sopenharmony_ci struct arm_pmu *cpu_pmu = to_arm_pmu(perf_event->pmu); 4778c2ecf20Sopenharmony_ci struct kvm_vcpu *vcpu = kvm_pmc_to_vcpu(pmc); 4788c2ecf20Sopenharmony_ci int idx = pmc->idx; 4798c2ecf20Sopenharmony_ci u64 period; 4808c2ecf20Sopenharmony_ci 4818c2ecf20Sopenharmony_ci cpu_pmu->pmu.stop(perf_event, PERF_EF_UPDATE); 4828c2ecf20Sopenharmony_ci 4838c2ecf20Sopenharmony_ci /* 4848c2ecf20Sopenharmony_ci * Reset the sample period to the architectural limit, 4858c2ecf20Sopenharmony_ci * i.e. the point where the counter overflows. 4868c2ecf20Sopenharmony_ci */ 4878c2ecf20Sopenharmony_ci period = -(local64_read(&perf_event->count)); 4888c2ecf20Sopenharmony_ci 4898c2ecf20Sopenharmony_ci if (!kvm_pmu_idx_is_64bit(vcpu, pmc->idx)) 4908c2ecf20Sopenharmony_ci period &= GENMASK(31, 0); 4918c2ecf20Sopenharmony_ci 4928c2ecf20Sopenharmony_ci local64_set(&perf_event->hw.period_left, 0); 4938c2ecf20Sopenharmony_ci perf_event->attr.sample_period = period; 4948c2ecf20Sopenharmony_ci perf_event->hw.sample_period = period; 4958c2ecf20Sopenharmony_ci 4968c2ecf20Sopenharmony_ci __vcpu_sys_reg(vcpu, PMOVSSET_EL0) |= BIT(idx); 4978c2ecf20Sopenharmony_ci 4988c2ecf20Sopenharmony_ci if (kvm_pmu_overflow_status(vcpu)) { 4998c2ecf20Sopenharmony_ci kvm_make_request(KVM_REQ_IRQ_PENDING, vcpu); 5008c2ecf20Sopenharmony_ci 5018c2ecf20Sopenharmony_ci if (!in_nmi()) 5028c2ecf20Sopenharmony_ci kvm_vcpu_kick(vcpu); 5038c2ecf20Sopenharmony_ci else 5048c2ecf20Sopenharmony_ci irq_work_queue(&vcpu->arch.pmu.overflow_work); 5058c2ecf20Sopenharmony_ci } 5068c2ecf20Sopenharmony_ci 5078c2ecf20Sopenharmony_ci cpu_pmu->pmu.start(perf_event, PERF_EF_RELOAD); 5088c2ecf20Sopenharmony_ci} 5098c2ecf20Sopenharmony_ci 5108c2ecf20Sopenharmony_ci/** 5118c2ecf20Sopenharmony_ci * kvm_pmu_software_increment - do software increment 5128c2ecf20Sopenharmony_ci * @vcpu: The vcpu pointer 5138c2ecf20Sopenharmony_ci * @val: the value guest writes to PMSWINC register 5148c2ecf20Sopenharmony_ci */ 5158c2ecf20Sopenharmony_civoid kvm_pmu_software_increment(struct kvm_vcpu *vcpu, u64 val) 5168c2ecf20Sopenharmony_ci{ 5178c2ecf20Sopenharmony_ci struct kvm_pmu *pmu = &vcpu->arch.pmu; 5188c2ecf20Sopenharmony_ci int i; 5198c2ecf20Sopenharmony_ci 5208c2ecf20Sopenharmony_ci if (!(__vcpu_sys_reg(vcpu, PMCR_EL0) & ARMV8_PMU_PMCR_E)) 5218c2ecf20Sopenharmony_ci return; 5228c2ecf20Sopenharmony_ci 5238c2ecf20Sopenharmony_ci /* Weed out disabled counters */ 5248c2ecf20Sopenharmony_ci val &= __vcpu_sys_reg(vcpu, PMCNTENSET_EL0); 5258c2ecf20Sopenharmony_ci 5268c2ecf20Sopenharmony_ci for (i = 0; i < ARMV8_PMU_CYCLE_IDX; i++) { 5278c2ecf20Sopenharmony_ci u64 type, reg; 5288c2ecf20Sopenharmony_ci 5298c2ecf20Sopenharmony_ci if (!(val & BIT(i))) 5308c2ecf20Sopenharmony_ci continue; 5318c2ecf20Sopenharmony_ci 5328c2ecf20Sopenharmony_ci /* PMSWINC only applies to ... SW_INC! */ 5338c2ecf20Sopenharmony_ci type = __vcpu_sys_reg(vcpu, PMEVTYPER0_EL0 + i); 5348c2ecf20Sopenharmony_ci type &= kvm_pmu_event_mask(vcpu->kvm); 5358c2ecf20Sopenharmony_ci if (type != ARMV8_PMUV3_PERFCTR_SW_INCR) 5368c2ecf20Sopenharmony_ci continue; 5378c2ecf20Sopenharmony_ci 5388c2ecf20Sopenharmony_ci /* increment this even SW_INC counter */ 5398c2ecf20Sopenharmony_ci reg = __vcpu_sys_reg(vcpu, PMEVCNTR0_EL0 + i) + 1; 5408c2ecf20Sopenharmony_ci reg = lower_32_bits(reg); 5418c2ecf20Sopenharmony_ci __vcpu_sys_reg(vcpu, PMEVCNTR0_EL0 + i) = reg; 5428c2ecf20Sopenharmony_ci 5438c2ecf20Sopenharmony_ci if (reg) /* no overflow on the low part */ 5448c2ecf20Sopenharmony_ci continue; 5458c2ecf20Sopenharmony_ci 5468c2ecf20Sopenharmony_ci if (kvm_pmu_pmc_is_chained(&pmu->pmc[i])) { 5478c2ecf20Sopenharmony_ci /* increment the high counter */ 5488c2ecf20Sopenharmony_ci reg = __vcpu_sys_reg(vcpu, PMEVCNTR0_EL0 + i + 1) + 1; 5498c2ecf20Sopenharmony_ci reg = lower_32_bits(reg); 5508c2ecf20Sopenharmony_ci __vcpu_sys_reg(vcpu, PMEVCNTR0_EL0 + i + 1) = reg; 5518c2ecf20Sopenharmony_ci if (!reg) /* mark overflow on the high counter */ 5528c2ecf20Sopenharmony_ci __vcpu_sys_reg(vcpu, PMOVSSET_EL0) |= BIT(i + 1); 5538c2ecf20Sopenharmony_ci } else { 5548c2ecf20Sopenharmony_ci /* mark overflow on low counter */ 5558c2ecf20Sopenharmony_ci __vcpu_sys_reg(vcpu, PMOVSSET_EL0) |= BIT(i); 5568c2ecf20Sopenharmony_ci } 5578c2ecf20Sopenharmony_ci } 5588c2ecf20Sopenharmony_ci} 5598c2ecf20Sopenharmony_ci 5608c2ecf20Sopenharmony_ci/** 5618c2ecf20Sopenharmony_ci * kvm_pmu_handle_pmcr - handle PMCR register 5628c2ecf20Sopenharmony_ci * @vcpu: The vcpu pointer 5638c2ecf20Sopenharmony_ci * @val: the value guest writes to PMCR register 5648c2ecf20Sopenharmony_ci */ 5658c2ecf20Sopenharmony_civoid kvm_pmu_handle_pmcr(struct kvm_vcpu *vcpu, u64 val) 5668c2ecf20Sopenharmony_ci{ 5678c2ecf20Sopenharmony_ci unsigned long mask = kvm_pmu_valid_counter_mask(vcpu); 5688c2ecf20Sopenharmony_ci int i; 5698c2ecf20Sopenharmony_ci 5708c2ecf20Sopenharmony_ci if (val & ARMV8_PMU_PMCR_E) { 5718c2ecf20Sopenharmony_ci kvm_pmu_enable_counter_mask(vcpu, 5728c2ecf20Sopenharmony_ci __vcpu_sys_reg(vcpu, PMCNTENSET_EL0) & mask); 5738c2ecf20Sopenharmony_ci } else { 5748c2ecf20Sopenharmony_ci kvm_pmu_disable_counter_mask(vcpu, mask); 5758c2ecf20Sopenharmony_ci } 5768c2ecf20Sopenharmony_ci 5778c2ecf20Sopenharmony_ci if (val & ARMV8_PMU_PMCR_C) 5788c2ecf20Sopenharmony_ci kvm_pmu_set_counter_value(vcpu, ARMV8_PMU_CYCLE_IDX, 0); 5798c2ecf20Sopenharmony_ci 5808c2ecf20Sopenharmony_ci if (val & ARMV8_PMU_PMCR_P) { 5818c2ecf20Sopenharmony_ci mask &= ~BIT(ARMV8_PMU_CYCLE_IDX); 5828c2ecf20Sopenharmony_ci for_each_set_bit(i, &mask, 32) 5838c2ecf20Sopenharmony_ci kvm_pmu_set_counter_value(vcpu, i, 0); 5848c2ecf20Sopenharmony_ci } 5858c2ecf20Sopenharmony_ci} 5868c2ecf20Sopenharmony_ci 5878c2ecf20Sopenharmony_cistatic bool kvm_pmu_counter_is_enabled(struct kvm_vcpu *vcpu, u64 select_idx) 5888c2ecf20Sopenharmony_ci{ 5898c2ecf20Sopenharmony_ci return (__vcpu_sys_reg(vcpu, PMCR_EL0) & ARMV8_PMU_PMCR_E) && 5908c2ecf20Sopenharmony_ci (__vcpu_sys_reg(vcpu, PMCNTENSET_EL0) & BIT(select_idx)); 5918c2ecf20Sopenharmony_ci} 5928c2ecf20Sopenharmony_ci 5938c2ecf20Sopenharmony_ci/** 5948c2ecf20Sopenharmony_ci * kvm_pmu_create_perf_event - create a perf event for a counter 5958c2ecf20Sopenharmony_ci * @vcpu: The vcpu pointer 5968c2ecf20Sopenharmony_ci * @select_idx: The number of selected counter 5978c2ecf20Sopenharmony_ci */ 5988c2ecf20Sopenharmony_cistatic void kvm_pmu_create_perf_event(struct kvm_vcpu *vcpu, u64 select_idx) 5998c2ecf20Sopenharmony_ci{ 6008c2ecf20Sopenharmony_ci struct kvm_pmu *pmu = &vcpu->arch.pmu; 6018c2ecf20Sopenharmony_ci struct kvm_pmc *pmc; 6028c2ecf20Sopenharmony_ci struct perf_event *event; 6038c2ecf20Sopenharmony_ci struct perf_event_attr attr; 6048c2ecf20Sopenharmony_ci u64 eventsel, counter, reg, data; 6058c2ecf20Sopenharmony_ci 6068c2ecf20Sopenharmony_ci /* 6078c2ecf20Sopenharmony_ci * For chained counters the event type and filtering attributes are 6088c2ecf20Sopenharmony_ci * obtained from the low/even counter. We also use this counter to 6098c2ecf20Sopenharmony_ci * determine if the event is enabled/disabled. 6108c2ecf20Sopenharmony_ci */ 6118c2ecf20Sopenharmony_ci pmc = kvm_pmu_get_canonical_pmc(&pmu->pmc[select_idx]); 6128c2ecf20Sopenharmony_ci 6138c2ecf20Sopenharmony_ci reg = (pmc->idx == ARMV8_PMU_CYCLE_IDX) 6148c2ecf20Sopenharmony_ci ? PMCCFILTR_EL0 : PMEVTYPER0_EL0 + pmc->idx; 6158c2ecf20Sopenharmony_ci data = __vcpu_sys_reg(vcpu, reg); 6168c2ecf20Sopenharmony_ci 6178c2ecf20Sopenharmony_ci kvm_pmu_stop_counter(vcpu, pmc); 6188c2ecf20Sopenharmony_ci if (pmc->idx == ARMV8_PMU_CYCLE_IDX) 6198c2ecf20Sopenharmony_ci eventsel = ARMV8_PMUV3_PERFCTR_CPU_CYCLES; 6208c2ecf20Sopenharmony_ci else 6218c2ecf20Sopenharmony_ci eventsel = data & kvm_pmu_event_mask(vcpu->kvm); 6228c2ecf20Sopenharmony_ci 6238c2ecf20Sopenharmony_ci /* Software increment event doesn't need to be backed by a perf event */ 6248c2ecf20Sopenharmony_ci if (eventsel == ARMV8_PMUV3_PERFCTR_SW_INCR) 6258c2ecf20Sopenharmony_ci return; 6268c2ecf20Sopenharmony_ci 6278c2ecf20Sopenharmony_ci /* 6288c2ecf20Sopenharmony_ci * If we have a filter in place and that the event isn't allowed, do 6298c2ecf20Sopenharmony_ci * not install a perf event either. 6308c2ecf20Sopenharmony_ci */ 6318c2ecf20Sopenharmony_ci if (vcpu->kvm->arch.pmu_filter && 6328c2ecf20Sopenharmony_ci !test_bit(eventsel, vcpu->kvm->arch.pmu_filter)) 6338c2ecf20Sopenharmony_ci return; 6348c2ecf20Sopenharmony_ci 6358c2ecf20Sopenharmony_ci memset(&attr, 0, sizeof(struct perf_event_attr)); 6368c2ecf20Sopenharmony_ci attr.type = PERF_TYPE_RAW; 6378c2ecf20Sopenharmony_ci attr.size = sizeof(attr); 6388c2ecf20Sopenharmony_ci attr.pinned = 1; 6398c2ecf20Sopenharmony_ci attr.disabled = !kvm_pmu_counter_is_enabled(vcpu, pmc->idx); 6408c2ecf20Sopenharmony_ci attr.exclude_user = data & ARMV8_PMU_EXCLUDE_EL0 ? 1 : 0; 6418c2ecf20Sopenharmony_ci attr.exclude_kernel = data & ARMV8_PMU_EXCLUDE_EL1 ? 1 : 0; 6428c2ecf20Sopenharmony_ci attr.exclude_hv = 1; /* Don't count EL2 events */ 6438c2ecf20Sopenharmony_ci attr.exclude_host = 1; /* Don't count host events */ 6448c2ecf20Sopenharmony_ci attr.config = eventsel; 6458c2ecf20Sopenharmony_ci 6468c2ecf20Sopenharmony_ci counter = kvm_pmu_get_pair_counter_value(vcpu, pmc); 6478c2ecf20Sopenharmony_ci 6488c2ecf20Sopenharmony_ci if (kvm_pmu_pmc_is_chained(pmc)) { 6498c2ecf20Sopenharmony_ci /** 6508c2ecf20Sopenharmony_ci * The initial sample period (overflow count) of an event. For 6518c2ecf20Sopenharmony_ci * chained counters we only support overflow interrupts on the 6528c2ecf20Sopenharmony_ci * high counter. 6538c2ecf20Sopenharmony_ci */ 6548c2ecf20Sopenharmony_ci attr.sample_period = (-counter) & GENMASK(63, 0); 6558c2ecf20Sopenharmony_ci attr.config1 |= PERF_ATTR_CFG1_KVM_PMU_CHAINED; 6568c2ecf20Sopenharmony_ci 6578c2ecf20Sopenharmony_ci event = perf_event_create_kernel_counter(&attr, -1, current, 6588c2ecf20Sopenharmony_ci kvm_pmu_perf_overflow, 6598c2ecf20Sopenharmony_ci pmc + 1); 6608c2ecf20Sopenharmony_ci } else { 6618c2ecf20Sopenharmony_ci /* The initial sample period (overflow count) of an event. */ 6628c2ecf20Sopenharmony_ci if (kvm_pmu_idx_is_64bit(vcpu, pmc->idx)) 6638c2ecf20Sopenharmony_ci attr.sample_period = (-counter) & GENMASK(63, 0); 6648c2ecf20Sopenharmony_ci else 6658c2ecf20Sopenharmony_ci attr.sample_period = (-counter) & GENMASK(31, 0); 6668c2ecf20Sopenharmony_ci 6678c2ecf20Sopenharmony_ci event = perf_event_create_kernel_counter(&attr, -1, current, 6688c2ecf20Sopenharmony_ci kvm_pmu_perf_overflow, pmc); 6698c2ecf20Sopenharmony_ci } 6708c2ecf20Sopenharmony_ci 6718c2ecf20Sopenharmony_ci if (IS_ERR(event)) { 6728c2ecf20Sopenharmony_ci pr_err_once("kvm: pmu event creation failed %ld\n", 6738c2ecf20Sopenharmony_ci PTR_ERR(event)); 6748c2ecf20Sopenharmony_ci return; 6758c2ecf20Sopenharmony_ci } 6768c2ecf20Sopenharmony_ci 6778c2ecf20Sopenharmony_ci pmc->perf_event = event; 6788c2ecf20Sopenharmony_ci} 6798c2ecf20Sopenharmony_ci 6808c2ecf20Sopenharmony_ci/** 6818c2ecf20Sopenharmony_ci * kvm_pmu_update_pmc_chained - update chained bitmap 6828c2ecf20Sopenharmony_ci * @vcpu: The vcpu pointer 6838c2ecf20Sopenharmony_ci * @select_idx: The number of selected counter 6848c2ecf20Sopenharmony_ci * 6858c2ecf20Sopenharmony_ci * Update the chained bitmap based on the event type written in the 6868c2ecf20Sopenharmony_ci * typer register and the enable state of the odd register. 6878c2ecf20Sopenharmony_ci */ 6888c2ecf20Sopenharmony_cistatic void kvm_pmu_update_pmc_chained(struct kvm_vcpu *vcpu, u64 select_idx) 6898c2ecf20Sopenharmony_ci{ 6908c2ecf20Sopenharmony_ci struct kvm_pmu *pmu = &vcpu->arch.pmu; 6918c2ecf20Sopenharmony_ci struct kvm_pmc *pmc = &pmu->pmc[select_idx], *canonical_pmc; 6928c2ecf20Sopenharmony_ci bool new_state, old_state; 6938c2ecf20Sopenharmony_ci 6948c2ecf20Sopenharmony_ci old_state = kvm_pmu_pmc_is_chained(pmc); 6958c2ecf20Sopenharmony_ci new_state = kvm_pmu_idx_has_chain_evtype(vcpu, pmc->idx) && 6968c2ecf20Sopenharmony_ci kvm_pmu_counter_is_enabled(vcpu, pmc->idx | 0x1); 6978c2ecf20Sopenharmony_ci 6988c2ecf20Sopenharmony_ci if (old_state == new_state) 6998c2ecf20Sopenharmony_ci return; 7008c2ecf20Sopenharmony_ci 7018c2ecf20Sopenharmony_ci canonical_pmc = kvm_pmu_get_canonical_pmc(pmc); 7028c2ecf20Sopenharmony_ci kvm_pmu_stop_counter(vcpu, canonical_pmc); 7038c2ecf20Sopenharmony_ci if (new_state) { 7048c2ecf20Sopenharmony_ci /* 7058c2ecf20Sopenharmony_ci * During promotion from !chained to chained we must ensure 7068c2ecf20Sopenharmony_ci * the adjacent counter is stopped and its event destroyed 7078c2ecf20Sopenharmony_ci */ 7088c2ecf20Sopenharmony_ci kvm_pmu_stop_counter(vcpu, kvm_pmu_get_alternate_pmc(pmc)); 7098c2ecf20Sopenharmony_ci set_bit(pmc->idx >> 1, vcpu->arch.pmu.chained); 7108c2ecf20Sopenharmony_ci return; 7118c2ecf20Sopenharmony_ci } 7128c2ecf20Sopenharmony_ci clear_bit(pmc->idx >> 1, vcpu->arch.pmu.chained); 7138c2ecf20Sopenharmony_ci} 7148c2ecf20Sopenharmony_ci 7158c2ecf20Sopenharmony_ci/** 7168c2ecf20Sopenharmony_ci * kvm_pmu_set_counter_event_type - set selected counter to monitor some event 7178c2ecf20Sopenharmony_ci * @vcpu: The vcpu pointer 7188c2ecf20Sopenharmony_ci * @data: The data guest writes to PMXEVTYPER_EL0 7198c2ecf20Sopenharmony_ci * @select_idx: The number of selected counter 7208c2ecf20Sopenharmony_ci * 7218c2ecf20Sopenharmony_ci * When OS accesses PMXEVTYPER_EL0, that means it wants to set a PMC to count an 7228c2ecf20Sopenharmony_ci * event with given hardware event number. Here we call perf_event API to 7238c2ecf20Sopenharmony_ci * emulate this action and create a kernel perf event for it. 7248c2ecf20Sopenharmony_ci */ 7258c2ecf20Sopenharmony_civoid kvm_pmu_set_counter_event_type(struct kvm_vcpu *vcpu, u64 data, 7268c2ecf20Sopenharmony_ci u64 select_idx) 7278c2ecf20Sopenharmony_ci{ 7288c2ecf20Sopenharmony_ci u64 reg, mask; 7298c2ecf20Sopenharmony_ci 7308c2ecf20Sopenharmony_ci mask = ARMV8_PMU_EVTYPE_MASK; 7318c2ecf20Sopenharmony_ci mask &= ~ARMV8_PMU_EVTYPE_EVENT; 7328c2ecf20Sopenharmony_ci mask |= kvm_pmu_event_mask(vcpu->kvm); 7338c2ecf20Sopenharmony_ci 7348c2ecf20Sopenharmony_ci reg = (select_idx == ARMV8_PMU_CYCLE_IDX) 7358c2ecf20Sopenharmony_ci ? PMCCFILTR_EL0 : PMEVTYPER0_EL0 + select_idx; 7368c2ecf20Sopenharmony_ci 7378c2ecf20Sopenharmony_ci __vcpu_sys_reg(vcpu, reg) = data & mask; 7388c2ecf20Sopenharmony_ci 7398c2ecf20Sopenharmony_ci kvm_pmu_update_pmc_chained(vcpu, select_idx); 7408c2ecf20Sopenharmony_ci kvm_pmu_create_perf_event(vcpu, select_idx); 7418c2ecf20Sopenharmony_ci} 7428c2ecf20Sopenharmony_ci 7438c2ecf20Sopenharmony_cistatic int kvm_pmu_probe_pmuver(void) 7448c2ecf20Sopenharmony_ci{ 7458c2ecf20Sopenharmony_ci struct perf_event_attr attr = { }; 7468c2ecf20Sopenharmony_ci struct perf_event *event; 7478c2ecf20Sopenharmony_ci struct arm_pmu *pmu; 7488c2ecf20Sopenharmony_ci int pmuver = 0xf; 7498c2ecf20Sopenharmony_ci 7508c2ecf20Sopenharmony_ci /* 7518c2ecf20Sopenharmony_ci * Create a dummy event that only counts user cycles. As we'll never 7528c2ecf20Sopenharmony_ci * leave this function with the event being live, it will never 7538c2ecf20Sopenharmony_ci * count anything. But it allows us to probe some of the PMU 7548c2ecf20Sopenharmony_ci * details. Yes, this is terrible. 7558c2ecf20Sopenharmony_ci */ 7568c2ecf20Sopenharmony_ci attr.type = PERF_TYPE_RAW; 7578c2ecf20Sopenharmony_ci attr.size = sizeof(attr); 7588c2ecf20Sopenharmony_ci attr.pinned = 1; 7598c2ecf20Sopenharmony_ci attr.disabled = 0; 7608c2ecf20Sopenharmony_ci attr.exclude_user = 0; 7618c2ecf20Sopenharmony_ci attr.exclude_kernel = 1; 7628c2ecf20Sopenharmony_ci attr.exclude_hv = 1; 7638c2ecf20Sopenharmony_ci attr.exclude_host = 1; 7648c2ecf20Sopenharmony_ci attr.config = ARMV8_PMUV3_PERFCTR_CPU_CYCLES; 7658c2ecf20Sopenharmony_ci attr.sample_period = GENMASK(63, 0); 7668c2ecf20Sopenharmony_ci 7678c2ecf20Sopenharmony_ci event = perf_event_create_kernel_counter(&attr, -1, current, 7688c2ecf20Sopenharmony_ci kvm_pmu_perf_overflow, &attr); 7698c2ecf20Sopenharmony_ci 7708c2ecf20Sopenharmony_ci if (IS_ERR(event)) { 7718c2ecf20Sopenharmony_ci pr_err_once("kvm: pmu event creation failed %ld\n", 7728c2ecf20Sopenharmony_ci PTR_ERR(event)); 7738c2ecf20Sopenharmony_ci return 0xf; 7748c2ecf20Sopenharmony_ci } 7758c2ecf20Sopenharmony_ci 7768c2ecf20Sopenharmony_ci if (event->pmu) { 7778c2ecf20Sopenharmony_ci pmu = to_arm_pmu(event->pmu); 7788c2ecf20Sopenharmony_ci if (pmu->pmuver) 7798c2ecf20Sopenharmony_ci pmuver = pmu->pmuver; 7808c2ecf20Sopenharmony_ci } 7818c2ecf20Sopenharmony_ci 7828c2ecf20Sopenharmony_ci perf_event_disable(event); 7838c2ecf20Sopenharmony_ci perf_event_release_kernel(event); 7848c2ecf20Sopenharmony_ci 7858c2ecf20Sopenharmony_ci return pmuver; 7868c2ecf20Sopenharmony_ci} 7878c2ecf20Sopenharmony_ci 7888c2ecf20Sopenharmony_ciu64 kvm_pmu_get_pmceid(struct kvm_vcpu *vcpu, bool pmceid1) 7898c2ecf20Sopenharmony_ci{ 7908c2ecf20Sopenharmony_ci unsigned long *bmap = vcpu->kvm->arch.pmu_filter; 7918c2ecf20Sopenharmony_ci u64 val, mask = 0; 7928c2ecf20Sopenharmony_ci int base, i, nr_events; 7938c2ecf20Sopenharmony_ci 7948c2ecf20Sopenharmony_ci if (!pmceid1) { 7958c2ecf20Sopenharmony_ci val = read_sysreg(pmceid0_el0); 7968c2ecf20Sopenharmony_ci base = 0; 7978c2ecf20Sopenharmony_ci } else { 7988c2ecf20Sopenharmony_ci val = read_sysreg(pmceid1_el0); 7998c2ecf20Sopenharmony_ci base = 32; 8008c2ecf20Sopenharmony_ci } 8018c2ecf20Sopenharmony_ci 8028c2ecf20Sopenharmony_ci if (!bmap) 8038c2ecf20Sopenharmony_ci return val; 8048c2ecf20Sopenharmony_ci 8058c2ecf20Sopenharmony_ci nr_events = kvm_pmu_event_mask(vcpu->kvm) + 1; 8068c2ecf20Sopenharmony_ci 8078c2ecf20Sopenharmony_ci for (i = 0; i < 32; i += 8) { 8088c2ecf20Sopenharmony_ci u64 byte; 8098c2ecf20Sopenharmony_ci 8108c2ecf20Sopenharmony_ci byte = bitmap_get_value8(bmap, base + i); 8118c2ecf20Sopenharmony_ci mask |= byte << i; 8128c2ecf20Sopenharmony_ci if (nr_events >= (0x4000 + base + 32)) { 8138c2ecf20Sopenharmony_ci byte = bitmap_get_value8(bmap, 0x4000 + base + i); 8148c2ecf20Sopenharmony_ci mask |= byte << (32 + i); 8158c2ecf20Sopenharmony_ci } 8168c2ecf20Sopenharmony_ci } 8178c2ecf20Sopenharmony_ci 8188c2ecf20Sopenharmony_ci return val & mask; 8198c2ecf20Sopenharmony_ci} 8208c2ecf20Sopenharmony_ci 8218c2ecf20Sopenharmony_cibool kvm_arm_support_pmu_v3(void) 8228c2ecf20Sopenharmony_ci{ 8238c2ecf20Sopenharmony_ci /* 8248c2ecf20Sopenharmony_ci * Check if HW_PERF_EVENTS are supported by checking the number of 8258c2ecf20Sopenharmony_ci * hardware performance counters. This could ensure the presence of 8268c2ecf20Sopenharmony_ci * a physical PMU and CONFIG_PERF_EVENT is selected. 8278c2ecf20Sopenharmony_ci */ 8288c2ecf20Sopenharmony_ci return (perf_num_counters() > 0); 8298c2ecf20Sopenharmony_ci} 8308c2ecf20Sopenharmony_ci 8318c2ecf20Sopenharmony_ciint kvm_arm_pmu_v3_enable(struct kvm_vcpu *vcpu) 8328c2ecf20Sopenharmony_ci{ 8338c2ecf20Sopenharmony_ci if (!vcpu->arch.pmu.created) 8348c2ecf20Sopenharmony_ci return 0; 8358c2ecf20Sopenharmony_ci 8368c2ecf20Sopenharmony_ci /* 8378c2ecf20Sopenharmony_ci * A valid interrupt configuration for the PMU is either to have a 8388c2ecf20Sopenharmony_ci * properly configured interrupt number and using an in-kernel 8398c2ecf20Sopenharmony_ci * irqchip, or to not have an in-kernel GIC and not set an IRQ. 8408c2ecf20Sopenharmony_ci */ 8418c2ecf20Sopenharmony_ci if (irqchip_in_kernel(vcpu->kvm)) { 8428c2ecf20Sopenharmony_ci int irq = vcpu->arch.pmu.irq_num; 8438c2ecf20Sopenharmony_ci if (!kvm_arm_pmu_irq_initialized(vcpu)) 8448c2ecf20Sopenharmony_ci return -EINVAL; 8458c2ecf20Sopenharmony_ci 8468c2ecf20Sopenharmony_ci /* 8478c2ecf20Sopenharmony_ci * If we are using an in-kernel vgic, at this point we know 8488c2ecf20Sopenharmony_ci * the vgic will be initialized, so we can check the PMU irq 8498c2ecf20Sopenharmony_ci * number against the dimensions of the vgic and make sure 8508c2ecf20Sopenharmony_ci * it's valid. 8518c2ecf20Sopenharmony_ci */ 8528c2ecf20Sopenharmony_ci if (!irq_is_ppi(irq) && !vgic_valid_spi(vcpu->kvm, irq)) 8538c2ecf20Sopenharmony_ci return -EINVAL; 8548c2ecf20Sopenharmony_ci } else if (kvm_arm_pmu_irq_initialized(vcpu)) { 8558c2ecf20Sopenharmony_ci return -EINVAL; 8568c2ecf20Sopenharmony_ci } 8578c2ecf20Sopenharmony_ci 8588c2ecf20Sopenharmony_ci kvm_pmu_vcpu_reset(vcpu); 8598c2ecf20Sopenharmony_ci vcpu->arch.pmu.ready = true; 8608c2ecf20Sopenharmony_ci 8618c2ecf20Sopenharmony_ci return 0; 8628c2ecf20Sopenharmony_ci} 8638c2ecf20Sopenharmony_ci 8648c2ecf20Sopenharmony_cistatic int kvm_arm_pmu_v3_init(struct kvm_vcpu *vcpu) 8658c2ecf20Sopenharmony_ci{ 8668c2ecf20Sopenharmony_ci if (irqchip_in_kernel(vcpu->kvm)) { 8678c2ecf20Sopenharmony_ci int ret; 8688c2ecf20Sopenharmony_ci 8698c2ecf20Sopenharmony_ci /* 8708c2ecf20Sopenharmony_ci * If using the PMU with an in-kernel virtual GIC 8718c2ecf20Sopenharmony_ci * implementation, we require the GIC to be already 8728c2ecf20Sopenharmony_ci * initialized when initializing the PMU. 8738c2ecf20Sopenharmony_ci */ 8748c2ecf20Sopenharmony_ci if (!vgic_initialized(vcpu->kvm)) 8758c2ecf20Sopenharmony_ci return -ENODEV; 8768c2ecf20Sopenharmony_ci 8778c2ecf20Sopenharmony_ci if (!kvm_arm_pmu_irq_initialized(vcpu)) 8788c2ecf20Sopenharmony_ci return -ENXIO; 8798c2ecf20Sopenharmony_ci 8808c2ecf20Sopenharmony_ci ret = kvm_vgic_set_owner(vcpu, vcpu->arch.pmu.irq_num, 8818c2ecf20Sopenharmony_ci &vcpu->arch.pmu); 8828c2ecf20Sopenharmony_ci if (ret) 8838c2ecf20Sopenharmony_ci return ret; 8848c2ecf20Sopenharmony_ci } 8858c2ecf20Sopenharmony_ci 8868c2ecf20Sopenharmony_ci init_irq_work(&vcpu->arch.pmu.overflow_work, 8878c2ecf20Sopenharmony_ci kvm_pmu_perf_overflow_notify_vcpu); 8888c2ecf20Sopenharmony_ci 8898c2ecf20Sopenharmony_ci vcpu->arch.pmu.created = true; 8908c2ecf20Sopenharmony_ci return 0; 8918c2ecf20Sopenharmony_ci} 8928c2ecf20Sopenharmony_ci 8938c2ecf20Sopenharmony_ci/* 8948c2ecf20Sopenharmony_ci * For one VM the interrupt type must be same for each vcpu. 8958c2ecf20Sopenharmony_ci * As a PPI, the interrupt number is the same for all vcpus, 8968c2ecf20Sopenharmony_ci * while as an SPI it must be a separate number per vcpu. 8978c2ecf20Sopenharmony_ci */ 8988c2ecf20Sopenharmony_cistatic bool pmu_irq_is_valid(struct kvm *kvm, int irq) 8998c2ecf20Sopenharmony_ci{ 9008c2ecf20Sopenharmony_ci int i; 9018c2ecf20Sopenharmony_ci struct kvm_vcpu *vcpu; 9028c2ecf20Sopenharmony_ci 9038c2ecf20Sopenharmony_ci kvm_for_each_vcpu(i, vcpu, kvm) { 9048c2ecf20Sopenharmony_ci if (!kvm_arm_pmu_irq_initialized(vcpu)) 9058c2ecf20Sopenharmony_ci continue; 9068c2ecf20Sopenharmony_ci 9078c2ecf20Sopenharmony_ci if (irq_is_ppi(irq)) { 9088c2ecf20Sopenharmony_ci if (vcpu->arch.pmu.irq_num != irq) 9098c2ecf20Sopenharmony_ci return false; 9108c2ecf20Sopenharmony_ci } else { 9118c2ecf20Sopenharmony_ci if (vcpu->arch.pmu.irq_num == irq) 9128c2ecf20Sopenharmony_ci return false; 9138c2ecf20Sopenharmony_ci } 9148c2ecf20Sopenharmony_ci } 9158c2ecf20Sopenharmony_ci 9168c2ecf20Sopenharmony_ci return true; 9178c2ecf20Sopenharmony_ci} 9188c2ecf20Sopenharmony_ci 9198c2ecf20Sopenharmony_ciint kvm_arm_pmu_v3_set_attr(struct kvm_vcpu *vcpu, struct kvm_device_attr *attr) 9208c2ecf20Sopenharmony_ci{ 9218c2ecf20Sopenharmony_ci if (!kvm_arm_support_pmu_v3() || 9228c2ecf20Sopenharmony_ci !test_bit(KVM_ARM_VCPU_PMU_V3, vcpu->arch.features)) 9238c2ecf20Sopenharmony_ci return -ENODEV; 9248c2ecf20Sopenharmony_ci 9258c2ecf20Sopenharmony_ci if (vcpu->arch.pmu.created) 9268c2ecf20Sopenharmony_ci return -EBUSY; 9278c2ecf20Sopenharmony_ci 9288c2ecf20Sopenharmony_ci if (!vcpu->kvm->arch.pmuver) 9298c2ecf20Sopenharmony_ci vcpu->kvm->arch.pmuver = kvm_pmu_probe_pmuver(); 9308c2ecf20Sopenharmony_ci 9318c2ecf20Sopenharmony_ci if (vcpu->kvm->arch.pmuver == 0xf) 9328c2ecf20Sopenharmony_ci return -ENODEV; 9338c2ecf20Sopenharmony_ci 9348c2ecf20Sopenharmony_ci switch (attr->attr) { 9358c2ecf20Sopenharmony_ci case KVM_ARM_VCPU_PMU_V3_IRQ: { 9368c2ecf20Sopenharmony_ci int __user *uaddr = (int __user *)(long)attr->addr; 9378c2ecf20Sopenharmony_ci int irq; 9388c2ecf20Sopenharmony_ci 9398c2ecf20Sopenharmony_ci if (!irqchip_in_kernel(vcpu->kvm)) 9408c2ecf20Sopenharmony_ci return -EINVAL; 9418c2ecf20Sopenharmony_ci 9428c2ecf20Sopenharmony_ci if (get_user(irq, uaddr)) 9438c2ecf20Sopenharmony_ci return -EFAULT; 9448c2ecf20Sopenharmony_ci 9458c2ecf20Sopenharmony_ci /* The PMU overflow interrupt can be a PPI or a valid SPI. */ 9468c2ecf20Sopenharmony_ci if (!(irq_is_ppi(irq) || irq_is_spi(irq))) 9478c2ecf20Sopenharmony_ci return -EINVAL; 9488c2ecf20Sopenharmony_ci 9498c2ecf20Sopenharmony_ci if (!pmu_irq_is_valid(vcpu->kvm, irq)) 9508c2ecf20Sopenharmony_ci return -EINVAL; 9518c2ecf20Sopenharmony_ci 9528c2ecf20Sopenharmony_ci if (kvm_arm_pmu_irq_initialized(vcpu)) 9538c2ecf20Sopenharmony_ci return -EBUSY; 9548c2ecf20Sopenharmony_ci 9558c2ecf20Sopenharmony_ci kvm_debug("Set kvm ARM PMU irq: %d\n", irq); 9568c2ecf20Sopenharmony_ci vcpu->arch.pmu.irq_num = irq; 9578c2ecf20Sopenharmony_ci return 0; 9588c2ecf20Sopenharmony_ci } 9598c2ecf20Sopenharmony_ci case KVM_ARM_VCPU_PMU_V3_FILTER: { 9608c2ecf20Sopenharmony_ci struct kvm_pmu_event_filter __user *uaddr; 9618c2ecf20Sopenharmony_ci struct kvm_pmu_event_filter filter; 9628c2ecf20Sopenharmony_ci int nr_events; 9638c2ecf20Sopenharmony_ci 9648c2ecf20Sopenharmony_ci nr_events = kvm_pmu_event_mask(vcpu->kvm) + 1; 9658c2ecf20Sopenharmony_ci 9668c2ecf20Sopenharmony_ci uaddr = (struct kvm_pmu_event_filter __user *)(long)attr->addr; 9678c2ecf20Sopenharmony_ci 9688c2ecf20Sopenharmony_ci if (copy_from_user(&filter, uaddr, sizeof(filter))) 9698c2ecf20Sopenharmony_ci return -EFAULT; 9708c2ecf20Sopenharmony_ci 9718c2ecf20Sopenharmony_ci if (((u32)filter.base_event + filter.nevents) > nr_events || 9728c2ecf20Sopenharmony_ci (filter.action != KVM_PMU_EVENT_ALLOW && 9738c2ecf20Sopenharmony_ci filter.action != KVM_PMU_EVENT_DENY)) 9748c2ecf20Sopenharmony_ci return -EINVAL; 9758c2ecf20Sopenharmony_ci 9768c2ecf20Sopenharmony_ci mutex_lock(&vcpu->kvm->lock); 9778c2ecf20Sopenharmony_ci 9788c2ecf20Sopenharmony_ci if (!vcpu->kvm->arch.pmu_filter) { 9798c2ecf20Sopenharmony_ci vcpu->kvm->arch.pmu_filter = bitmap_alloc(nr_events, GFP_KERNEL); 9808c2ecf20Sopenharmony_ci if (!vcpu->kvm->arch.pmu_filter) { 9818c2ecf20Sopenharmony_ci mutex_unlock(&vcpu->kvm->lock); 9828c2ecf20Sopenharmony_ci return -ENOMEM; 9838c2ecf20Sopenharmony_ci } 9848c2ecf20Sopenharmony_ci 9858c2ecf20Sopenharmony_ci /* 9868c2ecf20Sopenharmony_ci * The default depends on the first applied filter. 9878c2ecf20Sopenharmony_ci * If it allows events, the default is to deny. 9888c2ecf20Sopenharmony_ci * Conversely, if the first filter denies a set of 9898c2ecf20Sopenharmony_ci * events, the default is to allow. 9908c2ecf20Sopenharmony_ci */ 9918c2ecf20Sopenharmony_ci if (filter.action == KVM_PMU_EVENT_ALLOW) 9928c2ecf20Sopenharmony_ci bitmap_zero(vcpu->kvm->arch.pmu_filter, nr_events); 9938c2ecf20Sopenharmony_ci else 9948c2ecf20Sopenharmony_ci bitmap_fill(vcpu->kvm->arch.pmu_filter, nr_events); 9958c2ecf20Sopenharmony_ci } 9968c2ecf20Sopenharmony_ci 9978c2ecf20Sopenharmony_ci if (filter.action == KVM_PMU_EVENT_ALLOW) 9988c2ecf20Sopenharmony_ci bitmap_set(vcpu->kvm->arch.pmu_filter, filter.base_event, filter.nevents); 9998c2ecf20Sopenharmony_ci else 10008c2ecf20Sopenharmony_ci bitmap_clear(vcpu->kvm->arch.pmu_filter, filter.base_event, filter.nevents); 10018c2ecf20Sopenharmony_ci 10028c2ecf20Sopenharmony_ci mutex_unlock(&vcpu->kvm->lock); 10038c2ecf20Sopenharmony_ci 10048c2ecf20Sopenharmony_ci return 0; 10058c2ecf20Sopenharmony_ci } 10068c2ecf20Sopenharmony_ci case KVM_ARM_VCPU_PMU_V3_INIT: 10078c2ecf20Sopenharmony_ci return kvm_arm_pmu_v3_init(vcpu); 10088c2ecf20Sopenharmony_ci } 10098c2ecf20Sopenharmony_ci 10108c2ecf20Sopenharmony_ci return -ENXIO; 10118c2ecf20Sopenharmony_ci} 10128c2ecf20Sopenharmony_ci 10138c2ecf20Sopenharmony_ciint kvm_arm_pmu_v3_get_attr(struct kvm_vcpu *vcpu, struct kvm_device_attr *attr) 10148c2ecf20Sopenharmony_ci{ 10158c2ecf20Sopenharmony_ci switch (attr->attr) { 10168c2ecf20Sopenharmony_ci case KVM_ARM_VCPU_PMU_V3_IRQ: { 10178c2ecf20Sopenharmony_ci int __user *uaddr = (int __user *)(long)attr->addr; 10188c2ecf20Sopenharmony_ci int irq; 10198c2ecf20Sopenharmony_ci 10208c2ecf20Sopenharmony_ci if (!irqchip_in_kernel(vcpu->kvm)) 10218c2ecf20Sopenharmony_ci return -EINVAL; 10228c2ecf20Sopenharmony_ci 10238c2ecf20Sopenharmony_ci if (!test_bit(KVM_ARM_VCPU_PMU_V3, vcpu->arch.features)) 10248c2ecf20Sopenharmony_ci return -ENODEV; 10258c2ecf20Sopenharmony_ci 10268c2ecf20Sopenharmony_ci if (!kvm_arm_pmu_irq_initialized(vcpu)) 10278c2ecf20Sopenharmony_ci return -ENXIO; 10288c2ecf20Sopenharmony_ci 10298c2ecf20Sopenharmony_ci irq = vcpu->arch.pmu.irq_num; 10308c2ecf20Sopenharmony_ci return put_user(irq, uaddr); 10318c2ecf20Sopenharmony_ci } 10328c2ecf20Sopenharmony_ci } 10338c2ecf20Sopenharmony_ci 10348c2ecf20Sopenharmony_ci return -ENXIO; 10358c2ecf20Sopenharmony_ci} 10368c2ecf20Sopenharmony_ci 10378c2ecf20Sopenharmony_ciint kvm_arm_pmu_v3_has_attr(struct kvm_vcpu *vcpu, struct kvm_device_attr *attr) 10388c2ecf20Sopenharmony_ci{ 10398c2ecf20Sopenharmony_ci switch (attr->attr) { 10408c2ecf20Sopenharmony_ci case KVM_ARM_VCPU_PMU_V3_IRQ: 10418c2ecf20Sopenharmony_ci case KVM_ARM_VCPU_PMU_V3_INIT: 10428c2ecf20Sopenharmony_ci case KVM_ARM_VCPU_PMU_V3_FILTER: 10438c2ecf20Sopenharmony_ci if (kvm_arm_support_pmu_v3() && 10448c2ecf20Sopenharmony_ci test_bit(KVM_ARM_VCPU_PMU_V3, vcpu->arch.features)) 10458c2ecf20Sopenharmony_ci return 0; 10468c2ecf20Sopenharmony_ci } 10478c2ecf20Sopenharmony_ci 10488c2ecf20Sopenharmony_ci return -ENXIO; 10498c2ecf20Sopenharmony_ci} 1050