Blame src/libpfm4/perf_examples/task_smpl.c

Packit Service a1973e
/*
Packit Service a1973e
 * task_smpl.c - example of a task sampling another one using a randomized sampling period
Packit Service a1973e
 *
Packit Service a1973e
 * Copyright (c) 2009 Google, Inc
Packit Service a1973e
 * Contributed by Stephane Eranian <eranian@gmail.com>
Packit Service a1973e
 *
Packit Service a1973e
 * Based on:
Packit Service a1973e
 * Copyright (c) 2003-2006 Hewlett-Packard Development Company, L.P.
Packit Service a1973e
 * Contributed by Stephane Eranian <eranian@hpl.hp.com>
Packit Service a1973e
 *
Packit Service a1973e
 * Permission is hereby granted, free of charge, to any person obtaining a copy
Packit Service a1973e
 * of this software and associated documentation files (the "Software"), to deal
Packit Service a1973e
 * in the Software without restriction, including without limitation the rights
Packit Service a1973e
 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies
Packit Service a1973e
 * of the Software, and to permit persons to whom the Software is furnished to do so,
Packit Service a1973e
 * subject to the following conditions:
Packit Service a1973e
 *
Packit Service a1973e
 * The above copyright notice and this permission notice shall be included in all
Packit Service a1973e
 * copies or substantial portions of the Software.
Packit Service a1973e
 *
Packit Service a1973e
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED,
Packit Service a1973e
 * INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A
Packit Service a1973e
 * PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
Packit Service a1973e
 * HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF
Packit Service a1973e
 * CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE
Packit Service a1973e
 * OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
Packit Service a1973e
 */
