cregit-Linux how code gets into the kernel

Release 4.11 tools/power/cpupower/utils/cpufreq-set.c

/*
 *  (C) 2004-2009  Dominik Brodowski <linux@dominikbrodowski.de>
 *
 *  Licensed under the terms of the GNU GPL License version 2.
 */


#include <unistd.h>
#include <stdio.h>
#include <errno.h>
#include <stdlib.h>
#include <limits.h>
#include <string.h>
#include <ctype.h>

#include <getopt.h>

#include "cpufreq.h"
#include "cpuidle.h"
#include "helpers/helpers.h"


#define NORM_FREQ_LEN 32


static struct option set_opts[] = {
	{"min",		required_argument,	NULL, 'd'},
	{"max",		required_argument,	NULL, 'u'},
	{"governor",	required_argument,	NULL, 'g'},
	{"freq",	required_argument,	NULL, 'f'},
	{"related",	no_argument,		NULL, 'r'},
	{ },
};


static void print_error(void) { printf(_("Error setting new values. Common errors:\n" "- Do you have proper administration rights? (super-user?)\n" "- Is the governor you requested available and modprobed?\n" "- Trying to set an invalid policy?\n" "- Trying to set a specific frequency, but userspace governor is not available,\n" " for example because of hardware which cannot be set to a specific frequency\n" " or because the userspace governor isn't loaded?\n")); }

Contributors

PersonTokensPropCommitsCommitProp
Dominik Brodowski22100.00%1100.00%
Total22100.00%1100.00%

; struct freq_units { char *str_unit; int power_of_ten; }; const struct freq_units def_units[] = { {"hz", -3}, {"khz", 0}, /* default */ {"mhz", 3}, {"ghz", 6}, {"thz", 9}, {NULL, 0} };
static void print_unknown_arg(void) { printf(_("invalid or unknown argument\n")); }

Contributors

PersonTokensPropCommitsCommitProp
Dominik Brodowski16100.00%1100.00%
Total16100.00%1100.00%


static unsigned long string_to_frequency(const char *str) { char normalized[NORM_FREQ_LEN]; const struct freq_units *unit; const char *scan; char *end; unsigned long freq; int power = 0, match_count = 0, i, cp, pad; while (*str == '0') str++; for (scan = str; isdigit(*scan) || *scan == '.'; scan++) { if (*scan == '.' && match_count == 0) match_count = 1; else if (*scan == '.' && match_count == 1) return 0; } if (*scan) { match_count = 0; for (unit = def_units; unit->str_unit; unit++) { for (i = 0; scan[i] && tolower(scan[i]) == unit->str_unit[i]; ++i) continue; if (scan[i]) continue; match_count++; power = unit->power_of_ten; } if (match_count != 1) return 0; } /* count the number of digits to be copied */ for (cp = 0; isdigit(str[cp]); cp++) continue; if (str[cp] == '.') { while (power > -1 && isdigit(str[cp+1])) cp++, power--; } if (power >= -1) /* not enough => pad */ pad = power + 1; else /* to much => strip */ pad = 0, cp += power + 1; /* check bounds */ if (cp <= 0 || cp + pad > NORM_FREQ_LEN - 1) return 0; /* copy digits */ for (i = 0; i < cp; i++, str++) { if (*str == '.') str++; normalized[i] = *str; } /* and pad */ for (; i < cp + pad; i++) normalized[i] = '0'; /* round up, down ? */ match_count = (normalized[i-1] >= '5'); /* and drop the decimal part */ normalized[i-1] = 0; /* cp > 0 && pad >= 0 ==> i > 0 */ /* final conversion (and applying rounding) */ errno = 0; freq = strtoul(normalized, &end, 10); if (errno) return 0; else { if (match_count && freq != ULONG_MAX) freq++; return freq; } }

Contributors

PersonTokensPropCommitsCommitProp
Dominik Brodowski416100.00%1100.00%
Total416100.00%1100.00%


static int do_new_policy(unsigned int cpu, struct cpufreq_policy *new_pol) { struct cpufreq_policy *cur_pol = cpufreq_get_policy(cpu); int ret; if (!cur_pol) { printf(_("wrong, unknown or unhandled CPU?\n")); return -EINVAL; } if (!new_pol->min) new_pol->min = cur_pol->min; if (!new_pol->max) new_pol->max = cur_pol->max; if (!new_pol->governor) new_pol->governor = cur_pol->governor; ret = cpufreq_set_policy(cpu, new_pol); cpufreq_put_policy(cur_pol); return ret; }

Contributors

PersonTokensPropCommitsCommitProp
Dominik Brodowski109100.00%1100.00%
Total109100.00%1100.00%


