Commit 324bcf54 authored by Jens Axboe's avatar Jens Axboe

mm: use limited read-ahead to satisfy read

For the case where read-ahead is disabled on the file, or if the cgroup
is congested, ensure that we can at least do 1 page of read-ahead to
make progress on the read in an async fashion. This could potentially be
larger, but it's not needed in terms of functionality, so let's error on
the side of caution as larger counts of pages may run into reclaim
issues (particularly if we're congested).

This makes sure we're not hitting the potentially sync ->readpage() path
for IO that is marked IOCB_WAITQ, which could cause us to block. It also
means we'll use the same path for IO, regardless of whether or not
read-ahead happens to be disabled on the lower level device.
Acked-by: default avatarJohannes Weiner <hannes@cmpxchg.org>
Reported-by: default avatarMatthew Wilcox (Oracle) <willy@infradead.org>
Reported-by: default avatarHao_Xu <haoxu@linux.alibaba.com>
[axboe: updated for new ractl API]
Signed-off-by: default avatarJens Axboe <axboe@kernel.dk>
parent 13bd6914
...@@ -552,15 +552,23 @@ static void ondemand_readahead(struct readahead_control *ractl, ...@@ -552,15 +552,23 @@ static void ondemand_readahead(struct readahead_control *ractl,
void page_cache_sync_ra(struct readahead_control *ractl, void page_cache_sync_ra(struct readahead_control *ractl,
struct file_ra_state *ra, unsigned long req_count) struct file_ra_state *ra, unsigned long req_count)
{ {
/* no read-ahead */ bool do_forced_ra = ractl->file && (ractl->file->f_mode & FMODE_RANDOM);
if (!ra->ra_pages)
return;
if (blk_cgroup_congested()) /*
* Even if read-ahead is disabled, issue this request as read-ahead
* as we'll need it to satisfy the requested range. The forced
* read-ahead will do the right thing and limit the read to just the
* requested range, which we'll set to 1 page for this case.
*/
if (!ra->ra_pages || blk_cgroup_congested()) {
if (!ractl->file)
return; return;
req_count = 1;
do_forced_ra = true;
}
/* be dumb */ /* be dumb */
if (ractl->file && (ractl->file->f_mode & FMODE_RANDOM)) { if (do_forced_ra) {
force_page_cache_ra(ractl, ra, req_count); force_page_cache_ra(ractl, ra, req_count);
return; return;
} }
......
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