On 2018/8/13 19:04, Dan Carpenter wrote: > On Sun, Aug 12, 2018 at 10:01:44PM +0800, Chao Yu wrote: >> --- a/drivers/staging/erofs/data.c >> +++ b/drivers/staging/erofs/data.c >> @@ -39,31 +39,44 @@ static inline void read_endio(struct bio *bio) >> } >> >> /* prio -- true is used for dir */ >> -struct page *erofs_get_meta_page(struct super_block *sb, >> - erofs_blk_t blkaddr, bool prio) >> +struct page *__erofs_get_meta_page(struct super_block *sb, >> + erofs_blk_t blkaddr, bool prio, bool nofail) >> { >> - struct inode *bd_inode = sb->s_bdev->bd_inode; >> - struct address_space *mapping = bd_inode->i_mapping; >> + struct inode *const bd_inode = sb->s_bdev->bd_inode; >> + struct address_space *const mapping = bd_inode->i_mapping; >> + /* prefer retrying in the allocator to blindly looping below */ >> + const gfp_t gfp = mapping_gfp_constraint(mapping, ~__GFP_FS) | >> + (nofail ? __GFP_NOFAIL : 0); >> + unsigned int io_retries = nofail ? EROFS_IO_MAX_RETRIES_NOFAIL : 0; >> struct page *page; >> >> repeat: >> - page = find_or_create_page(mapping, blkaddr, >> - /* >> - * Prefer looping in the allocator rather than here, >> - * at least that code knows what it's doing. >> - */ >> - mapping_gfp_constraint(mapping, ~__GFP_FS) | __GFP_NOFAIL); >> - >> - BUG_ON(!page || !PageLocked(page)); >> + page = find_or_create_page(mapping, blkaddr, gfp); >> + if (unlikely(page == NULL)) { >> + DBG_BUGON(nofail); >> + return ERR_PTR(-ENOMEM); >> + } >> + DBG_BUGON(!PageLocked(page)); >> >> if (!PageUptodate(page)) { >> struct bio *bio; >> int err; >> >> - bio = erofs_grab_bio(sb, blkaddr, 1, read_endio, true); >> + bio = erofs_grab_bio(sb, blkaddr, 1, read_endio, nofail); >> + if (unlikely(bio == NULL)) { >> + DBG_BUGON(nofail); >> + err = -ENOMEM; >> +err_out: >> + unlock_page(page); >> + put_page(page); >> + return ERR_PTR(err); > > > Put this err_out stuff at the bottom of the function so that we don't > have to do backward hops to get to it. Agreed, we can move error path at the bottom of this function. > >> + } >> >> err = bio_add_page(bio, page, PAGE_SIZE, 0); >> - BUG_ON(err != PAGE_SIZE); >> + if (unlikely(err != PAGE_SIZE)) { >> + err = -EFAULT; >> + goto err_out; > ^^^^^^^^^^^^ > Like this. Generally avoid backwards hops if you can. > >> + } >> >> __submit_bio(bio, REQ_OP_READ, >> REQ_META | (prio ? REQ_PRIO : 0)); >> @@ -72,6 +85,7 @@ struct page *erofs_get_meta_page(struct super_block *sb, >> >> /* the page has been truncated by others? */ >> if (unlikely(page->mapping != mapping)) { >> +unlock_repeat: > > The question mark in the "truncated by others?" is a little concerning. Yup, we can remove the question mark. > It's slightly weird that we don't check "io_retries" on this path. We don't need to cover this path since io_retries is used for accounting retry time only when IO occurs. Thanks, > >> unlock_page(page); >> put_page(page); >> goto repeat; >> @@ -79,10 +93,12 @@ struct page *erofs_get_meta_page(struct super_block *sb, >> >> /* more likely a read error */ >> if (unlikely(!PageUptodate(page))) { >> - unlock_page(page); >> - put_page(page); >> - >> - page = ERR_PTR(-EIO); >> + if (io_retries) { >> + --io_retries; >> + goto unlock_repeat; >> + } >> + err = -EIO; >> + goto err_out; > > regards, > dan carpenter > _______________________________________________ devel mailing list devel@xxxxxxxxxxxxxxxxxxxxxx http://driverdev.linuxdriverproject.org/mailman/listinfo/driverdev-devel