xref: /aosp_15_r20/external/autotest/client/profilers/powertop/src/intelcstates.c (revision 9c5db1993ded3edbeafc8092d69fe5de2ee02df7)
1*9c5db199SXin Li /*
2*9c5db199SXin Li  * Copyright 2008, Intel Corporation
3*9c5db199SXin Li  *
4*9c5db199SXin Li  * This file is part of PowerTOP
5*9c5db199SXin Li  *
6*9c5db199SXin Li  * This program file is free software; you can redistribute it and/or modify it
7*9c5db199SXin Li  * under the terms of the GNU General Public License as published by the
8*9c5db199SXin Li  * Free Software Foundation; version 2 of the License.
9*9c5db199SXin Li  *
10*9c5db199SXin Li  * This program is distributed in the hope that it will be useful, but WITHOUT
11*9c5db199SXin Li  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
12*9c5db199SXin Li  * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
13*9c5db199SXin Li  * for more details.
14*9c5db199SXin Li  *
15*9c5db199SXin Li  * You should have received a copy of the GNU General Public License
16*9c5db199SXin Li  * along with this program in a file named COPYING; if not, write to the
17*9c5db199SXin Li  * Free Software Foundation, Inc.,
18*9c5db199SXin Li  * 51 Franklin Street, Fifth Floor,
19*9c5db199SXin Li  * Boston, MA 02110-1301 USA
20*9c5db199SXin Li  *
21*9c5db199SXin Li  * Authors:
22*9c5db199SXin Li  * 	Arjan van de Ven <[email protected]>
23*9c5db199SXin Li  */
24*9c5db199SXin Li 
25*9c5db199SXin Li #include <unistd.h>
26*9c5db199SXin Li #include <stdio.h>
27*9c5db199SXin Li #include <stdlib.h>
28*9c5db199SXin Li #include <string.h>
29*9c5db199SXin Li #include <stdint.h>
30*9c5db199SXin Li #include <sys/types.h>
31*9c5db199SXin Li #include <dirent.h>
32*9c5db199SXin Li #include <ctype.h>
33*9c5db199SXin Li 
34*9c5db199SXin Li #include "powertop.h"
35*9c5db199SXin Li 
36*9c5db199SXin Li #ifdef __i386
37*9c5db199SXin Li 
38*9c5db199SXin Li 
39*9c5db199SXin Li /*
40*9c5db199SXin Li  * Perform a CPU ID operation; with various registers set
41*9c5db199SXin Li  */
cpuid(unsigned int * eax,unsigned int * ebx,unsigned int * ecx,unsigned int * edx)42*9c5db199SXin Li static void cpuid(      unsigned int *eax,
43*9c5db199SXin Li                         unsigned int *ebx,
44*9c5db199SXin Li                         unsigned int *ecx,
45*9c5db199SXin Li                         unsigned int *edx)
46*9c5db199SXin Li {
47*9c5db199SXin Li 	/* call the cpuid instruction with the registers as input and output
48*9c5db199SXin Li 	 * modification by Dwokfur based on Sam Hocevar's discussion on
49*9c5db199SXin Li 	 * how to make Assemly code PIC compatible:
50*9c5db199SXin Li 	 * http://sam.zoy.org/blog/2007-04-13-shlib-with-non-pic-code-have-inline-assembly-and-pic-mix-well
51*9c5db199SXin Li 	 */
52*9c5db199SXin Li 	__asm__("pushl %%ebx	\n\t" /* save %ebx */
53*9c5db199SXin Li 		"cpuid		\n\t"
54*9c5db199SXin Li 		"movl %%ebx, %1	\n\t" /* save what cpuid just put in %ebx */
55*9c5db199SXin Li 		"popl %%ebx	\n\t" /* restore the old %ebx */
56*9c5db199SXin Li 		: "=a" (*eax),
57*9c5db199SXin Li 		  "=r" (*ebx),
58*9c5db199SXin Li 		  "=c" (*ecx),
59*9c5db199SXin Li 		  "=d" (*edx)
60*9c5db199SXin Li 		: "0" (*eax),
61*9c5db199SXin Li 		  "1" (*ebx),
62*9c5db199SXin Li 		  "2" (*ecx),
63*9c5db199SXin Li 		  "3" (*edx)
64*9c5db199SXin Li 		);
65*9c5db199SXin Li }
66*9c5db199SXin Li 
67*9c5db199SXin Li #endif
68*9c5db199SXin Li 
69*9c5db199SXin Li 
print_intel_cstates(void)70*9c5db199SXin Li void print_intel_cstates(void)
71*9c5db199SXin Li {
72*9c5db199SXin Li #ifdef __i386__
73*9c5db199SXin Li 
74*9c5db199SXin Li         int bios_table[8];
75*9c5db199SXin Li         int bioscount = 0;
76*9c5db199SXin Li 	DIR *cpudir;
77*9c5db199SXin Li 	DIR *dir;
78*9c5db199SXin Li 	struct dirent *entry;
79*9c5db199SXin Li 	FILE *file = NULL;
80*9c5db199SXin Li 	char line[4096];
81*9c5db199SXin Li 	char filename[128], *f;
82*9c5db199SXin Li 	int len, i;
83*9c5db199SXin Li 	unsigned int eax, ebx, ecx, edx;
84*9c5db199SXin Li 
85*9c5db199SXin Li 	memset(bios_table, 0, sizeof(bios_table));
86*9c5db199SXin Li 
87*9c5db199SXin Li 
88*9c5db199SXin Li 	cpudir = opendir("/sys/devices/system/cpu");
89*9c5db199SXin Li 	if (!cpudir)
90*9c5db199SXin Li 		return;
91*9c5db199SXin Li 
92*9c5db199SXin Li 	/* Loop over cpuN entries */
93*9c5db199SXin Li 	while ((entry = readdir(cpudir))) {
94*9c5db199SXin Li 		if (strlen(entry->d_name) < 3)
95*9c5db199SXin Li 			continue;
96*9c5db199SXin Li 
97*9c5db199SXin Li 		if (!isdigit(entry->d_name[3]))
98*9c5db199SXin Li 			continue;
99*9c5db199SXin Li 
100*9c5db199SXin Li 		len = sprintf(filename, "/sys/devices/system/cpu/%s/cpuidle",
101*9c5db199SXin Li 			      entry->d_name);
102*9c5db199SXin Li 
103*9c5db199SXin Li 		dir = opendir(filename);
104*9c5db199SXin Li 		if (!dir)
105*9c5db199SXin Li 			return;
106*9c5db199SXin Li 
107*9c5db199SXin Li 		/* For each C-state, there is a stateX directory which
108*9c5db199SXin Li 		 * contains a 'usage' and a 'time' (duration) file */
109*9c5db199SXin Li 		while ((entry = readdir(dir))) {
110*9c5db199SXin Li 			if (strlen(entry->d_name) < 3)
111*9c5db199SXin Li 				continue;
112*9c5db199SXin Li 			sprintf(filename + len, "/%s/desc", entry->d_name);
113*9c5db199SXin Li 			file = fopen(filename, "r");
114*9c5db199SXin Li 			if (file) {
115*9c5db199SXin Li 
116*9c5db199SXin Li 				memset(line, 0, 4096);
117*9c5db199SXin Li 				f = fgets(line, 4096, file);
118*9c5db199SXin Li 				fclose(file);
119*9c5db199SXin Li 				if (f == NULL)
120*9c5db199SXin Li 					break;
121*9c5db199SXin Li 
122*9c5db199SXin Li 				f = strstr(line, "MWAIT ");
123*9c5db199SXin Li 				if (f) {
124*9c5db199SXin Li 					f += 6;
125*9c5db199SXin Li 					bios_table[(strtoull(f, NULL, 16)>>4) + 1]++;
126*9c5db199SXin Li 					bioscount++;
127*9c5db199SXin Li 				}
128*9c5db199SXin Li 			}
129*9c5db199SXin Li 		}
130*9c5db199SXin Li 		closedir(dir);
131*9c5db199SXin Li 
132*9c5db199SXin Li 	}
133*9c5db199SXin Li 	closedir(cpudir);
134*9c5db199SXin Li 	if (!bioscount)
135*9c5db199SXin Li 		return;
136*9c5db199SXin Li 
137*9c5db199SXin Li 	eax = 5;
138*9c5db199SXin Li 	ebx = 0; ecx = 0; edx = 0;
139*9c5db199SXin Li 	cpuid(&eax, &ebx, &ecx, &edx);
140*9c5db199SXin Li 	if (!edx || ((ecx&1) == 0))
141*9c5db199SXin Li 		return;
142*9c5db199SXin Li 
143*9c5db199SXin Li 	printf(_("Your CPU supports the following C-states : "));
144*9c5db199SXin Li 	i = 0;
145*9c5db199SXin Li 	while (edx) {
146*9c5db199SXin Li 		if (edx&7)
147*9c5db199SXin Li 			printf("C%i ", i);
148*9c5db199SXin Li 		edx = edx >> 4;
149*9c5db199SXin Li 		i++;
150*9c5db199SXin Li 	}
151*9c5db199SXin Li 	printf("\n");
152*9c5db199SXin Li 	printf(_("Your BIOS reports the following C-states : "));
153*9c5db199SXin Li 	for (i = 0; i < 8; i++)
154*9c5db199SXin Li 		if (bios_table[i])
155*9c5db199SXin Li 			printf("C%i ", i);
156*9c5db199SXin Li 	printf("\n");
157*9c5db199SXin Li #endif
158*9c5db199SXin Li }
159