472 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			472 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| // SPDX-License-Identifier: GPL-2.0
 | |
| /*
 | |
|  * perf iostat
 | |
|  *
 | |
|  * Copyright (C) 2020, Intel Corporation
 | |
|  *
 | |
|  * Authors: Alexander Antonov <alexander.antonov@linux.intel.com>
 | |
|  */
 | |
| 
 | |
| #include <api/fs/fs.h>
 | |
| #include <linux/kernel.h>
 | |
| #include <linux/err.h>
 | |
| #include <linux/zalloc.h>
 | |
| #include <limits.h>
 | |
| #include <stdio.h>
 | |
| #include <string.h>
 | |
| #include <errno.h>
 | |
| #include <sys/types.h>
 | |
| #include <sys/stat.h>
 | |
| #include <fcntl.h>
 | |
| #include <dirent.h>
 | |
| #include <unistd.h>
 | |
| #include <stdlib.h>
 | |
| #include <regex.h>
 | |
| #include "util/cpumap.h"
 | |
| #include "util/debug.h"
 | |
| #include "util/iostat.h"
 | |
| #include "util/counts.h"
 | |
| #include "path.h"
 | |
| 
 | |
| #ifndef MAX_PATH
 | |
| #define MAX_PATH 1024
 | |
| #endif
 | |
| 
 | |
| #define UNCORE_IIO_PMU_PATH	"devices/uncore_iio_%d"
 | |
| #define SYSFS_UNCORE_PMU_PATH	"%s/"UNCORE_IIO_PMU_PATH
 | |
| #define PLATFORM_MAPPING_PATH	UNCORE_IIO_PMU_PATH"/die%d"
 | |
| 
 | |
| /*
 | |
|  * Each metric requiries one IIO event which increments at every 4B transfer
 | |
|  * in corresponding direction. The formulas to compute metrics are generic:
 | |
|  *     #EventCount * 4B / (1024 * 1024)
 | |
|  */
 | |
| static const char * const iostat_metrics[] = {
 | |
| 	"Inbound Read(MB)",
 | |
| 	"Inbound Write(MB)",
 | |
| 	"Outbound Read(MB)",
 | |
| 	"Outbound Write(MB)",
 | |
| };
 | |
| 
 | |
| static inline int iostat_metrics_count(void)
 | |
| {
 | |
| 	return sizeof(iostat_metrics) / sizeof(char *);
 | |
| }
 | |
| 
 | |
| static const char *iostat_metric_by_idx(int idx)
 | |
| {
 | |
| 	return *(iostat_metrics + idx % iostat_metrics_count());
 | |
| }
 | |
| 
 | |
| struct iio_root_port {
 | |
| 	u32 domain;
 | |
| 	u8 bus;
 | |
| 	u8 die;
 | |
| 	u8 pmu_idx;
 | |
| 	int idx;
 | |
| };
 | |
| 
 | |
| struct iio_root_ports_list {
 | |
| 	struct iio_root_port **rps;
 | |
| 	int nr_entries;
 | |
| };
 | |
| 
 | |
| static struct iio_root_ports_list *root_ports;
 | |
| 
 | |
| static void iio_root_port_show(FILE *output,
 | |
| 			       const struct iio_root_port * const rp)
 | |
| {
 | |
| 	if (output && rp)
 | |
| 		fprintf(output, "S%d-uncore_iio_%d<%04x:%02x>\n",
 | |
| 			rp->die, rp->pmu_idx, rp->domain, rp->bus);
 | |
| }
 | |
| 
 | |
| static struct iio_root_port *iio_root_port_new(u32 domain, u8 bus,
 | |
| 					       u8 die, u8 pmu_idx)
 | |
| {
 | |
| 	struct iio_root_port *p = calloc(1, sizeof(*p));
 | |
| 
 | |
| 	if (p) {
 | |
| 		p->domain = domain;
 | |
| 		p->bus = bus;
 | |
| 		p->die = die;
 | |
| 		p->pmu_idx = pmu_idx;
 | |
| 	}
 | |
| 	return p;
 | |
| }
 | |
| 
 | |
| static void iio_root_ports_list_free(struct iio_root_ports_list *list)
 | |