static int do_one_cpu(unsigned int cpu, struct cpufreq_policy *new_pol, unsigned long freq, unsigned int pc) { switch (pc) { case 0: return cpufreq_set_frequency(cpu, freq); case 1: /* if only one value of a policy is to be changed, we can * use a "fast path". */ if (new_pol->min) return cpufreq_modify_policy_min(cpu, new_pol->min); else if (new_pol->max) return cpufreq_modify_policy_max(cpu, new_pol->max); else if (new_pol->governor) return cpufreq_modify_policy_governor(cpu, new_pol->governor); default: /* slow path */ return do_new_policy(cpu, new_pol); } }

Contributors

PersonTokensPropCommitsCommitProp
Dominik Brodowski104100.00%1100.00%
Total104100.00%1100.00%


int cmd_freq_set(int argc, char **argv) { extern char *optarg; extern int optind, opterr, optopt; int ret = 0, cont = 1; int double_parm = 0, related = 0, policychange = 0; unsigned long freq = 0; char gov[20]; unsigned int cpu; struct cpufreq_policy new_pol = { .min = 0, .max = 0, .governor = NULL, }; /* parameter parsing */ do { ret = getopt_long(argc, argv, "d:u:g:f:r", set_opts, NULL); switch (ret) { case '?': print_unknown_arg(); return -EINVAL; case -1: cont = 0; break; case 'r': if (related) double_parm++; related++; break; case 'd': if (new_pol.min) double_parm++; policychange++; new_pol.min = string_to_frequency(optarg); if (new_pol.min == 0) { print_unknown_arg(); return -EINVAL; } break; case 'u': if (new_pol.max) double_parm++; policychange++; new_pol.max = string_to_frequency(optarg); if (new_pol.max == 0) { print_unknown_arg(); return -EINVAL; } break; case 'f': if (freq) double_parm++; freq = string_to_frequency(optarg); if (freq == 0) { print_unknown_arg(); return -EINVAL; } break; case 'g': if (new_pol.governor) double_parm++; policychange++; if ((strlen(optarg) < 3) || (strlen(optarg) > 18)) { print_unknown_arg(); return -EINVAL; } if ((sscanf(optarg, "%19s", gov)) != 1) { print_unknown_arg(); return -EINVAL; } new_pol.governor = gov; break; } } while (cont); /* parameter checking */ if (double_parm) { printf("the same parameter was passed more than once\n"); return -EINVAL; } if (freq && policychange) { printf(_("the -f/--freq parameter cannot be combined with -d/--min, -u/--max or\n" "-g/--governor parameters\n")); return -EINVAL; } if (!freq && !policychange) { printf(_("At least one parameter out of -f/--freq, -d/--min, -u/--max, and\n" "-g/--governor must be passed\n")); return -EINVAL; } /* Default is: set all CPUs */ if (bitmask_isallclear(cpus_chosen)) bitmask_setall(cpus_chosen); /* Also set frequency settings for related CPUs if -r is passed */ if (related) { for (cpu = bitmask_first(cpus_chosen); cpu <= bitmask_last(cpus_chosen); cpu++) { struct cpufreq_affected_cpus *cpus; if (!bitmask_isbitset(cpus_chosen, cpu) || cpupower_is_cpu_online(cpu) != 1) continue; cpus = cpufreq_get_related_cpus(cpu); if (!cpus) break; while (cpus->next) { bitmask_setbit(cpus_chosen, cpus->cpu); cpus = cpus->next; } cpufreq_put_related_cpus(cpus); } } /* loop over CPUs */ for (cpu = bitmask_first(cpus_chosen); cpu <= bitmask_last(cpus_chosen); cpu++) { if (!bitmask_isbitset(cpus_chosen, cpu) || cpupower_is_cpu_online(cpu) != 1) continue; printf(_("Setting cpu: %d\n"), cpu); ret = do_one_cpu(cpu, &new_pol, freq, policychange); if (ret) { print_error(); return ret; } } return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Dominik Brodowski56997.43%228.57%
Peter Senna Tschudin61.03%114.29%
Shilpasri G Bhat40.68%114.29%
Laura Abbott20.34%114.29%
Thomas Renninger20.34%114.29%
Alan One Thousand Gnomes10.17%114.29%
Total584100.00%7100.00%


Overall Contributors

PersonTokensPropCommitsCommitProp
Dominik Brodowski138998.65%228.57%
Peter Senna Tschudin60.43%114.29%
Shilpasri G Bhat60.43%114.29%
Thomas Renninger40.28%114.29%
Laura Abbott20.14%114.29%
Alan One Thousand Gnomes10.07%114.29%
Total1408100.00%7100.00%
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.
Created with cregit.