diff options
| author | Linus Torvalds <torvalds@linux-foundation.org> | 2011-10-26 08:39:47 -0400 |
|---|---|---|
| committer | Linus Torvalds <torvalds@linux-foundation.org> | 2011-10-26 08:39:47 -0400 |
| commit | 2355e4290336fcda4b4a799448f745155a000226 (patch) | |
| tree | 81814353573f5a47ac8f96d75221cfee46a0f92c | |
| parent | e33bae14fd8da449d735552d78e6dd33ece0458c (diff) | |
| parent | b1e4d20cbf2ef8e27515da032b95fdcbb5b06bf1 (diff) | |
Merge git://github.com/rustyrussell/linux
* git://github.com/rustyrussell/linux:
params: make dashes and underscores in parameter names truly equal
kmod: prevent kmod_loop_msg overflow in __request_module()
| -rw-r--r-- | include/linux/moduleparam.h | 20 | ||||
| -rw-r--r-- | init/main.c | 4 | ||||
| -rw-r--r-- | kernel/kmod.c | 4 | ||||
| -rw-r--r-- | kernel/params.c | 21 |
4 files changed, 39 insertions, 10 deletions
diff --git a/include/linux/moduleparam.h b/include/linux/moduleparam.h index ddaae98c53f9..fffb10bd5514 100644 --- a/include/linux/moduleparam.h +++ b/include/linux/moduleparam.h | |||
| @@ -262,6 +262,26 @@ static inline void __kernel_param_unlock(void) | |||
| 262 | .str = &__param_string_##name, 0, perm); \ | 262 | .str = &__param_string_##name, 0, perm); \ |
| 263 | __MODULE_PARM_TYPE(name, "string") | 263 | __MODULE_PARM_TYPE(name, "string") |
| 264 | 264 | ||
| 265 | /** | ||
| 266 | * parameq - checks if two parameter names match | ||
| 267 | * @name1: parameter name 1 | ||
| 268 | * @name2: parameter name 2 | ||
| 269 | * | ||
| 270 | * Returns true if the two parameter names are equal. | ||
| 271 | * Dashes (-) are considered equal to underscores (_). | ||
| 272 | */ | ||
| 273 | extern bool parameq(const char *name1, const char *name2); | ||
| 274 | |||
| 275 | /** | ||
| 276 | * parameqn - checks if two parameter names match | ||
| 277 | * @name1: parameter name 1 | ||
| 278 | * @name2: parameter name 2 | ||
| 279 | * @n: the length to compare | ||
| 280 | * | ||
| 281 | * Similar to parameq(), except it compares @n characters. | ||
| 282 | */ | ||
| 283 | extern bool parameqn(const char *name1, const char *name2, size_t n); | ||
| 284 | |||
| 265 | /* Called on module insert or kernel boot */ | 285 | /* Called on module insert or kernel boot */ |
| 266 | extern int parse_args(const char *name, | 286 | extern int parse_args(const char *name, |
| 267 | char *args, | 287 | char *args, |
diff --git a/init/main.c b/init/main.c index 03b408dff825..63f5f6f8dc3b 100644 --- a/init/main.c +++ b/init/main.c | |||
| @@ -163,7 +163,7 @@ static int __init obsolete_checksetup(char *line) | |||
| 163 | p = __setup_start; | 163 | p = __setup_start; |
| 164 | do { | 164 | do { |
| 165 | int n = strlen(p->str); | 165 | int n = strlen(p->str); |
| 166 | if (!strncmp(line, p->str, n)) { | 166 | if (parameqn(line, p->str, n)) { |
| 167 | if (p->early) { | 167 | if (p->early) { |
| 168 | /* Already done in parse_early_param? | 168 | /* Already done in parse_early_param? |
| 169 | * (Needs exact match on param part). | 169 | * (Needs exact match on param part). |
| @@ -392,7 +392,7 @@ static int __init do_early_param(char *param, char *val) | |||
| 392 | const struct obs_kernel_param *p; | 392 | const struct obs_kernel_param *p; |
| 393 | 393 | ||
| 394 | for (p = __setup_start; p < __setup_end; p++) { | 394 | for (p = __setup_start; p < __setup_end; p++) { |
| 395 | if ((p->early && strcmp(param, p->str) == 0) || | 395 | if ((p->early && parameq(param, p->str)) || |
| 396 | (strcmp(param, "console") == 0 && | 396 | (strcmp(param, "console") == 0 && |
| 397 | strcmp(p->str, "earlycon") == 0) | 397 | strcmp(p->str, "earlycon") == 0) |
| 398 | ) { | 398 | ) { |
diff --git a/kernel/kmod.c b/kernel/kmod.c index ddc7644c1305..a4bea97c75b6 100644 --- a/kernel/kmod.c +++ b/kernel/kmod.c | |||
| @@ -114,10 +114,12 @@ int __request_module(bool wait, const char *fmt, ...) | |||
| 114 | atomic_inc(&kmod_concurrent); | 114 | atomic_inc(&kmod_concurrent); |
| 115 | if (atomic_read(&kmod_concurrent) > max_modprobes) { | 115 | if (atomic_read(&kmod_concurrent) > max_modprobes) { |
| 116 | /* We may be blaming an innocent here, but unlikely */ | 116 | /* We may be blaming an innocent here, but unlikely */ |
| 117 | if (kmod_loop_msg++ < 5) | 117 | if (kmod_loop_msg < 5) { |
| 118 | printk(KERN_ERR | 118 | printk(KERN_ERR |
| 119 | "request_module: runaway loop modprobe %s\n", | 119 | "request_module: runaway loop modprobe %s\n", |
| 120 | module_name); | 120 | module_name); |
| 121 | kmod_loop_msg++; | ||
| 122 | } | ||
| 121 | atomic_dec(&kmod_concurrent); | 123 | atomic_dec(&kmod_concurrent); |
| 122 | return -ENOMEM; | 124 | return -ENOMEM; |
| 123 | } | 125 | } |
diff --git a/kernel/params.c b/kernel/params.c index 22df3e0d142a..821788947e40 100644 --- a/kernel/params.c +++ b/kernel/params.c | |||
| @@ -67,20 +67,27 @@ static void maybe_kfree_parameter(void *param) | |||
| 67 | } | 67 | } |
| 68 | } | 68 | } |
| 69 | 69 | ||
| 70 | static inline char dash2underscore(char c) | 70 | static char dash2underscore(char c) |
| 71 | { | 71 | { |
| 72 | if (c == '-') | 72 | if (c == '-') |
| 73 | return '_'; | 73 | return '_'; |
| 74 | return c; | 74 | return c; |
| 75 | } | 75 | } |
| 76 | 76 | ||
| 77 | static inline int parameq(const char *input, const char *paramname) | 77 | bool parameqn(const char *a, const char *b, size_t n) |
| 78 | { | 78 | { |
| 79 | unsigned int i; | 79 | size_t i; |
| 80 | for (i = 0; dash2underscore(input[i]) == paramname[i]; i++) | 80 | |
| 81 | if (input[i] == '\0') | 81 | for (i = 0; i < n; i++) { |
| 82 | return 1; | 82 | if (dash2underscore(a[i]) != dash2underscore(b[i])) |
| 83 | return 0; | 83 | return false; |
| 84 | } | ||
| 85 | return true; | ||
| 86 | } | ||
| 87 | |||
| 88 | bool parameq(const char *a, const char *b) | ||
| 89 | { | ||
| 90 | return parameqn(a, b, strlen(a)+1); | ||
| 84 | } | 91 | } |
| 85 | 92 | ||
| 86 | static int parse_one(char *param, | 93 | static int parse_one(char *param, |
