提交 53cbf3b1 编写于 作者: M Ming Lei 提交者: Jens Axboe

fs: direct-io: don't dirtying pages for ITER_BVEC/ITER_KVEC direct read

When direct read IO is submitted from kernel, it is often
unnecessary to dirty pages, for example of loop, dirtying pages
have been considered in the upper filesystem(over loop) side
already, and they don't need to be dirtied again.

So this patch doesn't dirtying pages for ITER_BVEC/ITER_KVEC
direct read, and loop should be the 1st case to use ITER_BVEC/ITER_KVEC
for direct read I/O.

The patch is based on previous Dave's patch.
Reviewed-by: NDave Kleikamp <dave.kleikamp@oracle.com>
Reviewed-by: NChristoph Hellwig <hch@lst.de>
Signed-off-by: NMing Lei <ming.lei@canonical.com>
Signed-off-by: NJens Axboe <axboe@fb.com>
上级 5948edbc
...@@ -120,6 +120,7 @@ struct dio { ...@@ -120,6 +120,7 @@ struct dio {
int page_errors; /* errno from get_user_pages() */ int page_errors; /* errno from get_user_pages() */
int is_async; /* is IO async ? */ int is_async; /* is IO async ? */
bool defer_completion; /* defer AIO completion to workqueue? */ bool defer_completion; /* defer AIO completion to workqueue? */
bool should_dirty; /* if pages should be dirtied */
int io_error; /* IO error in completion path */ int io_error; /* IO error in completion path */
unsigned long refcount; /* direct_io_worker() and bios */ unsigned long refcount; /* direct_io_worker() and bios */
struct bio *bio_list; /* singly linked via bi_private */ struct bio *bio_list; /* singly linked via bi_private */
...@@ -393,7 +394,7 @@ static inline void dio_bio_submit(struct dio *dio, struct dio_submit *sdio) ...@@ -393,7 +394,7 @@ static inline void dio_bio_submit(struct dio *dio, struct dio_submit *sdio)
dio->refcount++; dio->refcount++;
spin_unlock_irqrestore(&dio->bio_lock, flags); spin_unlock_irqrestore(&dio->bio_lock, flags);
if (dio->is_async && dio->rw == READ) if (dio->is_async && dio->rw == READ && dio->should_dirty)
bio_set_pages_dirty(bio); bio_set_pages_dirty(bio);
if (sdio->submit_io) if (sdio->submit_io)
...@@ -464,14 +465,15 @@ static int dio_bio_complete(struct dio *dio, struct bio *bio) ...@@ -464,14 +465,15 @@ static int dio_bio_complete(struct dio *dio, struct bio *bio)
if (bio->bi_error) if (bio->bi_error)
dio->io_error = -EIO; dio->io_error = -EIO;
if (dio->is_async && dio->rw == READ) { if (dio->is_async && dio->rw == READ && dio->should_dirty) {
bio_check_pages_dirty(bio); /* transfers ownership */ bio_check_pages_dirty(bio); /* transfers ownership */
err = bio->bi_error; err = bio->bi_error;
} else { } else {
bio_for_each_segment_all(bvec, bio, i) { bio_for_each_segment_all(bvec, bio, i) {
struct page *page = bvec->bv_page; struct page *page = bvec->bv_page;
if (dio->rw == READ && !PageCompound(page)) if (dio->rw == READ && !PageCompound(page) &&
dio->should_dirty)
set_page_dirty_lock(page); set_page_dirty_lock(page);
page_cache_release(page); page_cache_release(page);
} }
...@@ -1219,6 +1221,7 @@ do_blockdev_direct_IO(struct kiocb *iocb, struct inode *inode, ...@@ -1219,6 +1221,7 @@ do_blockdev_direct_IO(struct kiocb *iocb, struct inode *inode,
spin_lock_init(&dio->bio_lock); spin_lock_init(&dio->bio_lock);
dio->refcount = 1; dio->refcount = 1;
dio->should_dirty = (iter->type == ITER_IOVEC);
sdio.iter = iter; sdio.iter = iter;
sdio.final_block_in_request = sdio.final_block_in_request =
(offset + iov_iter_count(iter)) >> blkbits; (offset + iov_iter_count(iter)) >> blkbits;
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册