mirror of https://gitee.com/openkylin/linux.git
-----BEGIN PGP SIGNATURE----- iQEzBAABCAAdFiEEq1nRK9aeMoq1VSgcnJ2qBz9kQNkFAl+IUDwACgkQnJ2qBz9k QNnjoAf8CVCf/N2kblZywcy/Ks/2WzlZDU4S/dVr93O2VLXEb6kT+onZq4tc/+sY QIV8kUMGvt0Ez/KpwY6/wOUVEPZn8bSqM0vUsB+Xi8CwSz/CwcDKTbU/tm1667J0 eSFX96OL2hT5BockTzuESFohtykiPU3CvW10ae02N6k6XKTS0+frkn3cebJKno2O NDt0j2HkVjvpO6DqS7lnceFxo4t6DV6YXJHMtkJqRk8hYItzjyjOB6Df+1WHJJOA 4jxlobY4qR2MHh1y2ncryPvvQ91sMIxx1beMMXXUIXr+8Zu8aFVWBLgSwyoxBgRP G+ctJu3zMCmeeGnkpLfSlJK9ipYUMw== =nXw4 -----END PGP SIGNATURE----- Merge tag 'dio_for_v5.10-rc1' of git://git.kernel.org/pub/scm/linux/kernel/git/jack/linux-fs Pull direct-io fix from Jan Kara: "Fix for unaligned direct IO read past EOF in legacy DIO code" * tag 'dio_for_v5.10-rc1' of git://git.kernel.org/pub/scm/linux/kernel/git/jack/linux-fs: direct-io: defer alignment check until after the EOF check direct-io: don't force writeback for reads beyond EOF direct-io: clean up error paths of do_blockdev_direct_IO
This commit is contained in:
commit
4a165feba2
|
@ -1146,22 +1146,13 @@ do_blockdev_direct_IO(struct kiocb *iocb, struct inode *inode,
|
|||
* the early prefetch in the caller enough time.
|
||||
*/
|
||||
|
||||
if (align & blocksize_mask) {
|
||||
if (bdev)
|
||||
blkbits = blksize_bits(bdev_logical_block_size(bdev));
|
||||
blocksize_mask = (1 << blkbits) - 1;
|
||||
if (align & blocksize_mask)
|
||||
goto out;
|
||||
}
|
||||
|
||||
/* watch out for a 0 len io from a tricksy fs */
|
||||
if (iov_iter_rw(iter) == READ && !count)
|
||||
return 0;
|
||||
|
||||
dio = kmem_cache_alloc(dio_cache, GFP_KERNEL);
|
||||
retval = -ENOMEM;
|
||||
if (!dio)
|
||||
goto out;
|
||||
return -ENOMEM;
|
||||
/*
|
||||
* Believe it or not, zeroing out the page array caused a .5%
|
||||
* performance regression in a database benchmark. So, we take
|
||||
|
@ -1170,32 +1161,32 @@ do_blockdev_direct_IO(struct kiocb *iocb, struct inode *inode,
|
|||
memset(dio, 0, offsetof(struct dio, pages));
|
||||
|
||||
dio->flags = flags;
|
||||
if (dio->flags & DIO_LOCKING) {
|
||||
if (iov_iter_rw(iter) == READ) {
|
||||
struct address_space *mapping =
|
||||
iocb->ki_filp->f_mapping;
|
||||
|
||||
/* will be released by direct_io_worker */
|
||||
inode_lock(inode);
|
||||
|
||||
retval = filemap_write_and_wait_range(mapping, offset,
|
||||
end - 1);
|
||||
if (retval) {
|
||||
inode_unlock(inode);
|
||||
kmem_cache_free(dio_cache, dio);
|
||||
goto out;
|
||||
}
|
||||
}
|
||||
if (dio->flags & DIO_LOCKING && iov_iter_rw(iter) == READ) {
|
||||
/* will be released by direct_io_worker */
|
||||
inode_lock(inode);
|
||||
}
|
||||
|
||||
/* Once we sampled i_size check for reads beyond EOF */
|
||||
dio->i_size = i_size_read(inode);
|
||||
if (iov_iter_rw(iter) == READ && offset >= dio->i_size) {
|
||||
if (dio->flags & DIO_LOCKING)
|
||||
inode_unlock(inode);
|
||||
kmem_cache_free(dio_cache, dio);
|
||||
retval = 0;
|
||||
goto out;
|
||||
goto fail_dio;
|
||||
}
|
||||
|
||||
if (align & blocksize_mask) {
|
||||
if (bdev)
|
||||
blkbits = blksize_bits(bdev_logical_block_size(bdev));
|
||||
blocksize_mask = (1 << blkbits) - 1;
|
||||
if (align & blocksize_mask)
|
||||
goto fail_dio;
|
||||
}
|
||||
|
||||
if (dio->flags & DIO_LOCKING && iov_iter_rw(iter) == READ) {
|
||||
struct address_space *mapping = iocb->ki_filp->f_mapping;
|
||||
|
||||
retval = filemap_write_and_wait_range(mapping, offset, end - 1);
|
||||
if (retval)
|
||||
goto fail_dio;
|
||||
}
|
||||
|
||||
/*
|
||||
|
@ -1239,14 +1230,8 @@ do_blockdev_direct_IO(struct kiocb *iocb, struct inode *inode,
|
|||
*/
|
||||
retval = sb_init_dio_done_wq(dio->inode->i_sb);
|
||||
}
|
||||
if (retval) {
|
||||
/*
|
||||
* We grab i_mutex only for reads so we don't have
|
||||
* to release it here
|
||||
*/
|
||||
kmem_cache_free(dio_cache, dio);
|
||||
goto out;
|
||||
}
|
||||
if (retval)
|
||||
goto fail_dio;
|
||||
}
|
||||
|
||||
/*
|
||||
|
@ -1349,7 +1334,13 @@ do_blockdev_direct_IO(struct kiocb *iocb, struct inode *inode,
|
|||
} else
|
||||
BUG_ON(retval != -EIOCBQUEUED);
|
||||
|
||||
out:
|
||||
return retval;
|
||||
|
||||
fail_dio:
|
||||
if (dio->flags & DIO_LOCKING && iov_iter_rw(iter) == READ)
|
||||
inode_unlock(inode);
|
||||
|
||||
kmem_cache_free(dio_cache, dio);
|
||||
return retval;
|
||||
}
|
||||
|
||||
|
|
Loading…
Reference in New Issue