Skip to content

Commit da9c988

Browse files
lxbszgregkh
authored andcommitted
ceph: switch to vfs_inode_has_locks() to fix file lock bug
[ Upstream commit 461ab10 ] For the POSIX locks they are using the same owner, which is the thread id. And multiple POSIX locks could be merged into single one, so when checking whether the 'file' has locks may fail. For a file where some openers use locking and others don't is a really odd usage pattern though. Locks are like stoplights -- they only work if everyone pays attention to them. Just switch ceph_get_caps() to check whether any locks are set on the inode. If there are POSIX/OFD/FLOCK locks on the file at the time, we should set CHECK_FILELOCK, regardless of what fd was used to set the lock. Fixes: ff5d913 ("ceph: return -EIO if read/write against filp that lost file locks") Signed-off-by: Xiubo Li <xiubli@redhat.com> Reviewed-by: Jeff Layton <jlayton@kernel.org> Reviewed-by: Ilya Dryomov <idryomov@gmail.com> Signed-off-by: Ilya Dryomov <idryomov@gmail.com> Signed-off-by: Sasha Levin <sashal@kernel.org>
1 parent 54e72ce commit da9c988

File tree

3 files changed

+1
-6
lines changed

3 files changed

+1
-6
lines changed

fs/ceph/caps.c

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -2872,7 +2872,7 @@ int ceph_get_caps(struct file *filp, int need, int want, loff_t endoff, int *got
28722872

28732873
while (true) {
28742874
flags &= CEPH_FILE_MODE_MASK;
2875-
if (atomic_read(&fi->num_locks))
2875+
if (vfs_inode_has_locks(inode))
28762876
flags |= CHECK_FILELOCK;
28772877
_got = 0;
28782878
ret = try_get_cap_refs(inode, need, want, endoff,

fs/ceph/locks.c

Lines changed: 0 additions & 4 deletions
Original file line numberDiff line numberDiff line change
@@ -32,18 +32,14 @@ void __init ceph_flock_init(void)
3232

3333
static void ceph_fl_copy_lock(struct file_lock *dst, struct file_lock *src)
3434
{
35-
struct ceph_file_info *fi = dst->fl_file->private_data;
3635
struct inode *inode = file_inode(dst->fl_file);
3736
atomic_inc(&ceph_inode(inode)->i_filelock_ref);
38-
atomic_inc(&fi->num_locks);
3937
}
4038

4139
static void ceph_fl_release_lock(struct file_lock *fl)
4240
{
43-
struct ceph_file_info *fi = fl->fl_file->private_data;
4441
struct inode *inode = file_inode(fl->fl_file);
4542
struct ceph_inode_info *ci = ceph_inode(inode);
46-
atomic_dec(&fi->num_locks);
4743
if (atomic_dec_and_test(&ci->i_filelock_ref)) {
4844
/* clear error when all locks are released */
4945
spin_lock(&ci->i_ceph_lock);

fs/ceph/super.h

Lines changed: 0 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -773,7 +773,6 @@ struct ceph_file_info {
773773
struct list_head rw_contexts;
774774

775775
u32 filp_gen;
776-
atomic_t num_locks;
777776
};
778777

779778
struct ceph_dir_file_info {

0 commit comments

Comments
 (0)