xref: /freebsd/usr.sbin/pmcstat/pmcstat.c (revision 49874f6ea3d305366e9e829e2b93832c99a9c40d)
1ebccf1e3SJoseph Koshy /*-
249874f6eSJoseph Koshy  * Copyright (c) 2003-2006, Joseph Koshy
3ebccf1e3SJoseph Koshy  * All rights reserved.
4ebccf1e3SJoseph Koshy  *
5ebccf1e3SJoseph Koshy  * Redistribution and use in source and binary forms, with or without
6ebccf1e3SJoseph Koshy  * modification, are permitted provided that the following conditions
7ebccf1e3SJoseph Koshy  * are met:
8ebccf1e3SJoseph Koshy  * 1. Redistributions of source code must retain the above copyright
9ebccf1e3SJoseph Koshy  *    notice, this list of conditions and the following disclaimer.
10ebccf1e3SJoseph Koshy  * 2. Redistributions in binary form must reproduce the above copyright
11ebccf1e3SJoseph Koshy  *    notice, this list of conditions and the following disclaimer in the
12ebccf1e3SJoseph Koshy  *    documentation and/or other materials provided with the distribution.
13ebccf1e3SJoseph Koshy  *
14ebccf1e3SJoseph Koshy  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
15ebccf1e3SJoseph Koshy  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
16ebccf1e3SJoseph Koshy  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
17ebccf1e3SJoseph Koshy  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
18ebccf1e3SJoseph Koshy  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
19ebccf1e3SJoseph Koshy  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
20ebccf1e3SJoseph Koshy  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
21ebccf1e3SJoseph Koshy  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
22ebccf1e3SJoseph Koshy  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
23ebccf1e3SJoseph Koshy  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
24ebccf1e3SJoseph Koshy  * SUCH DAMAGE.
25ebccf1e3SJoseph Koshy  */
26ebccf1e3SJoseph Koshy 
27ebccf1e3SJoseph Koshy #include <sys/cdefs.h>
28ebccf1e3SJoseph Koshy __FBSDID("$FreeBSD$");
29ebccf1e3SJoseph Koshy 
30ebccf1e3SJoseph Koshy #include <sys/types.h>
31ebccf1e3SJoseph Koshy #include <sys/event.h>
32ebccf1e3SJoseph Koshy #include <sys/queue.h>
3315139246SJoseph Koshy #include <sys/socket.h>
3415139246SJoseph Koshy #include <sys/stat.h>
35ebccf1e3SJoseph Koshy #include <sys/time.h>
36ebccf1e3SJoseph Koshy #include <sys/ttycom.h>
37ebccf1e3SJoseph Koshy #include <sys/wait.h>
38ebccf1e3SJoseph Koshy 
39ebccf1e3SJoseph Koshy #include <assert.h>
40ebccf1e3SJoseph Koshy #include <err.h>
41ebccf1e3SJoseph Koshy #include <errno.h>
42ebccf1e3SJoseph Koshy #include <fcntl.h>
4349874f6eSJoseph Koshy #include <libgen.h>
44ebccf1e3SJoseph Koshy #include <limits.h>
45ebccf1e3SJoseph Koshy #include <math.h>
46ebccf1e3SJoseph Koshy #include <pmc.h>
47f263522aSJoseph Koshy #include <pmclog.h>
48ebccf1e3SJoseph Koshy #include <signal.h>
49ebccf1e3SJoseph Koshy #include <stdarg.h>
50ebccf1e3SJoseph Koshy #include <stdint.h>
5115139246SJoseph Koshy #include <stdio.h>
52ebccf1e3SJoseph Koshy #include <stdlib.h>
53ebccf1e3SJoseph Koshy #include <string.h>
54ebccf1e3SJoseph Koshy #include <sysexits.h>
55ebccf1e3SJoseph Koshy #include <unistd.h>
56ebccf1e3SJoseph Koshy 
5715139246SJoseph Koshy #include "pmcstat.h"
5815139246SJoseph Koshy 
59f263522aSJoseph Koshy /*
60f263522aSJoseph Koshy  * A given invocation of pmcstat(8) can manage multiple PMCs of both
61f263522aSJoseph Koshy  * the system-wide and per-process variety.  Each of these could be in
62f263522aSJoseph Koshy  * 'counting mode' or in 'sampling mode'.
63f263522aSJoseph Koshy  *
64f263522aSJoseph Koshy  * For 'counting mode' PMCs, pmcstat(8) will periodically issue a
65f263522aSJoseph Koshy  * pmc_read() at the configured time interval and print out the value
66f263522aSJoseph Koshy  * of the requested PMCs.
67f263522aSJoseph Koshy  *
68f263522aSJoseph Koshy  * For 'sampling mode' PMCs it can log to a file for offline analysis,
69f263522aSJoseph Koshy  * or can analyse sampling data "on the fly", either by converting
70f263522aSJoseph Koshy  * samples to printed textual form or by creating gprof(1) compatible
71f263522aSJoseph Koshy  * profiles, one per program executed.  When creating gprof(1)
72f263522aSJoseph Koshy  * profiles it can optionally merge entries from multiple processes
73f263522aSJoseph Koshy  * for a given executable into a single profile file.
74f263522aSJoseph Koshy  */
75f263522aSJoseph Koshy 
7615139246SJoseph Koshy /* Globals */
77ebccf1e3SJoseph Koshy 
78ebccf1e3SJoseph Koshy int	pmcstat_interrupt = 0;
79ebccf1e3SJoseph Koshy int	pmcstat_displayheight = DEFAULT_DISPLAY_HEIGHT;
80ebccf1e3SJoseph Koshy int	pmcstat_pipefd[NPIPEFD];
81ebccf1e3SJoseph Koshy int	pmcstat_kq;
82ebccf1e3SJoseph Koshy 
83ebccf1e3SJoseph Koshy /*
84ebccf1e3SJoseph Koshy  * cleanup
85ebccf1e3SJoseph Koshy  */
86ebccf1e3SJoseph Koshy 
87ebccf1e3SJoseph Koshy void
88ebccf1e3SJoseph Koshy pmcstat_cleanup(struct pmcstat_args *a)
89ebccf1e3SJoseph Koshy {
90ebccf1e3SJoseph Koshy 	struct pmcstat_ev *ev, *tmp;
91ebccf1e3SJoseph Koshy 
92ebccf1e3SJoseph Koshy 	/* release allocated PMCs. */
93ebccf1e3SJoseph Koshy 	STAILQ_FOREACH_SAFE(ev, &a->pa_head, ev_next, tmp)
94ebccf1e3SJoseph Koshy 	    if (ev->ev_pmcid != PMC_ID_INVALID) {
95ebccf1e3SJoseph Koshy 		if (pmc_release(ev->ev_pmcid) < 0)
96ebccf1e3SJoseph Koshy 			err(EX_OSERR, "ERROR: cannot release pmc "
97f263522aSJoseph Koshy 			    "0x%x \"%s\"", ev->ev_pmcid, ev->ev_name);
98ebccf1e3SJoseph Koshy 		free(ev->ev_name);
99ebccf1e3SJoseph Koshy 		free(ev->ev_spec);
100ebccf1e3SJoseph Koshy 		STAILQ_REMOVE(&a->pa_head, ev, pmcstat_ev, ev_next);
101ebccf1e3SJoseph Koshy 		free(ev);
102ebccf1e3SJoseph Koshy 	    }
103f263522aSJoseph Koshy 
104dc1d9d2eSJoseph Koshy 	/* de-configure the log file if present. */
105dc1d9d2eSJoseph Koshy 	if (a->pa_flags & (FLAG_HAS_PIPE | FLAG_HAS_OUTPUT_LOGFILE))
106dc1d9d2eSJoseph Koshy 		(void) pmc_configure_logfile(-1);
107dc1d9d2eSJoseph Koshy 
108f263522aSJoseph Koshy 	if (a->pa_logparser) {
109f263522aSJoseph Koshy 		pmclog_close(a->pa_logparser);
110f263522aSJoseph Koshy 		a->pa_logparser = NULL;
111f263522aSJoseph Koshy 	}
11215139246SJoseph Koshy 
11315139246SJoseph Koshy 	if (a->pa_flags & (FLAG_HAS_PIPE | FLAG_HAS_OUTPUT_LOGFILE))
11449874f6eSJoseph Koshy 		pmcstat_shutdown_logging(a);
115ebccf1e3SJoseph Koshy }
116ebccf1e3SJoseph Koshy 
117ebccf1e3SJoseph Koshy void
118ebccf1e3SJoseph Koshy pmcstat_start_pmcs(struct pmcstat_args *a)
119ebccf1e3SJoseph Koshy {
120ebccf1e3SJoseph Koshy 	struct pmcstat_ev *ev;
121ebccf1e3SJoseph Koshy 
122ebccf1e3SJoseph Koshy 	STAILQ_FOREACH(ev, &args.pa_head, ev_next) {
123ebccf1e3SJoseph Koshy 
124ebccf1e3SJoseph Koshy 	    assert(ev->ev_pmcid != PMC_ID_INVALID);
125ebccf1e3SJoseph Koshy 
126ebccf1e3SJoseph Koshy 	    if (pmc_start(ev->ev_pmcid) < 0) {
127f263522aSJoseph Koshy 	        warn("ERROR: Cannot start pmc 0x%x \"%s\"",
128ebccf1e3SJoseph Koshy 		    ev->ev_pmcid, ev->ev_name);
129ebccf1e3SJoseph Koshy 		pmcstat_cleanup(a);
130f263522aSJoseph Koshy 		exit(EX_OSERR);
131ebccf1e3SJoseph Koshy 	    }
132ebccf1e3SJoseph Koshy 	}
133ebccf1e3SJoseph Koshy 
134ebccf1e3SJoseph Koshy }
135ebccf1e3SJoseph Koshy 
136ebccf1e3SJoseph Koshy void
137ebccf1e3SJoseph Koshy pmcstat_print_headers(struct pmcstat_args *a)
138ebccf1e3SJoseph Koshy {
139ebccf1e3SJoseph Koshy 	struct pmcstat_ev *ev;
140ebccf1e3SJoseph Koshy 	int c;
141ebccf1e3SJoseph Koshy 
14215139246SJoseph Koshy 	(void) fprintf(a->pa_printfile, PRINT_HEADER_PREFIX);
143ebccf1e3SJoseph Koshy 
144ebccf1e3SJoseph Koshy 	STAILQ_FOREACH(ev, &a->pa_head, ev_next) {
145ebccf1e3SJoseph Koshy 		if (PMC_IS_SAMPLING_MODE(ev->ev_mode))
146ebccf1e3SJoseph Koshy 			continue;
147ebccf1e3SJoseph Koshy 
148ebccf1e3SJoseph Koshy 		c = PMC_IS_SYSTEM_MODE(ev->ev_mode) ? 's' : 'p';
149ebccf1e3SJoseph Koshy 
150ebccf1e3SJoseph Koshy 		if (ev->ev_fieldskip != 0) {
15115139246SJoseph Koshy 			(void) fprintf(a->pa_printfile, "%*s%c/%*s ",
152ebccf1e3SJoseph Koshy 			    ev->ev_fieldskip, "", c,
153ebccf1e3SJoseph Koshy 			    ev->ev_fieldwidth - ev->ev_fieldskip - 2,
154ebccf1e3SJoseph Koshy 			    ev->ev_name);
155ebccf1e3SJoseph Koshy 		} else
15615139246SJoseph Koshy 			(void) fprintf(a->pa_printfile, "%c/%*s ",
157ebccf1e3SJoseph Koshy 			    c, ev->ev_fieldwidth - 2, ev->ev_name);
158ebccf1e3SJoseph Koshy 	}
159ebccf1e3SJoseph Koshy 
16015139246SJoseph Koshy 	(void) fflush(a->pa_printfile);
161ebccf1e3SJoseph Koshy }
162ebccf1e3SJoseph Koshy 
163ebccf1e3SJoseph Koshy void
164ebccf1e3SJoseph Koshy pmcstat_print_counters(struct pmcstat_args *a)
165ebccf1e3SJoseph Koshy {
166ebccf1e3SJoseph Koshy 	int extra_width;
167ebccf1e3SJoseph Koshy 	struct pmcstat_ev *ev;
168ebccf1e3SJoseph Koshy 	pmc_value_t value;
169ebccf1e3SJoseph Koshy 
170ebccf1e3SJoseph Koshy 	extra_width = sizeof(PRINT_HEADER_PREFIX) - 1;
171ebccf1e3SJoseph Koshy 
172ebccf1e3SJoseph Koshy 	STAILQ_FOREACH(ev, &a->pa_head, ev_next) {
173ebccf1e3SJoseph Koshy 
174ebccf1e3SJoseph Koshy 		/* skip sampling mode counters */
175ebccf1e3SJoseph Koshy 		if (PMC_IS_SAMPLING_MODE(ev->ev_mode))
176ebccf1e3SJoseph Koshy 			continue;
177ebccf1e3SJoseph Koshy 
178ebccf1e3SJoseph Koshy 		if (pmc_read(ev->ev_pmcid, &value) < 0)
179ebccf1e3SJoseph Koshy 			err(EX_OSERR, "ERROR: Cannot read pmc "
180ebccf1e3SJoseph Koshy 			    "\"%s\"", ev->ev_name);
181ebccf1e3SJoseph Koshy 
18215139246SJoseph Koshy 		(void) fprintf(a->pa_printfile, "%*ju ",
18315139246SJoseph Koshy 		    ev->ev_fieldwidth + extra_width,
18415139246SJoseph Koshy 		    (uintmax_t) ev->ev_cumulative ? value :
18515139246SJoseph Koshy 		    (value - ev->ev_saved));
18615139246SJoseph Koshy 
187ebccf1e3SJoseph Koshy 		if (ev->ev_cumulative == 0)
188ebccf1e3SJoseph Koshy 			ev->ev_saved = value;
189ebccf1e3SJoseph Koshy 		extra_width = 0;
190ebccf1e3SJoseph Koshy 	}
191ebccf1e3SJoseph Koshy 
19215139246SJoseph Koshy 	(void) fflush(a->pa_printfile);
193ebccf1e3SJoseph Koshy }
194ebccf1e3SJoseph Koshy 
195ebccf1e3SJoseph Koshy /*
196ebccf1e3SJoseph Koshy  * Print output
197ebccf1e3SJoseph Koshy  */
198ebccf1e3SJoseph Koshy 
199ebccf1e3SJoseph Koshy void
200ebccf1e3SJoseph Koshy pmcstat_print_pmcs(struct pmcstat_args *a)
201ebccf1e3SJoseph Koshy {
202ebccf1e3SJoseph Koshy 	static int linecount = 0;
203ebccf1e3SJoseph Koshy 
20415139246SJoseph Koshy 	/* check if we need to print a header line */
205ebccf1e3SJoseph Koshy 	if (++linecount > pmcstat_displayheight) {
20615139246SJoseph Koshy 		(void) fprintf(a->pa_printfile, "\n");
207ebccf1e3SJoseph Koshy 		linecount = 1;
208ebccf1e3SJoseph Koshy 	}
209ebccf1e3SJoseph Koshy 	if (linecount == 1)
210ebccf1e3SJoseph Koshy 		pmcstat_print_headers(a);
21115139246SJoseph Koshy 	(void) fprintf(a->pa_printfile, "\n");
212ebccf1e3SJoseph Koshy 
213ebccf1e3SJoseph Koshy 	pmcstat_print_counters(a);
214ebccf1e3SJoseph Koshy 
215ebccf1e3SJoseph Koshy 	return;
216ebccf1e3SJoseph Koshy }
217ebccf1e3SJoseph Koshy 
218ebccf1e3SJoseph Koshy /*
219ebccf1e3SJoseph Koshy  * Do process profiling
220ebccf1e3SJoseph Koshy  *
221ebccf1e3SJoseph Koshy  * If a pid was specified, attach each allocated PMC to the target
222ebccf1e3SJoseph Koshy  * process.  Otherwise, fork a child and attach the PMCs to the child,
223ebccf1e3SJoseph Koshy  * and have the child exec() the target program.
224ebccf1e3SJoseph Koshy  */
225ebccf1e3SJoseph Koshy 
226ebccf1e3SJoseph Koshy void
227ebccf1e3SJoseph Koshy pmcstat_setup_process(struct pmcstat_args *a)
228ebccf1e3SJoseph Koshy {
229ebccf1e3SJoseph Koshy 	char token;
230ebccf1e3SJoseph Koshy 	struct pmcstat_ev *ev;
231ebccf1e3SJoseph Koshy 	struct kevent kev;
232ebccf1e3SJoseph Koshy 
233ebccf1e3SJoseph Koshy 	if (a->pa_flags & FLAG_HAS_PID) {
234f263522aSJoseph Koshy 		STAILQ_FOREACH(ev, &a->pa_head, ev_next)
235ebccf1e3SJoseph Koshy 		    if (pmc_attach(ev->ev_pmcid, a->pa_pid) != 0)
236ebccf1e3SJoseph Koshy 			    err(EX_OSERR, "ERROR: cannot attach pmc \"%s\" to "
237ebccf1e3SJoseph Koshy 				"process %d", ev->ev_name, (int) a->pa_pid);
238ebccf1e3SJoseph Koshy 	} else {
239ebccf1e3SJoseph Koshy 
240ebccf1e3SJoseph Koshy 		/*
241ebccf1e3SJoseph Koshy 		 * We need to fork a new process and startup the child
242ebccf1e3SJoseph Koshy 		 * using execvp().  Before doing the exec() the child
243ebccf1e3SJoseph Koshy 		 * process reads its pipe for a token so that the parent
244ebccf1e3SJoseph Koshy 		 * can finish doing its pmc_attach() calls.
245ebccf1e3SJoseph Koshy 		 */
246ebccf1e3SJoseph Koshy 		if (pipe(pmcstat_pipefd) < 0)
247ebccf1e3SJoseph Koshy 			err(EX_OSERR, "ERROR: cannot create pipe");
248ebccf1e3SJoseph Koshy 
249ebccf1e3SJoseph Koshy 		switch (a->pa_pid = fork()) {
250ebccf1e3SJoseph Koshy 		case -1:
251ebccf1e3SJoseph Koshy 			err(EX_OSERR, "ERROR: cannot fork");
252ebccf1e3SJoseph Koshy 			/*NOTREACHED*/
253ebccf1e3SJoseph Koshy 
254ebccf1e3SJoseph Koshy 		case 0:		/* child */
255ebccf1e3SJoseph Koshy 
256ebccf1e3SJoseph Koshy 			/* wait for our parent to signal us */
257ebccf1e3SJoseph Koshy 			(void) close(pmcstat_pipefd[WRITEPIPEFD]);
258ebccf1e3SJoseph Koshy 			if (read(pmcstat_pipefd[READPIPEFD], &token, 1) < 0)
259ebccf1e3SJoseph Koshy 				err(EX_OSERR, "ERROR (child): cannot read "
260ebccf1e3SJoseph Koshy 				    "token");
261ebccf1e3SJoseph Koshy 			(void) close(pmcstat_pipefd[READPIPEFD]);
262ebccf1e3SJoseph Koshy 
263ebccf1e3SJoseph Koshy 			/* exec() the program requested */
264f263522aSJoseph Koshy 			execvp(*a->pa_argv, a->pa_argv);
265f263522aSJoseph Koshy 			/* and if that fails, notify the parent */
266f263522aSJoseph Koshy 			kill(getppid(), SIGCHLD);
267f263522aSJoseph Koshy 			err(EX_OSERR, "ERROR: execvp \"%s\" failed",
268f263522aSJoseph Koshy 			    *a->pa_argv);
269ebccf1e3SJoseph Koshy 			/*NOTREACHED*/
270ebccf1e3SJoseph Koshy 
271ebccf1e3SJoseph Koshy 		default:	/* parent */
272ebccf1e3SJoseph Koshy 
273ebccf1e3SJoseph Koshy 			(void) close(pmcstat_pipefd[READPIPEFD]);
274ebccf1e3SJoseph Koshy 
275ebccf1e3SJoseph Koshy 			/* attach all our PMCs to the child */
276ebccf1e3SJoseph Koshy 			STAILQ_FOREACH(ev, &args.pa_head, ev_next)
277ebccf1e3SJoseph Koshy 			    if (PMC_IS_VIRTUAL_MODE(ev->ev_mode) &&
278ebccf1e3SJoseph Koshy 				pmc_attach(ev->ev_pmcid, a->pa_pid) != 0)
279ebccf1e3SJoseph Koshy 				    err(EX_OSERR, "ERROR: cannot attach pmc "
280ebccf1e3SJoseph Koshy 					"\"%s\" to process %d", ev->ev_name,
281ebccf1e3SJoseph Koshy 					(int) a->pa_pid);
282ebccf1e3SJoseph Koshy 
283ebccf1e3SJoseph Koshy 		}
284ebccf1e3SJoseph Koshy 	}
285ebccf1e3SJoseph Koshy 
286f263522aSJoseph Koshy 	/* Ask to be notified via a kevent when the target process exits */
287f263522aSJoseph Koshy 	EV_SET(&kev, a->pa_pid, EVFILT_PROC, EV_ADD|EV_ONESHOT, NOTE_EXIT, 0,
288f263522aSJoseph Koshy 	    NULL);
289ebccf1e3SJoseph Koshy 	if (kevent(pmcstat_kq, &kev, 1, NULL, 0, NULL) < 0)
290f263522aSJoseph Koshy 		err(EX_OSERR, "ERROR: cannot monitor child process %d",
291ebccf1e3SJoseph Koshy 		    a->pa_pid);
292ebccf1e3SJoseph Koshy 	return;
293ebccf1e3SJoseph Koshy }
294ebccf1e3SJoseph Koshy 
295ebccf1e3SJoseph Koshy void
296ebccf1e3SJoseph Koshy pmcstat_start_process(struct pmcstat_args *a)
297ebccf1e3SJoseph Koshy {
298ebccf1e3SJoseph Koshy 
299ebccf1e3SJoseph Koshy 	/* nothing to do: target is already running */
300ebccf1e3SJoseph Koshy 	if (a->pa_flags & FLAG_HAS_PID)
301ebccf1e3SJoseph Koshy 		return;
302ebccf1e3SJoseph Koshy 
303ebccf1e3SJoseph Koshy 	/* write token to child to state that we are ready */
304ebccf1e3SJoseph Koshy 	if (write(pmcstat_pipefd[WRITEPIPEFD], "+", 1) != 1)
305ebccf1e3SJoseph Koshy 		err(EX_OSERR, "ERROR: write failed");
306ebccf1e3SJoseph Koshy 
307ebccf1e3SJoseph Koshy 	(void) close(pmcstat_pipefd[WRITEPIPEFD]);
308ebccf1e3SJoseph Koshy }
309ebccf1e3SJoseph Koshy 
310ebccf1e3SJoseph Koshy void
311ebccf1e3SJoseph Koshy pmcstat_show_usage(void)
312ebccf1e3SJoseph Koshy {
313ebccf1e3SJoseph Koshy 	errx(EX_USAGE,
314ebccf1e3SJoseph Koshy 	    "[options] [commandline]\n"
315ebccf1e3SJoseph Koshy 	    "\t Measure process and/or system performance using hardware\n"
316ebccf1e3SJoseph Koshy 	    "\t performance monitoring counters.\n"
317ebccf1e3SJoseph Koshy 	    "\t Options include:\n"
318f263522aSJoseph Koshy 	    "\t -C\t\t (toggle) show cumulative counts\n"
319f263522aSJoseph Koshy 	    "\t -D path\t create profiles in directory \"path\"\n"
320f263522aSJoseph Koshy 	    "\t -E\t\t (toggle) show counts at process exit\n"
32149874f6eSJoseph Koshy 	    "\t -M file\t print executable/gmon file map to \"file\"\n"
322f263522aSJoseph Koshy 	    "\t -O file\t send log output to \"file\"\n"
323f263522aSJoseph Koshy 	    "\t -P spec\t allocate a process-private sampling PMC\n"
324f263522aSJoseph Koshy 	    "\t -R file\t read events from \"file\"\n"
325f263522aSJoseph Koshy 	    "\t -S spec\t allocate a system-wide sampling PMC\n"
326f263522aSJoseph Koshy 	    "\t -W\t\t (toggle) show counts per context switch\n"
327f263522aSJoseph Koshy 	    "\t -c cpu\t\t set cpu for subsequent system-wide PMCs\n"
328f263522aSJoseph Koshy 	    "\t -d\t\t (toggle) track descendants\n"
329f263522aSJoseph Koshy 	    "\t -g\t\t produce gprof(1) compatible profiles\n"
33049874f6eSJoseph Koshy 	    "\t -k dir\t set the path to the kernel\n"
331ebccf1e3SJoseph Koshy 	    "\t -n rate\t set sampling rate\n"
332ebccf1e3SJoseph Koshy 	    "\t -o file\t send print output to \"file\"\n"
333f263522aSJoseph Koshy 	    "\t -p spec\t allocate a process-private counting PMC\n"
33449874f6eSJoseph Koshy 	    "\t -q\t\t suppress verbosity\n"
33549874f6eSJoseph Koshy 	    "\t -r fsroot\t specify FS root directory\n"
336f263522aSJoseph Koshy 	    "\t -s spec\t allocate a system-wide counting PMC\n"
33739f4e0fcSJoseph Koshy 	    "\t -t pid\t\t attach to running process with pid \"pid\"\n"
33849874f6eSJoseph Koshy 	    "\t -v\t\t increase verbosity\n"
339ebccf1e3SJoseph Koshy 	    "\t -w secs\t set printing time interval"
340ebccf1e3SJoseph Koshy 	);
341ebccf1e3SJoseph Koshy }
342ebccf1e3SJoseph Koshy 
343ebccf1e3SJoseph Koshy /*
344ebccf1e3SJoseph Koshy  * Main
345ebccf1e3SJoseph Koshy  */
346ebccf1e3SJoseph Koshy 
347ebccf1e3SJoseph Koshy int
348ebccf1e3SJoseph Koshy main(int argc, char **argv)
349ebccf1e3SJoseph Koshy {
350ebccf1e3SJoseph Koshy 	double interval;
351ebccf1e3SJoseph Koshy 	int option, npmc, ncpu;
3527fbf3da2SJoseph Koshy 	int c, check_driver_stats, current_cpu, current_sampling_count;
353f263522aSJoseph Koshy 	int do_print, do_descendants;
354f263522aSJoseph Koshy 	int do_logproccsw, do_logprocexit;
355f263522aSJoseph Koshy 	int pipefd[2];
356f263522aSJoseph Koshy 	int use_cumulative_counts;
357ebccf1e3SJoseph Koshy 	pid_t pid;
35849874f6eSJoseph Koshy 	char *end, *tmp;
359f263522aSJoseph Koshy 	const char *errmsg;
360f263522aSJoseph Koshy 	enum pmcstat_state runstate;
3617fbf3da2SJoseph Koshy 	struct pmc_driverstats ds_start, ds_end;
362ebccf1e3SJoseph Koshy 	struct pmcstat_ev *ev;
363ebccf1e3SJoseph Koshy 	struct sigaction sa;
364ebccf1e3SJoseph Koshy 	struct kevent kev;
365ebccf1e3SJoseph Koshy 	struct winsize ws;
36615139246SJoseph Koshy 	struct stat sb;
36749874f6eSJoseph Koshy 	char buffer[PATH_MAX];
368ebccf1e3SJoseph Koshy 
3697fbf3da2SJoseph Koshy 	check_driver_stats      = 0;
370ebccf1e3SJoseph Koshy 	current_cpu 		= 0;
371ebccf1e3SJoseph Koshy 	current_sampling_count  = DEFAULT_SAMPLE_COUNT;
372ebccf1e3SJoseph Koshy 	do_descendants          = 0;
373f263522aSJoseph Koshy 	do_logproccsw           = 0;
374f263522aSJoseph Koshy 	do_logprocexit          = 0;
375ebccf1e3SJoseph Koshy 	use_cumulative_counts   = 0;
376f263522aSJoseph Koshy 	args.pa_required	= 0;
377ebccf1e3SJoseph Koshy 	args.pa_flags		= 0;
37849874f6eSJoseph Koshy 	args.pa_verbosity	= 1;
379ebccf1e3SJoseph Koshy 	args.pa_pid		= (pid_t) -1;
38015139246SJoseph Koshy 	args.pa_logfd		= -1;
38149874f6eSJoseph Koshy 	args.pa_fsroot		= "";
38249874f6eSJoseph Koshy 	args.pa_kernel		= strdup("/boot/kernel");
38315139246SJoseph Koshy 	args.pa_samplesdir	= ".";
38415139246SJoseph Koshy 	args.pa_printfile	= stderr;
385ebccf1e3SJoseph Koshy 	args.pa_interval	= DEFAULT_WAIT_INTERVAL;
38649874f6eSJoseph Koshy 	args.pa_mapfilename	= NULL;
387ebccf1e3SJoseph Koshy 	STAILQ_INIT(&args.pa_head);
388bea6aa93SJoseph Koshy 	bzero(&ds_start, sizeof(ds_start));
389bea6aa93SJoseph Koshy 	bzero(&ds_end, sizeof(ds_end));
390ebccf1e3SJoseph Koshy 	ev = NULL;
391ebccf1e3SJoseph Koshy 
39249874f6eSJoseph Koshy 	while ((option = getopt(argc, argv,
39349874f6eSJoseph Koshy 	    "CD:EM:O:P:R:S:Wc:dgk:n:o:p:qr:s:t:vw:")) != -1)
394ebccf1e3SJoseph Koshy 		switch (option) {
395ebccf1e3SJoseph Koshy 		case 'C':	/* cumulative values */
396ebccf1e3SJoseph Koshy 			use_cumulative_counts = !use_cumulative_counts;
397f263522aSJoseph Koshy 			args.pa_required |= FLAG_HAS_COUNTING_PMCS;
398ebccf1e3SJoseph Koshy 			break;
399ebccf1e3SJoseph Koshy 
400ebccf1e3SJoseph Koshy 		case 'c':	/* CPU */
401ebccf1e3SJoseph Koshy 			current_cpu = strtol(optarg, &end, 0);
402ebccf1e3SJoseph Koshy 			if (*end != '\0' || current_cpu < 0)
403ebccf1e3SJoseph Koshy 				errx(EX_USAGE,
404f263522aSJoseph Koshy 				    "ERROR: Illegal CPU number \"%s\".",
405ebccf1e3SJoseph Koshy 				    optarg);
406f263522aSJoseph Koshy 			args.pa_required |= FLAG_HAS_SYSTEM_PMCS;
407ebccf1e3SJoseph Koshy 			break;
408ebccf1e3SJoseph Koshy 
40915139246SJoseph Koshy 		case 'D':
41015139246SJoseph Koshy 			if (stat(optarg, &sb) < 0)
41115139246SJoseph Koshy 				err(EX_OSERR, "ERROR: Cannot stat \"%s\"",
41215139246SJoseph Koshy 				    optarg);
41315139246SJoseph Koshy 			if (!S_ISDIR(sb.st_mode))
41415139246SJoseph Koshy 				errx(EX_USAGE, "ERROR: \"%s\" is not a "
41515139246SJoseph Koshy 				    "directory", optarg);
41615139246SJoseph Koshy 			args.pa_samplesdir = optarg;
41715139246SJoseph Koshy 			args.pa_flags     |= FLAG_HAS_SAMPLESDIR;
41815139246SJoseph Koshy 			args.pa_required  |= FLAG_DO_GPROF;
41915139246SJoseph Koshy 			break;
42015139246SJoseph Koshy 
421ebccf1e3SJoseph Koshy 		case 'd':	/* toggle descendents */
422ebccf1e3SJoseph Koshy 			do_descendants = !do_descendants;
423f263522aSJoseph Koshy 			args.pa_required |= FLAG_HAS_PROCESS_PMCS;
424f263522aSJoseph Koshy 			break;
425f263522aSJoseph Koshy 
426f263522aSJoseph Koshy 		case 'g':	/* produce gprof compatible profiles */
427f263522aSJoseph Koshy 			args.pa_flags |= FLAG_DO_GPROF;
428f263522aSJoseph Koshy 			break;
429f263522aSJoseph Koshy 
43015139246SJoseph Koshy 		case 'k':	/* pathname to the kernel */
43149874f6eSJoseph Koshy 			free(args.pa_kernel);
43249874f6eSJoseph Koshy 			args.pa_kernel = strdup(optarg);
43315139246SJoseph Koshy 			args.pa_required |= FLAG_DO_GPROF;
43415139246SJoseph Koshy 			args.pa_flags    |= FLAG_HAS_KERNELPATH;
435f263522aSJoseph Koshy 			break;
436f263522aSJoseph Koshy 
437f263522aSJoseph Koshy 		case 'E':	/* log process exit */
438f263522aSJoseph Koshy 			do_logprocexit = !do_logprocexit;
439f263522aSJoseph Koshy 			args.pa_required |= (FLAG_HAS_PROCESS_PMCS |
44015139246SJoseph Koshy 			    FLAG_HAS_COUNTING_PMCS | FLAG_HAS_OUTPUT_LOGFILE);
441ebccf1e3SJoseph Koshy 			break;
442ebccf1e3SJoseph Koshy 
44349874f6eSJoseph Koshy 		case 'M':	/* mapfile */
44449874f6eSJoseph Koshy 			args.pa_mapfilename = optarg;
44549874f6eSJoseph Koshy 			break;
44649874f6eSJoseph Koshy 
447ebccf1e3SJoseph Koshy 		case 'p':	/* process virtual counting PMC */
448ebccf1e3SJoseph Koshy 		case 's':	/* system-wide counting PMC */
449ebccf1e3SJoseph Koshy 		case 'P':	/* process virtual sampling PMC */
450ebccf1e3SJoseph Koshy 		case 'S':	/* system-wide sampling PMC */
451ebccf1e3SJoseph Koshy 			if ((ev = malloc(sizeof(*ev))) == NULL)
452f263522aSJoseph Koshy 				errx(EX_SOFTWARE, "ERROR: Out of memory.");
453ebccf1e3SJoseph Koshy 
454ebccf1e3SJoseph Koshy 			switch (option) {
455ebccf1e3SJoseph Koshy 			case 'p': ev->ev_mode = PMC_MODE_TC; break;
456ebccf1e3SJoseph Koshy 			case 's': ev->ev_mode = PMC_MODE_SC; break;
457ebccf1e3SJoseph Koshy 			case 'P': ev->ev_mode = PMC_MODE_TS; break;
458ebccf1e3SJoseph Koshy 			case 'S': ev->ev_mode = PMC_MODE_SS; break;
459ebccf1e3SJoseph Koshy 			}
460ebccf1e3SJoseph Koshy 
461f263522aSJoseph Koshy 			if (option == 'P' || option == 'p') {
462f263522aSJoseph Koshy 				args.pa_flags |= FLAG_HAS_PROCESS_PMCS;
46315139246SJoseph Koshy 				args.pa_required |= (FLAG_HAS_COMMANDLINE |
464f263522aSJoseph Koshy 				    FLAG_HAS_PID);
465f263522aSJoseph Koshy 			}
466ebccf1e3SJoseph Koshy 
467f263522aSJoseph Koshy 			if (option == 'P' || option == 'S') {
468f263522aSJoseph Koshy 				args.pa_flags |= FLAG_HAS_SAMPLING_PMCS;
46915139246SJoseph Koshy 				args.pa_required |= (FLAG_HAS_PIPE |
47015139246SJoseph Koshy 				    FLAG_HAS_OUTPUT_LOGFILE);
471f263522aSJoseph Koshy 			}
472ebccf1e3SJoseph Koshy 
473ebccf1e3SJoseph Koshy 			if (option == 'p' || option == 's')
474f263522aSJoseph Koshy 				args.pa_flags |= FLAG_HAS_COUNTING_PMCS;
475f263522aSJoseph Koshy 
476f263522aSJoseph Koshy 			if (option == 's' || option == 'S')
477f263522aSJoseph Koshy 				args.pa_flags |= FLAG_HAS_SYSTEM_PMCS;
478ebccf1e3SJoseph Koshy 
479ebccf1e3SJoseph Koshy 			ev->ev_spec  = strdup(optarg);
480ebccf1e3SJoseph Koshy 
481ebccf1e3SJoseph Koshy 			if (option == 'S' || option == 'P')
482ebccf1e3SJoseph Koshy 				ev->ev_count = current_sampling_count;
483ebccf1e3SJoseph Koshy 			else
484ebccf1e3SJoseph Koshy 				ev->ev_count = -1;
485ebccf1e3SJoseph Koshy 
486ebccf1e3SJoseph Koshy 			if (option == 'S' || option == 's')
487ebccf1e3SJoseph Koshy 				ev->ev_cpu = current_cpu;
488ebccf1e3SJoseph Koshy 			else
489ebccf1e3SJoseph Koshy 				ev->ev_cpu = PMC_CPU_ANY;
490ebccf1e3SJoseph Koshy 
491f263522aSJoseph Koshy 			ev->ev_flags = 0;
492f263522aSJoseph Koshy 			if (do_descendants)
493f263522aSJoseph Koshy 				ev->ev_flags |= PMC_F_DESCENDANTS;
494f263522aSJoseph Koshy 			if (do_logprocexit)
495f263522aSJoseph Koshy 				ev->ev_flags |= PMC_F_LOG_PROCEXIT;
496f263522aSJoseph Koshy 			if (do_logproccsw)
497f263522aSJoseph Koshy 				ev->ev_flags |= PMC_F_LOG_PROCCSW;
498f263522aSJoseph Koshy 
499ebccf1e3SJoseph Koshy 			ev->ev_cumulative  = use_cumulative_counts;
500ebccf1e3SJoseph Koshy 
501ebccf1e3SJoseph Koshy 			ev->ev_saved = 0LL;
502ebccf1e3SJoseph Koshy 			ev->ev_pmcid = PMC_ID_INVALID;
503ebccf1e3SJoseph Koshy 
504ebccf1e3SJoseph Koshy 			/* extract event name */
505ebccf1e3SJoseph Koshy 			c = strcspn(optarg, ", \t");
506ebccf1e3SJoseph Koshy 			ev->ev_name = malloc(c + 1);
507ebccf1e3SJoseph Koshy 			(void) strncpy(ev->ev_name, optarg, c);
508ebccf1e3SJoseph Koshy 			*(ev->ev_name + c) = '\0';
509ebccf1e3SJoseph Koshy 
510ebccf1e3SJoseph Koshy 			STAILQ_INSERT_TAIL(&args.pa_head, ev, ev_next);
511ebccf1e3SJoseph Koshy 
512ebccf1e3SJoseph Koshy 			break;
513ebccf1e3SJoseph Koshy 
514ebccf1e3SJoseph Koshy 		case 'n':	/* sampling count */
515ebccf1e3SJoseph Koshy 			current_sampling_count = strtol(optarg, &end, 0);
516ebccf1e3SJoseph Koshy 			if (*end != '\0' || current_sampling_count <= 0)
517ebccf1e3SJoseph Koshy 				errx(EX_USAGE,
518f263522aSJoseph Koshy 				    "ERROR: Illegal count value \"%s\".",
519ebccf1e3SJoseph Koshy 				    optarg);
520f263522aSJoseph Koshy 			args.pa_required |= FLAG_HAS_SAMPLING_PMCS;
521ebccf1e3SJoseph Koshy 			break;
522ebccf1e3SJoseph Koshy 
523ebccf1e3SJoseph Koshy 		case 'o':	/* outputfile */
52415139246SJoseph Koshy 			if (args.pa_printfile != NULL)
52515139246SJoseph Koshy 				(void) fclose(args.pa_printfile);
52615139246SJoseph Koshy 			if ((args.pa_printfile = fopen(optarg, "w")) == NULL)
527ebccf1e3SJoseph Koshy 				errx(EX_OSERR, "ERROR: cannot open \"%s\" for "
528f263522aSJoseph Koshy 				    "writing.", optarg);
52915139246SJoseph Koshy 			args.pa_flags |= FLAG_DO_PRINT;
530f263522aSJoseph Koshy 			break;
531ebccf1e3SJoseph Koshy 
532ebccf1e3SJoseph Koshy 		case 'O':	/* sampling output */
53315139246SJoseph Koshy 			if (args.pa_outputpath)
53415139246SJoseph Koshy 				errx(EX_USAGE, "ERROR: option -O may only be "
535f263522aSJoseph Koshy 				    "specified once.");
53615139246SJoseph Koshy 			args.pa_outputpath = optarg;
53715139246SJoseph Koshy 			args.pa_flags |= FLAG_HAS_OUTPUT_LOGFILE;
53815139246SJoseph Koshy 			break;
53915139246SJoseph Koshy 
54049874f6eSJoseph Koshy 		case 'q':	/* quiet mode */
54149874f6eSJoseph Koshy 			args.pa_verbosity = 0;
54249874f6eSJoseph Koshy 			break;
54349874f6eSJoseph Koshy 
54449874f6eSJoseph Koshy 		case 'r':	/* root FS path */
54549874f6eSJoseph Koshy 			args.pa_fsroot = optarg;
54649874f6eSJoseph Koshy 			break;
54749874f6eSJoseph Koshy 
54815139246SJoseph Koshy 		case 'R':	/* read an existing log file */
54915139246SJoseph Koshy 			if (args.pa_logparser != NULL)
55015139246SJoseph Koshy 				errx(EX_USAGE, "ERROR: option -R may only be "
55115139246SJoseph Koshy 				    "specified once.");
55215139246SJoseph Koshy 			args.pa_inputpath = optarg;
55315139246SJoseph Koshy 			if (args.pa_printfile == stderr)
55415139246SJoseph Koshy 				args.pa_printfile = stdout;
55515139246SJoseph Koshy 			args.pa_flags |= FLAG_READ_LOGFILE;
556ebccf1e3SJoseph Koshy 			break;
557ebccf1e3SJoseph Koshy 
558ebccf1e3SJoseph Koshy 		case 't':	/* target pid */
559ebccf1e3SJoseph Koshy 			pid = strtol(optarg, &end, 0);
560ebccf1e3SJoseph Koshy 			if (*end != '\0' || pid <= 0)
561ebccf1e3SJoseph Koshy 				errx(EX_USAGE, "ERROR: Illegal pid value "
562f263522aSJoseph Koshy 				    "\"%s\".", optarg);
563ebccf1e3SJoseph Koshy 
564ebccf1e3SJoseph Koshy 			args.pa_flags |= FLAG_HAS_PID;
565f263522aSJoseph Koshy 			args.pa_required |= FLAG_HAS_PROCESS_PMCS;
566ebccf1e3SJoseph Koshy 			args.pa_pid = pid;
567ebccf1e3SJoseph Koshy 			break;
568ebccf1e3SJoseph Koshy 
56949874f6eSJoseph Koshy 		case 'v':	/* verbose */
57049874f6eSJoseph Koshy 			args.pa_verbosity++;
57149874f6eSJoseph Koshy 			break;
57249874f6eSJoseph Koshy 
573ebccf1e3SJoseph Koshy 		case 'w':	/* wait interval */
574ebccf1e3SJoseph Koshy 			interval = strtod(optarg, &end);
575ebccf1e3SJoseph Koshy 			if (*end != '\0' || interval <= 0)
576ebccf1e3SJoseph Koshy 				errx(EX_USAGE, "ERROR: Illegal wait interval "
577f263522aSJoseph Koshy 				    "value \"%s\".", optarg);
578ebccf1e3SJoseph Koshy 			args.pa_flags |= FLAG_HAS_WAIT_INTERVAL;
57915139246SJoseph Koshy 			args.pa_required |= FLAG_HAS_COUNTING_PMCS;
580ebccf1e3SJoseph Koshy 			args.pa_interval = interval;
581f263522aSJoseph Koshy 			break;
582ebccf1e3SJoseph Koshy 
583f263522aSJoseph Koshy 		case 'W':	/* toggle LOG_CSW */
584f263522aSJoseph Koshy 			do_logproccsw = !do_logproccsw;
585f263522aSJoseph Koshy 			args.pa_required |= (FLAG_HAS_PROCESS_PMCS |
58615139246SJoseph Koshy 			    FLAG_HAS_COUNTING_PMCS | FLAG_HAS_OUTPUT_LOGFILE);
587ebccf1e3SJoseph Koshy 			break;
588ebccf1e3SJoseph Koshy 
589ebccf1e3SJoseph Koshy 		case '?':
590ebccf1e3SJoseph Koshy 		default:
591ebccf1e3SJoseph Koshy 			pmcstat_show_usage();
592ebccf1e3SJoseph Koshy 			break;
593ebccf1e3SJoseph Koshy 
594ebccf1e3SJoseph Koshy 		}
595ebccf1e3SJoseph Koshy 
596ebccf1e3SJoseph Koshy 	args.pa_argc = (argc -= optind);
597ebccf1e3SJoseph Koshy 	args.pa_argv = (argv += optind);
598ebccf1e3SJoseph Koshy 
59915139246SJoseph Koshy 	if (argc)	/* command line present */
60015139246SJoseph Koshy 		args.pa_flags |= FLAG_HAS_COMMANDLINE;
601ebccf1e3SJoseph Koshy 
602ebccf1e3SJoseph Koshy 	/*
603ebccf1e3SJoseph Koshy 	 * Check invocation syntax.
604ebccf1e3SJoseph Koshy 	 */
605ebccf1e3SJoseph Koshy 
60615139246SJoseph Koshy 	/* disallow -O and -R together */
60715139246SJoseph Koshy 	if (args.pa_outputpath && args.pa_inputpath)
60815139246SJoseph Koshy 		errx(EX_USAGE, "ERROR: options -O and -R are mutually "
60915139246SJoseph Koshy 		    "exclusive.");
61015139246SJoseph Koshy 
61115139246SJoseph Koshy 	if (args.pa_flags & FLAG_READ_LOGFILE) {
612f263522aSJoseph Koshy 		errmsg = NULL;
61315139246SJoseph Koshy 		if (args.pa_flags & FLAG_HAS_COMMANDLINE)
614f263522aSJoseph Koshy 			errmsg = "a command line specification";
615f263522aSJoseph Koshy 		else if (args.pa_flags & FLAG_HAS_PID)
616f263522aSJoseph Koshy 			errmsg = "option -t";
617f263522aSJoseph Koshy 		else if (!STAILQ_EMPTY(&args.pa_head))
618f263522aSJoseph Koshy 			errmsg = "a PMC event specification";
619f263522aSJoseph Koshy 		if (errmsg)
620f263522aSJoseph Koshy 			errx(EX_USAGE, "ERROR: option -R may not be used with "
621f263522aSJoseph Koshy 			    "%s.", errmsg);
62215139246SJoseph Koshy 	} else if (STAILQ_EMPTY(&args.pa_head))
62315139246SJoseph Koshy 		/* All other uses require a PMC spec. */
624ebccf1e3SJoseph Koshy 		pmcstat_show_usage();
625ebccf1e3SJoseph Koshy 
626f263522aSJoseph Koshy 	/* check for -t pid without a process PMC spec */
627f263522aSJoseph Koshy 	if ((args.pa_required & FLAG_HAS_PID) &&
628f263522aSJoseph Koshy 	    (args.pa_flags & FLAG_HAS_PROCESS_PMCS) == 0)
629f263522aSJoseph Koshy 		errx(EX_USAGE, "ERROR: option -t requires a process mode PMC "
630f263522aSJoseph Koshy 		    "to be specified.");
631f263522aSJoseph Koshy 
632f263522aSJoseph Koshy 	/* check for process-mode options without a command or -t pid */
633f263522aSJoseph Koshy 	if ((args.pa_required & FLAG_HAS_PROCESS_PMCS) &&
63415139246SJoseph Koshy 	    (args.pa_flags & (FLAG_HAS_COMMANDLINE | FLAG_HAS_PID)) == 0)
63515139246SJoseph Koshy 		errx(EX_USAGE, "ERROR: options -d, -E, -p, -P, and -W require "
63615139246SJoseph Koshy 		    "a command line or target process.");
637f263522aSJoseph Koshy 
638f263522aSJoseph Koshy 	/* check for -p | -P without a target process of some sort */
63915139246SJoseph Koshy 	if ((args.pa_required & (FLAG_HAS_COMMANDLINE | FLAG_HAS_PID)) &&
64015139246SJoseph Koshy 	    (args.pa_flags & (FLAG_HAS_COMMANDLINE | FLAG_HAS_PID)) == 0)
64115139246SJoseph Koshy 		errx(EX_USAGE, "ERROR: options -P and -p require a "
642f263522aSJoseph Koshy 		    "target process or a command line.");
643f263522aSJoseph Koshy 
644f263522aSJoseph Koshy 	/* check for process-mode options without a process-mode PMC */
645f263522aSJoseph Koshy 	if ((args.pa_required & FLAG_HAS_PROCESS_PMCS) &&
646f263522aSJoseph Koshy 	    (args.pa_flags & FLAG_HAS_PROCESS_PMCS) == 0)
64715139246SJoseph Koshy 		errx(EX_USAGE, "ERROR: options -d, -E, and -W require a "
648f263522aSJoseph Koshy 		    "process mode PMC to be specified.");
649f263522aSJoseph Koshy 
650f263522aSJoseph Koshy 	/* check for -c cpu and not system mode PMCs */
651f263522aSJoseph Koshy 	if ((args.pa_required & FLAG_HAS_SYSTEM_PMCS) &&
652f263522aSJoseph Koshy 	    (args.pa_flags & FLAG_HAS_SYSTEM_PMCS) == 0)
653f263522aSJoseph Koshy 		errx(EX_USAGE, "ERROR: option -c requires at least one "
654f263522aSJoseph Koshy 		    "system mode PMC to be specified.");
655f263522aSJoseph Koshy 
656f263522aSJoseph Koshy 	/* check for counting mode options without a counting PMC */
657f263522aSJoseph Koshy 	if ((args.pa_required & FLAG_HAS_COUNTING_PMCS) &&
658f263522aSJoseph Koshy 	    (args.pa_flags & FLAG_HAS_COUNTING_PMCS) == 0)
6594525db8cSJoseph Koshy 		errx(EX_USAGE, "ERROR: options -C, -W, -o and -w require at "
6604525db8cSJoseph Koshy 		    "least one counting mode PMC to be specified.");
661f263522aSJoseph Koshy 
662f263522aSJoseph Koshy 	/* check for sampling mode options without a sampling PMC spec */
663f263522aSJoseph Koshy 	if ((args.pa_required & FLAG_HAS_SAMPLING_PMCS) &&
664f263522aSJoseph Koshy 	    (args.pa_flags & FLAG_HAS_SAMPLING_PMCS) == 0)
66515139246SJoseph Koshy 		errx(EX_USAGE, "ERROR: options -n and -O require at least "
66615139246SJoseph Koshy 		    "one sampling mode PMC to be specified.");
667f263522aSJoseph Koshy 
66815139246SJoseph Koshy 	if ((args.pa_flags & (FLAG_HAS_PID | FLAG_HAS_COMMANDLINE)) ==
66915139246SJoseph Koshy 	    (FLAG_HAS_PID | FLAG_HAS_COMMANDLINE))
670ebccf1e3SJoseph Koshy 		errx(EX_USAGE,
671ebccf1e3SJoseph Koshy 		    "ERROR: option -t cannot be specified with a command "
672f263522aSJoseph Koshy 		    "line.");
673ebccf1e3SJoseph Koshy 
67415139246SJoseph Koshy 	/* check if -g is being used correctly */
67515139246SJoseph Koshy 	if ((args.pa_flags & FLAG_DO_GPROF) &&
67615139246SJoseph Koshy 	    !(args.pa_flags & (FLAG_HAS_SAMPLING_PMCS|FLAG_READ_LOGFILE)))
67715139246SJoseph Koshy 		errx(EX_USAGE, "ERROR: option -g requires sampling PMCs or -R "
67815139246SJoseph Koshy 		    "to be specified.");
67915139246SJoseph Koshy 
680f263522aSJoseph Koshy 	/* check if -O was spuriously specified */
68115139246SJoseph Koshy 	if ((args.pa_flags & FLAG_HAS_OUTPUT_LOGFILE) &&
68215139246SJoseph Koshy 	    (args.pa_required & FLAG_HAS_OUTPUT_LOGFILE) == 0)
683f263522aSJoseph Koshy 		errx(EX_USAGE,
684f263522aSJoseph Koshy 		    "ERROR: option -O is used only with options "
685f263522aSJoseph Koshy 		    "-E, -P, -S and -W.");
686f263522aSJoseph Koshy 
68749874f6eSJoseph Koshy 	/* -D dir and -k kernel path require -g or -R */
68815139246SJoseph Koshy 	if ((args.pa_flags & FLAG_HAS_KERNELPATH) &&
68949874f6eSJoseph Koshy 	    (args.pa_flags & FLAG_DO_GPROF) == 0 &&
69049874f6eSJoseph Koshy 	    (args.pa_flags & FLAG_READ_LOGFILE) == 0)
69149874f6eSJoseph Koshy 	    errx(EX_USAGE, "ERROR: option -k is only used with -g/-R.");
69215139246SJoseph Koshy 
69315139246SJoseph Koshy 	if ((args.pa_flags & FLAG_HAS_SAMPLESDIR) &&
69449874f6eSJoseph Koshy 	    (args.pa_flags & FLAG_DO_GPROF) == 0 &&
69549874f6eSJoseph Koshy 	    (args.pa_flags & FLAG_READ_LOGFILE) == 0)
69649874f6eSJoseph Koshy 	    errx(EX_USAGE, "ERROR: option -D is only used with -g/-R.");
69749874f6eSJoseph Koshy 
69849874f6eSJoseph Koshy 	/* -M mapfile requires -g or -R */
69949874f6eSJoseph Koshy 	if (args.pa_mapfilename != NULL &&
70049874f6eSJoseph Koshy 	    (args.pa_flags & FLAG_DO_GPROF) == 0 &&
70149874f6eSJoseph Koshy 	    (args.pa_flags & FLAG_READ_LOGFILE) == 0)
70249874f6eSJoseph Koshy 	    errx(EX_USAGE, "ERROR: option -M is only used with -g/-R.");
70315139246SJoseph Koshy 
70415139246SJoseph Koshy 	/*
70515139246SJoseph Koshy 	 * Disallow textual output of sampling PMCs if counting PMCs
70615139246SJoseph Koshy 	 * have also been asked for, mostly because the combined output
70715139246SJoseph Koshy 	 * is difficult to make sense of.
70815139246SJoseph Koshy 	 */
70915139246SJoseph Koshy 	if ((args.pa_flags & FLAG_HAS_COUNTING_PMCS) &&
71015139246SJoseph Koshy 	    (args.pa_flags & FLAG_HAS_SAMPLING_PMCS) &&
71115139246SJoseph Koshy 	    ((args.pa_required & FLAG_HAS_OUTPUT_LOGFILE) == 0))
71215139246SJoseph Koshy 		errx(EX_USAGE, "ERROR: option -O is required if counting and "
71315139246SJoseph Koshy 		    "sampling PMCs are specified together.");
71415139246SJoseph Koshy 
71549874f6eSJoseph Koshy 	/*
71649874f6eSJoseph Koshy 	 * Check if "-k kerneldir" was specified, and if whether 'kerneldir'
71749874f6eSJoseph Koshy 	 * actually refers to a a file.  If so, use `dirname path` to determine
71849874f6eSJoseph Koshy 	 * the kernel directory.
71949874f6eSJoseph Koshy 	 */
72049874f6eSJoseph Koshy 	if (args.pa_flags & FLAG_HAS_KERNELPATH) {
72149874f6eSJoseph Koshy 		(void) snprintf(buffer, sizeof(buffer), "%s%s", args.pa_fsroot,
72249874f6eSJoseph Koshy 		    args.pa_kernel);
72349874f6eSJoseph Koshy 		if (stat(buffer, &sb) < 0)
72449874f6eSJoseph Koshy 			err(EX_OSERR, "ERROR: Cannot locate kernel \"%s\"",
72549874f6eSJoseph Koshy 			    buffer);
72649874f6eSJoseph Koshy 		if (!S_ISREG(sb.st_mode) && !S_ISDIR(sb.st_mode))
72749874f6eSJoseph Koshy 			errx(EX_USAGE, "ERROR: \"%s\": Unsupported file type.",
72849874f6eSJoseph Koshy 			    buffer);
72949874f6eSJoseph Koshy 		if (!S_ISDIR(sb.st_mode)) {
73049874f6eSJoseph Koshy 			tmp = args.pa_kernel;
73149874f6eSJoseph Koshy 			args.pa_kernel = strdup(dirname(args.pa_kernel));
73249874f6eSJoseph Koshy 			free(tmp);
73349874f6eSJoseph Koshy 			(void) snprintf(buffer, sizeof(buffer), "%s%s",
73449874f6eSJoseph Koshy 			    args.pa_fsroot, args.pa_kernel);
73549874f6eSJoseph Koshy 			if (stat(buffer, &sb) < 0)
73649874f6eSJoseph Koshy 				err(EX_OSERR, "ERROR: Cannot stat \"%s\"",
73749874f6eSJoseph Koshy 				    buffer);
73849874f6eSJoseph Koshy 			if (!S_ISDIR(sb.st_mode))
73949874f6eSJoseph Koshy 				errx(EX_USAGE, "ERROR: \"%s\" is not a "
74049874f6eSJoseph Koshy 				    "directory.", buffer);
74149874f6eSJoseph Koshy 		}
74249874f6eSJoseph Koshy 	}
74349874f6eSJoseph Koshy 
744f263522aSJoseph Koshy 	/* if we've been asked to process a log file, do that and exit */
74515139246SJoseph Koshy 	if (args.pa_flags & FLAG_READ_LOGFILE) {
74615139246SJoseph Koshy 		/*
74715139246SJoseph Koshy 		 * Print the log in textual form if we haven't been
74815139246SJoseph Koshy 		 * asked to generate gmon.out files.
74915139246SJoseph Koshy 		 */
75015139246SJoseph Koshy 		if ((args.pa_flags & FLAG_DO_GPROF) == 0)
75115139246SJoseph Koshy 			args.pa_flags |= FLAG_DO_PRINT;
75215139246SJoseph Koshy 
75315139246SJoseph Koshy 		pmcstat_initialize_logging(&args);
75449874f6eSJoseph Koshy 		if ((args.pa_logfd = pmcstat_open_log(args.pa_inputpath,
75515139246SJoseph Koshy 		    PMCSTAT_OPEN_FOR_READ)) < 0)
75615139246SJoseph Koshy 			err(EX_OSERR, "ERROR: Cannot open \"%s\" for "
75715139246SJoseph Koshy 			    "reading", args.pa_inputpath);
75815139246SJoseph Koshy 		if ((args.pa_logparser = pmclog_open(args.pa_logfd)) == NULL)
75915139246SJoseph Koshy 			err(EX_OSERR, "ERROR: Cannot create parser");
760f263522aSJoseph Koshy 		pmcstat_process_log(&args);
76149874f6eSJoseph Koshy 		pmcstat_shutdown_logging(&args);
762f263522aSJoseph Koshy 		exit(EX_OK);
763f263522aSJoseph Koshy 	}
764f263522aSJoseph Koshy 
765f263522aSJoseph Koshy 	/* otherwise, we've been asked to collect data */
766ebccf1e3SJoseph Koshy 	if (pmc_init() < 0)
767ebccf1e3SJoseph Koshy 		err(EX_UNAVAILABLE,
768ebccf1e3SJoseph Koshy 		    "ERROR: Initialization of the pmc(3) library failed");
769ebccf1e3SJoseph Koshy 
770ebccf1e3SJoseph Koshy 	if ((ncpu = pmc_ncpu()) < 0)
771ebccf1e3SJoseph Koshy 		err(EX_OSERR, "ERROR: Cannot determine the number CPUs "
772ebccf1e3SJoseph Koshy 		    "on the system");
773ebccf1e3SJoseph Koshy 
774ebccf1e3SJoseph Koshy 	if ((npmc = pmc_npmc(0)) < 0) /* assume all CPUs are identical */
775ebccf1e3SJoseph Koshy 		err(EX_OSERR, "ERROR: Cannot determine the number of PMCs "
776ebccf1e3SJoseph Koshy 		    "on CPU %d", 0);
777ebccf1e3SJoseph Koshy 
77815139246SJoseph Koshy 	/* Allocate a kqueue */
77915139246SJoseph Koshy 	if ((pmcstat_kq = kqueue()) < 0)
78015139246SJoseph Koshy 		err(EX_OSERR, "ERROR: Cannot allocate kqueue");
78115139246SJoseph Koshy 
78215139246SJoseph Koshy 	/*
78315139246SJoseph Koshy 	 * Configure the specified log file or setup a default log
78415139246SJoseph Koshy 	 * consumer via a pipe.
78515139246SJoseph Koshy 	 */
78615139246SJoseph Koshy 	if (args.pa_required & FLAG_HAS_OUTPUT_LOGFILE) {
78715139246SJoseph Koshy 		if (args.pa_outputpath) {
78849874f6eSJoseph Koshy 			if ((args.pa_logfd =
78949874f6eSJoseph Koshy 			    pmcstat_open_log(args.pa_outputpath,
79015139246SJoseph Koshy 			    PMCSTAT_OPEN_FOR_WRITE)) < 0)
79115139246SJoseph Koshy 				err(EX_OSERR, "ERROR: Cannot open \"%s\" for "
79215139246SJoseph Koshy 				    "writing", args.pa_outputpath);
79315139246SJoseph Koshy 		} else {
79415139246SJoseph Koshy 			/*
79515139246SJoseph Koshy 			 * process the log on the fly by reading it in
79615139246SJoseph Koshy 			 * through a pipe.
79715139246SJoseph Koshy 			 */
79815139246SJoseph Koshy 			if (pipe(pipefd) < 0)
79915139246SJoseph Koshy 				err(EX_OSERR, "ERROR: pipe(2) failed");
80015139246SJoseph Koshy 
80115139246SJoseph Koshy 			if (fcntl(pipefd[READPIPEFD], F_SETFL, O_NONBLOCK) < 0)
80215139246SJoseph Koshy 				err(EX_OSERR, "ERROR: fcntl(2) failed");
80315139246SJoseph Koshy 
80415139246SJoseph Koshy 			EV_SET(&kev, pipefd[READPIPEFD], EVFILT_READ, EV_ADD,
80515139246SJoseph Koshy 			    0, 0, NULL);
80615139246SJoseph Koshy 
80715139246SJoseph Koshy 			if (kevent(pmcstat_kq, &kev, 1, NULL, 0, NULL) < 0)
80815139246SJoseph Koshy 				err(EX_OSERR, "ERROR: Cannot register kevent");
80915139246SJoseph Koshy 
81015139246SJoseph Koshy 			args.pa_logfd = pipefd[WRITEPIPEFD];
81115139246SJoseph Koshy 
81215139246SJoseph Koshy 			args.pa_flags |= (FLAG_HAS_PIPE | FLAG_DO_PRINT);
81315139246SJoseph Koshy 			args.pa_logparser = pmclog_open(pipefd[READPIPEFD]);
81415139246SJoseph Koshy 		}
81515139246SJoseph Koshy 
81615139246SJoseph Koshy 		if (pmc_configure_logfile(args.pa_logfd) < 0)
81715139246SJoseph Koshy 			err(EX_OSERR, "ERROR: Cannot configure log file");
81815139246SJoseph Koshy 	}
81915139246SJoseph Koshy 
8207fbf3da2SJoseph Koshy 	/* remember to check for driver errors if we are sampling or logging */
8217fbf3da2SJoseph Koshy 	check_driver_stats = (args.pa_flags & FLAG_HAS_SAMPLING_PMCS) ||
8227fbf3da2SJoseph Koshy 	    (args.pa_flags & FLAG_HAS_OUTPUT_LOGFILE);
8237fbf3da2SJoseph Koshy 
824ebccf1e3SJoseph Koshy 	/*
825ebccf1e3SJoseph Koshy 	 * Allocate PMCs.
826ebccf1e3SJoseph Koshy 	 */
827ebccf1e3SJoseph Koshy 
82815139246SJoseph Koshy 	STAILQ_FOREACH(ev, &args.pa_head, ev_next) {
829ebccf1e3SJoseph Koshy 	    if (pmc_allocate(ev->ev_spec, ev->ev_mode,
830f263522aSJoseph Koshy 		    ev->ev_flags, ev->ev_cpu, &ev->ev_pmcid) < 0)
831ebccf1e3SJoseph Koshy 		    err(EX_OSERR, "ERROR: Cannot allocate %s-mode pmc with "
832ebccf1e3SJoseph Koshy 			"specification \"%s\"",
833ebccf1e3SJoseph Koshy 			PMC_IS_SYSTEM_MODE(ev->ev_mode) ? "system" : "process",
834ebccf1e3SJoseph Koshy 			ev->ev_spec);
835ebccf1e3SJoseph Koshy 
83615139246SJoseph Koshy 	    if (PMC_IS_SAMPLING_MODE(ev->ev_mode) &&
83715139246SJoseph Koshy 		pmc_set(ev->ev_pmcid, ev->ev_count) < 0)
83815139246SJoseph Koshy 		    err(EX_OSERR, "ERROR: Cannot set sampling count "
83915139246SJoseph Koshy 			"for PMC \"%s\"", ev->ev_name);
84015139246SJoseph Koshy 	}
84115139246SJoseph Koshy 
842ebccf1e3SJoseph Koshy 	/* compute printout widths */
843ebccf1e3SJoseph Koshy 	STAILQ_FOREACH(ev, &args.pa_head, ev_next) {
844c5153e19SJoseph Koshy 		int counter_width;
845c5153e19SJoseph Koshy 		int display_width;
846c5153e19SJoseph Koshy 		int header_width;
847ebccf1e3SJoseph Koshy 
848c5153e19SJoseph Koshy 		(void) pmc_width(ev->ev_pmcid, &counter_width);
849c5153e19SJoseph Koshy 		header_width = strlen(ev->ev_name) + 2; /* prefix '%c|' */
850c5153e19SJoseph Koshy 		display_width = (int) floor(counter_width / 3.32193) + 1;
851ebccf1e3SJoseph Koshy 
852c5153e19SJoseph Koshy 		if (header_width > display_width) {
853ebccf1e3SJoseph Koshy 			ev->ev_fieldskip = 0;
854c5153e19SJoseph Koshy 			ev->ev_fieldwidth = header_width;
855ebccf1e3SJoseph Koshy 		} else {
856c5153e19SJoseph Koshy 			ev->ev_fieldskip = display_width -
857c5153e19SJoseph Koshy 			    header_width;
858c5153e19SJoseph Koshy 			ev->ev_fieldwidth = display_width;
859ebccf1e3SJoseph Koshy 		}
860ebccf1e3SJoseph Koshy 	}
861ebccf1e3SJoseph Koshy 
862ebccf1e3SJoseph Koshy 	/*
863ebccf1e3SJoseph Koshy 	 * If our output is being set to a terminal, register a handler
864ebccf1e3SJoseph Koshy 	 * for window size changes.
865ebccf1e3SJoseph Koshy 	 */
866ebccf1e3SJoseph Koshy 
86715139246SJoseph Koshy 	if (isatty(fileno(args.pa_printfile))) {
868ebccf1e3SJoseph Koshy 
86915139246SJoseph Koshy 		if (ioctl(fileno(args.pa_printfile), TIOCGWINSZ, &ws) < 0)
870ebccf1e3SJoseph Koshy 			err(EX_OSERR, "ERROR: Cannot determine window size");
871ebccf1e3SJoseph Koshy 
872ebccf1e3SJoseph Koshy 		pmcstat_displayheight = ws.ws_row - 1;
873ebccf1e3SJoseph Koshy 
874ebccf1e3SJoseph Koshy 		EV_SET(&kev, SIGWINCH, EVFILT_SIGNAL, EV_ADD, 0, 0, NULL);
875ebccf1e3SJoseph Koshy 
876ebccf1e3SJoseph Koshy 		if (kevent(pmcstat_kq, &kev, 1, NULL, 0, NULL) < 0)
877ebccf1e3SJoseph Koshy 			err(EX_OSERR, "ERROR: Cannot register kevent for "
878ebccf1e3SJoseph Koshy 			    "SIGWINCH");
879ebccf1e3SJoseph Koshy 	}
880ebccf1e3SJoseph Koshy 
881ebccf1e3SJoseph Koshy 	EV_SET(&kev, SIGINT, EVFILT_SIGNAL, EV_ADD, 0, 0, NULL);
882ebccf1e3SJoseph Koshy 	if (kevent(pmcstat_kq, &kev, 1, NULL, 0, NULL) < 0)
883ebccf1e3SJoseph Koshy 		err(EX_OSERR, "ERROR: Cannot register kevent for SIGINT");
884ebccf1e3SJoseph Koshy 
885f263522aSJoseph Koshy 	EV_SET(&kev, SIGIO, EVFILT_SIGNAL, EV_ADD, 0, 0, NULL);
886f263522aSJoseph Koshy 	if (kevent(pmcstat_kq, &kev, 1, NULL, 0, NULL) < 0)
887f263522aSJoseph Koshy 		err(EX_OSERR, "ERROR: Cannot register kevent for SIGIO");
888ebccf1e3SJoseph Koshy 
889ebccf1e3SJoseph Koshy 	/*
890f263522aSJoseph Koshy 	 * An exec() failure of a forked child is signalled by the
891f263522aSJoseph Koshy 	 * child sending the parent a SIGCHLD.  We don't register an
892f263522aSJoseph Koshy 	 * actual signal handler for SIGCHLD, but instead use our
893f263522aSJoseph Koshy 	 * kqueue to pick up the signal.
894ebccf1e3SJoseph Koshy 	 */
895f263522aSJoseph Koshy 	EV_SET(&kev, SIGCHLD, EVFILT_SIGNAL, EV_ADD, 0, 0, NULL);
896f263522aSJoseph Koshy 	if (kevent(pmcstat_kq, &kev, 1, NULL, 0, NULL) < 0)
897f263522aSJoseph Koshy 		err(EX_OSERR, "ERROR: Cannot register kevent for SIGCHLD");
898ebccf1e3SJoseph Koshy 
89915139246SJoseph Koshy 	/* setup a timer if we have counting mode PMCs needing to be printed */
90015139246SJoseph Koshy 	if ((args.pa_flags & FLAG_HAS_COUNTING_PMCS) &&
90115139246SJoseph Koshy 	    (args.pa_required & FLAG_HAS_OUTPUT_LOGFILE) == 0) {
902ebccf1e3SJoseph Koshy 		EV_SET(&kev, 0, EVFILT_TIMER, EV_ADD, 0,
903ebccf1e3SJoseph Koshy 		    args.pa_interval * 1000, NULL);
904ebccf1e3SJoseph Koshy 
905ebccf1e3SJoseph Koshy 		if (kevent(pmcstat_kq, &kev, 1, NULL, 0, NULL) < 0)
906ebccf1e3SJoseph Koshy 			err(EX_OSERR, "ERROR: Cannot register kevent for "
907ebccf1e3SJoseph Koshy 			    "timer");
908ebccf1e3SJoseph Koshy 	}
909ebccf1e3SJoseph Koshy 
910ebccf1e3SJoseph Koshy 	/* attach PMCs to the target process, starting it if specified */
91115139246SJoseph Koshy 	if (args.pa_flags & (FLAG_HAS_PID | FLAG_HAS_COMMANDLINE))
912ebccf1e3SJoseph Koshy 		pmcstat_setup_process(&args);
913ebccf1e3SJoseph Koshy 
9147fbf3da2SJoseph Koshy 	if (check_driver_stats && pmc_get_driver_stats(&ds_start) < 0)
9157fbf3da2SJoseph Koshy 		err(EX_OSERR, "ERROR: Cannot retrieve driver statistics");
9167fbf3da2SJoseph Koshy 
917ebccf1e3SJoseph Koshy 	/* start the pmcs */
918ebccf1e3SJoseph Koshy 	pmcstat_start_pmcs(&args);
919ebccf1e3SJoseph Koshy 
920ebccf1e3SJoseph Koshy 	/* start the (commandline) process if needed */
92115139246SJoseph Koshy 	if (args.pa_flags & FLAG_HAS_COMMANDLINE)
922ebccf1e3SJoseph Koshy 		pmcstat_start_process(&args);
923ebccf1e3SJoseph Koshy 
92415139246SJoseph Koshy 	/* initialize logging if printing the configured log */
92515139246SJoseph Koshy 	if ((args.pa_flags & FLAG_DO_PRINT) &&
92615139246SJoseph Koshy 	    (args.pa_flags & (FLAG_HAS_PIPE | FLAG_HAS_OUTPUT_LOGFILE)))
92715139246SJoseph Koshy 		pmcstat_initialize_logging(&args);
92815139246SJoseph Koshy 
929ebccf1e3SJoseph Koshy 	/* Handle SIGINT using the kqueue loop */
930ebccf1e3SJoseph Koshy 	sa.sa_handler = SIG_IGN;
931ebccf1e3SJoseph Koshy 	sa.sa_flags   = 0;
932ebccf1e3SJoseph Koshy 	(void) sigemptyset(&sa.sa_mask);
933ebccf1e3SJoseph Koshy 
934ebccf1e3SJoseph Koshy 	if (sigaction(SIGINT, &sa, NULL) < 0)
935ebccf1e3SJoseph Koshy 		err(EX_OSERR, "ERROR: Cannot install signal handler");
936ebccf1e3SJoseph Koshy 
937ebccf1e3SJoseph Koshy 	/*
938ebccf1e3SJoseph Koshy 	 * loop till either the target process (if any) exits, or we
939ebccf1e3SJoseph Koshy 	 * are killed by a SIGINT.
940ebccf1e3SJoseph Koshy 	 */
941f263522aSJoseph Koshy 	runstate = PMCSTAT_RUNNING;
942f263522aSJoseph Koshy 	do_print = 0;
943ebccf1e3SJoseph Koshy 	do {
944ebccf1e3SJoseph Koshy 		if ((c = kevent(pmcstat_kq, NULL, 0, &kev, 1, NULL)) <= 0) {
945ebccf1e3SJoseph Koshy 			if (errno != EINTR)
946ebccf1e3SJoseph Koshy 				err(EX_OSERR, "ERROR: kevent failed");
947ebccf1e3SJoseph Koshy 			else
948ebccf1e3SJoseph Koshy 				continue;
949ebccf1e3SJoseph Koshy 		}
950ebccf1e3SJoseph Koshy 
951ebccf1e3SJoseph Koshy 		if (kev.flags & EV_ERROR)
952ebccf1e3SJoseph Koshy 			errc(EX_OSERR, kev.data, "ERROR: kevent failed");
953ebccf1e3SJoseph Koshy 
954ebccf1e3SJoseph Koshy 		switch (kev.filter) {
955f263522aSJoseph Koshy 		case EVFILT_PROC:  /* target has exited */
95615139246SJoseph Koshy 			if (args.pa_flags & (FLAG_HAS_OUTPUT_LOGFILE |
95715139246SJoseph Koshy 				FLAG_HAS_PIPE))
958f263522aSJoseph Koshy 				runstate = pmcstat_close_log(&args);
95938fc5f34SJoseph Koshy 			else
96038fc5f34SJoseph Koshy 				runstate = PMCSTAT_FINISHED;
96138fc5f34SJoseph Koshy 			do_print = 1;
962f263522aSJoseph Koshy 			break;
963ebccf1e3SJoseph Koshy 
964f263522aSJoseph Koshy 		case EVFILT_READ:  /* log file data is present */
965dc1d9d2eSJoseph Koshy 			runstate = pmcstat_process_log(&args);
966ebccf1e3SJoseph Koshy 			break;
967ebccf1e3SJoseph Koshy 
968ebccf1e3SJoseph Koshy 		case EVFILT_SIGNAL:
969f263522aSJoseph Koshy 			if (kev.ident == SIGCHLD) {
970f263522aSJoseph Koshy 				/*
971f263522aSJoseph Koshy 				 * The child process sends us a
972f263522aSJoseph Koshy 				 * SIGCHLD if its exec() failed.  We
973f263522aSJoseph Koshy 				 * wait for it to exit and then exit
974f263522aSJoseph Koshy 				 * ourselves.
975f263522aSJoseph Koshy 				 */
976f263522aSJoseph Koshy 				(void) wait(&c);
977f263522aSJoseph Koshy 				runstate = PMCSTAT_FINISHED;
978f263522aSJoseph Koshy 			} else if (kev.ident == SIGIO) {
979f263522aSJoseph Koshy 				/*
980f263522aSJoseph Koshy 				 * We get a SIGIO if a PMC loses all
981f263522aSJoseph Koshy 				 * of its targets, or if logfile
982f263522aSJoseph Koshy 				 * writes encounter an error.
983f263522aSJoseph Koshy 				 */
98415139246SJoseph Koshy 				if (args.pa_flags & (FLAG_HAS_OUTPUT_LOGFILE |
98515139246SJoseph Koshy 				    FLAG_HAS_PIPE)) {
986f263522aSJoseph Koshy 					runstate = pmcstat_close_log(&args);
98715139246SJoseph Koshy 					if (args.pa_flags &
98815139246SJoseph Koshy 					    (FLAG_DO_PRINT|FLAG_DO_GPROF))
98915139246SJoseph Koshy 						pmcstat_process_log(&args);
99015139246SJoseph Koshy 				}
991f263522aSJoseph Koshy 				do_print = 1; /* print PMCs at exit */
992f263522aSJoseph Koshy 				runstate = PMCSTAT_FINISHED;
993f263522aSJoseph Koshy 			} else if (kev.ident == SIGINT) {
99415139246SJoseph Koshy 				/* Kill the child process if we started it */
99515139246SJoseph Koshy 				if (args.pa_flags & FLAG_HAS_COMMANDLINE)
996ebccf1e3SJoseph Koshy 					if (kill(args.pa_pid, SIGINT) != 0)
997f263522aSJoseph Koshy 						err(EX_OSERR, "ERROR: cannot "
998f263522aSJoseph Koshy 						    "signal child process");
999f263522aSJoseph Koshy 				runstate = PMCSTAT_FINISHED;
1000ebccf1e3SJoseph Koshy 			} else if (kev.ident == SIGWINCH) {
100115139246SJoseph Koshy 				if (ioctl(fileno(args.pa_printfile),
1002ebccf1e3SJoseph Koshy 					TIOCGWINSZ, &ws) < 0)
1003ebccf1e3SJoseph Koshy 				    err(EX_OSERR, "ERROR: Cannot determine "
1004ebccf1e3SJoseph Koshy 					"window size");
1005ebccf1e3SJoseph Koshy 				pmcstat_displayheight = ws.ws_row - 1;
1006ebccf1e3SJoseph Koshy 			} else
1007ebccf1e3SJoseph Koshy 				assert(0);
1008ebccf1e3SJoseph Koshy 
1009ebccf1e3SJoseph Koshy 			break;
1010f263522aSJoseph Koshy 
1011f263522aSJoseph Koshy 		case EVFILT_TIMER: /* print out counting PMCs */
1012f263522aSJoseph Koshy 			do_print = 1;
1013f263522aSJoseph Koshy 			break;
1014f263522aSJoseph Koshy 
1015ebccf1e3SJoseph Koshy 		}
1016ebccf1e3SJoseph Koshy 
101715139246SJoseph Koshy 		if (do_print &&
101815139246SJoseph Koshy 		    (args.pa_required & FLAG_HAS_OUTPUT_LOGFILE) == 0) {
1019f263522aSJoseph Koshy 			pmcstat_print_pmcs(&args);
102015139246SJoseph Koshy 			if (runstate == PMCSTAT_FINISHED && /* final newline */
102115139246SJoseph Koshy 			    (args.pa_flags & FLAG_DO_PRINT) == 0)
102215139246SJoseph Koshy 				(void) fprintf(args.pa_printfile, "\n");
1023f263522aSJoseph Koshy 			do_print = 0;
1024f263522aSJoseph Koshy 		}
1025f263522aSJoseph Koshy 
1026f263522aSJoseph Koshy 	} while (runstate != PMCSTAT_FINISHED);
1027f263522aSJoseph Koshy 
1028f263522aSJoseph Koshy 	/* flush any pending log entries */
102915139246SJoseph Koshy 	if (args.pa_flags & (FLAG_HAS_OUTPUT_LOGFILE | FLAG_HAS_PIPE))
1030f263522aSJoseph Koshy 		pmc_flush_logfile();
1031ebccf1e3SJoseph Koshy 
1032ebccf1e3SJoseph Koshy 	pmcstat_cleanup(&args);
1033ebccf1e3SJoseph Koshy 
103449874f6eSJoseph Koshy 	free(args.pa_kernel);
103549874f6eSJoseph Koshy 
10367fbf3da2SJoseph Koshy 	/* check if the driver lost any samples or events */
10377fbf3da2SJoseph Koshy 	if (check_driver_stats) {
10387fbf3da2SJoseph Koshy 		if (pmc_get_driver_stats(&ds_end) < 0)
10397fbf3da2SJoseph Koshy 			err(EX_OSERR, "ERROR: Cannot retrieve driver "
10407fbf3da2SJoseph Koshy 			    "statistics");
104149874f6eSJoseph Koshy 		if (ds_start.pm_intr_bufferfull != ds_end.pm_intr_bufferfull &&
104249874f6eSJoseph Koshy 		    args.pa_verbosity > 0)
1043bea6aa93SJoseph Koshy 			warnx("WARNING: some samples were dropped.  Please "
10447fbf3da2SJoseph Koshy 			    "consider tuning the \"kern.hwpmc.nsamples\" "
10457fbf3da2SJoseph Koshy 			    "tunable.");
10467fbf3da2SJoseph Koshy 		if (ds_start.pm_buffer_requests_failed !=
104749874f6eSJoseph Koshy 		    ds_end.pm_buffer_requests_failed &&
104849874f6eSJoseph Koshy 		    args.pa_verbosity > 0)
1049bea6aa93SJoseph Koshy 			warnx("WARNING: some events were discarded.  Please "
10507fbf3da2SJoseph Koshy 			    "consider tuning the \"kern.hwpmc.nbuffers\" "
10517fbf3da2SJoseph Koshy 			    "tunable.");
10527fbf3da2SJoseph Koshy 	}
10537fbf3da2SJoseph Koshy 
10547fbf3da2SJoseph Koshy 	exit(EX_OK);
1055ebccf1e3SJoseph Koshy }
1056