thermal: use %d to print S32 parameters
[linux/fpc-iii.git] / tools / power / cpupower / utils / cpufreq-info.c
blob590d12a25f6e0b8c625011e72d1aa2189e9e1c43
1 /*
2 * (C) 2004-2009 Dominik Brodowski <linux@dominikbrodowski.de>
4 * Licensed under the terms of the GNU GPL License version 2.
5 */
8 #include <unistd.h>
9 #include <stdio.h>
10 #include <errno.h>
11 #include <stdlib.h>
12 #include <string.h>
13 #include <limits.h>
15 #include <getopt.h>
17 #include "cpufreq.h"
18 #include "helpers/sysfs.h"
19 #include "helpers/helpers.h"
20 #include "helpers/bitmask.h"
22 #define LINE_LEN 10
24 static unsigned int count_cpus(void)
26 FILE *fp;
27 char value[LINE_LEN];
28 unsigned int ret = 0;
29 unsigned int cpunr = 0;
31 fp = fopen("/proc/stat", "r");
32 if (!fp) {
33 printf(_("Couldn't count the number of CPUs (%s: %s), assuming 1\n"), "/proc/stat", strerror(errno));
34 return 1;
37 while (!feof(fp)) {
38 if (!fgets(value, LINE_LEN, fp))
39 continue;
40 value[LINE_LEN - 1] = '\0';
41 if (strlen(value) < (LINE_LEN - 2))
42 continue;
43 if (strstr(value, "cpu "))
44 continue;
45 if (sscanf(value, "cpu%d ", &cpunr) != 1)
46 continue;
47 if (cpunr > ret)
48 ret = cpunr;
50 fclose(fp);
52 /* cpu count starts from 0, on error return 1 (UP) */
53 return ret + 1;
57 static void proc_cpufreq_output(void)
59 unsigned int cpu, nr_cpus;
60 struct cpufreq_policy *policy;
61 unsigned int min_pctg = 0;
62 unsigned int max_pctg = 0;
63 unsigned long min, max;
65 printf(_(" minimum CPU frequency - maximum CPU frequency - governor\n"));
67 nr_cpus = count_cpus();
68 for (cpu = 0; cpu < nr_cpus; cpu++) {
69 policy = cpufreq_get_policy(cpu);
70 if (!policy)
71 continue;
73 if (cpufreq_get_hardware_limits(cpu, &min, &max)) {
74 max = 0;
75 } else {
76 min_pctg = (policy->min * 100) / max;
77 max_pctg = (policy->max * 100) / max;
79 printf("CPU%3d %9lu kHz (%3d %%) - %9lu kHz (%3d %%) - %s\n",
80 cpu , policy->min, max ? min_pctg : 0, policy->max,
81 max ? max_pctg : 0, policy->governor);
83 cpufreq_put_policy(policy);
87 static int no_rounding;
88 static void print_speed(unsigned long speed)
90 unsigned long tmp;
92 if (no_rounding) {
93 if (speed > 1000000)
94 printf("%u.%06u GHz", ((unsigned int) speed/1000000),
95 ((unsigned int) speed%1000000));
96 else if (speed > 100000)
97 printf("%u MHz", (unsigned int) speed);
98 else if (speed > 1000)
99 printf("%u.%03u MHz", ((unsigned int) speed/1000),
100 (unsigned int) (speed%1000));
101 else
102 printf("%lu kHz", speed);
103 } else {
104 if (speed > 1000000) {
105 tmp = speed%10000;
106 if (tmp >= 5000)
107 speed += 10000;
108 printf("%u.%02u GHz", ((unsigned int) speed/1000000),
109 ((unsigned int) (speed%1000000)/10000));
110 } else if (speed > 100000) {
111 tmp = speed%1000;
112 if (tmp >= 500)
113 speed += 1000;
114 printf("%u MHz", ((unsigned int) speed/1000));
115 } else if (speed > 1000) {
116 tmp = speed%100;
117 if (tmp >= 50)
118 speed += 100;
119 printf("%u.%01u MHz", ((unsigned int) speed/1000),
120 ((unsigned int) (speed%1000)/100));
124 return;
127 static void print_duration(unsigned long duration)
129 unsigned long tmp;
131 if (no_rounding) {
132 if (duration > 1000000)
133 printf("%u.%06u ms", ((unsigned int) duration/1000000),
134 ((unsigned int) duration%1000000));
135 else if (duration > 100000)
136 printf("%u us", ((unsigned int) duration/1000));
137 else if (duration > 1000)
138 printf("%u.%03u us", ((unsigned int) duration/1000),
139 ((unsigned int) duration%1000));
140 else
141 printf("%lu ns", duration);
142 } else {
143 if (duration > 1000000) {
144 tmp = duration%10000;
145 if (tmp >= 5000)
146 duration += 10000;
147 printf("%u.%02u ms", ((unsigned int) duration/1000000),
148 ((unsigned int) (duration%1000000)/10000));
149 } else if (duration > 100000) {
150 tmp = duration%1000;
151 if (tmp >= 500)
152 duration += 1000;
153 printf("%u us", ((unsigned int) duration / 1000));
154 } else if (duration > 1000) {
155 tmp = duration%100;
156 if (tmp >= 50)
157 duration += 100;
158 printf("%u.%01u us", ((unsigned int) duration/1000),
159 ((unsigned int) (duration%1000)/100));
160 } else
161 printf("%lu ns", duration);
163 return;
166 /* --boost / -b */
168 static int get_boost_mode(unsigned int cpu)
170 int support, active, b_states = 0, ret, pstate_no, i;
171 /* ToDo: Make this more global */
172 unsigned long pstates[MAX_HW_PSTATES] = {0,};
174 if (cpupower_cpu_info.vendor != X86_VENDOR_AMD &&
175 cpupower_cpu_info.vendor != X86_VENDOR_INTEL)
176 return 0;
178 ret = cpufreq_has_boost_support(cpu, &support, &active, &b_states);
179 if (ret) {
180 printf(_("Error while evaluating Boost Capabilities"
181 " on CPU %d -- are you root?\n"), cpu);
182 return ret;
184 /* P state changes via MSR are identified via cpuid 80000007
185 on Intel and AMD, but we assume boost capable machines can do that
186 if (cpuid_eax(0x80000000) >= 0x80000007
187 && (cpuid_edx(0x80000007) & (1 << 7)))
190 printf(_(" boost state support:\n"));
192 printf(_(" Supported: %s\n"), support ? _("yes") : _("no"));
193 printf(_(" Active: %s\n"), active ? _("yes") : _("no"));
195 if (cpupower_cpu_info.vendor == X86_VENDOR_AMD &&
196 cpupower_cpu_info.family >= 0x10) {
197 ret = decode_pstates(cpu, cpupower_cpu_info.family, b_states,
198 pstates, &pstate_no);
199 if (ret)
200 return ret;
202 printf(_(" Boost States: %d\n"), b_states);
203 printf(_(" Total States: %d\n"), pstate_no);
204 for (i = 0; i < pstate_no; i++) {
205 if (i < b_states)
206 printf(_(" Pstate-Pb%d: %luMHz (boost state)"
207 "\n"), i, pstates[i]);
208 else
209 printf(_(" Pstate-P%d: %luMHz\n"),
210 i - b_states, pstates[i]);
212 } else if (cpupower_cpu_info.caps & CPUPOWER_CAP_HAS_TURBO_RATIO) {
213 double bclk;
214 unsigned long long intel_turbo_ratio = 0;
215 unsigned int ratio;
217 /* Any way to autodetect this ? */
218 if (cpupower_cpu_info.caps & CPUPOWER_CAP_IS_SNB)
219 bclk = 100.00;
220 else
221 bclk = 133.33;
222 intel_turbo_ratio = msr_intel_get_turbo_ratio(cpu);
223 dprint (" Ratio: 0x%llx - bclk: %f\n",
224 intel_turbo_ratio, bclk);
226 ratio = (intel_turbo_ratio >> 24) & 0xFF;
227 if (ratio)
228 printf(_(" %.0f MHz max turbo 4 active cores\n"),
229 ratio * bclk);
231 ratio = (intel_turbo_ratio >> 16) & 0xFF;
232 if (ratio)
233 printf(_(" %.0f MHz max turbo 3 active cores\n"),
234 ratio * bclk);
236 ratio = (intel_turbo_ratio >> 8) & 0xFF;
237 if (ratio)
238 printf(_(" %.0f MHz max turbo 2 active cores\n"),
239 ratio * bclk);
241 ratio = (intel_turbo_ratio >> 0) & 0xFF;
242 if (ratio)
243 printf(_(" %.0f MHz max turbo 1 active cores\n"),
244 ratio * bclk);
246 return 0;
249 /* --freq / -f */
251 static int get_freq_kernel(unsigned int cpu, unsigned int human)
253 unsigned long freq = cpufreq_get_freq_kernel(cpu);
254 printf(_(" current CPU frequency: "));
255 if (!freq) {
256 printf(_(" Unable to call to kernel\n"));
257 return -EINVAL;
259 if (human) {
260 print_speed(freq);
261 } else
262 printf("%lu", freq);
263 printf(_(" (asserted by call to kernel)\n"));
264 return 0;
268 /* --hwfreq / -w */
270 static int get_freq_hardware(unsigned int cpu, unsigned int human)
272 unsigned long freq = cpufreq_get_freq_hardware(cpu);
273 printf(_(" current CPU frequency: "));
274 if (!freq) {
275 printf("Unable to call hardware\n");
276 return -EINVAL;
278 if (human) {
279 print_speed(freq);
280 } else
281 printf("%lu", freq);
282 printf(_(" (asserted by call to hardware)\n"));
283 return 0;
286 /* --hwlimits / -l */
288 static int get_hardware_limits(unsigned int cpu)
290 unsigned long min, max;
292 printf(_(" hardware limits: "));
293 if (cpufreq_get_hardware_limits(cpu, &min, &max)) {
294 printf(_("Not Available\n"));
295 return -EINVAL;
298 print_speed(min);
299 printf(" - ");
300 print_speed(max);
301 printf("\n");
302 return 0;
305 /* --driver / -d */
307 static int get_driver(unsigned int cpu)
309 char *driver = cpufreq_get_driver(cpu);
310 if (!driver) {
311 printf(_(" no or unknown cpufreq driver is active on this CPU\n"));
312 return -EINVAL;
314 printf(" driver: %s\n", driver);
315 cpufreq_put_driver(driver);
316 return 0;
319 /* --policy / -p */
321 static int get_policy(unsigned int cpu)
323 struct cpufreq_policy *policy = cpufreq_get_policy(cpu);
324 if (!policy) {
325 printf(_(" Unable to determine current policy\n"));
326 return -EINVAL;
328 printf(_(" current policy: frequency should be within "));
329 print_speed(policy->min);
330 printf(_(" and "));
331 print_speed(policy->max);
333 printf(".\n ");
334 printf(_("The governor \"%s\" may decide which speed to use\n"
335 " within this range.\n"),
336 policy->governor);
337 cpufreq_put_policy(policy);
338 return 0;
341 /* --governors / -g */
343 static int get_available_governors(unsigned int cpu)
345 struct cpufreq_available_governors *governors =
346 cpufreq_get_available_governors(cpu);
348 printf(_(" available cpufreq governors: "));
349 if (!governors) {
350 printf(_("Not Available\n"));
351 return -EINVAL;
354 while (governors->next) {
355 printf("%s ", governors->governor);
356 governors = governors->next;
358 printf("%s\n", governors->governor);
359 cpufreq_put_available_governors(governors);
360 return 0;
364 /* --affected-cpus / -a */
366 static int get_affected_cpus(unsigned int cpu)
368 struct cpufreq_affected_cpus *cpus = cpufreq_get_affected_cpus(cpu);
370 printf(_(" CPUs which need to have their frequency coordinated by software: "));
371 if (!cpus) {
372 printf(_("Not Available\n"));
373 return -EINVAL;
376 while (cpus->next) {
377 printf("%d ", cpus->cpu);
378 cpus = cpus->next;
380 printf("%d\n", cpus->cpu);
381 cpufreq_put_affected_cpus(cpus);
382 return 0;
385 /* --related-cpus / -r */
387 static int get_related_cpus(unsigned int cpu)
389 struct cpufreq_affected_cpus *cpus = cpufreq_get_related_cpus(cpu);
391 printf(_(" CPUs which run at the same hardware frequency: "));
392 if (!cpus) {
393 printf(_("Not Available\n"));
394 return -EINVAL;
397 while (cpus->next) {
398 printf("%d ", cpus->cpu);
399 cpus = cpus->next;
401 printf("%d\n", cpus->cpu);
402 cpufreq_put_related_cpus(cpus);
403 return 0;
406 /* --stats / -s */
408 static int get_freq_stats(unsigned int cpu, unsigned int human)
410 unsigned long total_trans = cpufreq_get_transitions(cpu);
411 unsigned long long total_time;
412 struct cpufreq_stats *stats = cpufreq_get_stats(cpu, &total_time);
413 while (stats) {
414 if (human) {
415 print_speed(stats->frequency);
416 printf(":%.2f%%",
417 (100.0 * stats->time_in_state) / total_time);
418 } else
419 printf("%lu:%llu",
420 stats->frequency, stats->time_in_state);
421 stats = stats->next;
422 if (stats)
423 printf(", ");
425 cpufreq_put_stats(stats);
426 if (total_trans)
427 printf(" (%lu)\n", total_trans);
428 return 0;
431 /* --latency / -y */
433 static int get_latency(unsigned int cpu, unsigned int human)
435 unsigned long latency = cpufreq_get_transition_latency(cpu);
437 printf(_(" maximum transition latency: "));
438 if (!latency || latency == UINT_MAX) {
439 printf(_(" Cannot determine or is not supported.\n"));
440 return -EINVAL;
443 if (human) {
444 print_duration(latency);
445 printf("\n");
446 } else
447 printf("%lu\n", latency);
448 return 0;
451 static void debug_output_one(unsigned int cpu)
453 struct cpufreq_available_frequencies *freqs;
455 get_driver(cpu);
456 get_related_cpus(cpu);
457 get_affected_cpus(cpu);
458 get_latency(cpu, 1);
459 get_hardware_limits(cpu);
461 freqs = cpufreq_get_available_frequencies(cpu);
462 if (freqs) {
463 printf(_(" available frequency steps: "));
464 while (freqs->next) {
465 print_speed(freqs->frequency);
466 printf(", ");
467 freqs = freqs->next;
469 print_speed(freqs->frequency);
470 printf("\n");
471 cpufreq_put_available_frequencies(freqs);
474 get_available_governors(cpu);
475 get_policy(cpu);
476 if (get_freq_hardware(cpu, 1) < 0)
477 get_freq_kernel(cpu, 1);
478 get_boost_mode(cpu);
481 static struct option info_opts[] = {
482 {"debug", no_argument, NULL, 'e'},
483 {"boost", no_argument, NULL, 'b'},
484 {"freq", no_argument, NULL, 'f'},
485 {"hwfreq", no_argument, NULL, 'w'},
486 {"hwlimits", no_argument, NULL, 'l'},
487 {"driver", no_argument, NULL, 'd'},
488 {"policy", no_argument, NULL, 'p'},
489 {"governors", no_argument, NULL, 'g'},
490 {"related-cpus", no_argument, NULL, 'r'},
491 {"affected-cpus", no_argument, NULL, 'a'},
492 {"stats", no_argument, NULL, 's'},
493 {"latency", no_argument, NULL, 'y'},
494 {"proc", no_argument, NULL, 'o'},
495 {"human", no_argument, NULL, 'm'},
496 {"no-rounding", no_argument, NULL, 'n'},
497 { },
500 int cmd_freq_info(int argc, char **argv)
502 extern char *optarg;
503 extern int optind, opterr, optopt;
504 int ret = 0, cont = 1;
505 unsigned int cpu = 0;
506 unsigned int human = 0;
507 int output_param = 0;
509 do {
510 ret = getopt_long(argc, argv, "oefwldpgrasmybn", info_opts,
511 NULL);
512 switch (ret) {
513 case '?':
514 output_param = '?';
515 cont = 0;
516 break;
517 case -1:
518 cont = 0;
519 break;
520 case 'b':
521 case 'o':
522 case 'a':
523 case 'r':
524 case 'g':
525 case 'p':
526 case 'd':
527 case 'l':
528 case 'w':
529 case 'f':
530 case 'e':
531 case 's':
532 case 'y':
533 if (output_param) {
534 output_param = -1;
535 cont = 0;
536 break;
538 output_param = ret;
539 break;
540 case 'm':
541 if (human) {
542 output_param = -1;
543 cont = 0;
544 break;
546 human = 1;
547 break;
548 case 'n':
549 no_rounding = 1;
550 break;
551 default:
552 fprintf(stderr, "invalid or unknown argument\n");
553 return EXIT_FAILURE;
555 } while (cont);
557 switch (output_param) {
558 case 'o':
559 if (!bitmask_isallclear(cpus_chosen)) {
560 printf(_("The argument passed to this tool can't be "
561 "combined with passing a --cpu argument\n"));
562 return -EINVAL;
564 break;
565 case 0:
566 output_param = 'e';
569 ret = 0;
571 /* Default is: show output of CPU 0 only */
572 if (bitmask_isallclear(cpus_chosen))
573 bitmask_setbit(cpus_chosen, 0);
575 switch (output_param) {
576 case -1:
577 printf(_("You can't specify more than one --cpu parameter and/or\n"
578 "more than one output-specific argument\n"));
579 return -EINVAL;
580 case '?':
581 printf(_("invalid or unknown argument\n"));
582 return -EINVAL;
583 case 'o':
584 proc_cpufreq_output();
585 return EXIT_SUCCESS;
588 for (cpu = bitmask_first(cpus_chosen);
589 cpu <= bitmask_last(cpus_chosen); cpu++) {
591 if (!bitmask_isbitset(cpus_chosen, cpu))
592 continue;
594 printf(_("analyzing CPU %d:\n"), cpu);
596 if (sysfs_is_cpu_online(cpu) != 1) {
597 printf(_(" *is offline\n"));
598 printf("\n");
599 continue;
602 switch (output_param) {
603 case 'b':
604 get_boost_mode(cpu);
605 break;
606 case 'e':
607 debug_output_one(cpu);
608 break;
609 case 'a':
610 ret = get_affected_cpus(cpu);
611 break;
612 case 'r':
613 ret = get_related_cpus(cpu);
614 break;
615 case 'g':
616 ret = get_available_governors(cpu);
617 break;
618 case 'p':
619 ret = get_policy(cpu);
620 break;
621 case 'd':
622 ret = get_driver(cpu);
623 break;
624 case 'l':
625 ret = get_hardware_limits(cpu);
626 break;
627 case 'w':
628 ret = get_freq_hardware(cpu, human);
629 break;
630 case 'f':
631 ret = get_freq_kernel(cpu, human);
632 break;
633 case 's':
634 ret = get_freq_stats(cpu, human);
635 break;
636 case 'y':
637 ret = get_latency(cpu, human);
638 break;
640 if (ret)
641 return ret;
642 printf("\n");
644 return ret;