diff options
Diffstat (limited to 'tools/perf/arch/x86/util')
| -rw-r--r-- | tools/perf/arch/x86/util/header.c | 66 | ||||
| -rw-r--r-- | tools/perf/arch/x86/util/intel-pt.c | 11 | ||||
| -rw-r--r-- | tools/perf/arch/x86/util/kvm-stat.c | 2 | 
3 files changed, 77 insertions, 2 deletions
| diff --git a/tools/perf/arch/x86/util/header.c b/tools/perf/arch/x86/util/header.c index fb0d71afee8b..af9a9f2600be 100644 --- a/tools/perf/arch/x86/util/header.c +++ b/tools/perf/arch/x86/util/header.c @@ -4,6 +4,7 @@  #include <stdio.h>  #include <stdlib.h>  #include <string.h> +#include <regex.h>  #include "../../util/header.h" @@ -70,9 +71,72 @@ get_cpuid_str(struct perf_pmu *pmu __maybe_unused)  {  	char *buf = malloc(128); -	if (buf && __get_cpuid(buf, 128, "%s-%u-%X$") < 0) { +	if (buf && __get_cpuid(buf, 128, "%s-%u-%X-%X$") < 0) {  		free(buf);  		return NULL;  	}  	return buf;  } + +/* Full CPUID format for x86 is vendor-family-model-stepping */ +static bool is_full_cpuid(const char *id) +{ +	const char *tmp = id; +	int count = 0; + +	while ((tmp = strchr(tmp, '-')) != NULL) { +		count++; +		tmp++; +	} + +	if (count == 3) +		return true; + +	return false; +} + +int strcmp_cpuid_str(const char *mapcpuid, const char *id) +{ +	regex_t re; +	regmatch_t pmatch[1]; +	int match; +	bool full_mapcpuid = is_full_cpuid(mapcpuid); +	bool full_cpuid = is_full_cpuid(id); + +	/* +	 * Full CPUID format is required to identify a platform. +	 * Error out if the cpuid string is incomplete. +	 */ +	if (full_mapcpuid && !full_cpuid) { +		pr_info("Invalid CPUID %s. Full CPUID is required, " +			"vendor-family-model-stepping\n", id); +		return 1; +	} + +	if (regcomp(&re, mapcpuid, REG_EXTENDED) != 0) { +		/* Warn unable to generate match particular string. */ +		pr_info("Invalid regular expression %s\n", mapcpuid); +		return 1; +	} + +	match = !regexec(&re, id, 1, pmatch, 0); +	regfree(&re); +	if (match) { +		size_t match_len = (pmatch[0].rm_eo - pmatch[0].rm_so); +		size_t cpuid_len; + +		/* If the full CPUID format isn't required, +		 * ignoring the stepping. +		 */ +		if (!full_mapcpuid && full_cpuid) +			cpuid_len = strrchr(id, '-') - id; +		else +			cpuid_len = strlen(id); + +		/* Verify the entire string matched. */ +		if (match_len == cpuid_len) +			return 0; +	} + +	return 1; +} diff --git a/tools/perf/arch/x86/util/intel-pt.c b/tools/perf/arch/x86/util/intel-pt.c index db0ba8caf5a2..ba8ecaf52200 100644 --- a/tools/perf/arch/x86/util/intel-pt.c +++ b/tools/perf/arch/x86/util/intel-pt.c @@ -524,10 +524,21 @@ static int intel_pt_validate_config(struct perf_pmu *intel_pt_pmu,  				    struct perf_evsel *evsel)  {  	int err; +	char c;  	if (!evsel)  		return 0; +	/* +	 * If supported, force pass-through config term (pt=1) even if user +	 * sets pt=0, which avoids senseless kernel errors. +	 */ +	if (perf_pmu__scan_file(intel_pt_pmu, "format/pt", "%c", &c) == 1 && +	    !(evsel->attr.config & 1)) { +		pr_warning("pt=0 doesn't make sense, forcing pt=1\n"); +		evsel->attr.config |= 1; +	} +  	err = intel_pt_val_config_term(intel_pt_pmu, "caps/cycle_thresholds",  				       "cyc_thresh", "caps/psb_cyc",  				       evsel->attr.config); diff --git a/tools/perf/arch/x86/util/kvm-stat.c b/tools/perf/arch/x86/util/kvm-stat.c index b32409a0e546..081353d7b095 100644 --- a/tools/perf/arch/x86/util/kvm-stat.c +++ b/tools/perf/arch/x86/util/kvm-stat.c @@ -156,7 +156,7 @@ int cpu_isa_init(struct perf_kvm_stat *kvm, const char *cpuid)  	if (strstr(cpuid, "Intel")) {  		kvm->exit_reasons = vmx_exit_reasons;  		kvm->exit_reasons_isa = "VMX"; -	} else if (strstr(cpuid, "AMD")) { +	} else if (strstr(cpuid, "AMD") || strstr(cpuid, "Hygon")) {  		kvm->exit_reasons = svm_exit_reasons;  		kvm->exit_reasons_isa = "SVM";  	} else | 
