The patch titled Subject: kernel/sysctl.c: add flags to support min/max range clamping has been removed from the -mm tree. Its filename was sysctl-add-flags-to-support-min-max-range-clamping.patch This patch was dropped because an updated version will be merged ------------------------------------------------------ From: Waiman Long <longman@xxxxxxxxxx> Subject: kernel/sysctl.c: add flags to support min/max range clamping When minimum/maximum values are specified for a sysctl parameter in the ctl_table structure with proc_dointvec_minmax() handler, updates to that parameter will fail with error if the given value is outside of the required range. There are use cases where it may be better to clamp the value of the sysctl parameter to the given range without failing the update, especially if the users are not aware of the actual range limits. Reading the value back after the update will now be good practice to see if the provided value exceeds the range limits. To provide this less restrictive form of range checking a new flags field is added to the ctl_table structure. The new field is a 16-bit value that just fits into the hole left by the 16-bit umode_t field without increasing the size of the structure. When the CTL_FLAGS_CLAMP_RANGE flag is set in the ctl_table entry, any update from userspace will be clamped to the given range without error. [akpm@xxxxxxxxxxxxxxxxxxxx: clarify that ctl_table.flags is of type `enum ctl_table_flags'] Link: http://lkml.kernel.org/r/1519926220-7453-4-git-send-email-longman@xxxxxxxxxx Signed-off-by: Waiman Long <longman@xxxxxxxxxx> Reviewed-by: Andrew Morton <akpm@xxxxxxxxxxxxxxxxxxxx> Cc: Al Viro <viro@xxxxxxxxxxxxxxxxxx> Cc: Davidlohr Bueso <dave@xxxxxxxxxxxx> Cc: Kees Cook <keescook@xxxxxxxxxxxx> Cc: "Luis R. Rodriguez" <mcgrof@xxxxxxxxxx> Cc: Manfred Spraul <manfred@xxxxxxxxxxxxxxxx> Cc: Matthew Wilcox <willy@xxxxxxxxxxxxx> Cc: Randy Dunlap <rdunlap@xxxxxxxxxxxxx> Signed-off-by: Andrew Morton <akpm@xxxxxxxxxxxxxxxxxxxx> --- diff -puN include/linux/sysctl.h~sysctl-add-flags-to-support-min-max-range-clamping include/linux/sysctl.h --- a/include/linux/sysctl.h~sysctl-add-flags-to-support-min-max-range-clamping +++ a/include/linux/sysctl.h @@ -116,6 +116,7 @@ struct ctl_table void *data; int maxlen; umode_t mode; + uint16_t flags; /* enum ctl_table_flags */ struct ctl_table *child; /* Deprecated */ proc_handler *proc_handler; /* Callback for text formatting */ struct ctl_table_poll *poll; @@ -123,6 +124,19 @@ struct ctl_table void *extra2; } __randomize_layout; +/** + * enum ctl_table_flags - flags for the ctl table (struct ctl_table.flags) + * + * @CTL_FLAGS_CLAMP_RANGE: Set to indicate that the entry should be + * flexibly clamped to min/max range in case the user provided + * an incorrect value. + * + * At most 16 different flags will be allowed. + */ +enum ctl_table_flags { + CTL_FLAGS_CLAMP_RANGE = BIT(0), +}; + struct ctl_node { struct rb_node node; struct ctl_table_header *header; diff -puN kernel/sysctl.c~sysctl-add-flags-to-support-min-max-range-clamping kernel/sysctl.c --- a/kernel/sysctl.c~sysctl-add-flags-to-support-min-max-range-clamping +++ a/kernel/sysctl.c @@ -2504,6 +2504,7 @@ static int proc_dointvec_minmax_sysadmin * struct do_proc_dointvec_minmax_conv_param - proc_dointvec_minmax() range checking structure * @min: pointer to minimum allowable value * @max: pointer to maximum allowable value + * @flags: pointer to flags * * The do_proc_dointvec_minmax_conv_param structure provides the * minimum and maximum values for doing range checking for those sysctl @@ -2512,6 +2513,7 @@ static int proc_dointvec_minmax_sysadmin struct do_proc_dointvec_minmax_conv_param { int *min; int *max; + uint16_t *flags; }; static int do_proc_dointvec_minmax_conv(bool *negp, unsigned long *lvalp, @@ -2521,9 +2523,21 @@ static int do_proc_dointvec_minmax_conv( struct do_proc_dointvec_minmax_conv_param *param = data; if (write) { int val = *negp ? -*lvalp : *lvalp; - if ((param->min && *param->min > val) || - (param->max && *param->max < val)) - return -EINVAL; + bool clamp = param->flags && + (*param->flags & CTL_FLAGS_CLAMP_RANGE); + + if (param->min && *param->min > val) { + if (clamp) + val = *param->min; + else + return -EINVAL; + } + if (param->max && *param->max < val) { + if (clamp) + val = *param->max; + else + return -EINVAL; + } *valp = val; } else { int val = *valp; @@ -2552,7 +2566,8 @@ static int do_proc_dointvec_minmax_conv( * This routine will ensure the values are within the range specified by * table->extra1 (min) and table->extra2 (max). * - * Returns 0 on success or -EINVAL on write when the range check fails. + * Returns 0 on success or -EINVAL on write when the range check fails + * without the CTL_FLAGS_CLAMP_RANGE flag. */ int proc_dointvec_minmax(struct ctl_table *table, int write, void __user *buffer, size_t *lenp, loff_t *ppos) @@ -2560,6 +2575,7 @@ int proc_dointvec_minmax(struct ctl_tabl struct do_proc_dointvec_minmax_conv_param param = { .min = (int *) table->extra1, .max = (int *) table->extra2, + .flags = &table->flags, }; return do_proc_dointvec(table, write, buffer, lenp, ppos, do_proc_dointvec_minmax_conv, ¶m); @@ -2569,6 +2585,7 @@ int proc_dointvec_minmax(struct ctl_tabl * struct do_proc_douintvec_minmax_conv_param - proc_douintvec_minmax() range checking structure * @min: pointer to minimum allowable value * @max: pointer to maximum allowable value + * @flags: pointer to flags * * The do_proc_douintvec_minmax_conv_param structure provides the * minimum and maximum values for doing range checking for those sysctl @@ -2577,6 +2594,7 @@ int proc_dointvec_minmax(struct ctl_tabl struct do_proc_douintvec_minmax_conv_param { unsigned int *min; unsigned int *max; + uint16_t *flags; }; static int do_proc_douintvec_minmax_conv(unsigned long *lvalp, @@ -2587,14 +2605,24 @@ static int do_proc_douintvec_minmax_conv if (write) { unsigned int val = *lvalp; + bool clamp = param->flags && + (*param->flags & CTL_FLAGS_CLAMP_RANGE); if (*lvalp > UINT_MAX) return -EINVAL; - if ((param->min && *param->min > val) || - (param->max && *param->max < val)) - return -ERANGE; - + if (param->min && *param->min > val) { + if (clamp) + val = *param->min; + else + return -ERANGE; + } + if (param->max && *param->max < val) { + if (clamp) + val = *param->max; + else + return -ERANGE; + } *valp = val; } else { unsigned int val = *valp; @@ -2621,7 +2649,8 @@ static int do_proc_douintvec_minmax_conv * check for UINT_MAX to avoid having to support wrap around uses from * userspace. * - * Returns 0 on success or -ERANGE on write when the range check fails. + * Returns 0 on success or -ERANGE on write when the range check fails + * without the CTL_FLAGS_CLAMP_RANGE flag. */ int proc_douintvec_minmax(struct ctl_table *table, int write, void __user *buffer, size_t *lenp, loff_t *ppos) @@ -2629,6 +2658,7 @@ int proc_douintvec_minmax(struct ctl_tab struct do_proc_douintvec_minmax_conv_param param = { .min = (unsigned int *) table->extra1, .max = (unsigned int *) table->extra2, + .flags = &table->flags, }; return do_proc_douintvec(table, write, buffer, lenp, ppos, do_proc_douintvec_minmax_conv, ¶m); _ Patches currently in -mm which might be from longman@xxxxxxxxxx are list_lru-prefetch-neighboring-list-entries-before-acquiring-lock.patch proc-sysctl-fix-typo-in-sysctl_check_table_array.patch sysctl-add-kdoc-comments-to-do_proc_douintvec_minmax_conv_param.patch sysctl-warn-when-a-clamped-sysctl-parameter-is-set-out-of-range.patch ipc-clamp-msgmni-and-shmmni-to-the-real-ipcmni-limit.patch ipc-clamp-semmni-to-the-real-ipcmni-limit.patch -- To unsubscribe from this list: send the line "unsubscribe mm-commits" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html