|  | // SPDX-License-Identifier: GPL-2.0 | 
|  | #include "debug.h" | 
|  | #include "evsel.h" | 
|  | #include "kvm-stat.h" | 
|  |  | 
|  | #if defined(HAVE_KVM_STAT_SUPPORT) && defined(HAVE_LIBTRACEEVENT) | 
|  |  | 
|  | bool kvm_exit_event(struct evsel *evsel) | 
|  | { | 
|  | return evsel__name_is(evsel, kvm_exit_trace); | 
|  | } | 
|  |  | 
|  | void exit_event_get_key(struct evsel *evsel, | 
|  | struct perf_sample *sample, | 
|  | struct event_key *key) | 
|  | { | 
|  | key->info = 0; | 
|  | key->key  = evsel__intval(evsel, sample, kvm_exit_reason); | 
|  | } | 
|  |  | 
|  |  | 
|  | bool exit_event_begin(struct evsel *evsel, | 
|  | struct perf_sample *sample, struct event_key *key) | 
|  | { | 
|  | if (kvm_exit_event(evsel)) { | 
|  | exit_event_get_key(evsel, sample, key); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | return false; | 
|  | } | 
|  |  | 
|  | bool kvm_entry_event(struct evsel *evsel) | 
|  | { | 
|  | return evsel__name_is(evsel, kvm_entry_trace); | 
|  | } | 
|  |  | 
|  | bool exit_event_end(struct evsel *evsel, | 
|  | struct perf_sample *sample __maybe_unused, | 
|  | struct event_key *key __maybe_unused) | 
|  | { | 
|  | return kvm_entry_event(evsel); | 
|  | } | 
|  |  | 
|  | static const char *get_exit_reason(struct perf_kvm_stat *kvm, | 
|  | struct exit_reasons_table *tbl, | 
|  | u64 exit_code) | 
|  | { | 
|  | while (tbl->reason != NULL) { | 
|  | if (tbl->exit_code == exit_code) | 
|  | return tbl->reason; | 
|  | tbl++; | 
|  | } | 
|  |  | 
|  | pr_err("unknown kvm exit code:%lld on %s\n", | 
|  | (unsigned long long)exit_code, kvm->exit_reasons_isa); | 
|  | return "UNKNOWN"; | 
|  | } | 
|  |  | 
|  | void exit_event_decode_key(struct perf_kvm_stat *kvm, | 
|  | struct event_key *key, | 
|  | char *decode) | 
|  | { | 
|  | const char *exit_reason = get_exit_reason(kvm, key->exit_reasons, | 
|  | key->key); | 
|  |  | 
|  | scnprintf(decode, KVM_EVENT_NAME_LEN, "%s", exit_reason); | 
|  | } | 
|  |  | 
|  | #endif |