Commit c078bd9e authored by Christoph Hellwig's avatar Christoph Hellwig Committed by Linus Torvalds

[PATCH] Direct write vs truncate deadlock

On Fri, Dec 10, 2004 at 11:04:22AM +1100, Nathan Scott wrote:
> The bug# for the write deadlock I mentioned earlier -- 925836.

I finally found some time to look at it, and this fix is almost trivial.

XFS doesn't need the i_alloc_sem at all, so we should avoid taking it in
direct-io.c completely.  As a side-effect it makes the code a little bit
simpler even.
Signed-off-by: default avatarAndrew Morton <akpm@osdl.org>
Signed-off-by: default avatarLinus Torvalds <torvalds@osdl.org>
parent d88971f9
...@@ -215,7 +215,7 @@ static void dio_complete(struct dio *dio, loff_t offset, ssize_t bytes) ...@@ -215,7 +215,7 @@ static void dio_complete(struct dio *dio, loff_t offset, ssize_t bytes)
{ {
if (dio->end_io && dio->result) if (dio->end_io && dio->result)
dio->end_io(dio->inode, offset, bytes, dio->map_bh.b_private); dio->end_io(dio->inode, offset, bytes, dio->map_bh.b_private);
if (dio->lock_type != DIO_NO_LOCKING) if (dio->lock_type == DIO_LOCKING)
up_read(&dio->inode->i_alloc_sem); up_read(&dio->inode->i_alloc_sem);
} }
...@@ -1201,8 +1201,8 @@ __blockdev_direct_IO(int rw, struct kiocb *iocb, struct inode *inode, ...@@ -1201,8 +1201,8 @@ __blockdev_direct_IO(int rw, struct kiocb *iocb, struct inode *inode,
* readers need to grab i_sem and i_alloc_sem * readers need to grab i_sem and i_alloc_sem
* writers need to grab i_alloc_sem only (i_sem is already held) * writers need to grab i_alloc_sem only (i_sem is already held)
* For regular files using DIO_OWN_LOCKING, * For regular files using DIO_OWN_LOCKING,
* readers need to grab i_alloc_sem only (i_sem is already held) * neither readers nor writers take any locks here
* writers need to grab i_alloc_sem only * (i_sem is already held and release for writers here)
*/ */
dio->lock_type = dio_lock_type; dio->lock_type = dio_lock_type;
if (dio_lock_type != DIO_NO_LOCKING) { if (dio_lock_type != DIO_NO_LOCKING) {
...@@ -1219,14 +1219,15 @@ __blockdev_direct_IO(int rw, struct kiocb *iocb, struct inode *inode, ...@@ -1219,14 +1219,15 @@ __blockdev_direct_IO(int rw, struct kiocb *iocb, struct inode *inode,
kfree(dio); kfree(dio);
goto out; goto out;
} }
down_read(&inode->i_alloc_sem);
if (dio_lock_type == DIO_OWN_LOCKING) { if (dio_lock_type == DIO_OWN_LOCKING) {
up(&inode->i_sem); up(&inode->i_sem);
reader_with_isem = 0; reader_with_isem = 0;
} }
} else {
down_read(&inode->i_alloc_sem);
} }
if (dio_lock_type == DIO_LOCKING)
down_read(&inode->i_alloc_sem);
} }
/* /*
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment