Commit 5946c431 authored by NeilBrown's avatar NeilBrown Committed by Jeff Layton

fs/locks: allow a lock request to block other requests.

Currently, a lock can block pending requests, but all pending
requests are equal.  If lots of pending requests are
mutually exclusive, this means they will all be woken up
and all but one will fail.  This can hurt performance.

So we will allow pending requests to block other requests.
Only the first request will be woken, and it will wake the others.

This patch doesn't implement this fully, but prepares the way.

- It acknowledges that a request might be blocking other requests,
  and when the request is converted to a lock, those blocked
  requests are moved across.
- When a request is requeued or discarded, all blocked requests are
  woken.
- When deadlock-detection looks for the lock which blocks a
  given request, we follow the chain of ->fl_blocker all
  the way to the top.
Tested-by: default avatarkernel test robot <rong.a.chen@intel.com>
Signed-off-by: default avatarNeilBrown <neilb@suse.com>
Reviewed-by: default avatarJ. Bruce Fields <bfields@redhat.com>
Signed-off-by: default avatarJeff Layton <jlayton@kernel.org>
parent d6367d62
...@@ -402,6 +402,24 @@ void locks_copy_lock(struct file_lock *new, struct file_lock *fl) ...@@ -402,6 +402,24 @@ void locks_copy_lock(struct file_lock *new, struct file_lock *fl)
EXPORT_SYMBOL(locks_copy_lock); EXPORT_SYMBOL(locks_copy_lock);
static void locks_move_blocks(struct file_lock *new, struct file_lock *fl)
{
struct file_lock *f;
/*
* As ctx->flc_lock is held, new requests cannot be added to
* ->fl_blocked_requests, so we don't need a lock to check if it
* is empty.
*/
if (list_empty(&fl->fl_blocked_requests))
return;
spin_lock(&blocked_lock_lock);
list_splice_init(&fl->fl_blocked_requests, &new->fl_blocked_requests);
list_for_each_entry(f, &fl->fl_blocked_requests, fl_blocked_member)
f->fl_blocker = new;
spin_unlock(&blocked_lock_lock);
}
static inline int flock_translate_cmd(int cmd) { static inline int flock_translate_cmd(int cmd) {
if (cmd & LOCK_MAND) if (cmd & LOCK_MAND)
return cmd & (LOCK_MAND | LOCK_RW); return cmd & (LOCK_MAND | LOCK_RW);
...@@ -693,6 +711,7 @@ static void __locks_wake_up_blocks(struct file_lock *blocker) ...@@ -693,6 +711,7 @@ static void __locks_wake_up_blocks(struct file_lock *blocker)
static void locks_delete_block(struct file_lock *waiter) static void locks_delete_block(struct file_lock *waiter)
{ {
spin_lock(&blocked_lock_lock); spin_lock(&blocked_lock_lock);
__locks_wake_up_blocks(waiter);
__locks_delete_block(waiter); __locks_delete_block(waiter);
spin_unlock(&blocked_lock_lock); spin_unlock(&blocked_lock_lock);
} }
...@@ -716,6 +735,12 @@ static void __locks_insert_block(struct file_lock *blocker, ...@@ -716,6 +735,12 @@ static void __locks_insert_block(struct file_lock *blocker,
list_add_tail(&waiter->fl_blocked_member, &blocker->fl_blocked_requests); list_add_tail(&waiter->fl_blocked_member, &blocker->fl_blocked_requests);
if (IS_POSIX(blocker) && !IS_OFDLCK(blocker)) if (IS_POSIX(blocker) && !IS_OFDLCK(blocker))
locks_insert_global_blocked(waiter); locks_insert_global_blocked(waiter);
/* The requests in waiter->fl_blocked are known to conflict with
* waiter, but might not conflict with blocker, or the requests
* and lock which block it. So they all need to be woken.
*/
__locks_wake_up_blocks(waiter);
} }
/* Must be called with flc_lock held. */ /* Must be called with flc_lock held. */
...@@ -888,8 +913,11 @@ static struct file_lock *what_owner_is_waiting_for(struct file_lock *block_fl) ...@@ -888,8 +913,11 @@ static struct file_lock *what_owner_is_waiting_for(struct file_lock *block_fl)
struct file_lock *fl; struct file_lock *fl;
hash_for_each_possible(blocked_hash, fl, fl_link, posix_owner_key(block_fl)) { hash_for_each_possible(blocked_hash, fl, fl_link, posix_owner_key(block_fl)) {
if (posix_same_owner(fl, block_fl)) if (posix_same_owner(fl, block_fl)) {
return fl->fl_blocker; while (fl->fl_blocker)
fl = fl->fl_blocker;
return fl;
}
} }
return NULL; return NULL;
} }
...@@ -982,6 +1010,7 @@ static int flock_lock_inode(struct inode *inode, struct file_lock *request) ...@@ -982,6 +1010,7 @@ static int flock_lock_inode(struct inode *inode, struct file_lock *request)
if (request->fl_flags & FL_ACCESS) if (request->fl_flags & FL_ACCESS)
goto out; goto out;
locks_copy_lock(new_fl, request); locks_copy_lock(new_fl, request);
locks_move_blocks(new_fl, request);
locks_insert_lock_ctx(new_fl, &ctx->flc_flock); locks_insert_lock_ctx(new_fl, &ctx->flc_flock);
new_fl = NULL; new_fl = NULL;
error = 0; error = 0;
...@@ -1175,6 +1204,7 @@ static int posix_lock_inode(struct inode *inode, struct file_lock *request, ...@@ -1175,6 +1204,7 @@ static int posix_lock_inode(struct inode *inode, struct file_lock *request,
goto out; goto out;
} }
locks_copy_lock(new_fl, request); locks_copy_lock(new_fl, request);
locks_move_blocks(new_fl, request);
locks_insert_lock_ctx(new_fl, &fl->fl_list); locks_insert_lock_ctx(new_fl, &fl->fl_list);
fl = new_fl; fl = new_fl;
new_fl = NULL; new_fl = NULL;
...@@ -2586,13 +2616,14 @@ void locks_remove_file(struct file *filp) ...@@ -2586,13 +2616,14 @@ void locks_remove_file(struct file *filp)
int int
posix_unblock_lock(struct file_lock *waiter) posix_unblock_lock(struct file_lock *waiter)
{ {
int status = 0; int status = -ENOENT;
spin_lock(&blocked_lock_lock); spin_lock(&blocked_lock_lock);
if (waiter->fl_blocker) if (waiter->fl_blocker) {
__locks_wake_up_blocks(waiter);
__locks_delete_block(waiter); __locks_delete_block(waiter);
else status = 0;
status = -ENOENT; }
spin_unlock(&blocked_lock_lock); spin_unlock(&blocked_lock_lock);
return status; return status;
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment