diff options
author | Jeff Layton <jeff.layton@primarydata.com> | 2015-04-03 09:04:03 -0400 |
---|---|---|
committer | Jeff Layton <jeff.layton@primarydata.com> | 2015-04-03 09:04:03 -0400 |
commit | 5c1c669a1b2435e071d566b6db1a8e6b26542ba1 (patch) | |
tree | 512cf17f4056133293b863e38d81097f921037e0 /fs | |
parent | 663d5af750b8c025d0dfea2cf2a4b4a78cafa3a7 (diff) |
locks: don't allocate a lock context for an F_UNLCK request
In the event that we get an F_UNLCK request on an inode that has no lock
context, there is no reason to allocate one. Change
locks_get_lock_context to take a "type" pointer and avoid allocating a
new context if it's F_UNLCK.
Then, fix the callers to return appropriately if that function returns
NULL.
Signed-off-by: Jeff Layton <jlayton@primarydata.com>
Diffstat (limited to 'fs')
-rw-r--r-- | fs/locks.c | 20 |
1 files changed, 12 insertions, 8 deletions
diff --git a/fs/locks.c b/fs/locks.c index 36cf93f165a8..54a79883a7f9 100644 --- a/fs/locks.c +++ b/fs/locks.c | |||
@@ -203,11 +203,11 @@ static struct kmem_cache *flctx_cache __read_mostly; | |||
203 | static struct kmem_cache *filelock_cache __read_mostly; | 203 | static struct kmem_cache *filelock_cache __read_mostly; |
204 | 204 | ||
205 | static struct file_lock_context * | 205 | static struct file_lock_context * |
206 | locks_get_lock_context(struct inode *inode) | 206 | locks_get_lock_context(struct inode *inode, int type) |
207 | { | 207 | { |
208 | struct file_lock_context *new; | 208 | struct file_lock_context *new; |
209 | 209 | ||
210 | if (likely(inode->i_flctx)) | 210 | if (likely(inode->i_flctx) || type == F_UNLCK) |
211 | goto out; | 211 | goto out; |
212 | 212 | ||
213 | new = kmem_cache_alloc(flctx_cache, GFP_KERNEL); | 213 | new = kmem_cache_alloc(flctx_cache, GFP_KERNEL); |
@@ -877,9 +877,12 @@ static int flock_lock_file(struct file *filp, struct file_lock *request) | |||
877 | bool found = false; | 877 | bool found = false; |
878 | LIST_HEAD(dispose); | 878 | LIST_HEAD(dispose); |
879 | 879 | ||
880 | ctx = locks_get_lock_context(inode); | 880 | ctx = locks_get_lock_context(inode, request->fl_type); |
881 | if (!ctx) | 881 | if (!ctx) { |
882 | return -ENOMEM; | 882 | if (request->fl_type != F_UNLCK) |
883 | return -ENOMEM; | ||
884 | return (request->fl_flags & FL_EXISTS) ? -ENOENT : 0; | ||
885 | } | ||
883 | 886 | ||
884 | if (!(request->fl_flags & FL_ACCESS) && (request->fl_type != F_UNLCK)) { | 887 | if (!(request->fl_flags & FL_ACCESS) && (request->fl_type != F_UNLCK)) { |
885 | new_fl = locks_alloc_lock(); | 888 | new_fl = locks_alloc_lock(); |
@@ -945,9 +948,9 @@ static int __posix_lock_file(struct inode *inode, struct file_lock *request, str | |||
945 | bool added = false; | 948 | bool added = false; |
946 | LIST_HEAD(dispose); | 949 | LIST_HEAD(dispose); |
947 | 950 | ||
948 | ctx = locks_get_lock_context(inode); | 951 | ctx = locks_get_lock_context(inode, request->fl_type); |
949 | if (!ctx) | 952 | if (!ctx) |
950 | return -ENOMEM; | 953 | return (request->fl_type == F_UNLCK) ? 0 : -ENOMEM; |
951 | 954 | ||
952 | /* | 955 | /* |
953 | * We may need two file_lock structures for this operation, | 956 | * We may need two file_lock structures for this operation, |
@@ -1609,7 +1612,8 @@ generic_add_lease(struct file *filp, long arg, struct file_lock **flp, void **pr | |||
1609 | lease = *flp; | 1612 | lease = *flp; |
1610 | trace_generic_add_lease(inode, lease); | 1613 | trace_generic_add_lease(inode, lease); |
1611 | 1614 | ||
1612 | ctx = locks_get_lock_context(inode); | 1615 | /* Note that arg is never F_UNLCK here */ |
1616 | ctx = locks_get_lock_context(inode, arg); | ||
1613 | if (!ctx) | 1617 | if (!ctx) |
1614 | return -ENOMEM; | 1618 | return -ENOMEM; |
1615 | 1619 | ||