1*9c5db199SXin Li /*
2*9c5db199SXin Li * Copyright 2007, 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
33*9c5db199SXin Li #include "powertop.h"
34*9c5db199SXin Li
35*9c5db199SXin Li struct cpufreqdata {
36*9c5db199SXin Li uint64_t frequency;
37*9c5db199SXin Li uint64_t count;
38*9c5db199SXin Li };
39*9c5db199SXin Li
40*9c5db199SXin Li struct cpufreqdata freqs[16];
41*9c5db199SXin Li struct cpufreqdata oldfreqs[16];
42*9c5db199SXin Li
43*9c5db199SXin Li struct cpufreqdata delta[16];
44*9c5db199SXin Li
45*9c5db199SXin Li char cpufreqstrings[6][80];
46*9c5db199SXin Li int topfreq = -1;
47*9c5db199SXin Li
zap(void)48*9c5db199SXin Li static void zap(void)
49*9c5db199SXin Li {
50*9c5db199SXin Li memset(freqs, 0, sizeof(freqs));
51*9c5db199SXin Li }
52*9c5db199SXin Li
sort_by_count(const void * av,const void * bv)53*9c5db199SXin Li int sort_by_count (const void *av, const void *bv)
54*9c5db199SXin Li {
55*9c5db199SXin Li const struct cpufreqdata *a = av, *b = bv;
56*9c5db199SXin Li return b->count - a->count;
57*9c5db199SXin Li }
58*9c5db199SXin Li
sort_by_freq(const void * av,const void * bv)59*9c5db199SXin Li int sort_by_freq (const void *av, const void *bv)
60*9c5db199SXin Li {
61*9c5db199SXin Li const struct cpufreqdata *a = av, *b = bv;
62*9c5db199SXin Li return b->frequency - a->frequency;
63*9c5db199SXin Li }
64*9c5db199SXin Li
HzToHuman(unsigned long hz)65*9c5db199SXin Li static char *HzToHuman(unsigned long hz)
66*9c5db199SXin Li {
67*9c5db199SXin Li static char buffer[1024];
68*9c5db199SXin Li memset(buffer, 0, 1024);
69*9c5db199SXin Li unsigned long long Hz;
70*9c5db199SXin Li
71*9c5db199SXin Li Hz = hz;
72*9c5db199SXin Li
73*9c5db199SXin Li /* default: just put the Number in */
74*9c5db199SXin Li sprintf(buffer,_("%9lli"), Hz);
75*9c5db199SXin Li
76*9c5db199SXin Li if (Hz>1000)
77*9c5db199SXin Li sprintf(buffer, _("%6lli Mhz"), (Hz+500)/1000);
78*9c5db199SXin Li
79*9c5db199SXin Li if (Hz>1500000)
80*9c5db199SXin Li sprintf(buffer, _("%6.2f Ghz"), (Hz+5000.0)/1000000);
81*9c5db199SXin Li
82*9c5db199SXin Li
83*9c5db199SXin Li return buffer;
84*9c5db199SXin Li }
85*9c5db199SXin Li
86*9c5db199SXin Li
do_cpufreq_stats(void)87*9c5db199SXin Li void do_cpufreq_stats(void)
88*9c5db199SXin Li {
89*9c5db199SXin Li DIR *dir;
90*9c5db199SXin Li struct dirent *dirent;
91*9c5db199SXin Li FILE *file;
92*9c5db199SXin Li char filename[PATH_MAX];
93*9c5db199SXin Li char line[1024];
94*9c5db199SXin Li
95*9c5db199SXin Li int ret = 0;
96*9c5db199SXin Li int maxfreq = 0;
97*9c5db199SXin Li uint64_t total_time = 0;
98*9c5db199SXin Li
99*9c5db199SXin Li memcpy(&oldfreqs, &freqs, sizeof(freqs));
100*9c5db199SXin Li memset(&cpufreqstrings, 0, sizeof(cpufreqstrings));
101*9c5db199SXin Li sprintf(cpufreqstrings[0], _("P-states (frequencies)\n"));
102*9c5db199SXin Li
103*9c5db199SXin Li for (ret = 0; ret<16; ret++)
104*9c5db199SXin Li freqs[ret].count = 0;
105*9c5db199SXin Li
106*9c5db199SXin Li dir = opendir("/sys/devices/system/cpu");
107*9c5db199SXin Li if (!dir)
108*9c5db199SXin Li return;
109*9c5db199SXin Li
110*9c5db199SXin Li while ((dirent = readdir(dir))) {
111*9c5db199SXin Li int i;
112*9c5db199SXin Li if (dirent->d_name[0]=='.')
113*9c5db199SXin Li continue;
114*9c5db199SXin Li sprintf(filename, "/sys/devices/system/cpu/%s/cpufreq/stats/time_in_state", dirent->d_name);
115*9c5db199SXin Li file = fopen(filename, "r");
116*9c5db199SXin Li if (!file)
117*9c5db199SXin Li continue;
118*9c5db199SXin Li memset(line, 0, 1024);
119*9c5db199SXin Li
120*9c5db199SXin Li i = 0;
121*9c5db199SXin Li while (!feof(file)) {
122*9c5db199SXin Li uint64_t f,count;
123*9c5db199SXin Li char *c;
124*9c5db199SXin Li if (fgets(line, 1023,file)==NULL)
125*9c5db199SXin Li break;
126*9c5db199SXin Li f = strtoull(line, &c, 10);
127*9c5db199SXin Li if (!c)
128*9c5db199SXin Li break;
129*9c5db199SXin Li count = strtoull(c, NULL, 10);
130*9c5db199SXin Li
131*9c5db199SXin Li if (freqs[i].frequency && freqs[i].frequency != f) {
132*9c5db199SXin Li zap();
133*9c5db199SXin Li break;
134*9c5db199SXin Li }
135*9c5db199SXin Li
136*9c5db199SXin Li freqs[i].frequency = f;
137*9c5db199SXin Li freqs[i].count += count;
138*9c5db199SXin Li
139*9c5db199SXin Li if (f && maxfreq < i)
140*9c5db199SXin Li maxfreq = i;
141*9c5db199SXin Li i++;
142*9c5db199SXin Li if (i>15)
143*9c5db199SXin Li break;
144*9c5db199SXin Li }
145*9c5db199SXin Li fclose(file);
146*9c5db199SXin Li }
147*9c5db199SXin Li
148*9c5db199SXin Li closedir(dir);
149*9c5db199SXin Li
150*9c5db199SXin Li for (ret = 0; ret < 16; ret++) {
151*9c5db199SXin Li delta[ret].count = freqs[ret].count - oldfreqs[ret].count;
152*9c5db199SXin Li total_time += delta[ret].count;
153*9c5db199SXin Li delta[ret].frequency = freqs[ret].frequency;
154*9c5db199SXin Li if (freqs[ret].frequency != oldfreqs[ret].frequency)
155*9c5db199SXin Li return; /* duff data */
156*9c5db199SXin Li }
157*9c5db199SXin Li
158*9c5db199SXin Li
159*9c5db199SXin Li if (!total_time)
160*9c5db199SXin Li return;
161*9c5db199SXin Li
162*9c5db199SXin Li qsort(&delta, maxfreq+1, sizeof(struct cpufreqdata), sort_by_count);
163*9c5db199SXin Li if (maxfreq>4)
164*9c5db199SXin Li maxfreq=4;
165*9c5db199SXin Li qsort(&delta, maxfreq+1, sizeof(struct cpufreqdata), sort_by_freq);
166*9c5db199SXin Li
167*9c5db199SXin Li topfreq = -1;
168*9c5db199SXin Li for (ret = 0 ; ret<=maxfreq; ret++) {
169*9c5db199SXin Li sprintf(cpufreqstrings[ret+1], "%6s %5.1f%%\n", HzToHuman(delta[ret].frequency), delta[ret].count * 100.0 / total_time);
170*9c5db199SXin Li if (delta[ret].count > total_time/2)
171*9c5db199SXin Li topfreq = ret;
172*9c5db199SXin Li }
173*9c5db199SXin Li
174*9c5db199SXin Li }
175