xref: /onnv-gate/usr/src/cmd/powertop/common/cpuidle.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 <dtrace.h>
429338Srafael.vanoni@sun.com #include "powertop.h"
439338Srafael.vanoni@sun.com 
449711Srafael.vanoni@sun.com #define	S2NS(x)		((x) * (NANOSEC))
459711Srafael.vanoni@sun.com 
469338Srafael.vanoni@sun.com static dtrace_hdl_t 	*dtp;
479338Srafael.vanoni@sun.com 
489338Srafael.vanoni@sun.com /*
499338Srafael.vanoni@sun.com  * Buffer containing DTrace program to track CPU idle state transitions
509338Srafael.vanoni@sun.com  */
519338Srafael.vanoni@sun.com static const char *dtp_cpuidle =
529338Srafael.vanoni@sun.com ":::idle-state-transition"
539338Srafael.vanoni@sun.com "/arg0 != 0/"
549338Srafael.vanoni@sun.com "{"
559338Srafael.vanoni@sun.com "	self->start = timestamp;"
569338Srafael.vanoni@sun.com "	self->state = arg0;"
579338Srafael.vanoni@sun.com "}"
589338Srafael.vanoni@sun.com ""
599338Srafael.vanoni@sun.com ":::idle-state-transition"
609338Srafael.vanoni@sun.com "/arg0 == 0 && self->start/"
619338Srafael.vanoni@sun.com "{"
629338Srafael.vanoni@sun.com "	@number[self->state] = count();"
639711Srafael.vanoni@sun.com "	@times[self->state] = sum(timestamp - self->start);"
649338Srafael.vanoni@sun.com "	self->start = 0;"
659338Srafael.vanoni@sun.com "	self->state = 0;"
669338Srafael.vanoni@sun.com "}";
679338Srafael.vanoni@sun.com 
689338Srafael.vanoni@sun.com /*
699338Srafael.vanoni@sun.com  * Same as above but only for a specific CPU
709338Srafael.vanoni@sun.com  */
719338Srafael.vanoni@sun.com static const char *dtp_cpuidle_c =
729338Srafael.vanoni@sun.com ":::idle-state-transition"
739338Srafael.vanoni@sun.com "/cpu == $0 &&"
749338Srafael.vanoni@sun.com " arg0 != 0/"
759338Srafael.vanoni@sun.com "{"
769338Srafael.vanoni@sun.com "	self->start = timestamp;"
779338Srafael.vanoni@sun.com "	self->state = arg0;"
789338Srafael.vanoni@sun.com "}"
799338Srafael.vanoni@sun.com ""
809338Srafael.vanoni@sun.com ":::idle-state-transition"
819338Srafael.vanoni@sun.com "/cpu == $0 &&"
829338Srafael.vanoni@sun.com " arg0 == 0 && self->start/"
839338Srafael.vanoni@sun.com "{"
849338Srafael.vanoni@sun.com "	@number[self->state] = count();"
859711Srafael.vanoni@sun.com "	@times[self->state] = sum(timestamp - self->start);"
869338Srafael.vanoni@sun.com "	self->start = 0;"
879338Srafael.vanoni@sun.com "	self->state = 0;"
889338Srafael.vanoni@sun.com "}";
899338Srafael.vanoni@sun.com 
909338Srafael.vanoni@sun.com static int 	pt_cpuidle_dtrace_walk(const dtrace_aggdata_t *, void *);
919338Srafael.vanoni@sun.com 
929338Srafael.vanoni@sun.com /*
939338Srafael.vanoni@sun.com  * Perform setup necessary to track CPU idle state transitions
949338Srafael.vanoni@sun.com  */
959338Srafael.vanoni@sun.com int
pt_cpuidle_stat_prepare(void)969338Srafael.vanoni@sun.com pt_cpuidle_stat_prepare(void)
979338Srafael.vanoni@sun.com {
989338Srafael.vanoni@sun.com 	dtrace_prog_t 		*prog;
999338Srafael.vanoni@sun.com 	dtrace_proginfo_t 	info;
1009338Srafael.vanoni@sun.com 	dtrace_optval_t 	statustime;
1019338Srafael.vanoni@sun.com 	int 			err;
1029338Srafael.vanoni@sun.com 	char			*prog_ptr;
1039338Srafael.vanoni@sun.com 
1049338Srafael.vanoni@sun.com 	if ((dtp = dtrace_open(DTRACE_VERSION, 0, &err)) == NULL) {
105*11122Srafael.vanoni@sun.com 		pt_error("cannot open dtrace library for the %s report: %s\n",
106*11122Srafael.vanoni@sun.com 		    g_msg_idle_state, dtrace_errmsg(NULL, err));
1079338Srafael.vanoni@sun.com 		return (-1);
1089338Srafael.vanoni@sun.com 	}
1099338Srafael.vanoni@sun.com 
1109338Srafael.vanoni@sun.com 	/*
1119338Srafael.vanoni@sun.com 	 * Execute different scripts (defined above) depending on
1129338Srafael.vanoni@sun.com 	 * user specified options.
1139338Srafael.vanoni@sun.com 	 */
1149711Srafael.vanoni@sun.com 	if (PT_ON_CPU)
1159338Srafael.vanoni@sun.com 		prog_ptr = (char *)dtp_cpuidle_c;
1169338Srafael.vanoni@sun.com 	else
1179338Srafael.vanoni@sun.com 		prog_ptr = (char *)dtp_cpuidle;
1189338Srafael.vanoni@sun.com 
1199338Srafael.vanoni@sun.com 	if ((prog = dtrace_program_strcompile(dtp, prog_ptr,
1209338Srafael.vanoni@sun.com 	    DTRACE_PROBESPEC_NAME, 0, g_argc, g_argv)) == NULL) {
121*11122Srafael.vanoni@sun.com 		pt_error("failed to compile %s program\n", g_msg_idle_state);
1229338Srafael.vanoni@sun.com 		return (dtrace_errno(dtp));
1239338Srafael.vanoni@sun.com 	}
1249338Srafael.vanoni@sun.com 
1259338Srafael.vanoni@sun.com 	if (dtrace_program_exec(dtp, prog, &info) == -1) {
126*11122Srafael.vanoni@sun.com 		pt_error("failed to enable %s probes\n", g_msg_idle_state);
1279338Srafael.vanoni@sun.com 		return (dtrace_errno(dtp));
1289338Srafael.vanoni@sun.com 	}
1299338Srafael.vanoni@sun.com 
130*11122Srafael.vanoni@sun.com 	if (dtrace_setopt(dtp, "aggsize", "128k") == -1)
131*11122Srafael.vanoni@sun.com 		pt_error("failed to set %s 'aggsize'\n", g_msg_idle_state);
1329338Srafael.vanoni@sun.com 
133*11122Srafael.vanoni@sun.com 	if (dtrace_setopt(dtp, "aggrate", "0") == -1)
134*11122Srafael.vanoni@sun.com 		pt_error("failed to set %s 'aggrate'\n", g_msg_idle_state);
1359338Srafael.vanoni@sun.com 
136*11122Srafael.vanoni@sun.com 	if (dtrace_setopt(dtp, "aggpercpu", 0) == -1)
137*11122Srafael.vanoni@sun.com 		pt_error("failed to set %s 'aggpercpu'\n", g_msg_idle_state);
1389338Srafael.vanoni@sun.com 
1399338Srafael.vanoni@sun.com 	if (dtrace_go(dtp) != 0) {
140*11122Srafael.vanoni@sun.com 		pt_error("failed to start %s observation\n", g_msg_idle_state);
1419338Srafael.vanoni@sun.com 		return (dtrace_errno(dtp));
1429338Srafael.vanoni@sun.com 	}
1439338Srafael.vanoni@sun.com 
1449338Srafael.vanoni@sun.com 	if (dtrace_getopt(dtp, "statusrate", &statustime) == -1) {
145*11122Srafael.vanoni@sun.com 		pt_error("failed to get %s 'statusrate'\n", g_msg_idle_state);
1469338Srafael.vanoni@sun.com 		return (dtrace_errno(dtp));
1479338Srafael.vanoni@sun.com 	}
1489338Srafael.vanoni@sun.com 
1499338Srafael.vanoni@sun.com 	return (0);
1509338Srafael.vanoni@sun.com }
1519338Srafael.vanoni@sun.com 
1529338Srafael.vanoni@sun.com /*
1539338Srafael.vanoni@sun.com  * The DTrace probes have been enabled, and are tracking CPU idle state
1549338Srafael.vanoni@sun.com  * transitions. Take a snapshot of the aggregations, and invoke the aggregation
1559338Srafael.vanoni@sun.com  * walker to process any records. The walker does most of the accounting work
1569338Srafael.vanoni@sun.com  * chalking up time spent into the g_cstate_info structure.
1579338Srafael.vanoni@sun.com  */
1589338Srafael.vanoni@sun.com int
pt_cpuidle_stat_collect(double interval)1599338Srafael.vanoni@sun.com pt_cpuidle_stat_collect(double interval)
1609338Srafael.vanoni@sun.com {
161*11122Srafael.vanoni@sun.com 	int i;
162*11122Srafael.vanoni@sun.com 	hrtime_t t = 0;
1639338Srafael.vanoni@sun.com 
1649338Srafael.vanoni@sun.com 	/*
1659338Srafael.vanoni@sun.com 	 * Assume that all the time spent in this interval will
1669338Srafael.vanoni@sun.com 	 * be the default "0" state. The DTrace walker will reallocate
1679338Srafael.vanoni@sun.com 	 * time out of the default bucket as it processes aggregation
1689338Srafael.vanoni@sun.com 	 * records for time spent in other states.
1699338Srafael.vanoni@sun.com 	 */
1709711Srafael.vanoni@sun.com 	g_cstate_info[0].total_time = (uint64_t)S2NS(interval *
1719711Srafael.vanoni@sun.com 	    g_ncpus_observed);
1729338Srafael.vanoni@sun.com 
1739338Srafael.vanoni@sun.com 	if (dtrace_status(dtp) == -1)
1749338Srafael.vanoni@sun.com 		return (-1);
1759338Srafael.vanoni@sun.com 
1769338Srafael.vanoni@sun.com 	if (dtrace_aggregate_snap(dtp) != 0)
177*11122Srafael.vanoni@sun.com 		pt_error("failed to collect data for %s\n", g_msg_idle_state);
1789338Srafael.vanoni@sun.com 
1799338Srafael.vanoni@sun.com 	if (dtrace_aggregate_walk_keyvarsorted(dtp, pt_cpuidle_dtrace_walk,
1809338Srafael.vanoni@sun.com 	    NULL) != 0)
181*11122Srafael.vanoni@sun.com 		pt_error("failed to sort %s data\n", g_msg_idle_state);
1829338Srafael.vanoni@sun.com 
1839338Srafael.vanoni@sun.com 	dtrace_aggregate_clear(dtp);
1849338Srafael.vanoni@sun.com 
1859338Srafael.vanoni@sun.com 	/*
1869338Srafael.vanoni@sun.com 	 * Populate g_cstate_info with the correct amount of time spent
1879338Srafael.vanoni@sun.com 	 * in each C state and update the number of C states in g_max_cstate
1889338Srafael.vanoni@sun.com 	 */
1899338Srafael.vanoni@sun.com 	g_total_c_time = 0;
1909338Srafael.vanoni@sun.com 	for (i = 0; i < NSTATES; i++) {
1919338Srafael.vanoni@sun.com 		if (g_cstate_info[i].total_time > 0) {
1929338Srafael.vanoni@sun.com 			g_total_c_time += g_cstate_info[i].total_time;
1939338Srafael.vanoni@sun.com 			if (i > g_max_cstate)
1949338Srafael.vanoni@sun.com 				g_max_cstate = i;
1959338Srafael.vanoni@sun.com 			if (g_cstate_info[i].last_time > t) {
1969338Srafael.vanoni@sun.com 				t = g_cstate_info[i].last_time;
1979338Srafael.vanoni@sun.com 				g_longest_cstate = i;
1989338Srafael.vanoni@sun.com 			}
1999338Srafael.vanoni@sun.com 		}
2009338Srafael.vanoni@sun.com 	}
2019338Srafael.vanoni@sun.com 
2029338Srafael.vanoni@sun.com 	return (0);
2039338Srafael.vanoni@sun.com }
2049338Srafael.vanoni@sun.com 
2059338Srafael.vanoni@sun.com /*
2069338Srafael.vanoni@sun.com  * DTrace aggregation walker that sorts through a snapshot of data records
2079338Srafael.vanoni@sun.com  * collected during firings of the idle-state-transition probe.
2089338Srafael.vanoni@sun.com  *
2099338Srafael.vanoni@sun.com  * XXX A way of querying the current idle state for a CPU is needed in addition
2109338Srafael.vanoni@sun.com  *     to logic similar to that in cpufreq.c
2119338Srafael.vanoni@sun.com  */
2129338Srafael.vanoni@sun.com /*ARGSUSED*/
2139338Srafael.vanoni@sun.com static int
pt_cpuidle_dtrace_walk(const dtrace_aggdata_t * data,void * arg)2149338Srafael.vanoni@sun.com pt_cpuidle_dtrace_walk(const dtrace_aggdata_t *data, void *arg)
2159338Srafael.vanoni@sun.com {
2169338Srafael.vanoni@sun.com 	dtrace_aggdesc_t 	*aggdesc = data->dtada_desc;
2179338Srafael.vanoni@sun.com 	dtrace_recdesc_t 	*rec;
218*11122Srafael.vanoni@sun.com 	uint64_t 		n = 0, state;
2199338Srafael.vanoni@sun.com 	int 			i;
2209338Srafael.vanoni@sun.com 
2219338Srafael.vanoni@sun.com 	rec = &aggdesc->dtagd_rec[1];
222*11122Srafael.vanoni@sun.com 
223*11122Srafael.vanoni@sun.com 	switch (g_bit_depth) {
224*11122Srafael.vanoni@sun.com 		case 32:
225*11122Srafael.vanoni@sun.com 			/* LINTED - alignment */
226*11122Srafael.vanoni@sun.com 			state = *(uint32_t *)(data->dtada_data +
227*11122Srafael.vanoni@sun.com 			    rec->dtrd_offset);
228*11122Srafael.vanoni@sun.com 			break;
229*11122Srafael.vanoni@sun.com 		case 64:
230*11122Srafael.vanoni@sun.com 			/* LINTED - alignment */
231*11122Srafael.vanoni@sun.com 			state = *(uint64_t *)(data->dtada_data +
232*11122Srafael.vanoni@sun.com 			    rec->dtrd_offset);
233*11122Srafael.vanoni@sun.com 			break;
234*11122Srafael.vanoni@sun.com 	}
2359338Srafael.vanoni@sun.com 
2369338Srafael.vanoni@sun.com 	if (strcmp(aggdesc->dtagd_name, "number") == 0) {
2379338Srafael.vanoni@sun.com 		for (i = 0; i < g_ncpus; i++) {
2389338Srafael.vanoni@sun.com 			/* LINTED - alignment */
2399338Srafael.vanoni@sun.com 			n += *((uint64_t *)(data->dtada_percpu[i]));
2409338Srafael.vanoni@sun.com 		}
2419338Srafael.vanoni@sun.com 		g_total_events += n;
2429338Srafael.vanoni@sun.com 		g_cstate_info[state].events += n;
2439338Srafael.vanoni@sun.com 	}
2449338Srafael.vanoni@sun.com 	else
2459338Srafael.vanoni@sun.com 		if (strcmp(aggdesc->dtagd_name, "times") == 0) {
2469338Srafael.vanoni@sun.com 			for (i = 0; i < g_ncpus; i++) {
2479338Srafael.vanoni@sun.com 				/* LINTED - alignment */
2489338Srafael.vanoni@sun.com 				n += *((uint64_t *)(data->dtada_percpu[i]));
2499338Srafael.vanoni@sun.com 			}
2509338Srafael.vanoni@sun.com 			g_cstate_info[state].last_time = n;
2519338Srafael.vanoni@sun.com 			g_cstate_info[state].total_time += n;
2529338Srafael.vanoni@sun.com 			if (g_cstate_info[0].total_time >= n)
2539338Srafael.vanoni@sun.com 				g_cstate_info[0].total_time -= n;
2549711Srafael.vanoni@sun.com 			else
2559711Srafael.vanoni@sun.com 				g_cstate_info[0].total_time = 0;
2569338Srafael.vanoni@sun.com 		}
2579338Srafael.vanoni@sun.com 
2589338Srafael.vanoni@sun.com 	return (DTRACE_AGGWALK_NEXT);
2599338Srafael.vanoni@sun.com }
260