|
@@ -97,6 +97,7 @@ static struct inode *fuse_alloc_inode(struct super_block *sb)
|
|
INIT_LIST_HEAD(&fi->queued_writes);
|
|
INIT_LIST_HEAD(&fi->queued_writes);
|
|
INIT_LIST_HEAD(&fi->writepages);
|
|
INIT_LIST_HEAD(&fi->writepages);
|
|
init_waitqueue_head(&fi->page_waitq);
|
|
init_waitqueue_head(&fi->page_waitq);
|
|
|
|
+ mutex_init(&fi->mutex);
|
|
fi->forget = fuse_alloc_forget();
|
|
fi->forget = fuse_alloc_forget();
|
|
if (!fi->forget) {
|
|
if (!fi->forget) {
|
|
kmem_cache_free(fuse_inode_cachep, inode);
|
|
kmem_cache_free(fuse_inode_cachep, inode);
|
|
@@ -117,6 +118,7 @@ static void fuse_destroy_inode(struct inode *inode)
|
|
struct fuse_inode *fi = get_fuse_inode(inode);
|
|
struct fuse_inode *fi = get_fuse_inode(inode);
|
|
BUG_ON(!list_empty(&fi->write_files));
|
|
BUG_ON(!list_empty(&fi->write_files));
|
|
BUG_ON(!list_empty(&fi->queued_writes));
|
|
BUG_ON(!list_empty(&fi->queued_writes));
|
|
|
|
+ mutex_destroy(&fi->mutex);
|
|
kfree(fi->forget);
|
|
kfree(fi->forget);
|
|
call_rcu(&inode->i_rcu, fuse_i_callback);
|
|
call_rcu(&inode->i_rcu, fuse_i_callback);
|
|
}
|
|
}
|
|
@@ -351,6 +353,18 @@ int fuse_reverse_inval_inode(struct super_block *sb, u64 nodeid,
|
|
return 0;
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
+void fuse_lock_inode(struct inode *inode)
|
|
|
|
+{
|
|
|
|
+ if (!get_fuse_conn(inode)->parallel_dirops)
|
|
|
|
+ mutex_lock(&get_fuse_inode(inode)->mutex);
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+void fuse_unlock_inode(struct inode *inode)
|
|
|
|
+{
|
|
|
|
+ if (!get_fuse_conn(inode)->parallel_dirops)
|
|
|
|
+ mutex_unlock(&get_fuse_inode(inode)->mutex);
|
|
|
|
+}
|
|
|
|
+
|
|
static void fuse_umount_begin(struct super_block *sb)
|
|
static void fuse_umount_begin(struct super_block *sb)
|
|
{
|
|
{
|
|
fuse_abort_conn(get_fuse_conn_super(sb));
|
|
fuse_abort_conn(get_fuse_conn_super(sb));
|
|
@@ -898,6 +912,8 @@ static void process_init_reply(struct fuse_conn *fc, struct fuse_req *req)
|
|
fc->async_dio = 1;
|
|
fc->async_dio = 1;
|
|
if (arg->flags & FUSE_WRITEBACK_CACHE)
|
|
if (arg->flags & FUSE_WRITEBACK_CACHE)
|
|
fc->writeback_cache = 1;
|
|
fc->writeback_cache = 1;
|
|
|
|
+ if (arg->flags & FUSE_PARALLEL_DIROPS)
|
|
|
|
+ fc->parallel_dirops = 1;
|
|
if (arg->time_gran && arg->time_gran <= 1000000000)
|
|
if (arg->time_gran && arg->time_gran <= 1000000000)
|
|
fc->sb->s_time_gran = arg->time_gran;
|
|
fc->sb->s_time_gran = arg->time_gran;
|
|
} else {
|
|
} else {
|
|
@@ -928,7 +944,8 @@ static void fuse_send_init(struct fuse_conn *fc, struct fuse_req *req)
|
|
FUSE_SPLICE_WRITE | FUSE_SPLICE_MOVE | FUSE_SPLICE_READ |
|
|
FUSE_SPLICE_WRITE | FUSE_SPLICE_MOVE | FUSE_SPLICE_READ |
|
|
FUSE_FLOCK_LOCKS | FUSE_IOCTL_DIR | FUSE_AUTO_INVAL_DATA |
|
|
FUSE_FLOCK_LOCKS | FUSE_IOCTL_DIR | FUSE_AUTO_INVAL_DATA |
|
|
FUSE_DO_READDIRPLUS | FUSE_READDIRPLUS_AUTO | FUSE_ASYNC_DIO |
|
|
FUSE_DO_READDIRPLUS | FUSE_READDIRPLUS_AUTO | FUSE_ASYNC_DIO |
|
|
- FUSE_WRITEBACK_CACHE | FUSE_NO_OPEN_SUPPORT;
|
|
|
|
|
|
+ FUSE_WRITEBACK_CACHE | FUSE_NO_OPEN_SUPPORT |
|
|
|
|
+ FUSE_PARALLEL_DIROPS;
|
|
req->in.h.opcode = FUSE_INIT;
|
|
req->in.h.opcode = FUSE_INIT;
|
|
req->in.numargs = 1;
|
|
req->in.numargs = 1;
|
|
req->in.args[0].size = sizeof(*arg);
|
|
req->in.args[0].size = sizeof(*arg);
|