]> granicus.if.org Git - zfs/commitdiff
Handle NULL in nfsd .fsync() hook
authorBrian Behlendorf <behlendorf1@llnl.gov>
Fri, 6 May 2011 19:23:34 +0000 (12:23 -0700)
committerBrian Behlendorf <behlendorf1@llnl.gov>
Fri, 6 May 2011 19:33:45 +0000 (12:33 -0700)
How nfsd handles .fsync() has been changed a couple of times in the
recent kernels.  But basically there are three cases we need to
consider.

Linux 2.6.12 - 2.6.33
* The .fsync() hook takes 3 arguments
* The nfsd will call .fsync() with a NULL file struct pointer.

Linux 2.6.34
* The .fsync() hook takes 3 arguments
* The nfsd no longer calls .fsync() but instead used sync_inode()

Linux 2.6.35 - 2.6.x
* The .fsync() hook takes 2 arguments
* The nfsd no longer calls .fsync() but instead used sync_inode()

For once it looks like we've gotten lucky.  The first two cases can
actually be collased in to one if we stop using the file struct
pointer entirely.  Since the dentry is still passed in both cases
this is possible.  The last case can then be safely handled by
unconditionally using the dentry in the file struct pointer now
that we know the nfsd caller has been removed.

Closes #230

include/linux/vfs_compat.h
module/zfs/zpl_file.c

index c25cf44779435310bd87d64d02c3ea6cffd99fd1..dce2b6f28127bf6b52587f733b967c07a5bb60fc 100644 (file)
 #ifndef _ZFS_VFS_H
 #define _ZFS_VFS_H
 
-/*
- * 2.6.35 API change,
- * The dentry argument to the .fsync() vfs hook was deemed unused by
- * all filesystem consumers and dropped.  Add a compatibility prototype
- * to ensure correct usage when defining this callback.
- */
-#ifdef HAVE_2ARGS_FSYNC
-#define ZPL_FSYNC_PROTO(fn, x, y, z)   static int fn(struct file *x, int z)
-#else
-#define ZPL_FSYNC_PROTO(fn, x, y, z)   static int fn(struct file *x, \
-                                                     struct dentry *y, int z)
-#endif /* HAVE_2ARGS_FSYNC */
-
 /*
  * 2.6.28 API change,
  * Added insert_inode_locked() helper function, prior to this most callers
index c8a3fedb7c8e7ba9719b29f1d760b1854067313d..de66ff4b49200f6bf6d997bdfde8f9ced671cb9f 100644 (file)
@@ -76,13 +76,32 @@ zpl_readdir(struct file *filp, void *dirent, filldir_t filldir)
        return (error);
 }
 
-ZPL_FSYNC_PROTO(zpl_fsync, filp, unused_dentry, datasync)
+/*
+ * 2.6.35 API change,
+ * As of 2.6.35 the dentry argument to the .fsync() vfs hook was deemed
+ * redundant.  The dentry is still accessible via filp->f_path.dentry,
+ * and we are guaranteed that filp will never be NULL.
+ *
+ * 2.6.34 API change,
+ * Prior to 2.6.34 the nfsd kernel server would pass a NULL file struct *
+ * to the .fsync() hook.  For this reason, we must be careful not to use
+ * filp unconditionally in the 3 argument case.
+ */
+#ifdef HAVE_2ARGS_FSYNC
+static int
+zpl_fsync(struct file *filp, int datasync)
+{
+       struct dentry *dentry = filp->f_path.dentry;
+#else
+static int
+zpl_fsync(struct file *filp, struct dentry *dentry, int datasync)
 {
+#endif /* HAVE_2ARGS_FSYNC */
        cred_t *cr = CRED();
        int error;
 
        crhold(cr);
-       error = -zfs_fsync(filp->f_path.dentry->d_inode, datasync, cr);
+       error = -zfs_fsync(dentry->d_inode, datasync, cr);
        crfree(cr);
        ASSERT3S(error, <=, 0);