diff options
Diffstat (limited to 'fs/nfs/nfs4namespace.c')
| -rw-r--r-- | fs/nfs/nfs4namespace.c | 118 | 
1 files changed, 114 insertions, 4 deletions
| diff --git a/fs/nfs/nfs4namespace.c b/fs/nfs/nfs4namespace.c index 2288cd3c9278..c08cbf40c59e 100644 --- a/fs/nfs/nfs4namespace.c +++ b/fs/nfs/nfs4namespace.c @@ -137,6 +137,7 @@ static size_t nfs_parse_server_name(char *string, size_t len,  /**   * nfs_find_best_sec - Find a security mechanism supported locally + * @server: NFS server struct   * @flavors: List of security tuples returned by SECINFO procedure   *   * Return the pseudoflavor of the first security mechanism in @@ -145,7 +146,8 @@ static size_t nfs_parse_server_name(char *string, size_t len,   * is searched in the order returned from the server, per RFC 3530   * recommendation.   */ -rpc_authflavor_t nfs_find_best_sec(struct nfs4_secinfo_flavors *flavors) +static rpc_authflavor_t nfs_find_best_sec(struct nfs_server *server, +					  struct nfs4_secinfo_flavors *flavors)  {  	rpc_authflavor_t pseudoflavor;  	struct nfs4_secinfo4 *secinfo; @@ -160,12 +162,19 @@ rpc_authflavor_t nfs_find_best_sec(struct nfs4_secinfo_flavors *flavors)  		case RPC_AUTH_GSS:  			pseudoflavor = rpcauth_get_pseudoflavor(secinfo->flavor,  							&secinfo->flavor_info); -			if (pseudoflavor != RPC_AUTH_MAXFLAVOR) +			/* make sure pseudoflavor matches sec= mount opt */ +			if (pseudoflavor != RPC_AUTH_MAXFLAVOR && +			    nfs_auth_info_match(&server->auth_info, +						pseudoflavor))  				return pseudoflavor;  			break;  		}  	} +	/* if there were any sec= options then nothing matched */ +	if (server->auth_info.flavor_len > 0) +		return -EPERM; +  	return RPC_AUTH_UNIX;  } @@ -187,7 +196,7 @@ static rpc_authflavor_t nfs4_negotiate_security(struct inode *inode, struct qstr  		goto out;  	} -	flavor = nfs_find_best_sec(flavors); +	flavor = nfs_find_best_sec(NFS_SERVER(inode), flavors);  out:  	put_page(page); @@ -390,7 +399,7 @@ struct vfsmount *nfs4_submount(struct nfs_server *server, struct dentry *dentry,  	if (client->cl_auth->au_flavor != flavor)  		flavor = client->cl_auth->au_flavor; -	else if (!(server->flags & NFS_MOUNT_SECFLAVOUR)) { +	else {  		rpc_authflavor_t new = nfs4_negotiate_security(dir, name);  		if ((int)new >= 0)  			flavor = new; @@ -400,3 +409,104 @@ out:  	rpc_shutdown_client(client);  	return mnt;  } + +/* + * Try one location from the fs_locations array. + * + * Returns zero on success, or a negative errno value. + */ +static int nfs4_try_replacing_one_location(struct nfs_server *server, +		char *page, char *page2, +		const struct nfs4_fs_location *location) +{ +	const size_t addr_bufsize = sizeof(struct sockaddr_storage); +	struct sockaddr *sap; +	unsigned int s; +	size_t salen; +	int error; + +	sap = kmalloc(addr_bufsize, GFP_KERNEL); +	if (sap == NULL) +		return -ENOMEM; + +	error = -ENOENT; +	for (s = 0; s < location->nservers; s++) { +		const struct nfs4_string *buf = &location->servers[s]; +		char *hostname; + +		if (buf->len <= 0 || buf->len > PAGE_SIZE) +			continue; + +		if (memchr(buf->data, IPV6_SCOPE_DELIMITER, buf->len) != NULL) +			continue; + +		salen = nfs_parse_server_name(buf->data, buf->len, +						sap, addr_bufsize, server); +		if (salen == 0) +			continue; +		rpc_set_port(sap, NFS_PORT); + +		error = -ENOMEM; +		hostname = kstrndup(buf->data, buf->len, GFP_KERNEL); +		if (hostname == NULL) +			break; + +		error = nfs4_update_server(server, hostname, sap, salen); +		kfree(hostname); +		if (error == 0) +			break; +	} + +	kfree(sap); +	return error; +} + +/** + * nfs4_replace_transport - set up transport to destination server + * + * @server: export being migrated + * @locations: fs_locations array + * + * Returns zero on success, or a negative errno value. + * + * The client tries all the entries in the "locations" array, in the + * order returned by the server, until one works or the end of the + * array is reached. + */ +int nfs4_replace_transport(struct nfs_server *server, +			   const struct nfs4_fs_locations *locations) +{ +	char *page = NULL, *page2 = NULL; +	int loc, error; + +	error = -ENOENT; +	if (locations == NULL || locations->nlocations <= 0) +		goto out; + +	error = -ENOMEM; +	page = (char *) __get_free_page(GFP_USER); +	if (!page) +		goto out; +	page2 = (char *) __get_free_page(GFP_USER); +	if (!page2) +		goto out; + +	for (loc = 0; loc < locations->nlocations; loc++) { +		const struct nfs4_fs_location *location = +						&locations->locations[loc]; + +		if (location == NULL || location->nservers <= 0 || +		    location->rootpath.ncomponents == 0) +			continue; + +		error = nfs4_try_replacing_one_location(server, page, +							page2, location); +		if (error == 0) +			break; +	} + +out: +	free_page((unsigned long)page); +	free_page((unsigned long)page2); +	return error; +} | 