Packit Service a1973e
#include <sys/types.h>
Packit Service a1973e
#include <linux/types.h>
Packit Service a1973e
#include <stdio.h>
Packit Service a1973e
#include <stdlib.h>
Packit Service a1973e
#include <stdarg.h>
Packit Service a1973e
#include <unistd.h>
Packit Service a1973e
#include <errno.h>
Packit Service a1973e
#include <string.h>
Packit Service a1973e
#include <signal.h>
Packit Service a1973e
#include <getopt.h>
Packit Service a1973e
#include <setjmp.h>
Packit Service a1973e
#include <sys/wait.h>
Packit Service a1973e
#include <sys/poll.h>
Packit Service a1973e
#include <sys/mman.h>
Packit Service a1973e
#include <locale.h>
Packit Service a1973e
#include <sys/ioctl.h>
Packit Service a1973e
#include <err.h>
Packit Service a1973e
Packit Service a1973e
#include "perf_util.h"
Packit Service a1973e
Packit Service a1973e
#define SMPL_PERIOD	240000000ULL
Packit Service a1973e
Packit Service a1973e
typedef struct {
Packit Service a1973e
	int opt_no_show;
Packit Service a1973e
	int opt_inherit;
Packit Service a1973e
	int mem_mode;
Packit Service a1973e
	int branch_mode;
Packit Service a1973e
	int cpu;
Packit Service a1973e
	int mmap_pages;
Packit Service a1973e
	char *events;
Packit Service a1973e
	FILE *output_file;
Packit Service a1973e
} options_t;
Packit Service a1973e
Packit Service a1973e
static jmp_buf jbuf;
Packit Service a1973e
static uint64_t collected_samples, lost_samples;
Packit Service a1973e
static perf_event_desc_t *fds;
Packit Service a1973e
static int num_fds;
Packit Service a1973e
static options_t options;
Packit Service a1973e
Packit Service a1973e
static struct option the_options[]={
Packit Service a1973e
	{ "help", 0, 0,  1},
Packit Service a1973e
	{ "no-show", 0, &options.opt_no_show, 1},
Packit Service a1973e
	{ 0, 0, 0, 0}
Packit Service a1973e
};
Packit Service a1973e
Packit Service a1973e
static char *gen_events = "cycles,instructions";
Packit Service a1973e
Packit Service a1973e
static void
Packit Service a1973e
cld_handler(int n)
Packit Service a1973e
{
Packit Service a1973e
	longjmp(jbuf, 1);
Packit Service a1973e
}
Packit Service a1973e
Packit Service a1973e
int
Packit Service a1973e
child(char **arg)
Packit Service a1973e
{
Packit Service a1973e
	execvp(arg[0], arg);
Packit Service a1973e
	/* not reached */
Packit Service a1973e
	return -1;
Packit Service a1973e
}
Packit Service a1973e
Packit Service a1973e
struct timeval last_read, this_read;
Packit Service a1973e
Packit Service a1973e
static void
Packit Service a1973e
process_smpl_buf(perf_event_desc_t *hw)
Packit Service a1973e
{
Packit Service a1973e
	struct perf_event_header ehdr;
Packit Service a1973e
	int ret;
Packit Service a1973e
Packit Service a1973e
	for(;;) {
Packit Service a1973e
		ret = perf_read_buffer(hw, &ehdr, sizeof(ehdr));
Packit Service a1973e
		if (ret)
Packit Service a1973e
			return; /* nothing to read */
Packit Service a1973e
Packit Service a1973e
		if (options.opt_no_show) {
Packit Service a1973e
			perf_skip_buffer(hw, ehdr.size - sizeof(ehdr));
Packit Service a1973e
			continue;
Packit Service a1973e
		}
Packit Service a1973e
Packit Service a1973e
		switch(ehdr.type) {
Packit Service a1973e
			case PERF_RECORD_SAMPLE:
Packit Service a1973e
				collected_samples++;
Packit Service a1973e
				ret = perf_display_sample(fds, num_fds, hw - fds, &ehdr, options.output_file);
Packit Service a1973e
				if (ret)
Packit Service a1973e
					errx(1, "cannot parse sample");
Packit Service a1973e
				break;
Packit Service a1973e
			case PERF_RECORD_EXIT:
Packit Service a1973e
				display_exit(hw, options.output_file);
Packit Service a1973e
				break;
Packit Service a1973e
			case PERF_RECORD_LOST:
Packit Service a1973e
				lost_samples += display_lost(hw, fds, num_fds, options.output_file);
Packit Service a1973e
				break;
Packit Service a1973e
			case PERF_RECORD_THROTTLE:
Packit Service a1973e
				display_freq(1, hw, options.output_file);
Packit Service a1973e
				break;
Packit Service a1973e
			case PERF_RECORD_UNTHROTTLE:
Packit Service a1973e
				display_freq(0, hw, options.output_file);
Packit Service a1973e
				break;
Packit Service a1973e
			default:
Packit Service a1973e
				printf("unknown sample type %d\n", ehdr.type);
Packit Service a1973e
				perf_skip_buffer(hw, ehdr.size - sizeof(ehdr));
Packit Service a1973e
		}
Packit Service a1973e
	}
Packit Service a1973e
}
Packit Service a1973e
Packit Service a1973e
int
Packit Service a1973e
mainloop(char **arg)
Packit Service a1973e
{
Packit Service a1973e
	static uint64_t ovfl_count; /* static to avoid setjmp issue */
Packit Service a1973e
	struct pollfd pollfds[1];
Packit Service a1973e
	sigset_t bmask;
Packit Service a1973e
	int go[2], ready[2];
Packit Service a1973e
	size_t pgsz;
Packit Service a1973e
	size_t map_size = 0;
Packit Service a1973e
	pid_t pid;
Packit Service a1973e
	int status, ret;
Packit Service a1973e
	int i;
Packit Service a1973e
	char buf;
Packit Service a1973e
Packit Service a1973e
	if (pfm_initialize() != PFM_SUCCESS)
Packit Service a1973e
		errx(1, "libpfm initialization failed\n");
Packit Service a1973e
Packit Service a1973e
	pgsz = sysconf(_SC_PAGESIZE);
Packit Service a1973e
	map_size = (options.mmap_pages+1)*pgsz;
Packit Service a1973e
Packit Service a1973e
	/*
Packit Service a1973e
	 * does allocate fds
Packit Service a1973e
	 */
Packit Service a1973e
	ret  = perf_setup_list_events(options.events, &fds, &num_fds);
Packit Service a1973e
	if (ret || !num_fds)
Packit Service a1973e
		errx(1, "cannot setup event list");
Packit Service a1973e
Packit Service a1973e
	memset(pollfds, 0, sizeof(pollfds));
Packit Service a1973e
Packit Service a1973e
	ret = pipe(ready);
Packit Service a1973e
	if (ret)
Packit Service a1973e
		err(1, "cannot create pipe ready");
Packit Service a1973e
Packit Service a1973e
	ret = pipe(go);
Packit Service a1973e
	if (ret)
Packit Service a1973e
		err(1, "cannot create pipe go");
Packit Service a1973e
Packit Service a1973e
	/*
Packit Service a1973e
	 * Create the child task
Packit Service a1973e
	 */
Packit Service a1973e
	if ((pid=fork()) == -1)
Packit Service a1973e
		err(1, "cannot fork process\n");
Packit Service a1973e
Packit Service a1973e
	if (pid == 0) {
Packit Service a1973e
		close(ready[0]);
Packit Service a1973e
		close(go[1]);
Packit Service a1973e
Packit Service a1973e
		/*
Packit Service a1973e
		 * let the parent know we exist
Packit Service a1973e
		 */
Packit Service a1973e
		close(ready[1]);
Packit Service a1973e
		if (read(go[0], &buf, 1) == -1)
Packit Service a1973e
			err(1, "unable to read go_pipe");
Packit Service a1973e
Packit Service a1973e
		exit(child(arg));
Packit Service a1973e
	}
Packit Service a1973e
	close(ready[1]);
Packit Service a1973e
	close(go[0]);
Packit Service a1973e
Packit Service a1973e
	if (read(ready[0], &buf, 1) == -1)
Packit Service a1973e
		err(1, "unable to read child_ready_pipe");
Packit Service a1973e
Packit Service a1973e
	close(ready[0]);
Packit Service a1973e
Packit Service a1973e
	fds[0].fd = -1;
Packit Service a1973e
Packit Service a1973e
	if (!fds[0].hw.sample_period)
Packit Service a1973e
		errx(1, "need to set sampling period or freq on first event, use :period= or :freq=");
Packit Service a1973e
Packit Service a1973e
	for(i=0; i < num_fds; i++) {
Packit Service a1973e
Packit Service a1973e
		if (i == 0) {
Packit Service a1973e
			fds[i].hw.disabled = 1;
Packit Service a1973e
			fds[i].hw.enable_on_exec = 1; /* start immediately */
Packit Service a1973e
		} else
Packit Service a1973e
			fds[i].hw.disabled = 0;
Packit Service a1973e
Packit Service a1973e
Packit Service a1973e
		if (options.opt_inherit)
Packit Service a1973e
			fds[i].hw.inherit = 1;
Packit Service a1973e
Packit Service a1973e
		if (fds[i].hw.sample_period) {
Packit Service a1973e
			/*
Packit Service a1973e
			 * set notification threshold to be halfway through the buffer
Packit Service a1973e
			 */
Packit Service a1973e
			fds[i].hw.wakeup_watermark = (options.mmap_pages*pgsz) / 2;
Packit Service a1973e
			fds[i].hw.watermark = 1;
Packit Service a1973e
Packit Service a1973e
			fds[i].hw.sample_type = PERF_SAMPLE_IP|PERF_SAMPLE_TID|PERF_SAMPLE_READ|PERF_SAMPLE_TIME|PERF_SAMPLE_PERIOD;
Packit Service a1973e
			/*
Packit Service a1973e
			 * if we have more than one event, then record event identifier to help with parsing
Packit Service a1973e
			 */
Packit Service a1973e
			if (num_fds > 1)
Packit Service a1973e
				fds[i].hw.sample_type |= PERF_SAMPLE_IDENTIFIER;
Packit Service a1973e
Packit Service a1973e
			fprintf(options.output_file,"%s period=%"PRIu64" freq=%d\n", fds[i].name, fds[i].hw.sample_period, fds[i].hw.freq);
Packit Service a1973e
Packit Service a1973e
			fds[i].hw.read_format = PERF_FORMAT_SCALE;
Packit Service a1973e
Packit Service a1973e
			if (fds[i].hw.freq)
Packit Service a1973e
				fds[i].hw.sample_type |= PERF_SAMPLE_PERIOD;
Packit Service a1973e
Packit Service a1973e
			if (options.mem_mode)
Packit Service a1973e
				fds[i].hw.sample_type |= PERF_SAMPLE_WEIGHT | PERF_SAMPLE_DATA_SRC | PERF_SAMPLE_ADDR;
Packit Service a1973e
			if (options.branch_mode) {
Packit Service a1973e
				fds[i].hw.sample_type |= PERF_SAMPLE_BRANCH_STACK;
Packit Service a1973e
				fds[i].hw.branch_sample_type = PERF_SAMPLE_BRANCH_ANY;
Packit Service a1973e
			}
Packit Service a1973e
		}
Packit Service a1973e
		/*
Packit Service a1973e
		 * we are grouping the events, so there may be a limit
Packit Service a1973e
		 */
Packit Service a1973e
		fds[i].fd = perf_event_open(&fds[i].hw, pid, options.cpu, fds[0].fd, 0);
Packit Service a1973e
		if (fds[i].fd == -1) {
Packit Service a1973e
			if (fds[i].hw.precise_ip)
Packit Service a1973e
				err(1, "cannot attach event %s: precise mode may not be supported", fds[i].name);
Packit Service a1973e
			err(1, "cannot attach event %s", fds[i].name);
Packit Service a1973e
		}
Packit Service a1973e
	}
Packit Service a1973e
Packit Service a1973e
	/*
Packit Service a1973e
	 * kernel adds the header page to the size of the mmapped region
Packit Service a1973e
	 */
Packit Service a1973e
	fds[0].buf = mmap(NULL, map_size, PROT_READ|PROT_WRITE, MAP_SHARED, fds[0].fd, 0);
Packit Service a1973e
	if (fds[0].buf == MAP_FAILED)
Packit Service a1973e
		err(1, "cannot mmap buffer");
Packit Service a1973e
Packit Service a1973e
	/* does not include header page */
Packit Service a1973e
	fds[0].pgmsk = (options.mmap_pages*pgsz)-1;
Packit Service a1973e
Packit Service a1973e
	/*
Packit Service a1973e
	 * send samples for all events to first event's buffer
Packit Service a1973e
	 */
Packit Service a1973e
	for (i = 1; i < num_fds; i++) {
Packit Service a1973e
		if (!fds[i].hw.sample_period)
Packit Service a1973e
			continue;
Packit Service a1973e
		ret = ioctl(fds[i].fd, PERF_EVENT_IOC_SET_OUTPUT, fds[0].fd);
Packit Service a1973e
		if (ret)
Packit Service a1973e
			err(1, "cannot redirect sampling output");
Packit Service a1973e
	}
Packit Service a1973e
Packit Service a1973e
	if (num_fds > 1 && fds[0].fd > -1) {
Packit Service a1973e
		for(i = 0; i < num_fds; i++) {
Packit Service a1973e
			/*
Packit Service a1973e
			 * read the event identifier using ioctl
Packit Service a1973e
			 * new method replaced the trick with PERF_FORMAT_GROUP + PERF_FORMAT_ID + read()
Packit Service a1973e
			 */
Packit Service a1973e
			ret = ioctl(fds[i].fd, PERF_EVENT_IOC_ID, &fds[i].id);
Packit Service a1973e
			if (ret == -1)
Packit Service a1973e
				err(1, "cannot read ID");
Packit Service a1973e
			fprintf(options.output_file,"ID %"PRIu64"  %s\n", fds[i].id, fds[i].name);
Packit Service a1973e
		}
Packit Service a1973e
	}
Packit Service a1973e
Packit Service a1973e
	pollfds[0].fd = fds[0].fd;
Packit Service a1973e
	pollfds[0].events = POLLIN;
Packit Service a1973e
	
Packit Service a1973e
	for(i=0; i < num_fds; i++) {
Packit Service a1973e
		ret = ioctl(fds[i].fd, PERF_EVENT_IOC_ENABLE, 0);
Packit Service a1973e
		if (ret)
Packit Service a1973e
			err(1, "cannot enable event %s\n", fds[i].name);
Packit Service a1973e
	}
Packit Service a1973e
	signal(SIGCHLD, cld_handler);
Packit Service a1973e
Packit Service a1973e
	close(go[1]);
Packit Service a1973e
Packit Service a1973e
	if (setjmp(jbuf) == 1)
Packit Service a1973e
		goto terminate_session;
Packit Service a1973e
Packit Service a1973e
	sigemptyset(&bmask);
Packit Service a1973e
	sigaddset(&bmask, SIGCHLD);
Packit Service a1973e
Packit Service a1973e
	/*
Packit Service a1973e
	 * core loop
Packit Service a1973e
	 */
Packit Service a1973e
	for(;;) {
Packit Service a1973e
		ret = poll(pollfds, 1, -1);
Packit Service a1973e
		if (ret < 0 && errno == EINTR)
Packit Service a1973e
			break;
Packit Service a1973e
		ovfl_count++;
Packit Service a1973e
		ret = sigprocmask(SIG_SETMASK, &bmask, NULL);
Packit Service a1973e
		if (ret)
Packit Service a1973e
			err(1, "setmask");
Packit Service a1973e
		process_smpl_buf(&fds[0]);
Packit Service a1973e
		ret = sigprocmask(SIG_UNBLOCK, &bmask, NULL);
Packit Service a1973e
		if (ret)
Packit Service a1973e
			err(1, "unblock");
Packit Service a1973e
	}
Packit Service a1973e
terminate_session:
Packit Service a1973e
	/*
Packit Service a1973e
	 * cleanup child
Packit Service a1973e
	 */
Packit Service a1973e
	wait4(pid, &status, 0, NULL);
Packit Service a1973e
Packit Service a1973e
	for(i=0; i < num_fds; i++)
Packit Service a1973e
		close(fds[i].fd);
Packit Service a1973e
Packit Service a1973e
	/* check for partial event buffer */
Packit Service a1973e
	process_smpl_buf(&fds[0]);
Packit Service a1973e
	munmap(fds[0].buf, map_size);
Packit Service a1973e
Packit Service a1973e
	perf_free_fds(fds, num_fds);
Packit Service a1973e
Packit Service a1973e
	fprintf(options.output_file,
Packit Service a1973e
		"%"PRIu64" samples collected in %"PRIu64" poll events, %"PRIu64" lost samples\n",
Packit Service a1973e
		collected_samples,
Packit Service a1973e
		ovfl_count, lost_samples);
Packit Service a1973e
Packit Service a1973e
	/* free libpfm resources cleanly */
Packit Service a1973e
	pfm_terminate();
Packit Service a1973e
Packit Service a1973e
	fclose(options.output_file);
Packit Service a1973e
Packit Service a1973e
	return 0;
Packit Service a1973e
}
Packit Service a1973e
Packit Service a1973e
static void
Packit Service a1973e
usage(void)
Packit Service a1973e
{
Packit Service a1973e
	printf("usage: task_smpl [-h] [--help] [-i] [-c cpu] [-m mmap_pages] [-M] [-b] [-o output_file] [-e event1,...,eventn] cmd\n");
Packit Service a1973e
}
Packit Service a1973e
Packit Service a1973e
int
Packit Service a1973e
main(int argc, char **argv)
Packit Service a1973e
{
Packit Service a1973e
	int c;
Packit Service a1973e
Packit Service a1973e
	setlocale(LC_ALL, "");
Packit Service a1973e
Packit Service a1973e
	options.cpu = -1;
Packit Service a1973e
	options.output_file=stdout;
Packit Service a1973e
Packit Service a1973e
	while ((c=getopt_long(argc, argv,"+he:m:ic:o:Mb", the_options, 0)) != -1) {
Packit Service a1973e
		switch(c) {
Packit Service a1973e
			case 0: continue;
Packit Service a1973e
			case 'e':
Packit Service a1973e
				if (options.events)
Packit Service a1973e
					errx(1, "events specified twice\n");
Packit Service a1973e
				options.events = optarg;
Packit Service a1973e
				break;
Packit Service a1973e
			case 'i':
Packit Service a1973e
				options.opt_inherit = 1;
Packit Service a1973e
				break;
Packit Service a1973e
			case 'm':
Packit Service a1973e
				if (options.mmap_pages)
Packit Service a1973e
					errx(1, "mmap pages already set\n");
Packit Service a1973e
				options.mmap_pages = atoi(optarg);
Packit Service a1973e
				break;
Packit Service a1973e
			case 'M':
Packit Service a1973e
				options.mem_mode = 1;
Packit Service a1973e
				break;
Packit Service a1973e
			case 'b':
Packit Service a1973e
				options.branch_mode = 1;
Packit Service a1973e
				break;
Packit Service a1973e
			case 'c':
Packit Service a1973e
				options.cpu = atoi(optarg);
Packit Service a1973e
				break;
Packit Service a1973e
			case 'o':
Packit Service a1973e
				options.output_file=fopen(optarg,"w");
Packit Service a1973e
				if (options.output_file==NULL) {
Packit Service a1973e
					printf("Invalid filename %s\n",
Packit Service a1973e
						optarg);
Packit Service a1973e
					exit(0);
Packit Service a1973e
				}
Packit Service a1973e
				break;
Packit Service a1973e
			case 'h':
Packit Service a1973e
				usage();
Packit Service a1973e
				exit(0);
Packit Service a1973e
			default:
Packit Service a1973e
				errx(1, "unknown option");
Packit Service a1973e
		}
Packit Service a1973e
	}
Packit Service a1973e
Packit Service a1973e
	if (argv[optind] == NULL)
Packit Service a1973e
		errx(1, "you must specify a command to execute\n");
Packit Service a1973e
	if (!options.events)
Packit Service a1973e
		options.events = strdup(gen_events);
Packit Service a1973e
Packit Service a1973e
	if (!options.mmap_pages)
Packit Service a1973e
		options.mmap_pages = 1;
Packit Service a1973e
	
Packit Service a1973e
	if (options.mmap_pages > 1 && ((options.mmap_pages) & 0x1))
Packit Service a1973e
		errx(1, "number of pages must be power of 2\n");
Packit Service a1973e
Packit Service a1973e
	return mainloop(argv+optind);
Packit Service a1973e
}