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