xref: /onnv-gate/usr/src/cmd/powertop/common/events.c (revision 11122:393b5ac48d9b)
19338Srafael.vanoni@sun.com /*
29338Srafael.vanoni@sun.com  * Copyright 2009, Intel Corporation
39338Srafael.vanoni@sun.com  * Copyright 2009, Sun Microsystems, Inc
49338Srafael.vanoni@sun.com  *
59338Srafael.vanoni@sun.com  * This file is part of PowerTOP
69338Srafael.vanoni@sun.com  *
79338Srafael.vanoni@sun.com  * This program file is free software; you can redistribute it and/or modify it
89338Srafael.vanoni@sun.com  * under the terms of the GNU General Public License as published by the
99338Srafael.vanoni@sun.com  * Free Software Foundation; version 2 of the License.
109338Srafael.vanoni@sun.com  *
119338Srafael.vanoni@sun.com  * This program is distributed in the hope that it will be useful, but WITHOUT
129338Srafael.vanoni@sun.com  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
139338Srafael.vanoni@sun.com  * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
149338Srafael.vanoni@sun.com  * for more details.
159338Srafael.vanoni@sun.com  *
169338Srafael.vanoni@sun.com  * You should have received a copy of the GNU General Public License
179338Srafael.vanoni@sun.com  * along with this program in a file named COPYING; if not, write to the
189338Srafael.vanoni@sun.com  * Free Software Foundation, Inc.,
199338Srafael.vanoni@sun.com  * 51 Franklin Street, Fifth Floor,
209338Srafael.vanoni@sun.com  * Boston, MA 02110-1301 USA
219338Srafael.vanoni@sun.com  *
229338Srafael.vanoni@sun.com  * Authors:
239338Srafael.vanoni@sun.com  *	Arjan van de Ven <arjan@linux.intel.com>
249338Srafael.vanoni@sun.com  *	Eric C Saxe <eric.saxe@sun.com>
259338Srafael.vanoni@sun.com  *	Aubrey Li <aubrey.li@intel.com>
269338Srafael.vanoni@sun.com  */
279338Srafael.vanoni@sun.com 
289338Srafael.vanoni@sun.com /*
299338Srafael.vanoni@sun.com  * GPL Disclaimer
309338Srafael.vanoni@sun.com  *
319338Srafael.vanoni@sun.com  * For the avoidance of doubt, except that if any license choice other
329338Srafael.vanoni@sun.com  * than GPL or LGPL is available it will apply instead, Sun elects to
339338Srafael.vanoni@sun.com  * use only the General Public License version 2 (GPLv2) at this time
349338Srafael.vanoni@sun.com  * for any software where a choice of GPL license versions is made
359338Srafael.vanoni@sun.com  * available with the language indicating that GPLv2 or any later
369338Srafael.vanoni@sun.com  * version may be used, or where a choice of which version of the GPL
379338Srafael.vanoni@sun.com  * is applied is otherwise unspecified.
389338Srafael.vanoni@sun.com  */
399338Srafael.vanoni@sun.com 
409338Srafael.vanoni@sun.com #include <string.h>
419338Srafael.vanoni@sun.com #include <stdlib.h>
429338Srafael.vanoni@sun.com #include <dtrace.h>
439338Srafael.vanoni@sun.com #include "powertop.h"
449338Srafael.vanoni@sun.com 
459338Srafael.vanoni@sun.com static dtrace_hdl_t *dtp;
469711Srafael.vanoni@sun.com static event_info_t *event;
479338Srafael.vanoni@sun.com 
489338Srafael.vanoni@sun.com /*ARGSUSED*/
499338Srafael.vanoni@sun.com static int
pt_events_walk(const dtrace_aggdata_t * data,void * arg)509711Srafael.vanoni@sun.com pt_events_walk(const dtrace_aggdata_t *data, void *arg)
519338Srafael.vanoni@sun.com {
529338Srafael.vanoni@sun.com 	dtrace_aggdesc_t 	*aggdesc = data->dtada_desc;
539338Srafael.vanoni@sun.com 	dtrace_recdesc_t 	*rec1, *rec2, *rec3;
549338Srafael.vanoni@sun.com 	dtrace_syminfo_t 	dts;
559711Srafael.vanoni@sun.com 	GElf_Sym 		sym;
569338Srafael.vanoni@sun.com 	uint64_t		offender_addr;
579711Srafael.vanoni@sun.com 	uint64_t 		n = 0;
589338Srafael.vanoni@sun.com 	int32_t 		*instance, *offender_cpu;
599338Srafael.vanoni@sun.com 	int 			i;
609711Srafael.vanoni@sun.com 	char 			*offense_name;
619338Srafael.vanoni@sun.com 
629711Srafael.vanoni@sun.com 	if (g_top_events >= EVENT_NUM_MAX)
639338Srafael.vanoni@sun.com 		return (0);
649338Srafael.vanoni@sun.com 
659338Srafael.vanoni@sun.com 	rec1 = &aggdesc->dtagd_rec[1];
669338Srafael.vanoni@sun.com 	rec2 = &aggdesc->dtagd_rec[2];
679338Srafael.vanoni@sun.com 
689338Srafael.vanoni@sun.com 	/*
699338Srafael.vanoni@sun.com 	 * Report interrupts
709338Srafael.vanoni@sun.com 	 */
719338Srafael.vanoni@sun.com 	if (strcmp(aggdesc->dtagd_name, "interrupts") == 0) {
729338Srafael.vanoni@sun.com 		offense_name = data->dtada_data + rec1->dtrd_offset;
739338Srafael.vanoni@sun.com 
749338Srafael.vanoni@sun.com 		/* LINTED - alignment */
759338Srafael.vanoni@sun.com 		instance = (int32_t *)(data->dtada_data + rec2->dtrd_offset);
769711Srafael.vanoni@sun.com 		(void) snprintf((char *)(event->offender_name),
779338Srafael.vanoni@sun.com 		    EVENT_NAME_MAX, "%s", "<interrupt>");
789711Srafael.vanoni@sun.com 		(void) snprintf((char *)(event->offense_name),
799338Srafael.vanoni@sun.com 		    EVENT_NAME_MAX, "%s#%d", offense_name, *instance);
809338Srafael.vanoni@sun.com 	/*
819338Srafael.vanoni@sun.com 	 * Report kernel events
829338Srafael.vanoni@sun.com 	 */
839338Srafael.vanoni@sun.com 	} else if (strcmp(aggdesc->dtagd_name, "events_k") == 0) {
849338Srafael.vanoni@sun.com 
859711Srafael.vanoni@sun.com 		(void) snprintf((char *)(event->offender_name),
869338Srafael.vanoni@sun.com 		    EVENT_NAME_MAX, "%s", "<kernel>");
879338Srafael.vanoni@sun.com 
889338Srafael.vanoni@sun.com 		/*
899338Srafael.vanoni@sun.com 		 * Casting offender_addr to the wrong type will cause
909338Srafael.vanoni@sun.com 		 * dtrace_lookup_by_addr to return 0 and the report
919338Srafael.vanoni@sun.com 		 * to show an address instead of a name.
929338Srafael.vanoni@sun.com 		 */
939338Srafael.vanoni@sun.com 		switch (g_bit_depth) {
949338Srafael.vanoni@sun.com 		case 32:
959338Srafael.vanoni@sun.com 			/* LINTED - alignment */
969338Srafael.vanoni@sun.com 			offender_addr = *(uint32_t *)(data->dtada_data +
979338Srafael.vanoni@sun.com 			    rec1->dtrd_offset);
989338Srafael.vanoni@sun.com 			break;
999338Srafael.vanoni@sun.com 		case 64:
1009338Srafael.vanoni@sun.com 			/* LINTED - alignment */
1019338Srafael.vanoni@sun.com 			offender_addr = *(uint64_t *)(data->dtada_data +
1029338Srafael.vanoni@sun.com 			    rec1->dtrd_offset);
1039338Srafael.vanoni@sun.com 			break;
1049338Srafael.vanoni@sun.com 		}
1059338Srafael.vanoni@sun.com 
1069338Srafael.vanoni@sun.com 		/*
1079338Srafael.vanoni@sun.com 		 * We have the address of the kernel callout.
1089338Srafael.vanoni@sun.com 		 * Try to resolve it into a meaningful symbol
1099338Srafael.vanoni@sun.com 		 */
1109338Srafael.vanoni@sun.com 		if (offender_addr != NULL && dtrace_lookup_by_addr(dtp,
1119338Srafael.vanoni@sun.com 		    offender_addr, &sym, &dts) == 0) {
1129711Srafael.vanoni@sun.com 			(void) snprintf((char *)(event->offense_name),
1139338Srafael.vanoni@sun.com 			    EVENT_NAME_MAX, "%s`%s", dts.dts_object,
1149338Srafael.vanoni@sun.com 			    dts.dts_name);
1159338Srafael.vanoni@sun.com 		} else {
1169711Srafael.vanoni@sun.com 			(void) snprintf((char *)(event->offense_name),
1179338Srafael.vanoni@sun.com 			    EVENT_NAME_MAX, "0x%llx", offender_addr);
1189338Srafael.vanoni@sun.com 		}
1199338Srafael.vanoni@sun.com 	/*
1209338Srafael.vanoni@sun.com 	 * Report user events
1219338Srafael.vanoni@sun.com 	 */
1229338Srafael.vanoni@sun.com 	} else if (strcmp(aggdesc->dtagd_name, "events_u") == 0) {
1239338Srafael.vanoni@sun.com 		offense_name = data->dtada_data + rec1->dtrd_offset;
1249338Srafael.vanoni@sun.com 
1259711Srafael.vanoni@sun.com 		(void) snprintf((char *)(event->offender_name),
1269338Srafael.vanoni@sun.com 		    EVENT_NAME_MAX, "%s", offense_name);
1279711Srafael.vanoni@sun.com 		(void) snprintf((char *)(event->offense_name),
1289338Srafael.vanoni@sun.com 		    EVENT_NAME_MAX, "<scheduled timeout expiration>");
1299338Srafael.vanoni@sun.com 	/*
1309338Srafael.vanoni@sun.com 	 * Report cross calls
1319338Srafael.vanoni@sun.com 	 */
1329338Srafael.vanoni@sun.com 	} else if (strcmp(aggdesc->dtagd_name, "events_x") == 0) {
1339338Srafael.vanoni@sun.com 		offense_name = data->dtada_data + rec1->dtrd_offset;
1349338Srafael.vanoni@sun.com 
1359711Srafael.vanoni@sun.com 		(void) snprintf((char *)(event->offender_name),
1369338Srafael.vanoni@sun.com 		    EVENT_NAME_MAX, "%s", offense_name);
1379338Srafael.vanoni@sun.com 
1389338Srafael.vanoni@sun.com 		switch (g_bit_depth) {
1399338Srafael.vanoni@sun.com 		case 32:
1409338Srafael.vanoni@sun.com 			/* LINTED - alignment */
1419338Srafael.vanoni@sun.com 			offender_addr = *(uint32_t *)(data->dtada_data +
1429338Srafael.vanoni@sun.com 			    rec2->dtrd_offset);
1439338Srafael.vanoni@sun.com 			break;
1449338Srafael.vanoni@sun.com 		case 64:
1459338Srafael.vanoni@sun.com 			/* LINTED - alignment */
1469338Srafael.vanoni@sun.com 			offender_addr = *(uint64_t *)(data->dtada_data +
1479338Srafael.vanoni@sun.com 			    rec2->dtrd_offset);
1489338Srafael.vanoni@sun.com 			break;
1499338Srafael.vanoni@sun.com 		}
1509338Srafael.vanoni@sun.com 
1519338Srafael.vanoni@sun.com 		/*
1529338Srafael.vanoni@sun.com 		 * Try to resolve the address of the cross call function.
1539338Srafael.vanoni@sun.com 		 */
1549338Srafael.vanoni@sun.com 		if (offender_addr != NULL && dtrace_lookup_by_addr(dtp,
1559338Srafael.vanoni@sun.com 		    offender_addr, &sym, &dts) == 0) {
1569711Srafael.vanoni@sun.com 			(void) snprintf((char *)(event->offense_name),
1579338Srafael.vanoni@sun.com 			    EVENT_NAME_MAX, "<xcalls> %s`%s",
1589338Srafael.vanoni@sun.com 			    dts.dts_object, dts.dts_name);
1599338Srafael.vanoni@sun.com 		} else {
1609711Srafael.vanoni@sun.com 			(void) snprintf((char *)(event->offense_name),
1619338Srafael.vanoni@sun.com 			    EVENT_NAME_MAX, "<xcalls>");
1629338Srafael.vanoni@sun.com 		}
1639338Srafael.vanoni@sun.com 	/*
1649338Srafael.vanoni@sun.com 	 * Report cross calls from other CPUs than the one we're observing
1659338Srafael.vanoni@sun.com 	 * with the -C option
1669338Srafael.vanoni@sun.com 	 */
1679338Srafael.vanoni@sun.com 	} else if (strcmp(aggdesc->dtagd_name, "events_xc") == 0) {
1689338Srafael.vanoni@sun.com 		rec3 = &aggdesc->dtagd_rec[3];
1699338Srafael.vanoni@sun.com 		offense_name = data->dtada_data + rec1->dtrd_offset;
1709338Srafael.vanoni@sun.com 
1719711Srafael.vanoni@sun.com 		(void) snprintf((char *)(event->offender_name),
1729338Srafael.vanoni@sun.com 		    EVENT_NAME_MAX, "%s", offense_name);
1739338Srafael.vanoni@sun.com 
1749338Srafael.vanoni@sun.com 		switch (g_bit_depth) {
1759338Srafael.vanoni@sun.com 		case 32:
1769338Srafael.vanoni@sun.com 			/* LINTED - alignment */
1779338Srafael.vanoni@sun.com 			offender_addr = *(uint32_t *)(data->dtada_data +
1789338Srafael.vanoni@sun.com 			    rec2->dtrd_offset);
1799338Srafael.vanoni@sun.com 			break;
1809338Srafael.vanoni@sun.com 		case 64:
1819338Srafael.vanoni@sun.com 			/* LINTED - alignment */
1829338Srafael.vanoni@sun.com 			offender_addr = *(uint64_t *)(data->dtada_data +
1839338Srafael.vanoni@sun.com 			    rec2->dtrd_offset);
1849338Srafael.vanoni@sun.com 			break;
1859338Srafael.vanoni@sun.com 		}
1869338Srafael.vanoni@sun.com 		/* LINTED - alignment */
1879338Srafael.vanoni@sun.com 		offender_cpu = (int32_t *)(data->dtada_data +
1889338Srafael.vanoni@sun.com 		    rec3->dtrd_offset);
1899338Srafael.vanoni@sun.com 
1909338Srafael.vanoni@sun.com 		/*
1919338Srafael.vanoni@sun.com 		 * Try to resolve the address of the cross call function.
1929338Srafael.vanoni@sun.com 		 */
1939338Srafael.vanoni@sun.com 		if (offender_addr != NULL && dtrace_lookup_by_addr(dtp,
1949338Srafael.vanoni@sun.com 		    offender_addr, &sym, &dts) == 0) {
1959711Srafael.vanoni@sun.com 			(void) snprintf((char *)(event->offense_name),
1969338Srafael.vanoni@sun.com 			    EVENT_NAME_MAX, "<xcalls> %s`%s (CPU %d)",
1979338Srafael.vanoni@sun.com 			    dts.dts_object, dts.dts_name, *offender_cpu);
1989338Srafael.vanoni@sun.com 		} else {
1999711Srafael.vanoni@sun.com 			(void) snprintf((char *)(event->offense_name),
2009338Srafael.vanoni@sun.com 			    EVENT_NAME_MAX, "<xcalls> (CPU %d)",
2019338Srafael.vanoni@sun.com 			    *offender_cpu);
2029338Srafael.vanoni@sun.com 		}
2039338Srafael.vanoni@sun.com 	/*
2049338Srafael.vanoni@sun.com 	 * Report unknown events
2059338Srafael.vanoni@sun.com 	 */
2069338Srafael.vanoni@sun.com 	} else {
2079711Srafael.vanoni@sun.com 		(void) snprintf((char *)(event->offender_name),
2089338Srafael.vanoni@sun.com 		    EVENT_NAME_MAX, "%s", "<unknown>");
2099711Srafael.vanoni@sun.com 		(void) snprintf((char *)(event->offense_name),
2109338Srafael.vanoni@sun.com 		    EVENT_NAME_MAX, "%s", "<unknown>");
2119338Srafael.vanoni@sun.com 	}
2129338Srafael.vanoni@sun.com 
2139711Srafael.vanoni@sun.com 	for (i = 0; i < g_ncpus; i++) {
2149338Srafael.vanoni@sun.com 		/* LINTED - alignment */
2159338Srafael.vanoni@sun.com 		n += *((uint64_t *)(data->dtada_percpu[i]));
2169711Srafael.vanoni@sun.com 	}
2179338Srafael.vanoni@sun.com 
2189711Srafael.vanoni@sun.com 	event->total_count = n;
2199711Srafael.vanoni@sun.com 
2209711Srafael.vanoni@sun.com 	event++;
2219711Srafael.vanoni@sun.com 	g_top_events++;
2229338Srafael.vanoni@sun.com 
2239338Srafael.vanoni@sun.com 	return (DTRACE_AGGWALK_NEXT);
2249338Srafael.vanoni@sun.com }
2259338Srafael.vanoni@sun.com 
2269338Srafael.vanoni@sun.com int
pt_events_stat_prepare(void)2279338Srafael.vanoni@sun.com pt_events_stat_prepare(void)
2289338Srafael.vanoni@sun.com {
2299338Srafael.vanoni@sun.com 	dtrace_prog_t 		*prog;
2309338Srafael.vanoni@sun.com 	dtrace_proginfo_t 	info;
2319711Srafael.vanoni@sun.com 	dtrace_optval_t 	statustime;
2329338Srafael.vanoni@sun.com 	int 			err;
2339338Srafael.vanoni@sun.com 	char			*prog_ptr;
2349338Srafael.vanoni@sun.com 
2359711Srafael.vanoni@sun.com 	event = g_event_info;
2369338Srafael.vanoni@sun.com 
2379338Srafael.vanoni@sun.com 	if ((dtp = dtrace_open(DTRACE_VERSION, 0, &err)) == NULL) {
238*11122Srafael.vanoni@sun.com 		pt_error("cannot open dtrace library for the event report: "
239*11122Srafael.vanoni@sun.com 		    "%s\n", dtrace_errmsg(NULL, err));
2409338Srafael.vanoni@sun.com 		return (-1);
2419338Srafael.vanoni@sun.com 	}
2429338Srafael.vanoni@sun.com 
2439338Srafael.vanoni@sun.com 	/*
2449338Srafael.vanoni@sun.com 	 * Execute different scripts (defined in the platform specific file)
2459338Srafael.vanoni@sun.com 	 * depending on user specified options.
2469338Srafael.vanoni@sun.com 	 */
2479711Srafael.vanoni@sun.com 	if (PT_ON_VERBOSE) {
2489338Srafael.vanoni@sun.com 		prog_ptr = (char *)g_dtp_events_v;
2499338Srafael.vanoni@sun.com 	} else {
2509711Srafael.vanoni@sun.com 		if (PT_ON_CPU)
2519338Srafael.vanoni@sun.com 			prog_ptr = (char *)g_dtp_events_c;
2529338Srafael.vanoni@sun.com 		else
2539338Srafael.vanoni@sun.com 			prog_ptr = (char *)g_dtp_events;
2549338Srafael.vanoni@sun.com 	}
2559338Srafael.vanoni@sun.com 
2569338Srafael.vanoni@sun.com 	if ((prog = dtrace_program_strcompile(dtp, prog_ptr,
2579338Srafael.vanoni@sun.com 	    DTRACE_PROBESPEC_NAME, 0, g_argc, g_argv)) == NULL) {
258*11122Srafael.vanoni@sun.com 		pt_error("failed to compile the event report program\n");
2599338Srafael.vanoni@sun.com 		return (dtrace_errno(dtp));
2609338Srafael.vanoni@sun.com 	}
2619338Srafael.vanoni@sun.com 
2629338Srafael.vanoni@sun.com 	if (dtrace_program_exec(dtp, prog, &info) == -1) {
263*11122Srafael.vanoni@sun.com 		pt_error("failed to enable probes for the event report\n");
2649338Srafael.vanoni@sun.com 		return (dtrace_errno(dtp));
2659338Srafael.vanoni@sun.com 	}
2669338Srafael.vanoni@sun.com 
2679338Srafael.vanoni@sun.com 	if (dtrace_setopt(dtp, "aggsize", "128k") == -1) {
268*11122Srafael.vanoni@sun.com 		pt_error("failed to set 'aggsize' for the event report\n");
2699338Srafael.vanoni@sun.com 		return (dtrace_errno(dtp));
2709338Srafael.vanoni@sun.com 	}
2719338Srafael.vanoni@sun.com 
2729338Srafael.vanoni@sun.com 	if (dtrace_setopt(dtp, "aggrate", "0") == -1) {
273*11122Srafael.vanoni@sun.com 		pt_error("failed to set 'aggrate' for the event report\n");
2749338Srafael.vanoni@sun.com 		return (dtrace_errno(dtp));
2759338Srafael.vanoni@sun.com 	}
2769338Srafael.vanoni@sun.com 
2779338Srafael.vanoni@sun.com 	if (dtrace_setopt(dtp, "aggpercpu", 0) == -1) {
278*11122Srafael.vanoni@sun.com 		pt_error("failed to set 'aggpercpu' for the event report\n");
2799338Srafael.vanoni@sun.com 		return (dtrace_errno(dtp));
2809338Srafael.vanoni@sun.com 	}
2819338Srafael.vanoni@sun.com 
2829338Srafael.vanoni@sun.com 	if (dtrace_go(dtp) != 0) {
283*11122Srafael.vanoni@sun.com 		pt_error("failed to start the event report observation\n");
2849338Srafael.vanoni@sun.com 		return (dtrace_errno(dtp));
2859338Srafael.vanoni@sun.com 	}
2869338Srafael.vanoni@sun.com 
2879338Srafael.vanoni@sun.com 	if (dtrace_getopt(dtp, "statusrate", &statustime) == -1) {
288*11122Srafael.vanoni@sun.com 		pt_error("failed to get 'statusrate' for the event report\n");
2899338Srafael.vanoni@sun.com 		return (dtrace_errno(dtp));
2909338Srafael.vanoni@sun.com 	}
2919338Srafael.vanoni@sun.com 
2929338Srafael.vanoni@sun.com 	return (0);
2939338Srafael.vanoni@sun.com }
2949338Srafael.vanoni@sun.com 
2959338Srafael.vanoni@sun.com int
pt_events_stat_collect(void)2969338Srafael.vanoni@sun.com pt_events_stat_collect(void)
2979338Srafael.vanoni@sun.com {
2989711Srafael.vanoni@sun.com 	g_top_events = 0;
2999711Srafael.vanoni@sun.com 	event = g_event_info;
3009338Srafael.vanoni@sun.com 
3019338Srafael.vanoni@sun.com 	if (dtrace_status(dtp) == -1)
3029338Srafael.vanoni@sun.com 		return (-1);
3039338Srafael.vanoni@sun.com 
3049338Srafael.vanoni@sun.com 	if (dtrace_aggregate_snap(dtp) != 0)
305*11122Srafael.vanoni@sun.com 		pt_error("failed to collect data for the event report\n");
3069338Srafael.vanoni@sun.com 
3079711Srafael.vanoni@sun.com 	if (dtrace_aggregate_walk_keyvarsorted(dtp, pt_events_walk, NULL) != 0)
308*11122Srafael.vanoni@sun.com 		pt_error("failed to sort data for the event report\n");
3099338Srafael.vanoni@sun.com 
3109338Srafael.vanoni@sun.com 	dtrace_aggregate_clear(dtp);
3119338Srafael.vanoni@sun.com 
3129338Srafael.vanoni@sun.com 	return (0);
3139338Srafael.vanoni@sun.com }
314