diff options
author | Jan Kara <jack@suse.cz> | 2013-01-29 22:48:17 -0500 |
---|---|---|
committer | Theodore Ts'o <tytso@mit.edu> | 2013-01-29 22:48:17 -0500 |
commit | 091e26dfc156aeb3b73bc5c5f277e433ad39331c (patch) | |
tree | 03a95cea66597c0b7289ea6fbdc13785a5e21919 /fs/ext4 | |
parent | b1deefc99e668348f7c785c6ece5f6ff4c6d8b5c (diff) |
ext4: fix possible use-after-free with AIO
Running AIO is pinning inode in memory using file reference. Once AIO
is completed using aio_complete(), file reference is put and inode can
be freed from memory. So we have to be sure that calling aio_complete()
is the last thing we do with the inode.
CC: stable@vger.kernel.org
Reviewed-by: Carlos Maiolino <cmaiolino@redhat.com>
Acked-by: Jeff Moyer <jmoyer@redhat.com>
Signed-off-by: Jan Kara <jack@suse.cz>
Signed-off-by: "Theodore Ts'o" <tytso@mit.edu>
Diffstat (limited to 'fs/ext4')
-rw-r--r-- | fs/ext4/inode.c | 2 | ||||
-rw-r--r-- | fs/ext4/page-io.c | 9 |
2 files changed, 5 insertions, 6 deletions
diff --git a/fs/ext4/inode.c b/fs/ext4/inode.c index 86bf43d6dfcd..07d9defeaf8c 100644 --- a/fs/ext4/inode.c +++ b/fs/ext4/inode.c | |||
@@ -2850,9 +2850,9 @@ static void ext4_end_io_dio(struct kiocb *iocb, loff_t offset, | |||
2850 | if (!(io_end->flag & EXT4_IO_END_UNWRITTEN)) { | 2850 | if (!(io_end->flag & EXT4_IO_END_UNWRITTEN)) { |
2851 | ext4_free_io_end(io_end); | 2851 | ext4_free_io_end(io_end); |
2852 | out: | 2852 | out: |
2853 | inode_dio_done(inode); | ||
2853 | if (is_async) | 2854 | if (is_async) |
2854 | aio_complete(iocb, ret, 0); | 2855 | aio_complete(iocb, ret, 0); |
2855 | inode_dio_done(inode); | ||
2856 | return; | 2856 | return; |
2857 | } | 2857 | } |
2858 | 2858 | ||
diff --git a/fs/ext4/page-io.c b/fs/ext4/page-io.c index 5d8c66948e1b..809b31003ecc 100644 --- a/fs/ext4/page-io.c +++ b/fs/ext4/page-io.c | |||
@@ -102,14 +102,13 @@ static int ext4_end_io(ext4_io_end_t *io) | |||
102 | "(inode %lu, offset %llu, size %zd, error %d)", | 102 | "(inode %lu, offset %llu, size %zd, error %d)", |
103 | inode->i_ino, offset, size, ret); | 103 | inode->i_ino, offset, size, ret); |
104 | } | 104 | } |
105 | if (io->iocb) | ||
106 | aio_complete(io->iocb, io->result, 0); | ||
107 | |||
108 | if (io->flag & EXT4_IO_END_DIRECT) | ||
109 | inode_dio_done(inode); | ||
110 | /* Wake up anyone waiting on unwritten extent conversion */ | 105 | /* Wake up anyone waiting on unwritten extent conversion */ |
111 | if (atomic_dec_and_test(&EXT4_I(inode)->i_unwritten)) | 106 | if (atomic_dec_and_test(&EXT4_I(inode)->i_unwritten)) |
112 | wake_up_all(ext4_ioend_wq(inode)); | 107 | wake_up_all(ext4_ioend_wq(inode)); |
108 | if (io->flag & EXT4_IO_END_DIRECT) | ||
109 | inode_dio_done(inode); | ||
110 | if (io->iocb) | ||
111 | aio_complete(io->iocb, io->result, 0); | ||
113 | return ret; | 112 | return ret; |
114 | } | 113 | } |
115 | 114 | ||