Linux 4.7 compat: use iterate_shared for concurrent readdir

Register iterate_shared if it exists so the kernel will used shared
lock and allowing concurrent readdir.

Also, use shared lock when doing llseek with SEEK_DATA or SEEK_HOLE
to allow concurrent seeking.

Signed-off-by: Chunwei Chen <david.chen@osnexus.com>
Signed-off-by: Brian Behlendorf <behlendorf1@llnl.gov>
Closes #4664
Closes #4665
This commit is contained in:
Chunwei Chen 2016-05-18 14:30:20 -07:00 committed by Brian Behlendorf
parent e42d46664e
commit 9baaa7deae
4 changed files with 54 additions and 25 deletions

View File

@ -1,8 +1,8 @@
dnl #
dnl # 3.11 API change
dnl #
AC_DEFUN([ZFS_AC_KERNEL_VFS_ITERATE], [ AC_DEFUN([ZFS_AC_KERNEL_VFS_ITERATE], [
AC_MSG_CHECKING([whether fops->iterate() is available]) dnl #
dnl # 4.7 API change
dnl #
AC_MSG_CHECKING([whether fops->iterate_shared() is available])
ZFS_LINUX_TRY_COMPILE([ ZFS_LINUX_TRY_COMPILE([
#include <linux/fs.h> #include <linux/fs.h>
int iterate(struct file *filp, struct dir_context * context) int iterate(struct file *filp, struct dir_context * context)
@ -10,34 +10,55 @@ AC_DEFUN([ZFS_AC_KERNEL_VFS_ITERATE], [
static const struct file_operations fops static const struct file_operations fops
__attribute__ ((unused)) = { __attribute__ ((unused)) = {
.iterate = iterate, .iterate_shared = iterate,
}; };
],[ ],[
],[ ],[
AC_MSG_RESULT(yes) AC_MSG_RESULT(yes)
AC_DEFINE(HAVE_VFS_ITERATE, 1, AC_DEFINE(HAVE_VFS_ITERATE_SHARED, 1,
[fops->iterate() is available]) [fops->iterate_shared() is available])
],[ ],[
AC_MSG_RESULT(no) AC_MSG_RESULT(no)
AC_MSG_CHECKING([whether fops->readdir() is available]) dnl #
dnl # 3.11 API change
dnl #
AC_MSG_CHECKING([whether fops->iterate() is available])
ZFS_LINUX_TRY_COMPILE([ ZFS_LINUX_TRY_COMPILE([
#include <linux/fs.h> #include <linux/fs.h>
int readdir(struct file *filp, void *entry, filldir_t func) int iterate(struct file *filp, struct dir_context * context)
{ return 0; } { return 0; }
static const struct file_operations fops static const struct file_operations fops
__attribute__ ((unused)) = { __attribute__ ((unused)) = {
.readdir = readdir, .iterate = iterate,
}; };
],[ ],[
],[ ],[
AC_MSG_RESULT(yes) AC_MSG_RESULT(yes)
AC_DEFINE(HAVE_VFS_READDIR, 1, AC_DEFINE(HAVE_VFS_ITERATE, 1,
[fops->readdir() is available]) [fops->iterate() is available])
],[ ],[
AC_MSG_ERROR(no; file a bug report with ZFSOnLinux) AC_MSG_RESULT(no)
])
AC_MSG_CHECKING([whether fops->readdir() is available])
ZFS_LINUX_TRY_COMPILE([
#include <linux/fs.h>
int readdir(struct file *filp, void *entry, filldir_t func)
{ return 0; }
static const struct file_operations fops
__attribute__ ((unused)) = {
.readdir = readdir,
};
],[
],[
AC_MSG_RESULT(yes)
AC_DEFINE(HAVE_VFS_READDIR, 1,
[fops->readdir() is available])
],[
AC_MSG_ERROR(no; file a bug report with ZFSOnLinux)
])
])
]) ])
]) ])

View File

@ -123,7 +123,7 @@ extern const struct inode_operations zpl_ops_snapdirs;
extern const struct file_operations zpl_fops_shares; extern const struct file_operations zpl_fops_shares;
extern const struct inode_operations zpl_ops_shares; extern const struct inode_operations zpl_ops_shares;
#ifdef HAVE_VFS_ITERATE #if defined(HAVE_VFS_ITERATE) || defined(HAVE_VFS_ITERATE_SHARED)
#define DIR_CONTEXT_INIT(_dirent, _actor, _pos) { \ #define DIR_CONTEXT_INIT(_dirent, _actor, _pos) { \
.actor = _actor, \ .actor = _actor, \

View File

@ -81,7 +81,7 @@ out:
return (error); return (error);
} }
#if !defined(HAVE_VFS_ITERATE) #if !defined(HAVE_VFS_ITERATE) && !defined(HAVE_VFS_ITERATE_SHARED)
static int static int
zpl_root_readdir(struct file *filp, void *dirent, filldir_t filldir) zpl_root_readdir(struct file *filp, void *dirent, filldir_t filldir)
{ {
@ -144,7 +144,9 @@ const struct file_operations zpl_fops_root = {
.open = zpl_common_open, .open = zpl_common_open,
.llseek = generic_file_llseek, .llseek = generic_file_llseek,
.read = generic_read_dir, .read = generic_read_dir,
#ifdef HAVE_VFS_ITERATE #ifdef HAVE_VFS_ITERATE_SHARED
.iterate_shared = zpl_root_iterate,
#elif defined(HAVE_VFS_ITERATE)
.iterate = zpl_root_iterate, .iterate = zpl_root_iterate,
#else #else
.readdir = zpl_root_readdir, .readdir = zpl_root_readdir,
@ -285,7 +287,7 @@ out:
return (error); return (error);
} }
#if !defined(HAVE_VFS_ITERATE) #if !defined(HAVE_VFS_ITERATE) && !defined(HAVE_VFS_ITERATE_SHARED)
static int static int
zpl_snapdir_readdir(struct file *filp, void *dirent, filldir_t filldir) zpl_snapdir_readdir(struct file *filp, void *dirent, filldir_t filldir)
{ {
@ -385,7 +387,9 @@ const struct file_operations zpl_fops_snapdir = {
.open = zpl_common_open, .open = zpl_common_open,
.llseek = generic_file_llseek, .llseek = generic_file_llseek,
.read = generic_read_dir, .read = generic_read_dir,
#ifdef HAVE_VFS_ITERATE #ifdef HAVE_VFS_ITERATE_SHARED
.iterate_shared = zpl_snapdir_iterate,
#elif defined(HAVE_VFS_ITERATE)
.iterate = zpl_snapdir_iterate, .iterate = zpl_snapdir_iterate,
#else #else
.readdir = zpl_snapdir_readdir, .readdir = zpl_snapdir_readdir,
@ -472,7 +476,7 @@ out:
return (error); return (error);
} }
#if !defined(HAVE_VFS_ITERATE) #if !defined(HAVE_VFS_ITERATE) && !defined(HAVE_VFS_ITERATE_SHARED)
static int static int
zpl_shares_readdir(struct file *filp, void *dirent, filldir_t filldir) zpl_shares_readdir(struct file *filp, void *dirent, filldir_t filldir)
{ {
@ -525,7 +529,9 @@ const struct file_operations zpl_fops_shares = {
.open = zpl_common_open, .open = zpl_common_open,
.llseek = generic_file_llseek, .llseek = generic_file_llseek,
.read = generic_read_dir, .read = generic_read_dir,
#ifdef HAVE_VFS_ITERATE #ifdef HAVE_VFS_ITERATE_SHARED
.iterate_shared = zpl_shares_iterate,
#elif defined(HAVE_VFS_ITERATE)
.iterate = zpl_shares_iterate, .iterate = zpl_shares_iterate,
#else #else
.readdir = zpl_shares_readdir, .readdir = zpl_shares_readdir,

View File

@ -93,7 +93,7 @@ zpl_iterate(struct file *filp, struct dir_context *ctx)
return (error); return (error);
} }
#if !defined(HAVE_VFS_ITERATE) #if !defined(HAVE_VFS_ITERATE) && !defined(HAVE_VFS_ITERATE_SHARED)
static int static int
zpl_readdir(struct file *filp, void *dirent, filldir_t filldir) zpl_readdir(struct file *filp, void *dirent, filldir_t filldir)
{ {
@ -421,13 +421,13 @@ zpl_llseek(struct file *filp, loff_t offset, int whence)
loff_t maxbytes = ip->i_sb->s_maxbytes; loff_t maxbytes = ip->i_sb->s_maxbytes;
loff_t error; loff_t error;
spl_inode_lock(ip); spl_inode_lock_shared(ip);
cookie = spl_fstrans_mark(); cookie = spl_fstrans_mark();
error = -zfs_holey(ip, whence, &offset); error = -zfs_holey(ip, whence, &offset);
spl_fstrans_unmark(cookie); spl_fstrans_unmark(cookie);
if (error == 0) if (error == 0)
error = lseek_execute(filp, ip, offset, maxbytes); error = lseek_execute(filp, ip, offset, maxbytes);
spl_inode_unlock(ip); spl_inode_unlock_shared(ip);
return (error); return (error);
} }
@ -853,7 +853,9 @@ const struct file_operations zpl_file_operations = {
const struct file_operations zpl_dir_file_operations = { const struct file_operations zpl_dir_file_operations = {
.llseek = generic_file_llseek, .llseek = generic_file_llseek,
.read = generic_read_dir, .read = generic_read_dir,
#ifdef HAVE_VFS_ITERATE #ifdef HAVE_VFS_ITERATE_SHARED
.iterate_shared = zpl_iterate,
#elif defined(HAVE_VFS_ITERATE)
.iterate = zpl_iterate, .iterate = zpl_iterate,
#else #else
.readdir = zpl_readdir, .readdir = zpl_readdir,