diff options
author | Jeff Layton <jeff.layton@primarydata.com> | 2015-04-03 09:04:04 -0400 |
---|---|---|
committer | Jeff Layton <jeff.layton@primarydata.com> | 2015-04-03 09:04:04 -0400 |
commit | 0429c2b5c1c4c8ba6cd563c1964baf3ed238df26 (patch) | |
tree | 60bda1e72adfea7a0a4fe3cdceacc78a3f0e8844 /fs | |
parent | 3648888e90bb7fe6d0586ec177511e6678ee22c3 (diff) |
locks: use cmpxchg to assign i_flctx pointer
During the v3.20/v4.0 cycle, I had originally had the code manage the
inode->i_flctx pointer using a compare-and-swap operation instead of the
i_lock.
Sasha Levin though hit a problem while testing with trinity that made me
believe that that wasn't safe. At the time, changing the code to protect
the i_flctx pointer seemed to fix the issue, but I now think that was
just coincidence.
The issue was likely the same race that Kirill Shutemov hit while
testing the pre-rc1 v4.0 kernel and that Linus spotted. Due to the way
that the spinlock was dropped in the middle of flock_lock_file, you
could end up with multiple flock locks for the same struct file on the
inode.
Reinstate the use of a CAS operation to assign this pointer since it's
likely to be more efficient and gets the i_lock completely out of the
file locking business.
Signed-off-by: Jeff Layton <jeff.layton@primarydata.com>
Diffstat (limited to 'fs')
-rw-r--r-- | fs/locks.c | 9 |
1 files changed, 1 insertions, 8 deletions
diff --git a/fs/locks.c b/fs/locks.c index 16cae1a00851..52b780fb5258 100644 --- a/fs/locks.c +++ b/fs/locks.c | |||
@@ -223,14 +223,7 @@ locks_get_lock_context(struct inode *inode, int type) | |||
223 | * Assign the pointer if it's not already assigned. If it is, then | 223 | * Assign the pointer if it's not already assigned. If it is, then |
224 | * free the context we just allocated. | 224 | * free the context we just allocated. |
225 | */ | 225 | */ |
226 | spin_lock(&inode->i_lock); | 226 | if (cmpxchg(&inode->i_flctx, NULL, new)) |
227 | if (likely(!inode->i_flctx)) { | ||
228 | inode->i_flctx = new; | ||
229 | new = NULL; | ||
230 | } | ||
231 | spin_unlock(&inode->i_lock); | ||
232 | |||
233 | if (new) | ||
234 | kmem_cache_free(flctx_cache, new); | 227 | kmem_cache_free(flctx_cache, new); |
235 | out: | 228 | out: |
236 | return inode->i_flctx; | 229 | return inode->i_flctx; |