Skip to content
Snippets Groups Projects
Commit 6ded703c authored by Jens Axboe's avatar Jens Axboe
Browse files

brd: check for REQ_NOWAIT and set correct page allocation mask


If REQ_NOWAIT is set, then do a non-blocking allocation if the operation
is a write and we need to insert a new page. Currently REQ_NOWAIT cannot
be set as the queue isn't marked as supporting nowait, this change is in
preparation for allowing that.

radix_tree_preload() warns on attempting to call it with an allocation
mask that doesn't allow blocking. While that warning could arguably
be removed, we need to handle radix insertion failures anyway as they
are more likely if we cannot block to get memory.

Remove legacy BUG_ON()'s and turn them into proper errors instead, one
for the allocation failure and one for finding a page that doesn't
match the correct index.

Cc: stable@vger.kernel.org # 5.10+
Reviewed-by: default avatarChristoph Hellwig <hch@lst.de>
Signed-off-by: default avatarJens Axboe <axboe@kernel.dk>
parent db0ccc44
No related branches found
No related tags found
No related merge requests found
...@@ -80,26 +80,21 @@ static struct page *brd_lookup_page(struct brd_device *brd, sector_t sector) ...@@ -80,26 +80,21 @@ static struct page *brd_lookup_page(struct brd_device *brd, sector_t sector)
/* /*
* Insert a new page for a given sector, if one does not already exist. * Insert a new page for a given sector, if one does not already exist.
*/ */
static int brd_insert_page(struct brd_device *brd, sector_t sector) static int brd_insert_page(struct brd_device *brd, sector_t sector, gfp_t gfp)
{ {
pgoff_t idx; pgoff_t idx;
struct page *page; struct page *page;
gfp_t gfp_flags; int ret = 0;
page = brd_lookup_page(brd, sector); page = brd_lookup_page(brd, sector);
if (page) if (page)
return 0; return 0;
/* page = alloc_page(gfp | __GFP_ZERO | __GFP_HIGHMEM);
* Must use NOIO because we don't want to recurse back into the
* block or filesystem layers from page reclaim.
*/
gfp_flags = GFP_NOIO | __GFP_ZERO | __GFP_HIGHMEM;
page = alloc_page(gfp_flags);
if (!page) if (!page)
return -ENOMEM; return -ENOMEM;
if (radix_tree_preload(GFP_NOIO)) { if (gfpflags_allow_blocking(gfp) && radix_tree_preload(gfp)) {
__free_page(page); __free_page(page);
return -ENOMEM; return -ENOMEM;
} }
...@@ -110,15 +105,17 @@ static int brd_insert_page(struct brd_device *brd, sector_t sector) ...@@ -110,15 +105,17 @@ static int brd_insert_page(struct brd_device *brd, sector_t sector)
if (radix_tree_insert(&brd->brd_pages, idx, page)) { if (radix_tree_insert(&brd->brd_pages, idx, page)) {
__free_page(page); __free_page(page);
page = radix_tree_lookup(&brd->brd_pages, idx); page = radix_tree_lookup(&brd->brd_pages, idx);
BUG_ON(!page); if (!page)
BUG_ON(page->index != idx); ret = -ENOMEM;
else if (page->index != idx)
ret = -EIO;
} else { } else {
brd->brd_nr_pages++; brd->brd_nr_pages++;
} }
spin_unlock(&brd->brd_lock); spin_unlock(&brd->brd_lock);
radix_tree_preload_end(); radix_tree_preload_end();
return 0; return ret;
} }
/* /*
...@@ -167,19 +164,20 @@ static void brd_free_pages(struct brd_device *brd) ...@@ -167,19 +164,20 @@ static void brd_free_pages(struct brd_device *brd)
/* /*
* copy_to_brd_setup must be called before copy_to_brd. It may sleep. * copy_to_brd_setup must be called before copy_to_brd. It may sleep.
*/ */
static int copy_to_brd_setup(struct brd_device *brd, sector_t sector, size_t n) static int copy_to_brd_setup(struct brd_device *brd, sector_t sector, size_t n,
gfp_t gfp)
{ {
unsigned int offset = (sector & (PAGE_SECTORS-1)) << SECTOR_SHIFT; unsigned int offset = (sector & (PAGE_SECTORS-1)) << SECTOR_SHIFT;
size_t copy; size_t copy;
int ret; int ret;
copy = min_t(size_t, n, PAGE_SIZE - offset); copy = min_t(size_t, n, PAGE_SIZE - offset);
ret = brd_insert_page(brd, sector); ret = brd_insert_page(brd, sector, gfp);
if (ret) if (ret)
return ret; return ret;
if (copy < n) { if (copy < n) {
sector += copy >> SECTOR_SHIFT; sector += copy >> SECTOR_SHIFT;
ret = brd_insert_page(brd, sector); ret = brd_insert_page(brd, sector, gfp);
} }
return ret; return ret;
} }
...@@ -254,20 +252,26 @@ static void copy_from_brd(void *dst, struct brd_device *brd, ...@@ -254,20 +252,26 @@ static void copy_from_brd(void *dst, struct brd_device *brd,
* Process a single bvec of a bio. * Process a single bvec of a bio.
*/ */
static int brd_do_bvec(struct brd_device *brd, struct page *page, static int brd_do_bvec(struct brd_device *brd, struct page *page,
unsigned int len, unsigned int off, enum req_op op, unsigned int len, unsigned int off, blk_opf_t opf,
sector_t sector) sector_t sector)
{ {
void *mem; void *mem;
int err = 0; int err = 0;
if (op_is_write(op)) { if (op_is_write(opf)) {
err = copy_to_brd_setup(brd, sector, len); /*
* Must use NOIO because we don't want to recurse back into the
* block or filesystem layers from page reclaim.
*/
gfp_t gfp = opf & REQ_NOWAIT ? GFP_NOWAIT : GFP_NOIO;
err = copy_to_brd_setup(brd, sector, len, gfp);
if (err) if (err)
goto out; goto out;
} }
mem = kmap_atomic(page); mem = kmap_atomic(page);
if (!op_is_write(op)) { if (!op_is_write(opf)) {
copy_from_brd(mem + off, brd, sector, len); copy_from_brd(mem + off, brd, sector, len);
flush_dcache_page(page); flush_dcache_page(page);
} else { } else {
...@@ -296,8 +300,12 @@ static void brd_submit_bio(struct bio *bio) ...@@ -296,8 +300,12 @@ static void brd_submit_bio(struct bio *bio)
(len & (SECTOR_SIZE - 1))); (len & (SECTOR_SIZE - 1)));
err = brd_do_bvec(brd, bvec.bv_page, len, bvec.bv_offset, err = brd_do_bvec(brd, bvec.bv_page, len, bvec.bv_offset,
bio_op(bio), sector); bio->bi_opf, sector);
if (err) { if (err) {
if (err == -ENOMEM && bio->bi_opf & REQ_NOWAIT) {
bio_wouldblock_error(bio);
return;
}
bio_io_error(bio); bio_io_error(bio);
return; return;
} }
......
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment