blob: e076bebde1c449f62a41ef345b03002316f7e4a4 [file] [log] [blame]
Dominik Brodowski7fe2f632011-03-30 16:30:11 +02001/*
2 * (C) 2004-2009 Dominik Brodowski <linux@dominikbrodowski.de>
3 * (C) 2010 Thomas Renninger <trenn@suse.de>
4 *
5 * Licensed under the terms of the GNU GPL License version 2.
6 */
7
8
9#include <unistd.h>
10#include <stdio.h>
11#include <errno.h>
12#include <stdlib.h>
13#include <string.h>
14#include <getopt.h>
15#include <cpufreq.h>
16
17#include "helpers/helpers.h"
18#include "helpers/sysfs.h"
19#include "helpers/bitmask.h"
20
21#define LINE_LEN 10
22
23static void cpuidle_cpu_output(unsigned int cpu, int verbose)
24{
25 int idlestates, idlestate;
26 char *tmp;
27
28 printf(_ ("Analyzing CPU %d:\n"), cpu);
29
30 idlestates = sysfs_get_idlestate_count(cpu);
31 if (idlestates == 0) {
32 printf(_("CPU %u: No idle states\n"), cpu);
33 return;
Dominik Brodowskia1ce5ba2011-04-19 20:33:50 +020034 } else if (idlestates <= 0) {
Dominik Brodowski7fe2f632011-03-30 16:30:11 +020035 printf(_("CPU %u: Can't read idle state info\n"), cpu);
36 return;
37 }
38 tmp = sysfs_get_idlestate_name(cpu, idlestates - 1);
39 if (!tmp) {
40 printf(_("Could not determine max idle state %u\n"),
41 idlestates - 1);
42 return;
43 }
44
45 printf(_("Number of idle states: %d\n"), idlestates);
Dominik Brodowski7fe2f632011-03-30 16:30:11 +020046 printf(_("Available idle states:"));
Thomas Renninger0b37ee62011-12-16 15:35:53 +010047 for (idlestate = 0; idlestate < idlestates; idlestate++) {
Dominik Brodowski7fe2f632011-03-30 16:30:11 +020048 tmp = sysfs_get_idlestate_name(cpu, idlestate);
49 if (!tmp)
50 continue;
51 printf(" %s", tmp);
52 free(tmp);
53 }
54 printf("\n");
55
56 if (!verbose)
57 return;
58
Thomas Renninger0b37ee62011-12-16 15:35:53 +010059 for (idlestate = 0; idlestate < idlestates; idlestate++) {
Dominik Brodowski7fe2f632011-03-30 16:30:11 +020060 tmp = sysfs_get_idlestate_name(cpu, idlestate);
61 if (!tmp)
62 continue;
63 printf("%s:\n", tmp);
64 free(tmp);
65
66 tmp = sysfs_get_idlestate_desc(cpu, idlestate);
67 if (!tmp)
68 continue;
69 printf(_("Flags/Description: %s\n"), tmp);
70 free(tmp);
71
72 printf(_("Latency: %lu\n"),
73 sysfs_get_idlestate_latency(cpu, idlestate));
74 printf(_("Usage: %lu\n"),
75 sysfs_get_idlestate_usage(cpu, idlestate));
76 printf(_("Duration: %llu\n"),
77 sysfs_get_idlestate_time(cpu, idlestate));
78 }
79 printf("\n");
80}
81
82static void cpuidle_general_output(void)
83{
84 char *tmp;
85
86 tmp = sysfs_get_cpuidle_driver();
87 if (!tmp) {
88 printf(_("Could not determine cpuidle driver\n"));
89 return;
90 }
91
92 printf(_("CPUidle driver: %s\n"), tmp);
Dominik Brodowskia1ce5ba2011-04-19 20:33:50 +020093 free(tmp);
Dominik Brodowski7fe2f632011-03-30 16:30:11 +020094
95 tmp = sysfs_get_cpuidle_governor();
96 if (!tmp) {
97 printf(_("Could not determine cpuidle governor\n"));
98 return;
99 }
100
101 printf(_("CPUidle governor: %s\n"), tmp);
Dominik Brodowskia1ce5ba2011-04-19 20:33:50 +0200102 free(tmp);
Dominik Brodowski7fe2f632011-03-30 16:30:11 +0200103}
104
105static void proc_cpuidle_cpu_output(unsigned int cpu)
106{
107 long max_allowed_cstate = 2000000000;
108 int cstates, cstate;
109
110 cstates = sysfs_get_idlestate_count(cpu);
111 if (cstates == 0) {
112 /*
113 * Go on and print same useless info as you'd see with
114 * cat /proc/acpi/processor/../power
115 * printf(_("CPU %u: No C-states available\n"), cpu);
116 * return;
117 */
Dominik Brodowskia1ce5ba2011-04-19 20:33:50 +0200118 } else if (cstates <= 0) {
Dominik Brodowski7fe2f632011-03-30 16:30:11 +0200119 printf(_("CPU %u: Can't read C-state info\n"), cpu);
120 return;
121 }
122 /* printf("Cstates: %d\n", cstates); */
123
124 printf(_("active state: C0\n"));
125 printf(_("max_cstate: C%u\n"), cstates-1);
126 printf(_("maximum allowed latency: %lu usec\n"), max_allowed_cstate);
127 printf(_("states:\t\n"));
128 for (cstate = 1; cstate < cstates; cstate++) {
129 printf(_(" C%d: "
130 "type[C%d] "), cstate, cstate);
131 printf(_("promotion[--] demotion[--] "));
132 printf(_("latency[%03lu] "),
133 sysfs_get_idlestate_latency(cpu, cstate));
134 printf(_("usage[%08lu] "),
135 sysfs_get_idlestate_usage(cpu, cstate));
136 printf(_("duration[%020Lu] \n"),
137 sysfs_get_idlestate_time(cpu, cstate));
138 }
139}
140
Dominik Brodowski7fe2f632011-03-30 16:30:11 +0200141static struct option info_opts[] = {
Dominik Brodowskia1ce5ba2011-04-19 20:33:50 +0200142 { .name = "silent", .has_arg = no_argument, .flag = NULL, .val = 's'},
143 { .name = "proc", .has_arg = no_argument, .flag = NULL, .val = 'o'},
Dominik Brodowski7fe2f632011-03-30 16:30:11 +0200144 { },
145};
146
147static inline void cpuidle_exit(int fail)
148{
Dominik Brodowski7fe2f632011-03-30 16:30:11 +0200149 exit(EXIT_FAILURE);
150}
151
152int cmd_idle_info(int argc, char **argv)
153{
154 extern char *optarg;
155 extern int optind, opterr, optopt;
156 int ret = 0, cont = 1, output_param = 0, verbose = 1;
157 unsigned int cpu = 0;
158
159 do {
Dominik Brodowski498ca792011-08-06 18:11:43 +0200160 ret = getopt_long(argc, argv, "os", info_opts, NULL);
Dominik Brodowski7fe2f632011-03-30 16:30:11 +0200161 if (ret == -1)
162 break;
163 switch (ret) {
164 case '?':
165 output_param = '?';
166 cont = 0;
167 break;
Dominik Brodowski7fe2f632011-03-30 16:30:11 +0200168 case 's':
169 verbose = 0;
170 break;
171 case -1:
172 cont = 0;
173 break;
174 case 'o':
175 if (output_param) {
176 output_param = -1;
177 cont = 0;
178 break;
179 }
180 output_param = ret;
181 break;
182 }
Dominik Brodowskia1ce5ba2011-04-19 20:33:50 +0200183 } while (cont);
Dominik Brodowski7fe2f632011-03-30 16:30:11 +0200184
185 switch (output_param) {
186 case -1:
187 printf(_("You can't specify more than one "
188 "output-specific argument\n"));
189 cpuidle_exit(EXIT_FAILURE);
190 case '?':
191 printf(_("invalid or unknown argument\n"));
192 cpuidle_exit(EXIT_FAILURE);
Dominik Brodowski7fe2f632011-03-30 16:30:11 +0200193 }
194
195 /* Default is: show output of CPU 0 only */
196 if (bitmask_isallclear(cpus_chosen))
197 bitmask_setbit(cpus_chosen, 0);
Dominik Brodowskia1ce5ba2011-04-19 20:33:50 +0200198
Dominik Brodowski7fe2f632011-03-30 16:30:11 +0200199 if (output_param == 0)
200 cpuidle_general_output();
Dominik Brodowskia1ce5ba2011-04-19 20:33:50 +0200201
Dominik Brodowski7fe2f632011-03-30 16:30:11 +0200202 for (cpu = bitmask_first(cpus_chosen);
203 cpu <= bitmask_last(cpus_chosen); cpu++) {
204
205 if (!bitmask_isbitset(cpus_chosen, cpu) ||
206 cpufreq_cpu_exists(cpu))
207 continue;
208
209 switch (output_param) {
210
211 case 'o':
212 proc_cpuidle_cpu_output(cpu);
213 break;
214 case 0:
215 printf("\n");
216 cpuidle_cpu_output(cpu, verbose);
217 break;
218 }
219 }
Dominik Brodowskia1ce5ba2011-04-19 20:33:50 +0200220 return EXIT_SUCCESS;
Dominik Brodowski7fe2f632011-03-30 16:30:11 +0200221}