NFSD: further refinement of content of /proc/fs/nfsd/versions
Prior toe35659f1b0
("NFSD: correctly range-check v4.x minor version when setting versions.") v4.0 could not be disabled without disabling all NFSv4 protocols. So the 'versions' file contained ±4 ±4.1 ±4.2. Writing "-4" would disable all v4 completely. Writing +4 would enabled those minor versions that are currently enabled, either by default or otherwise. After that commit, it was possible to disable v4.0 independently. To maximize backward compatibility with use cases which never disabled v4.0, the "versions" file would never contain "+4.0" - that was implied by "+4", unless explicitly negated by "-4.0". This introduced an inconsistency in that it was possible to disable all minor versions, but still have the major version advertised. e.g. "-4.0 -4.1 -4.2 +4" would result in NFSv4 support being advertised, but all attempts to use it rejected. Commitd3635ff07e
("nfsd: fix configuration of supported minor versions") and following removed this inconsistency. If all minor version were disabled, the major would be disabled too. If any minor was enabled, the major would be disabled. This patch also treated "+4" as equivalent to "+4.0" and "-4" as "-4.0". A consequence of this is that writing "-4" would only disable 4.0. This is a regression against the earlier behaviour, in a use case that rpc.nfsd actually uses. The command "rpc.nfsd -N 4" will write "+2 +3 -4" to the versions files. Previously, that would disable v4 completely. Now it will only disable v4.0. Also "4.0" never appears in the "versions" file when read. So if only v4.1 is available, the previous kernel would have reported "+4 -4.0 +4.1 -4.2" the current kernel reports "-4 +4.1 -4.2" which could easily confuse. This patch restores the implication that "+4" and "-4" apply more globals and do not imply "4.0". Specifically: writing "-4" will disable all 4.x minor versions. writing "+4" will enable all 4.1 minor version if none are currently enabled. rpc.nfsd will list minor versions before major versions, so rpc.nfsd -V 4.2 -N 4.1 will write "-4.1 +4.2 +2 +3 +4" so it would be a regression for "+4" to enable always all versions. reading "-4" implies that no v4.x are enabled reading "+4" implies that some v4.x are enabled, and that v4.0 is enabled unless "-4.0" is also present. All other minor versions will explicitly be listed. Signed-off-by: NeilBrown <neilb@suse.com> Signed-off-by: J. Bruce Fields <bfields@redhat.com>
This commit is contained in:
parent
c952cd4e94
commit
abcb4dacb0
@ -538,13 +538,21 @@ out_free:
|
|||||||
|
|
||||||
static ssize_t
|
static ssize_t
|
||||||
nfsd_print_version_support(char *buf, int remaining, const char *sep,
|
nfsd_print_version_support(char *buf, int remaining, const char *sep,
|
||||||
unsigned vers, unsigned minor)
|
unsigned vers, int minor)
|
||||||
{
|
{
|
||||||
const char *format = (minor == 0) ? "%s%c%u" : "%s%c%u.%u";
|
const char *format = minor < 0 ? "%s%c%u" : "%s%c%u.%u";
|
||||||
bool supported = !!nfsd_vers(vers, NFSD_TEST);
|
bool supported = !!nfsd_vers(vers, NFSD_TEST);
|
||||||
|
|
||||||
if (vers == 4 && !nfsd_minorversion(minor, NFSD_TEST))
|
if (vers == 4 && minor >= 0 &&
|
||||||
|
!nfsd_minorversion(minor, NFSD_TEST))
|
||||||
supported = false;
|
supported = false;
|
||||||
|
if (minor == 0 && supported)
|
||||||
|
/*
|
||||||
|
* special case for backward compatability.
|
||||||
|
* +4.0 is never reported, it is implied by
|
||||||
|
* +4, unless -4.0 is present.
|
||||||
|
*/
|
||||||
|
return 0;
|
||||||
return snprintf(buf, remaining, format, sep,
|
return snprintf(buf, remaining, format, sep,
|
||||||
supported ? '+' : '-', vers, minor);
|
supported ? '+' : '-', vers, minor);
|
||||||
}
|
}
|
||||||
@ -554,7 +562,6 @@ static ssize_t __write_versions(struct file *file, char *buf, size_t size)
|
|||||||
char *mesg = buf;
|
char *mesg = buf;
|
||||||
char *vers, *minorp, sign;
|
char *vers, *minorp, sign;
|
||||||
int len, num, remaining;
|
int len, num, remaining;
|
||||||
unsigned minor;
|
|
||||||
ssize_t tlen = 0;
|
ssize_t tlen = 0;
|
||||||
char *sep;
|
char *sep;
|
||||||
struct nfsd_net *nn = net_generic(netns(file), nfsd_net_id);
|
struct nfsd_net *nn = net_generic(netns(file), nfsd_net_id);
|
||||||
@ -575,6 +582,7 @@ static ssize_t __write_versions(struct file *file, char *buf, size_t size)
|
|||||||
if (len <= 0) return -EINVAL;
|
if (len <= 0) return -EINVAL;
|
||||||
do {
|
do {
|
||||||
enum vers_op cmd;
|
enum vers_op cmd;
|
||||||
|
unsigned minor;
|
||||||
sign = *vers;
|
sign = *vers;
|
||||||
if (sign == '+' || sign == '-')
|
if (sign == '+' || sign == '-')
|
||||||
num = simple_strtol((vers+1), &minorp, 0);
|
num = simple_strtol((vers+1), &minorp, 0);
|
||||||
@ -585,8 +593,8 @@ static ssize_t __write_versions(struct file *file, char *buf, size_t size)
|
|||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
if (kstrtouint(minorp+1, 0, &minor) < 0)
|
if (kstrtouint(minorp+1, 0, &minor) < 0)
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
} else
|
}
|
||||||
minor = 0;
|
|
||||||
cmd = sign == '-' ? NFSD_CLEAR : NFSD_SET;
|
cmd = sign == '-' ? NFSD_CLEAR : NFSD_SET;
|
||||||
switch(num) {
|
switch(num) {
|
||||||
case 2:
|
case 2:
|
||||||
@ -594,8 +602,20 @@ static ssize_t __write_versions(struct file *file, char *buf, size_t size)
|
|||||||
nfsd_vers(num, cmd);
|
nfsd_vers(num, cmd);
|
||||||
break;
|
break;
|
||||||
case 4:
|
case 4:
|
||||||
if (nfsd_minorversion(minor, cmd) >= 0)
|
if (*minorp == '.') {
|
||||||
break;
|
if (nfsd_minorversion(minor, cmd) < 0)
|
||||||
|
return -EINVAL;
|
||||||
|
} else if ((cmd == NFSD_SET) != nfsd_vers(num, NFSD_TEST)) {
|
||||||
|
/*
|
||||||
|
* Either we have +4 and no minors are enabled,
|
||||||
|
* or we have -4 and at least one minor is enabled.
|
||||||
|
* In either case, propagate 'cmd' to all minors.
|
||||||
|
*/
|
||||||
|
minor = 0;
|
||||||
|
while (nfsd_minorversion(minor, cmd) >= 0)
|
||||||
|
minor++;
|
||||||
|
}
|
||||||
|
break;
|
||||||
default:
|
default:
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
@ -612,9 +632,11 @@ static ssize_t __write_versions(struct file *file, char *buf, size_t size)
|
|||||||
sep = "";
|
sep = "";
|
||||||
remaining = SIMPLE_TRANSACTION_LIMIT;
|
remaining = SIMPLE_TRANSACTION_LIMIT;
|
||||||
for (num=2 ; num <= 4 ; num++) {
|
for (num=2 ; num <= 4 ; num++) {
|
||||||
|
int minor;
|
||||||
if (!nfsd_vers(num, NFSD_AVAIL))
|
if (!nfsd_vers(num, NFSD_AVAIL))
|
||||||
continue;
|
continue;
|
||||||
minor = 0;
|
|
||||||
|
minor = -1;
|
||||||
do {
|
do {
|
||||||
len = nfsd_print_version_support(buf, remaining,
|
len = nfsd_print_version_support(buf, remaining,
|
||||||
sep, num, minor);
|
sep, num, minor);
|
||||||
@ -624,7 +646,8 @@ static ssize_t __write_versions(struct file *file, char *buf, size_t size)
|
|||||||
buf += len;
|
buf += len;
|
||||||
tlen += len;
|
tlen += len;
|
||||||
minor++;
|
minor++;
|
||||||
sep = " ";
|
if (len)
|
||||||
|
sep = " ";
|
||||||
} while (num == 4 && minor <= NFSD_SUPPORTED_MINOR_VERSION);
|
} while (num == 4 && minor <= NFSD_SUPPORTED_MINOR_VERSION);
|
||||||
}
|
}
|
||||||
out:
|
out:
|
||||||
|
Loading…
Reference in New Issue
Block a user