| {
 | |
| 	int idx;
 | |
| 
 | |
| 	if (list) {
 | |
| 		for (idx = 0; idx < list->nr_entries; idx++)
 | |
| 			zfree(&list->rps[idx]);
 | |
| 		zfree(&list->rps);
 | |
| 		free(list);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| static struct iio_root_port *iio_root_port_find_by_notation(
 | |
| 	const struct iio_root_ports_list * const list, u32 domain, u8 bus)
 | |
| {
 | |
| 	int idx;
 | |
| 	struct iio_root_port *rp;
 | |
| 
 | |
| 	if (list) {
 | |
| 		for (idx = 0; idx < list->nr_entries; idx++) {
 | |
| 			rp = list->rps[idx];
 | |
| 			if (rp && rp->domain == domain && rp->bus == bus)
 | |
| 				return rp;
 | |
| 		}
 | |
| 	}
 | |
| 	return NULL;
 | |
| }
 | |
| 
 | |
| static int iio_root_ports_list_insert(struct iio_root_ports_list *list,
 | |
| 				      struct iio_root_port * const rp)
 | |
| {
 | |
| 	struct iio_root_port **tmp_buf;
 | |
| 
 | |
| 	if (list && rp) {
 | |
| 		rp->idx = list->nr_entries++;
 | |
| 		tmp_buf = realloc(list->rps,
 | |
| 				  list->nr_entries * sizeof(*list->rps));
 | |
| 		if (!tmp_buf) {
 | |
| 			pr_err("Failed to realloc memory\n");
 | |
| 			return -ENOMEM;
 | |
| 		}
 | |
| 		tmp_buf[rp->idx] = rp;
 | |
| 		list->rps = tmp_buf;
 | |
| 	}
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int iio_mapping(u8 pmu_idx, struct iio_root_ports_list * const list)
 | |
| {
 | |
| 	char *buf;
 | |
| 	char path[MAX_PATH];
 | |
| 	u32 domain;
 | |
| 	u8 bus;
 | |
| 	struct iio_root_port *rp;
 | |
| 	size_t size;
 | |
| 	int ret;
 | |
| 
 | |
| 	for (int die = 0; die < cpu__max_node(); die++) {
 | |
| 		scnprintf(path, MAX_PATH, PLATFORM_MAPPING_PATH, pmu_idx, die);
 | |
| 		if (sysfs__read_str(path, &buf, &size) < 0) {
 | |
| 			if (pmu_idx)
 | |
| 				goto out;
 | |
| 			pr_err("Mode iostat is not supported\n");
 | |
| 			return -1;
 | |
| 		}
 | |
| 		ret = sscanf(buf, "%04x:%02hhx", &domain, &bus);
 | |
| 		free(buf);
 | |
| 		if (ret != 2) {
 | |
| 			pr_err("Invalid mapping data: iio_%d; die%d\n",
 | |
| 			       pmu_idx, die);
 | |
| 			return -1;
 | |
| 		}
 | |
| 		rp = iio_root_port_new(domain, bus, die, pmu_idx);
 | |
| 		if (!rp || iio_root_ports_list_insert(list, rp)) {
 | |
| 			free(rp);
 | |
| 			return -ENOMEM;
 | |
| 		}
 | |
| 	}
 | |
| out:
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static u8 iio_pmu_count(void)
 | |
| {
 | |
| 	u8 pmu_idx = 0;
 | |
| 	char path[MAX_PATH];
 | |
| 	const char *sysfs = sysfs__mountpoint();
 | |
| 
 | |
| 	if (sysfs) {
 | |
| 		for (;; pmu_idx++) {
 | |
| 			snprintf(path, sizeof(path), SYSFS_UNCORE_PMU_PATH,
 | |
| 				 sysfs, pmu_idx);
 | |
| 			if (access(path, F_OK) != 0)
 | |
| 				break;
 | |
| 		}
 | |
| 	}
 | |
| 	return pmu_idx;
 | |
| }
 | |
| 
 | |
| static int iio_root_ports_scan(struct iio_root_ports_list **list)
 | |
| {
 | |
| 	int ret = -ENOMEM;
 | |
| 	struct iio_root_ports_list *tmp_list;
 | |
| 	u8 pmu_count = iio_pmu_count();
 | |
| 
 | |
| 	if (!pmu_count) {
 | |
| 		pr_err("Unsupported uncore pmu configuration\n");
 | |
| 		return -1;
 | |
| 	}
 | |
| 
 | |
| 	tmp_list = calloc(1, sizeof(*tmp_list));
 | |
| 	if (!tmp_list)
 | |
| 		goto err;
 | |
| 
 | |
| 	for (u8 pmu_idx = 0; pmu_idx < pmu_count; pmu_idx++) {
 | |
| 		ret = iio_mapping(pmu_idx, tmp_list);
 | |
| 		if (ret)
 | |
| 			break;
 | |
| 	}
 | |
| err:
 | |
| 	if (!ret)
 | |
| 		*list = tmp_list;
 | |
| 	else
 | |
| 		iio_root_ports_list_free(tmp_list);
 | |
| 
 | |
| 	return ret;
 | |
| }
 | |
| 
 | |
| static int iio_root_port_parse_str(u32 *domain, u8 *bus, char *str)
 | |
| {
 | |
| 	int ret;
 | |
| 	regex_t regex;
 | |
| 	/*
 | |
| 	 * Expected format domain:bus:
 | |
| 	 * Valid domain range [0:ffff]
 | |
| 	 * Valid bus range [0:ff]
 | |
| 	 * Example: 0000:af, 0:3d, 01:7
 | |
| 	 */
 | |
| 	regcomp(®ex, "^([a-f0-9A-F]{1,}):([a-f0-9A-F]{1,2})", REG_EXTENDED);
 | |
| 	ret = regexec(®ex, str, 0, NULL, 0);
 | |
| 	if (ret || sscanf(str, "%08x:%02hhx", domain, bus) != 2)
 | |
| 		pr_warning("Unrecognized root port format: %s\n"
 | |
| 			   "Please use the following format:\n"
 | |
| 			   "\t [domain]:[bus]\n"
 | |
| 			   "\t for example: 0000:3d\n", str);
 | |
| 
 | |
| 	regfree(®ex);
 | |
| 	return ret;
 | |
| }
 | |
| 
 | |
| static int iio_root_ports_list_filter(struct iio_root_ports_list **list,
 | |
| 				      const char *filter)
 | |
| {
 | |
| 	char *tok, *tmp, *filter_copy = NULL;
 | |
| 	struct iio_root_port *rp;
 | |
| 	u32 domain;
 | |
| 	u8 bus;
 | |
| 	int ret = -ENOMEM;
 | |
| 	struct iio_root_ports_list *tmp_list = calloc(1, sizeof(*tmp_list));
 | |
| 
 | |
| 	if (!tmp_list)
 | |
| 		goto err;
 | |
| 
 | |
| 	filter_copy = strdup(filter);
 | |
| 	if (!filter_copy)
 | |
| 		goto err;
 | |
| 
 | |
| 	for (tok = strtok_r(filter_copy, ",", &tmp); tok;
 | |
| 	     tok = strtok_r(NULL, ",", &tmp)) {
 | |
| 		if (!iio_root_port_parse_str(&domain, &bus, tok)) {
 | |
| 			rp = iio_root_port_find_by_notation(*list, domain, bus);
 | |
| 			if (rp) {
 | |
| 				(*list)->rps[rp->idx] = NULL;
 | |
| 				ret = iio_root_ports_list_insert(tmp_list, rp);
 | |
| 				if (ret) {
 | |
| 					free(rp);
 | |
| 					goto err;
 | |
| 				}
 | |
| 			} else if (!iio_root_port_find_by_notation(tmp_list,
 | |
| 								   domain, bus))
 | |
| 				pr_warning("Root port %04x:%02x were not found\n",
 | |
| 					   domain, bus);
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if (tmp_list->nr_entries == 0) {
 | |
| 		pr_err("Requested root ports were not found\n");
 | |
| 		ret = -EINVAL;
 | |
| 	}
 | |
| err:
 | |
| 	iio_root_ports_list_free(*list);
 | |
| 	if (ret)
 | |
| 		iio_root_ports_list_free(tmp_list);
 | |
| 	else
 | |
| 		*list = tmp_list;
 | |
| 
 | |
| 	free(filter_copy);
 | |
| 	return ret;
 | |
| }
 | |
| 
 | |
| static int iostat_event_group(struct evlist *evl,
 | |
| 			      struct iio_root_ports_list *list)
 | |
| {
 | |
| 	int ret;
 | |
| 	int idx;
 | |
| 	const char *iostat_cmd_template =
 | |
| 	"{uncore_iio_%x/event=0x83,umask=0x04,ch_mask=0xF,fc_mask=0x07/,\
 | |
| 	  uncore_iio_%x/event=0x83,umask=0x01,ch_mask=0xF,fc_mask=0x07/,\
 | |
| 	  uncore_iio_%x/event=0xc0,umask=0x04,ch_mask=0xF,fc_mask=0x07/,\
 | |
| 	  uncore_iio_%x/event=0xc0,umask=0x01,ch_mask=0xF,fc_mask=0x07/}";
 | |
| 	const int len_template = strlen(iostat_cmd_template) + 1;
 | |
| 	struct evsel *evsel = NULL;
 | |
| 	int metrics_count = iostat_metrics_count();
 | |
| 	char *iostat_cmd = calloc(len_template, 1);
 | |
| 
 | |
| 	if (!iostat_cmd)
 | |
| 		return -ENOMEM;
 | |
| 
 | |
| 	for (idx = 0; idx < list->nr_entries; idx++) {
 | |
| 		sprintf(iostat_cmd, iostat_cmd_template,
 | |
| 			list->rps[idx]->pmu_idx, list->rps[idx]->pmu_idx,
 | |
| 			list->rps[idx]->pmu_idx, list->rps[idx]->pmu_idx);
 | |
| 		ret = parse_event(evl, iostat_cmd);
 | |
| 		if (ret)
 | |
| 			goto err;
 | |
| 	}
 | |
| 
 | |
| 	evlist__for_each_entry(evl, evsel) {
 | |
| 		evsel->priv = list->rps[evsel->core.idx / metrics_count];
 | |
| 	}
 | |
| 	list->nr_entries = 0;
 | |
| err:
 | |
| 	iio_root_ports_list_free(list);
 | |
| 	free(iostat_cmd);
 | |
| 	return ret;
 | |
| }
 | |
| 
 | |
| int iostat_prepare(struct evlist *evlist, struct perf_stat_config *config)
 | |
| {
 | |
| 	if (evlist->core.nr_entries > 0) {
 | |
| 		pr_warning("The -e and -M options are not supported."
 | |
| 			   "All chosen events/metrics will be dropped\n");
 | |
| 		evlist__delete(evlist);
 | |
| 		evlist = evlist__new();
 | |
| 		if (!evlist)
 | |
| 			return -ENOMEM;
 | |
| 	}
 | |
| 
 | |
| 	config->metric_only = true;
 | |
| 	config->aggr_mode = AGGR_GLOBAL;
 | |
| 
 | |
| 	return iostat_event_group(evlist, root_ports);
 | |
| }
 | |
| 
 | |
| int iostat_parse(const struct option *opt, const char *str,
 | |
| 		 int unset __maybe_unused)
 | |
| {
 | |
| 	int ret;
 | |
| 	struct perf_stat_config *config = (struct perf_stat_config *)opt->data;
 | |
| 
 | |
| 	ret = iio_root_ports_scan(&root_ports);
 | |
| 	if (!ret) {
 | |
| 		config->iostat_run = true;
 | |
| 		if (!str)
 | |
| 			iostat_mode = IOSTAT_RUN;
 | |
| 		else if (!strcmp(str, "list"))
 | |
| 			iostat_mode = IOSTAT_LIST;
 | |
| 		else {
 | |
| 			iostat_mode = IOSTAT_RUN;
 | |
| 			ret = iio_root_ports_list_filter(&root_ports, str);
 | |
| 		}
 | |
| 	}
 | |
| 	return ret;
 | |
| }
 | |
| 
 | |
| void iostat_list(struct evlist *evlist, struct perf_stat_config *config)
 | |
| {
 | |
| 	struct evsel *evsel;
 | |
| 	struct iio_root_port *rp = NULL;
 | |
| 
 | |
| 	evlist__for_each_entry(evlist, evsel) {
 | |
| 		if (rp != evsel->priv) {
 | |
| 			rp = evsel->priv;
 | |
| 			iio_root_port_show(config->output, rp);
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void iostat_release(struct evlist *evlist)
 | |
| {
 | |
| 	struct evsel *evsel;
 | |
| 	struct iio_root_port *rp = NULL;
 | |
| 
 | |
| 	evlist__for_each_entry(evlist, evsel) {
 | |
| 		if (rp != evsel->priv) {
 | |
| 			rp = evsel->priv;
 | |
| 			zfree(&evsel->priv);
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void iostat_prefix(struct evlist *evlist,
 | |
| 		   struct perf_stat_config *config,
 | |
| 		   char *prefix, struct timespec *ts)
 | |
| {
 | |
| 	struct iio_root_port *rp = evlist->selected->priv;
 | |
| 
 | |
| 	if (rp) {
 | |
| 		if (ts)
 | |
| 			sprintf(prefix, "%6lu.%09lu%s%04x:%02x%s",
 | |
| 				ts->tv_sec, ts->tv_nsec,
 | |
| 				config->csv_sep, rp->domain, rp->bus,
 | |
| 				config->csv_sep);
 | |
| 		else
 | |
| 			sprintf(prefix, "%04x:%02x%s", rp->domain, rp->bus,
 | |
| 				config->csv_sep);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void iostat_print_header_prefix(struct perf_stat_config *config)
 | |
| {
 | |
| 	if (config->csv_output)
 | |
| 		fputs("port,", config->output);
 | |
| 	else if (config->interval)
 | |
| 		fprintf(config->output, "#          time    port         ");
 | |
| 	else
 | |
| 		fprintf(config->output, "   port         ");
 | |
| }
 | |
| 
 | |
| void iostat_print_metric(struct perf_stat_config *config, struct evsel *evsel,
 | |
| 			 struct perf_stat_output_ctx *out)
 | |
| {
 | |
| 	double iostat_value = 0;
 | |
| 	u64 prev_count_val = 0;
 | |
| 	const char *iostat_metric = iostat_metric_by_idx(evsel->core.idx);
 | |
| 	u8 die = ((struct iio_root_port *)evsel->priv)->die;
 | |
| 	struct perf_counts_values *count = perf_counts(evsel->counts, die, 0);
 | |
| 
 | |
| 	if (count && count->run && count->ena) {
 | |
| 		if (evsel->prev_raw_counts && !out->force_header) {
 | |
| 			struct perf_counts_values *prev_count =
 | |
| 				perf_counts(evsel->prev_raw_counts, die, 0);
 | |
| 
 | |
| 			prev_count_val = prev_count->val;
 | |
| 			prev_count->val = count->val;
 | |
| 		}
 | |
| 		iostat_value = (count->val - prev_count_val) /
 | |
| 			       ((double) count->run / count->ena);
 | |
| 	}
 | |
| 	out->print_metric(config, out->ctx, NULL, "%8.0f", iostat_metric,
 | |
| 			  iostat_value / (256 * 1024));
 | |
| }
 | |
| 
 | |
| void iostat_print_counters(struct evlist *evlist,
 | |
| 			   struct perf_stat_config *config, struct timespec *ts,
 | |
| 			   char *prefix, iostat_print_counter_t print_cnt_cb, void *arg)
 | |
| {
 | |
| 	void *perf_device = NULL;
 | |
| 	struct evsel *counter = evlist__first(evlist);
 | |
| 
 | |
| 	evlist__set_selected(evlist, counter);
 | |
| 	iostat_prefix(evlist, config, prefix, ts);
 | |
| 	fprintf(config->output, "%s", prefix);
 | |
| 	evlist__for_each_entry(evlist, counter) {
 | |
| 		perf_device = evlist->selected->priv;
 | |
| 		if (perf_device && perf_device != counter->priv) {
 | |
| 			evlist__set_selected(evlist, counter);
 | |
| 			iostat_prefix(evlist, config, prefix, ts);
 | |
| 			fprintf(config->output, "\n%s", prefix);
 | |
| 		}
 | |
| 		print_cnt_cb(config, counter, arg);
 | |
| 	}
 | |
| 	fputc('\n', config->output);
 | |
| }
 |