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