/* * (C) 2011 Thomas Renninger , Novell Inc. * * Licensed under the terms of the GNU GPL License version 2. */ #include #include #include #include #include #include #include #include "helpers/helpers.h" #include "helpers/sysfs.h" #include "helpers/bitmask.h" void set_help(void) { printf(_("Usage: cpupower set [ -b val ] [ -m val ] [ -s val ]\n")); printf(_("Options:\n")); printf(_(" -b, --perf-bias [VAL] Sets CPU's power vs performance policy on some\n" " Intel models [0-15], see manpage for details\n")); printf(_(" -m, --sched-mc [VAL] Sets the kernel's multi core scheduler policy.\n")); printf(_(" -s, --sched-smt [VAL] Sets the kernel's thread sibling scheduler policy.\n")); printf(_(" -h, --help Prints out this screen\n")); printf("\n"); } static struct option set_opts[] = { { .name="perf-bias", .has_arg=optional_argument, .flag=NULL, .val='b'}, { .name="sched-mc", .has_arg=optional_argument, .flag=NULL, .val='m'}, { .name="sched-smt", .has_arg=optional_argument, .flag=NULL, .val='s'}, { .name="help", .has_arg=no_argument, .flag=NULL, .val='h'}, { }, }; static void print_wrong_arg_exit(void) { printf(_("invalid or unknown argument\n")); set_help(); exit(EXIT_FAILURE); } int cmd_set(int argc, char **argv) { extern char *optarg; extern int optind, opterr, optopt; unsigned int cpu; union { struct { int sched_mc:1; int sched_smt:1; int perf_bias:1; }; int params; } params; int sched_mc = 0, sched_smt = 0, perf_bias = 0; int ret = 0; setlocale(LC_ALL, ""); textdomain (PACKAGE); params.params = 0; /* parameter parsing */ while ((ret = getopt_long(argc, argv, "m:s:b:h", set_opts, NULL)) != -1) { switch (ret) { case 'h': set_help(); return 0; case 'b': if (params.perf_bias) print_wrong_arg_exit(); perf_bias = atoi(optarg); if (perf_bias < 0 || perf_bias > 15) { printf(_("--perf-bias param out " "of range [0-%d]\n"), 15); print_wrong_arg_exit(); } params.perf_bias = 1; break; case 'm': if (params.sched_mc) print_wrong_arg_exit(); sched_mc = atoi(optarg); if (sched_mc < 0 || sched_mc > 2) { printf(_("--sched-mc param out " "of range [0-%d]\n"), 2); print_wrong_arg_exit(); } params.sched_mc = 1; break; case 's': if (params.sched_smt) print_wrong_arg_exit(); sched_smt = atoi(optarg); if (sched_smt < 0 || sched_smt > 2) { printf(_("--sched-smt param out " "of range [0-%d]\n"), 2); print_wrong_arg_exit(); } params.sched_smt = 1; break; default: print_wrong_arg_exit(); } }; if (!params.params) { set_help(); return -EINVAL; } if (params.sched_mc) { ret = sysfs_set_sched("mc", sched_mc); if (ret) fprintf(stderr, _("Error setting sched-mc %s\n"), (ret == -ENODEV) ? "not supported" : ""); } if (params.sched_smt) { ret = sysfs_set_sched("smt", sched_smt); if (ret) fprintf(stderr, _("Error setting sched-smt %s\n"), (ret == -ENODEV) ? "not supported" : ""); } /* Default is: set all CPUs */ if (bitmask_isallclear(cpus_chosen)) bitmask_setall(cpus_chosen); /* loop over CPUs */ for (cpu = bitmask_first(cpus_chosen); cpu <= bitmask_last(cpus_chosen); cpu++) { if (!bitmask_isbitset(cpus_chosen, cpu) || cpufreq_cpu_exists(cpu)) continue; if (params.perf_bias) { ret = msr_intel_set_perf_bias(cpu, perf_bias); if (ret) { fprintf(stderr, _("Error setting perf-bias " "value on CPU %d\n"), cpu); break; } } } return ret; }