Received: by 2002:ab2:6203:0:b0:1f5:f2ab:c469 with SMTP id o3csp370063lqt; Thu, 18 Apr 2024 19:35:09 -0700 (PDT) X-Forwarded-Encrypted: i=3; AJvYcCWs6ic4TsVkotGsApEDcfj1qdg1EJWLx+ZzlXNHltIdCWRS1rtpT1oOT0whyxHkiuImEohpkb5QvqIBv5Y1o1DLYHOyM4RDGXIoabMWzQ== X-Google-Smtp-Source: AGHT+IF06DD/Xhw5DncClgiwzS8Q/5Cc3lPAS11UNZqsy1aI2NnYaC2kP11RfaE2ynYXElDgC4yx X-Received: by 2002:a05:6e02:1988:b0:36b:85e:7d6c with SMTP id g8-20020a056e02198800b0036b085e7d6cmr1010874ilf.28.1713494108960; Thu, 18 Apr 2024 19:35:08 -0700 (PDT) ARC-Seal: i=2; a=rsa-sha256; t=1713494108; cv=pass; d=google.com; s=arc-20160816; b=0Su3SEloBS0q3/ZL+DmQ6PiKFEhlmxI1J5xyAfCY0WYyhgPCQkoYFJc9iPrZlQGVKa z9SrdoruhqyY9GNxW7su+EXz5uGmzvoWDjk/C/A8kam3eIls+gBdq0IAvSfzFSYEqXfg UZPuEfX3hSl6Mhgx/pdBQot8tTZgb3Bfdm0fCmP6OyC4BeIUy1tKe0wGoWM2pIuvhSvo vxBuMVpxw4LdZhVl/p1wqGTnuzR36tav7UUPbGXYeWB8T4LKukDNWrl9Lw2SaYk0Pesd F3EA7n2xJeAuTuW6sgXTyAjd/lSn5uia3tuWTcSVoAnabeXzLDInEgFYeC+G4OozTW97 JSAQ== ARC-Message-Signature: i=2; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=content-transfer-encoding:content-language:in-reply-to:mime-version :list-unsubscribe:list-subscribe:list-id:precedence:user-agent:date :message-id:references:cc:to:from:subject; bh=XkMFczPinpre5YSH3aPyJtJyq4RQY/JuW4p9o7iXdBc=; fh=LqXlvClQMN9BJuI4yfS5cdvK2IGGfQ86riHReCnXhZQ=; b=SnLZScH5uyfckYd4inqn20UWr+JVbOs/hrK1IUta8M9GrruI1+jxpe2yUe9nejiHx0 yRzvZD2RWIRP2RIK1JjEfuucsH9nggvV/XlWVRv7qkaUbEQb83X36V0mwEWqaL3lY8SK YlK3r+MkfzusdOXO5j+xxD7MJcDCzeN0Ioly+VoVcdu9uw/RPj4Wng9BgN5+ZaEypp98 kyzVTJc/oe3FrSNPL3s5YyKr+vd+wK7iEw90fRypaYVnOxzZYlpp44aCyNyfE92ly0Mo 97x38xS12Cl7cR671NubUkIt2IKKRCkYKey0REHky1wv4TWrdkSJJ/oCjw0VQmTFFYK9 BNcA==; dara=google.com ARC-Authentication-Results: i=2; mx.google.com; arc=pass (i=1 spf=pass spfdomain=loongson.cn); spf=pass (google.com: domain of linux-kernel+bounces-150944-linux.lists.archive=gmail.com@vger.kernel.org designates 139.178.88.99 as permitted sender) smtp.mailfrom="linux-kernel+bounces-150944-linux.lists.archive=gmail.com@vger.kernel.org" Return-Path: Received: from sv.mirrors.kernel.org (sv.mirrors.kernel.org. [139.178.88.99]) by mx.google.com with ESMTPS id d12-20020a63f24c000000b005dc905460f4si2473756pgk.3.2024.04.18.19.35.08 for (version=TLS1_3 cipher=TLS_AES_256_GCM_SHA384 bits=256/256); Thu, 18 Apr 2024 19:35:08 -0700 (PDT) Received-SPF: pass (google.com: domain of linux-kernel+bounces-150944-linux.lists.archive=gmail.com@vger.kernel.org designates 139.178.88.99 as permitted sender) client-ip=139.178.88.99; Authentication-Results: mx.google.com; arc=pass (i=1 spf=pass spfdomain=loongson.cn); spf=pass (google.com: domain of linux-kernel+bounces-150944-linux.lists.archive=gmail.com@vger.kernel.org designates 139.178.88.99 as permitted sender) smtp.mailfrom="linux-kernel+bounces-150944-linux.lists.archive=gmail.com@vger.kernel.org" Received: from smtp.subspace.kernel.org (wormhole.subspace.kernel.org [52.25.139.140]) (using TLSv1.2 with cipher ECDHE-RSA-AES256-GCM-SHA384 (256/256 bits)) (No client certificate requested) by sv.mirrors.kernel.org (Postfix) with ESMTPS id 89BE5283B0A for ; Fri, 19 Apr 2024 02:35:08 +0000 (UTC) Received: from localhost.localdomain (localhost.localdomain [127.0.0.1]) by smtp.subspace.kernel.org (Postfix) with ESMTP id 9F8375664; Fri, 19 Apr 2024 02:35:01 +0000 (UTC) Received: from mail.loongson.cn (mail.loongson.cn [114.242.206.163]) by smtp.subspace.kernel.org (Postfix) with ESMTP id EC977EC2; Fri, 19 Apr 2024 02:34:56 +0000 (UTC) Authentication-Results: smtp.subspace.kernel.org; arc=none smtp.client-ip=114.242.206.163 ARC-Seal:i=1; a=rsa-sha256; d=subspace.kernel.org; s=arc-20240116; t=1713494100; cv=none; b=eDCJNoVAvM+alcRqNaLlVP90p0qh5QiyqxEU/f8I4Xn+ZdUCLx7t5lxBlMXD1Y2lr5Q7MSoLk3xNaoqUYzIvI8zUyjJq05aofWTn0c6dFUG8GLM22ljzLZJNZiH7y8DYDKwmtyEp8naq7BdF5W+95mS5uRtmY4tJEphAWSqDHNA= ARC-Message-Signature:i=1; a=rsa-sha256; d=subspace.kernel.org; s=arc-20240116; t=1713494100; c=relaxed/simple; bh=yvM8HF7cnA2nA+GphFhM2X2UQDN3BhWtVkoFXB7geUo=; h=Subject:From:To:Cc:References:Message-ID:Date:MIME-Version: In-Reply-To:Content-Type; b=B1MuMpimCJ6z8W7oV+cClHPSww3vspK3musQyyp2dwF1YQHGxCtv4hzUhR/U38JYlXJnAbQz2/4csr/QZILIqKWmsddwHyLpeBMswFMnBEGMPL8H1Qk8yru8NVNMa7cOMqH+kopmFTjwIY1f/n1MHyAQn1XgMz/TqFAjBepa4rY= ARC-Authentication-Results:i=1; smtp.subspace.kernel.org; dmarc=none (p=none dis=none) header.from=loongson.cn; spf=pass smtp.mailfrom=loongson.cn; arc=none smtp.client-ip=114.242.206.163 Authentication-Results: smtp.subspace.kernel.org; dmarc=none (p=none dis=none) header.from=loongson.cn Authentication-Results: smtp.subspace.kernel.org; spf=pass smtp.mailfrom=loongson.cn Received: from loongson.cn (unknown [10.20.42.173]) by gateway (Coremail) with SMTP id _____8Cx2uhO2CFmzJkpAA--.9058S3; Fri, 19 Apr 2024 10:34:54 +0800 (CST) Received: from [10.20.42.173] (unknown [10.20.42.173]) by localhost.localdomain (Coremail) with SMTP id AQAAf8CxPs9I2CFm6sR_AA--.45636S3; Fri, 19 Apr 2024 10:34:51 +0800 (CST) Subject: Re: [PATCH v3] perf kvm: Add kvm-stat for loongarch64 From: maobibo To: Ming Wang , Huacai Chen Cc: Peter Zijlstra , Ingo Molnar , linux-kernel@vger.kernel.org, linux-perf-users@vger.kernel.org, Ian Rogers References: <20240411121047.2005033-1-maobibo@loongson.cn> Message-ID: Date: Fri, 19 Apr 2024 10:34:48 +0800 User-Agent: Mozilla/5.0 (X11; Linux loongarch64; rv:68.0) Gecko/20100101 Thunderbird/68.7.0 Precedence: bulk X-Mailing-List: linux-kernel@vger.kernel.org List-Id: List-Subscribe: List-Unsubscribe: MIME-Version: 1.0 In-Reply-To: <20240411121047.2005033-1-maobibo@loongson.cn> Content-Type: text/plain; charset=utf-8; format=flowed Content-Language: en-US Content-Transfer-Encoding: 8bit X-CM-TRANSID:AQAAf8CxPs9I2CFm6sR_AA--.45636S3 X-CM-SenderInfo: xpdruxter6z05rqj20fqof0/ X-Coremail-Antispam: 1Uk129KBj93XoWxKFW5tw45WF1DAF4kJr1kJFc_yoWfZrWkpF WjyrnxKr4rJF17Cwn7AFn093Z8Jw4xuFyjga47XrW7AFsxZr9rGa1IqFnxGFyayw4kJF40 9w1F9w40kFW8JabCm3ZEXasCq-sJn29KB7ZKAUJUUUU7529EdanIXcx71UUUUU7KY7ZEXa sCq-sGcSsGvfJ3Ic02F40EFcxC0VAKzVAqx4xG6I80ebIjqfuFe4nvWSU5nxnvy29KBjDU 0xBIdaVrnRJUUUBYb4IE77IF4wAFF20E14v26r1j6r4UM7CY07I20VC2zVCF04k26cxKx2 IYs7xG6rWj6s0DM7CIcVAFz4kK6r1Y6r17M28lY4IEw2IIxxk0rwA2F7IY1VAKz4vEj48v e4kI8wA2z4x0Y4vE2Ix0cI8IcVAFwI0_Gr0_Xr1l84ACjcxK6xIIjxv20xvEc7CjxVAFwI 0_Gr0_Cr1l84ACjcxK6I8E87Iv67AKxVWxJVW8Jr1l84ACjcxK6I8E87Iv6xkF7I0E14v2 6r4UJVWxJr1ln4kS14v26r1Y6r17M2AIxVAIcxkEcVAq07x20xvEncxIr21l57IF6xkI12 xvs2x26I8E6xACxx1l5I8CrVACY4xI64kE6c02F40Ex7xfMcIj6xIIjxv20xvE14v26r12 6r1DMcIj6I8E87Iv67AKxVWUJVW8JwAm72CE4IkC6x0Yz7v_Jr0_Gr1lF7xvr2IY64vIr4 1lc7I2V7IY0VAS07AlzVAYIcxG8wCF04k20xvY0x0EwIxGrwCFx2IqxVCFs4IE7xkEbVWU JVW8JwCFI7km07C267AKxVWUXVWUAwC20s026c02F40E14v26r1j6r18MI8I3I0E7480Y4 vE14v26r106r1rMI8E67AF67kF1VAFwI0_JF0_Jw1lIxkGc2Ij64vIr41lIxAIcVC0I7IY x2IY67AKxVWUJVWUCwCI42IY6xIIjxv20xvEc7CjxVAFwI0_Jr0_Gr1lIxAIcVCF04k26c xKx2IYs7xG6r1j6r1xMIIF0xvEx4A2jsIE14v26r1j6r4UMIIF0xvEx4A2jsIEc7CjxVAF wI0_Jr0_GrUvcSsGvfC2KfnxnUUI43ZEXa7IU8q2NtUUUUU== slightly ping... On 2024/4/11 下午8:10, Bibo Mao wrote: > Add support for 'perf kvm stat' on loongarch64 platform, now only > kvm exit event is supported. > > Here is example output about "perf kvm --host stat report" command > > Event name Samples Sample% Time (ns) Time% Mean Time (ns) > Mem store 83969 51.00% 625697070 8.00% 7451 > Mem read 37641 22.00% 112485730 1.00% 2988 > Interrupt 15542 9.00% 20620190 0.00% 1326 > Iocsr 15207 9.00% 94296190 1.00% 6200 > Hypercall 4873 2.00% 12265280 0.00% 2516 > Idle 3713 2.00% 6322055860 87.00% 1702681 > FPU 1819 1.00% 2750300 0.00% 1511 > Ifecth 502 0.00% 1341740 0.00% 2672 > Mem modify 324 0.00% 602240 0.00% 1858 > Cpucfg 55 0.00% 77610 0.00% 1411 > Csr 12 0.00% 19690 0.00% 1640 > LASX 3 0.00% 4870 0.00% 1623 > LSX 2 0.00% 2100 0.00% 1050 > > Signed-off-by: Bibo Mao > --- > v2 --- v3: > 1. Add NULL check with cpuid in function get_cpuid() > 2. Add example output from /proc/cpuinfo before function get_cpuid() > v1 --- v2: > 1. Add child_ops for kvm exit event, split kvm:kvm_exit_gspr events > into cpucfg/csr/iocsr/idle child events by decoding detailed gspr > instruction. > 2. Remove some exception code type which does not happen in current > kvm implementation, such as meomry NR/NX/priviledge exception. > --- > tools/perf/arch/loongarch/Makefile | 1 + > tools/perf/arch/loongarch/util/Build | 2 + > tools/perf/arch/loongarch/util/header.c | 88 ++++++++++++++ > tools/perf/arch/loongarch/util/kvm-stat.c | 135 ++++++++++++++++++++++ > 4 files changed, 226 insertions(+) > create mode 100644 tools/perf/arch/loongarch/util/header.c > create mode 100644 tools/perf/arch/loongarch/util/kvm-stat.c > > diff --git a/tools/perf/arch/loongarch/Makefile b/tools/perf/arch/loongarch/Makefile > index 3992a67a87d9..c89d6bb6b184 100644 > --- a/tools/perf/arch/loongarch/Makefile > +++ b/tools/perf/arch/loongarch/Makefile > @@ -4,6 +4,7 @@ PERF_HAVE_DWARF_REGS := 1 > endif > PERF_HAVE_ARCH_REGS_QUERY_REGISTER_OFFSET := 1 > PERF_HAVE_JITDUMP := 1 > +HAVE_KVM_STAT_SUPPORT := 1 > > # > # Syscall table generation for perf > diff --git a/tools/perf/arch/loongarch/util/Build b/tools/perf/arch/loongarch/util/Build > index d776125a2d06..b12d374d7096 100644 > --- a/tools/perf/arch/loongarch/util/Build > +++ b/tools/perf/arch/loongarch/util/Build > @@ -1,5 +1,7 @@ > +perf-y += header.o > perf-y += perf_regs.o > > perf-$(CONFIG_DWARF) += dwarf-regs.o > perf-$(CONFIG_LOCAL_LIBUNWIND) += unwind-libunwind.o > perf-$(CONFIG_LIBDW_DWARF_UNWIND) += unwind-libdw.o > +perf-$(CONFIG_LIBTRACEEVENT) += kvm-stat.o > diff --git a/tools/perf/arch/loongarch/util/header.c b/tools/perf/arch/loongarch/util/header.c > new file mode 100644 > index 000000000000..a4ed732b49c6 > --- /dev/null > +++ b/tools/perf/arch/loongarch/util/header.c > @@ -0,0 +1,88 @@ > +// SPDX-License-Identifier: GPL-2.0-only > +/* > + * Implementation of get_cpuid(). > + * > + * Author: Nikita Shubin > + */ > + > +#include > +#include > +#include > +#include > +#include "util/debug.h" > +#include "util/header.h" > + > +/* > + * Output example from /proc/cpuinfo > + * CPU Family : Loongson-64bit > + * Model Name : Loongson-3C5000 > + * CPU Revision : 0x11 > + */ > +#define CPUINFO_MODEL "Model Name" > +#define CPUINFO "/proc/cpuinfo" > +static char *_get_field(const char *line) > +{ > + char *line2, *nl; > + > + line2 = strrchr(line, ' '); > + if (!line2) > + return NULL; > + > + line2++; > + nl = strrchr(line, '\n'); > + if (!nl) > + return NULL; > + > + return strndup(line2, nl - line2); > +} > + > +static char *__get_cpuid(void) > +{ > + char *line, *model, *cpuid; > + unsigned long line_sz; > + FILE *file; > + > + file = fopen(CPUINFO, "r"); > + if (file == NULL) > + return cpuid; > + > + line = model = cpuid = NULL; > + while (getline(&line, &line_sz, file) != -1) { > + if (strncmp(line, CPUINFO_MODEL, strlen(CPUINFO_MODEL))) > + continue; > + > + model = _get_field(line); > + if (!model) > + goto free; > + break; > + } > + > + if (model && (asprintf(&cpuid, "%s", model) < 0)) > + cpuid = NULL; > + > +free: > + fclose(file); > + free(model); > + return cpuid; > +} > + > +int get_cpuid(char *buffer, size_t sz) > +{ > + char *cpuid = __get_cpuid(); > + int ret = 0; > + > + if (!cpuid) > + return EINVAL; > + > + if (sz >= strlen(cpuid)) > + scnprintf(buffer, sz, "%s", cpuid); > + else > + ret = ENOBUFS; > + free(cpuid); > + return ret; > +} > + > +char *get_cpuid_str(struct perf_pmu *pmu __maybe_unused) > +{ > + return __get_cpuid(); > +} > diff --git a/tools/perf/arch/loongarch/util/kvm-stat.c b/tools/perf/arch/loongarch/util/kvm-stat.c > new file mode 100644 > index 000000000000..cc50adb0835a > --- /dev/null > +++ b/tools/perf/arch/loongarch/util/kvm-stat.c > @@ -0,0 +1,135 @@ > +// SPDX-License-Identifier: GPL-2.0 > +#include > +#include > +#include > +#include "util/kvm-stat.h" > +#include "util/parse-events.h" > +#include "util/debug.h" > +#include "util/evsel.h" > +#include "util/evlist.h" > +#include "util/pmus.h" > + > +#define LOONGARCH_EXCEPTION_INT 0 > +#define LOONGARCH_EXCEPTION_PIL 1 > +#define LOONGARCH_EXCEPTION_PIS 2 > +#define LOONGARCH_EXCEPTION_PIF 3 > +#define LOONGARCH_EXCEPTION_PME 4 > +#define LOONGARCH_EXCEPTION_FPD 15 > +#define LOONGARCH_EXCEPTION_SXD 16 > +#define LOONGARCH_EXCEPTION_ASXD 17 > +#define LOONGARCH_EXCEPTION_GSPR 22 > +#define LOONGARCH_EXCEPTION_CPUCFG 100 > +#define LOONGARCH_EXCEPTION_CSR 101 > +#define LOONGARCH_EXCEPTION_IOCSR 102 > +#define LOONGARCH_EXCEPTION_IDLE 103 > +#define LOONGARCH_EXCEPTION_OTHERS 104 > +#define LOONGARCH_EXCEPTION_HVC 23 > + > +#define loongarch_exception_type \ > + {LOONGARCH_EXCEPTION_INT, "Interrupt" }, \ > + {LOONGARCH_EXCEPTION_PIL, "Mem read" }, \ > + {LOONGARCH_EXCEPTION_PIS, "Mem store" }, \ > + {LOONGARCH_EXCEPTION_PIF, "Ifecth" }, \ > + {LOONGARCH_EXCEPTION_PME, "Mem modify" }, \ > + {LOONGARCH_EXCEPTION_FPD, "FPU" }, \ > + {LOONGARCH_EXCEPTION_SXD, "LSX" }, \ > + {LOONGARCH_EXCEPTION_ASXD, "LASX" }, \ > + {LOONGARCH_EXCEPTION_GSPR, "Privilege Error" }, \ > + {LOONGARCH_EXCEPTION_HVC, "Hypercall" }, \ > + {LOONGARCH_EXCEPTION_CPUCFG, "Cpucfg" }, \ > + {LOONGARCH_EXCEPTION_CSR, "Csr" }, \ > + {LOONGARCH_EXCEPTION_IOCSR, "Iocsr" }, \ > + {LOONGARCH_EXCEPTION_IDLE, "Idle" }, \ > + {LOONGARCH_EXCEPTION_OTHERS, "Others" } > + > +define_exit_reasons_table(loongarch_exit_reasons, loongarch_exception_type); > + > +const char *vcpu_id_str = "vcpu_id"; > +const char *kvm_exit_reason = "reason"; > +const char *kvm_entry_trace = "kvm:kvm_enter"; > +const char *kvm_reenter_trace = "kvm:kvm_reenter"; > +const char *kvm_exit_trace = "kvm:kvm_exit"; > +const char *kvm_events_tp[] = { > + "kvm:kvm_enter", > + "kvm:kvm_reenter", > + "kvm:kvm_exit", > + "kvm:kvm_exit_gspr", > + NULL, > +}; > + > +static bool event_end(struct evsel *evsel, > + struct perf_sample *sample __maybe_unused, > + struct event_key *key __maybe_unused) > +{ > + /* > + * LoongArch kvm is different with other architectures > + * > + * There is kvm:kvm_reenter or kvm:kvm_enter event adjacent with > + * kvm:kvm_exit event. > + * kvm:kvm_reenter means returning to guest immediately > + * kvm:kvm_enter means returning to vmm and then to guest > + */ > + return evsel__name_is(evsel, kvm_entry_trace) || > + evsel__name_is(evsel, kvm_reenter_trace); > +} > + > +static void event_gspr_get_key(struct evsel *evsel, > + struct perf_sample *sample, > + struct event_key *key) > +{ > + unsigned int insn; > + > + key->key = LOONGARCH_EXCEPTION_OTHERS; > + insn = evsel__intval(evsel, sample, "inst_word"); > + switch (insn >> 24) { > + case 0: > + /* cpucfg inst trap */ > + if ((insn >> 10) == 0x1b) > + key->key = LOONGARCH_EXCEPTION_CPUCFG; > + break; > + case 4: > + /* csr inst trap */ > + key->key = LOONGARCH_EXCEPTION_CSR; > + break; > + case 6: > + /* iocsr inst trap */ > + if ((insn >> 15) == 0xc90) > + key->key = LOONGARCH_EXCEPTION_IOCSR; > + else if ((insn >> 15) == 0xc91) > + /* idle inst trap */ > + key->key = LOONGARCH_EXCEPTION_IDLE; > + break; > + default: > + key->key = LOONGARCH_EXCEPTION_OTHERS; > + break; > + } > +} > + > +static struct child_event_ops child_events[] = { > + { .name = "kvm:kvm_exit_gspr", .get_key = event_gspr_get_key }, > + { NULL, NULL }, > +}; > + > +static struct kvm_events_ops exit_events = { > + .is_begin_event = exit_event_begin, > + .is_end_event = event_end, > + .child_ops = child_events, > + .decode_key = exit_event_decode_key, > + .name = "VM-EXIT" > +}; > + > +struct kvm_reg_events_ops kvm_reg_events_ops[] = { > + { .name = "vmexit", .ops = &exit_events, }, > + { NULL, NULL }, > +}; > + > +const char * const kvm_skip_events[] = { > + NULL, > +}; > + > +int cpu_isa_init(struct perf_kvm_stat *kvm, const char *cpuid __maybe_unused) > +{ > + kvm->exit_reasons_isa = "loongarch64"; > + kvm->exit_reasons = loongarch_exit_reasons; > + return 0; > +} > > base-commit: 2c71fdf02a95b3dd425b42f28fd47fb2b1d22702 >