162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0 262306a36Sopenharmony_ci#include <errno.h> 362306a36Sopenharmony_ci#include <memory.h> 462306a36Sopenharmony_ci#include "../../../util/evsel.h" 562306a36Sopenharmony_ci#include "../../../util/kvm-stat.h" 662306a36Sopenharmony_ci#include "arm64_exception_types.h" 762306a36Sopenharmony_ci#include "debug.h" 862306a36Sopenharmony_ci 962306a36Sopenharmony_cidefine_exit_reasons_table(arm64_exit_reasons, kvm_arm_exception_type); 1062306a36Sopenharmony_cidefine_exit_reasons_table(arm64_trap_exit_reasons, kvm_arm_exception_class); 1162306a36Sopenharmony_ci 1262306a36Sopenharmony_ciconst char *kvm_trap_exit_reason = "esr_ec"; 1362306a36Sopenharmony_ciconst char *vcpu_id_str = "id"; 1462306a36Sopenharmony_ciconst char *kvm_exit_reason = "ret"; 1562306a36Sopenharmony_ciconst char *kvm_entry_trace = "kvm:kvm_entry"; 1662306a36Sopenharmony_ciconst char *kvm_exit_trace = "kvm:kvm_exit"; 1762306a36Sopenharmony_ci 1862306a36Sopenharmony_ciconst char *kvm_events_tp[] = { 1962306a36Sopenharmony_ci "kvm:kvm_entry", 2062306a36Sopenharmony_ci "kvm:kvm_exit", 2162306a36Sopenharmony_ci NULL, 2262306a36Sopenharmony_ci}; 2362306a36Sopenharmony_ci 2462306a36Sopenharmony_cistatic void event_get_key(struct evsel *evsel, 2562306a36Sopenharmony_ci struct perf_sample *sample, 2662306a36Sopenharmony_ci struct event_key *key) 2762306a36Sopenharmony_ci{ 2862306a36Sopenharmony_ci key->info = 0; 2962306a36Sopenharmony_ci key->key = evsel__intval(evsel, sample, kvm_exit_reason); 3062306a36Sopenharmony_ci key->exit_reasons = arm64_exit_reasons; 3162306a36Sopenharmony_ci 3262306a36Sopenharmony_ci /* 3362306a36Sopenharmony_ci * TRAP exceptions carry exception class info in esr_ec field 3462306a36Sopenharmony_ci * and, hence, we need to use a different exit_reasons table to 3562306a36Sopenharmony_ci * properly decode event's est_ec. 3662306a36Sopenharmony_ci */ 3762306a36Sopenharmony_ci if (key->key == ARM_EXCEPTION_TRAP) { 3862306a36Sopenharmony_ci key->key = evsel__intval(evsel, sample, kvm_trap_exit_reason); 3962306a36Sopenharmony_ci key->exit_reasons = arm64_trap_exit_reasons; 4062306a36Sopenharmony_ci } 4162306a36Sopenharmony_ci} 4262306a36Sopenharmony_ci 4362306a36Sopenharmony_cistatic bool event_begin(struct evsel *evsel, 4462306a36Sopenharmony_ci struct perf_sample *sample __maybe_unused, 4562306a36Sopenharmony_ci struct event_key *key __maybe_unused) 4662306a36Sopenharmony_ci{ 4762306a36Sopenharmony_ci return evsel__name_is(evsel, kvm_entry_trace); 4862306a36Sopenharmony_ci} 4962306a36Sopenharmony_ci 5062306a36Sopenharmony_cistatic bool event_end(struct evsel *evsel, 5162306a36Sopenharmony_ci struct perf_sample *sample, 5262306a36Sopenharmony_ci struct event_key *key) 5362306a36Sopenharmony_ci{ 5462306a36Sopenharmony_ci if (evsel__name_is(evsel, kvm_exit_trace)) { 5562306a36Sopenharmony_ci event_get_key(evsel, sample, key); 5662306a36Sopenharmony_ci return true; 5762306a36Sopenharmony_ci } 5862306a36Sopenharmony_ci return false; 5962306a36Sopenharmony_ci} 6062306a36Sopenharmony_ci 6162306a36Sopenharmony_cistatic struct kvm_events_ops exit_events = { 6262306a36Sopenharmony_ci .is_begin_event = event_begin, 6362306a36Sopenharmony_ci .is_end_event = event_end, 6462306a36Sopenharmony_ci .decode_key = exit_event_decode_key, 6562306a36Sopenharmony_ci .name = "VM-EXIT" 6662306a36Sopenharmony_ci}; 6762306a36Sopenharmony_ci 6862306a36Sopenharmony_cistruct kvm_reg_events_ops kvm_reg_events_ops[] = { 6962306a36Sopenharmony_ci { 7062306a36Sopenharmony_ci .name = "vmexit", 7162306a36Sopenharmony_ci .ops = &exit_events, 7262306a36Sopenharmony_ci }, 7362306a36Sopenharmony_ci { NULL, NULL }, 7462306a36Sopenharmony_ci}; 7562306a36Sopenharmony_ci 7662306a36Sopenharmony_ciconst char * const kvm_skip_events[] = { 7762306a36Sopenharmony_ci NULL, 7862306a36Sopenharmony_ci}; 7962306a36Sopenharmony_ci 8062306a36Sopenharmony_ciint cpu_isa_init(struct perf_kvm_stat *kvm, const char *cpuid __maybe_unused) 8162306a36Sopenharmony_ci{ 8262306a36Sopenharmony_ci kvm->exit_reasons_isa = "arm64"; 8362306a36Sopenharmony_ci return 0; 8462306a36Sopenharmony_ci} 85