Donate to e Foundation | Murena handsets with /e/OS | Own a part of Murena! Learn more

Commit fd85b817 authored by Marc Eshel's avatar Marc Eshel Committed by J. Bruce Fields
Browse files

nfsd4: Convert NFSv4 to new lock interface



Convert NFSv4 to the new lock interface.  We don't define any callback for now,
so we're not taking advantage of the asynchronous feature--that's less critical
for the multi-threaded nfsd then it is for the single-threaded lockd.  But this
does allow a cluster filesystems to export cluster-coherent locking to NFS.

Note that it's cluster filesystems that are the issue--of the filesystems that
define lock methods (nfs, cifs, etc.), most are not exportable by nfsd.

Signed-off-by: default avatarMarc Eshel <eshel@almaden.ibm.com>
Signed-off-by: default avatarJ. Bruce Fields <bfields@citi.umich.edu>
parent 9b9d2ab4
Loading
Loading
Loading
Loading
+13 −7
Original line number Original line Diff line number Diff line
@@ -50,6 +50,7 @@
#include <linux/nfsd/xdr4.h>
#include <linux/nfsd/xdr4.h>
#include <linux/namei.h>
#include <linux/namei.h>
#include <linux/mutex.h>
#include <linux/mutex.h>
#include <linux/lockd/bind.h>


#define NFSDDBG_FACILITY                NFSDDBG_PROC
#define NFSDDBG_FACILITY                NFSDDBG_PROC


@@ -2773,7 +2774,7 @@ nfsd4_lock(struct svc_rqst *rqstp, struct nfsd4_compound_state *cstate,
	/* XXX?: Just to divert the locks_release_private at the start of
	/* XXX?: Just to divert the locks_release_private at the start of
	 * locks_copy_lock: */
	 * locks_copy_lock: */
	locks_init_lock(&conflock);
	locks_init_lock(&conflock);
	err = posix_lock_file(filp, &file_lock, &conflock);
	err = vfs_lock_file(filp, cmd, &file_lock, &conflock);
	switch (-err) {
	switch (-err) {
	case 0: /* success! */
	case 0: /* success! */
		update_stateid(&lock_stp->st_stateid);
		update_stateid(&lock_stp->st_stateid);
@@ -2790,7 +2791,7 @@ nfsd4_lock(struct svc_rqst *rqstp, struct nfsd4_compound_state *cstate,
		status = nfserr_deadlock;
		status = nfserr_deadlock;
		break;
		break;
	default:        
	default:        
		dprintk("NFSD: nfsd4_lock: posix_lock_file_conf() failed! status %d\n",err);
		dprintk("NFSD: nfsd4_lock: vfs_lock_file() failed! status %d\n",err);
		status = nfserr_resource;
		status = nfserr_resource;
		break;
		break;
	}
	}
@@ -2815,6 +2816,7 @@ nfsd4_lockt(struct svc_rqst *rqstp, struct nfsd4_compound_state *cstate,
	struct inode *inode;
	struct inode *inode;
	struct file file;
	struct file file;
	struct file_lock file_lock;
	struct file_lock file_lock;
	int error;
	__be32 status;
	__be32 status;


	if (nfs4_in_grace())
	if (nfs4_in_grace())
@@ -2870,16 +2872,20 @@ nfsd4_lockt(struct svc_rqst *rqstp, struct nfsd4_compound_state *cstate,


	nfs4_transform_lock_offset(&file_lock);
	nfs4_transform_lock_offset(&file_lock);


	/* posix_test_lock uses the struct file _only_ to resolve the inode.
	/* vfs_test_lock uses the struct file _only_ to resolve the inode.
	 * since LOCKT doesn't require an OPEN, and therefore a struct
	 * since LOCKT doesn't require an OPEN, and therefore a struct
	 * file may not exist, pass posix_test_lock a struct file with
	 * file may not exist, pass vfs_test_lock a struct file with
	 * only the dentry:inode set.
	 * only the dentry:inode set.
	 */
	 */
	memset(&file, 0, sizeof (struct file));
	memset(&file, 0, sizeof (struct file));
	file.f_path.dentry = cstate->current_fh.fh_dentry;
	file.f_path.dentry = cstate->current_fh.fh_dentry;


	status = nfs_ok;
	status = nfs_ok;
	posix_test_lock(&file, &file_lock);
	error = vfs_test_lock(&file, &file_lock);
	if (error) {
		status = nfserrno(error);
		goto out;
	}
	if (file_lock.fl_type != F_UNLCK) {
	if (file_lock.fl_type != F_UNLCK) {
		status = nfserr_denied;
		status = nfserr_denied;
		nfs4_set_lock_denied(&file_lock, &lockt->lt_denied);
		nfs4_set_lock_denied(&file_lock, &lockt->lt_denied);
@@ -2935,9 +2941,9 @@ nfsd4_locku(struct svc_rqst *rqstp, struct nfsd4_compound_state *cstate,
	/*
	/*
	*  Try to unlock the file in the VFS.
	*  Try to unlock the file in the VFS.
	*/
	*/
	err = posix_lock_file(filp, &file_lock, NULL);
	err = vfs_lock_file(filp, F_SETLK, &file_lock, NULL);
	if (err) {
	if (err) {
		dprintk("NFSD: nfs4_locku: posix_lock_file failed!\n");
		dprintk("NFSD: nfs4_locku: vfs_lock_file failed!\n");
		goto out_nfserr;
		goto out_nfserr;
	}
	}
	/*
	/*