aboutsummaryrefslogtreecommitdiffstats
path: root/fs
diff options
context:
space:
mode:
authorJeff Layton <jeff.layton@primarydata.com>2015-04-03 09:04:03 -0400
committerJeff Layton <jeff.layton@primarydata.com>2015-04-03 09:04:03 -0400
commit5c1c669a1b2435e071d566b6db1a8e6b26542ba1 (patch)
tree512cf17f4056133293b863e38d81097f921037e0 /fs
parent663d5af750b8c025d0dfea2cf2a4b4a78cafa3a7 (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.c20
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;
203static struct kmem_cache *filelock_cache __read_mostly; 203static struct kmem_cache *filelock_cache __read_mostly;
204 204
205static struct file_lock_context * 205static struct file_lock_context *
206locks_get_lock_context(struct inode *inode) 206locks_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