diff options
author | Pavel Shilovsky <pshilovsky@samba.org> | 2013-07-11 03:17:45 -0400 |
---|---|---|
committer | Steve French <smfrench@gmail.com> | 2013-07-11 19:05:41 -0400 |
commit | 689c3db4d57a73bee6c5ad7797fce7b54d32a87c (patch) | |
tree | c8a1f22b1fcdc1335d12a0b41889ec5098176d8b /fs/cifs | |
parent | b33fcf1c9d359374ef6e8e7ec1e0b6f7a8717e16 (diff) |
CIFS: Fix a deadlock when a file is reopened
If we request reading or writing on a file that needs to be
reopened, it causes the deadlock: we are already holding rw
semaphore for reading and then we try to acquire it for writing
in cifs_relock_file. Fix this by acquiring the semaphore for
reading in cifs_relock_file due to we don't make any changes in
locks and don't need a write access.
CC: <stable@vger.kernel.org>
Signed-off-by: Pavel Shilovsky <pshilovsky@samba.org>
Acked-by: Jeff Layton <jlayton@redhat.com>
Signed-off-by: Steve French <smfrench@gmail.com>
Diffstat (limited to 'fs/cifs')
-rw-r--r-- | fs/cifs/file.c | 9 |
1 files changed, 4 insertions, 5 deletions
diff --git a/fs/cifs/file.c b/fs/cifs/file.c index 1dc9dea2ae70..1e57f36ea1b2 100644 --- a/fs/cifs/file.c +++ b/fs/cifs/file.c | |||
@@ -561,11 +561,10 @@ cifs_relock_file(struct cifsFileInfo *cfile) | |||
561 | struct cifs_tcon *tcon = tlink_tcon(cfile->tlink); | 561 | struct cifs_tcon *tcon = tlink_tcon(cfile->tlink); |
562 | int rc = 0; | 562 | int rc = 0; |
563 | 563 | ||
564 | /* we are going to update can_cache_brlcks here - need a write access */ | 564 | down_read(&cinode->lock_sem); |
565 | down_write(&cinode->lock_sem); | ||
566 | if (cinode->can_cache_brlcks) { | 565 | if (cinode->can_cache_brlcks) { |
567 | /* can cache locks - no need to push them */ | 566 | /* can cache locks - no need to relock */ |
568 | up_write(&cinode->lock_sem); | 567 | up_read(&cinode->lock_sem); |
569 | return rc; | 568 | return rc; |
570 | } | 569 | } |
571 | 570 | ||
@@ -576,7 +575,7 @@ cifs_relock_file(struct cifsFileInfo *cfile) | |||
576 | else | 575 | else |
577 | rc = tcon->ses->server->ops->push_mand_locks(cfile); | 576 | rc = tcon->ses->server->ops->push_mand_locks(cfile); |
578 | 577 | ||
579 | up_write(&cinode->lock_sem); | 578 | up_read(&cinode->lock_sem); |
580 | return rc; | 579 | return rc; |
581 | } | 580 | } |
582 | 581 | ||