linux/tools/power/cpupower/utils/cpupower.c
<<
>>
Prefs
   1/*
   2 *  (C) 2010,2011       Thomas Renninger <trenn@suse.de>, Novell Inc.
   3 *
   4 *  Licensed under the terms of the GNU GPL License version 2.
   5 *
   6 *  Ideas taken over from the perf userspace tool (included in the Linus
   7 *  kernel git repo): subcommand builtins and param parsing.
   8 */
   9
  10#include <stdio.h>
  11#include <stdlib.h>
  12#include <string.h>
  13#include <unistd.h>
  14#include <errno.h>
  15#include <sched.h>
  16#include <sys/types.h>
  17#include <sys/stat.h>
  18#include <sys/utsname.h>
  19
  20#include "builtin.h"
  21#include "helpers/helpers.h"
  22#include "helpers/bitmask.h"
  23
  24#define ARRAY_SIZE(x) (sizeof(x)/sizeof(x[0]))
  25
  26static int cmd_help(int argc, const char **argv);
  27
  28/* Global cpu_info object available for all binaries
  29 * Info only retrieved from CPU 0
  30 *
  31 * Values will be zero/unknown on non X86 archs
  32 */
  33struct cpupower_cpu_info cpupower_cpu_info;
  34int run_as_root;
  35int base_cpu;
  36/* Affected cpus chosen by -c/--cpu param */
  37struct bitmask *cpus_chosen;
  38
  39#ifdef DEBUG
  40int be_verbose;
  41#endif
  42
  43static void print_help(void);
  44
  45struct cmd_struct {
  46        const char *cmd;
  47        int (*main)(int, const char **);
  48        int needs_root;
  49};
  50
  51static struct cmd_struct commands[] = {
  52        { "frequency-info",     cmd_freq_info,  0       },
  53        { "frequency-set",      cmd_freq_set,   1       },
  54        { "idle-info",          cmd_idle_info,  0       },
  55        { "idle-set",           cmd_idle_set,   1       },
  56        { "set",                cmd_set,        1       },
  57        { "info",               cmd_info,       0       },
  58        { "monitor",            cmd_monitor,    0       },
  59        { "help",               cmd_help,       0       },
  60        /*      { "bench",      cmd_bench,      1       }, */
  61};
  62
  63static void print_help(void)
  64{
  65        unsigned int i;
  66
  67#ifdef DEBUG
  68        printf(_("Usage:\tcpupower [-d|--debug] [-c|--cpu cpulist ] <command> [<args>]\n"));
  69#else
  70        printf(_("Usage:\tcpupower [-c|--cpu cpulist ] <command> [<args>]\n"));
  71#endif
  72        printf(_("Supported commands are:\n"));
  73        for (i = 0; i < ARRAY_SIZE(commands); i++)
  74                printf("\t%s\n", commands[i].cmd);
  75        printf(_("\nNot all commands can make use of the -c cpulist option.\n"));
  76        printf(_("\nUse 'cpupower help <command>' for getting help for above commands.\n"));
  77}
  78
  79static int print_man_page(const char *subpage)
  80{
  81        int len;
  82        char *page;
  83
  84        len = 10; /* enough for "cpupower-" */
  85        if (subpage != NULL)
  86                len += strlen(subpage);
  87
  88        page = malloc(len);
  89        if (!page)
  90                return -ENOMEM;
  91
  92        sprintf(page, "cpupower");
  93        if ((subpage != NULL) && strcmp(subpage, "help")) {
  94                strcat(page, "-");
  95                strcat(page, subpage);
  96        }
  97
  98        execlp("man", "man", page, NULL);
  99
 100        /* should not be reached */
 101        return -EINVAL;
 102}
 103
 104static int cmd_help(int argc, const char **argv)
 105{
 106        if (argc > 1) {
 107                print_man_page(argv[1]); /* exits within execlp() */
 108                return EXIT_FAILURE;
 109        }
 110
 111        print_help();
 112        return EXIT_SUCCESS;
 113}
 114
 115static void print_version(void)
 116{
 117        printf(PACKAGE " " VERSION "\n");
 118        printf(_("Report errors and bugs to %s, please.\n"), PACKAGE_BUGREPORT);
 119}
 120
 121static void handle_options(int *argc, const char ***argv)
 122{
 123        int ret, x, new_argc = 0;
 124
 125        if (*argc < 1)
 126                return;
 127
 128        for (x = 0;  x < *argc && ((*argv)[x])[0] == '-'; x++) {
 129                const char *param = (*argv)[x];
 130                if (!strcmp(param, "-h") || !strcmp(param, "--help")) {
 131                        print_help();
 132                        exit(EXIT_SUCCESS);
 133                } else if (!strcmp(param, "-c") || !strcmp(param, "--cpu")) {
 134                        if (*argc < 2) {
 135                                print_help();
 136                                exit(EXIT_FAILURE);
 137                        }
 138                        if (!strcmp((*argv)[x+1], "all"))
 139                                bitmask_setall(cpus_chosen);
 140                        else {
 141                                ret = bitmask_parselist(
 142                                                (*argv)[x+1], cpus_chosen);
 143                                if (ret < 0) {
 144                                        fprintf(stderr, _("Error parsing cpu "
 145                                                          "list\n"));
 146                                        exit(EXIT_FAILURE);
 147                                }
 148                        }
 149                        x += 1;
 150                        /* Cut out param: cpupower -c 1 info -> cpupower info */
 151                        new_argc += 2;
 152                        continue;
 153                } else if (!strcmp(param, "-v") ||
 154                        !strcmp(param, "--version")) {
 155                        print_version();
 156                        exit(EXIT_SUCCESS);
 157#ifdef DEBUG
 158                } else if (!strcmp(param, "-d") || !strcmp(param, "--debug")) {
 159                        be_verbose = 1;
 160                        new_argc++;
 161                        continue;
 162#endif
 163                } else {
 164                        fprintf(stderr, "Unknown option: %s\n", param);
 165                        print_help();
 166                        exit(EXIT_FAILURE);
 167                }
 168        }
 169        *argc -= new_argc;
 170        *argv += new_argc;
 171}
 172
 173int main(int argc, const char *argv[])
 174{
 175        const char *cmd;
 176        unsigned int i, ret;
 177        struct stat statbuf;
 178        struct utsname uts;
 179        char pathname[32];
 180
 181        cpus_chosen = bitmask_alloc(sysconf(_SC_NPROCESSORS_CONF));
 182
 183        argc--;
 184        argv += 1;
 185
 186        handle_options(&argc, &argv);
 187
 188        cmd = argv[0];
 189
 190        if (argc < 1) {
 191                print_help();
 192                return EXIT_FAILURE;
 193        }
 194
 195        setlocale(LC_ALL, "");
 196        textdomain(PACKAGE);
 197
 198        /* Turn "perf cmd --help" into "perf help cmd" */
 199        if (argc > 1 && !strcmp(argv[1], "--help")) {
 200                argv[1] = argv[0];
 201                argv[0] = cmd = "help";
 202        }
 203
 204        base_cpu = sched_getcpu();
 205        if (base_cpu < 0) {
 206                fprintf(stderr, _("No valid cpus found.\n"));
 207                return EXIT_FAILURE;
 208        }
 209
 210        get_cpu_info(&cpupower_cpu_info);
 211        run_as_root = !geteuid();
 212        if (run_as_root) {
 213                ret = uname(&uts);
 214                sprintf(pathname, "/dev/cpu/%d/msr", base_cpu);
 215                if (!ret && !strcmp(uts.machine, "x86_64") &&
 216                    stat(pathname, &statbuf) != 0) {
 217                        if (system("modprobe msr") == -1)
 218        fprintf(stderr, _("MSR access not available.\n"));
 219                }
 220        }
 221
 222        for (i = 0; i < ARRAY_SIZE(commands); i++) {
 223                struct cmd_struct *p = commands + i;
 224                if (strcmp(p->cmd, cmd))
 225                        continue;
 226                if (!run_as_root && p->needs_root) {
 227                        fprintf(stderr, _("Subcommand %s needs root "
 228                                          "privileges\n"), cmd);
 229                        return EXIT_FAILURE;
 230                }
 231                ret = p->main(argc, argv);
 232                if (cpus_chosen)
 233                        bitmask_free(cpus_chosen);
 234                return ret;
 235        }
 236        print_help();
 237        return EXIT_FAILURE;
 238}
 239