[PREVIEW] [PATCH RESEND 2/4] staging: erofs: introduce erofs_grab_bio

Gao Xiang gaoxiang25 at huawei.com
Fri Aug 10 13:01:26 AEST 2018


Hi Chao,

On 2018/8/10 10:46, Chao Yu wrote:
> On 2018/8/1 14:04, Gao Xiang wrote:
>> this patch renames prepare_bio to erofs_grab_bio, and
>> adds a nofail option in order to retry in the bio allocator.
>>
>> Signed-off-by: Gao Xiang <gaoxiang25 at huawei.com>
>> ---
>>  drivers/staging/erofs/data.c      | 12 ++++++++++--
>>  drivers/staging/erofs/internal.h  | 35 +++++++++++++++++------------------
>>  drivers/staging/erofs/unzip_vle.c |  4 ++--
>>  3 files changed, 29 insertions(+), 22 deletions(-)
>>
>> diff --git a/drivers/staging/erofs/data.c b/drivers/staging/erofs/data.c
>> index ac263a1..2426eda 100644
>> --- a/drivers/staging/erofs/data.c
>> +++ b/drivers/staging/erofs/data.c
>> @@ -60,7 +60,8 @@ struct page *erofs_get_meta_page(struct super_block *sb,
>>  		struct bio *bio;
>>  		int err;
>>  
>> -		bio = prepare_bio(sb, blkaddr, 1, read_endio);
>> +		bio = erofs_grab_bio(sb, blkaddr, 1, read_endio, true);
>> +
>>  		err = bio_add_page(bio, page, PAGE_SIZE, 0);
>>  		BUG_ON(err != PAGE_SIZE);
>>  
>> @@ -278,7 +279,14 @@ static inline struct bio *erofs_read_raw_page(
>>  		if (nblocks > BIO_MAX_PAGES)
>>  			nblocks = BIO_MAX_PAGES;
>>  
>> -		bio = prepare_bio(inode->i_sb, blknr, nblocks, read_endio);
>> +		bio = erofs_grab_bio(inode->i_sb,
>> +			blknr, nblocks, read_endio, false);
>> +
>> +		if (unlikely(IS_ERR(bio))) {
>> +			err = PTR_ERR(bio);
>> +			bio = NULL;
>> +			goto err_out;
>> +		}
>>  	}
>>  
>>  	err = bio_add_page(bio, page, PAGE_SIZE, 0);
>> diff --git a/drivers/staging/erofs/internal.h b/drivers/staging/erofs/internal.h
>> index 4518729..acce2d6 100644
>> --- a/drivers/staging/erofs/internal.h
>> +++ b/drivers/staging/erofs/internal.h
>> @@ -419,26 +419,25 @@ struct erofs_map_blocks {
>>  #define EROFS_GET_BLOCKS_RAW    0x0001
>>  
>>  /* data.c */
>> -static inline struct bio *prepare_bio(
>> -	struct super_block *sb,
>> -	erofs_blk_t blkaddr, unsigned nr_pages,
>> -	bio_end_io_t endio)
>> +static inline struct bio *
>> +erofs_grab_bio(struct super_block *sb,
>> +	       erofs_blk_t blkaddr, unsigned nr_pages,
>> +	       bio_end_io_t endio, bool nofail)
>>  {
>> -	gfp_t gfp = GFP_NOIO;
>> -	struct bio *bio = bio_alloc(gfp, nr_pages);
>> -
>> -	if (unlikely(bio == NULL) &&
>> -		(current->flags & PF_MEMALLOC)) {
>> -		do {
>> -			nr_pages /= 2;
>> -			if (unlikely(!nr_pages)) {
>> -				bio = bio_alloc(gfp | __GFP_NOFAIL, 1);
>> -				BUG_ON(bio == NULL);
>> -				break;
>> +	const gfp_t gfp = GFP_NOIO;
>> +	struct bio *bio;
>> +
>> +	do {
>> +		if (nr_pages == 1) {
>> +			bio = bio_alloc(gfp | (nofail ? __GFP_NOFAIL : 0), 1);
>> +			if (unlikely(bio == NULL)) {
>> +				DBG_BUGON(nofail);
>> +				return ERR_PTR(-ENOMEM);
>>  			}
>> -			bio = bio_alloc(gfp, nr_pages);
>> -		} while (bio == NULL);
>> -	}
>> +		}
>> +		bio = bio_alloc(gfp, nr_pages);
> 
> Need to cover this bio_alloc, so should initialize gfp as:
> 
> gfp_t gfp = GFP_NOIO | (nofail ? __GFP_NOFAIL : 0);
> 
> Thanks,
This patch means if bio = bio_alloc(gfp, nr_pages); fails, try to bio_alloc with nr_pages/2, nr_page/4... until 1...
and tag __GFP_NOFAIL with 1...

As you said offline, there is another bug :-(, let me send the next patch...

Thanks,
Gao Xiang


More information about the Linux-erofs mailing list