Re: [PATCH] NFSv4: Fix up mirror allocation

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



Looks good to me.

Reviewed-by: Weston Andros Adamson <dros@xxxxxxxxxxxxxxx>

-dros

> On Aug 20, 2017, at 10:32 AM, Trond Myklebust <trond.myklebust@xxxxxxxxxxxxxxx> wrote:
> 
> There are a number of callers of nfs_pageio_complete() that want to
> continue using the nfs_pageio_descriptor without needing to call
> nfs_pageio_init() again. Examples include nfs_pageio_resend() and
> nfs_pageio_cond_complete().
> 
> The problem is that nfs_pageio_complete() also calls
> nfs_pageio_cleanup_mirroring(), which frees up the array of mirrors.
> This can lead to writeback errors, in the next call to
> nfs_pageio_setup_mirroring().
> 
> Fix by simply moving the allocation of the mirrors to
> nfs_pageio_setup_mirroring().
> 
> Link: https://bugzilla.kernel.org/show_bug.cgi?id=196709
> Reported-by: JianhongYin <yin-jianhong@xxxxxxx>
> Cc: stable@xxxxxxxxxxxxxxx # 4.0+
> Signed-off-by: Trond Myklebust <trond.myklebust@xxxxxxxxxxxxxxx>
> ---
> fs/nfs/pagelist.c | 73 +++++++++++++++++++++++++++++--------------------------
> 1 file changed, 39 insertions(+), 34 deletions(-)
> 
> diff --git a/fs/nfs/pagelist.c b/fs/nfs/pagelist.c
> index 548ebc7256ff..1c7e625824dc 100644
> --- a/fs/nfs/pagelist.c
> +++ b/fs/nfs/pagelist.c
> @@ -685,9 +685,6 @@ void nfs_pageio_init(struct nfs_pageio_descriptor *desc,
> 		     int io_flags,
> 		     gfp_t gfp_flags)
> {
> -	struct nfs_pgio_mirror *new;
> -	int i;
> -
> 	desc->pg_moreio = 0;
> 	desc->pg_inode = inode;
> 	desc->pg_ops = pg_ops;
> @@ -703,21 +700,9 @@ void nfs_pageio_init(struct nfs_pageio_descriptor *desc,
> 	desc->pg_mirror_count = 1;
> 	desc->pg_mirror_idx = 0;
> 
> -	if (pg_ops->pg_get_mirror_count) {
> -		/* until we have a request, we don't have an lseg and no
> -		 * idea how many mirrors there will be */
> -		new = kcalloc(NFS_PAGEIO_DESCRIPTOR_MIRROR_MAX,
> -			      sizeof(struct nfs_pgio_mirror), gfp_flags);
> -		desc->pg_mirrors_dynamic = new;
> -		desc->pg_mirrors = new;
> -
> -		for (i = 0; i < NFS_PAGEIO_DESCRIPTOR_MIRROR_MAX; i++)
> -			nfs_pageio_mirror_init(&desc->pg_mirrors[i], bsize);
> -	} else {
> -		desc->pg_mirrors_dynamic = NULL;
> -		desc->pg_mirrors = desc->pg_mirrors_static;
> -		nfs_pageio_mirror_init(&desc->pg_mirrors[0], bsize);
> -	}
> +	desc->pg_mirrors_dynamic = NULL;
> +	desc->pg_mirrors = desc->pg_mirrors_static;
> +	nfs_pageio_mirror_init(&desc->pg_mirrors[0], bsize);
> }
> EXPORT_SYMBOL_GPL(nfs_pageio_init);
> 
> @@ -836,32 +821,52 @@ static int nfs_generic_pg_pgios(struct nfs_pageio_descriptor *desc)
> 	return ret;
> }
> 
> +static struct nfs_pgio_mirror *
> +nfs_pageio_alloc_mirrors(struct nfs_pageio_descriptor *desc,
> +		unsigned int mirror_count)
> +{
> +	struct nfs_pgio_mirror *ret;
> +	unsigned int i;
> +
> +	kfree(desc->pg_mirrors_dynamic);
> +	desc->pg_mirrors_dynamic = NULL;
> +	if (mirror_count == 1)
> +		return desc->pg_mirrors_static;
> +	ret = kmalloc_array(mirror_count, sizeof(*ret), GFP_NOFS);
> +	if (ret != NULL) {
> +		for (i = 0; i < mirror_count; i++)
> +			nfs_pageio_mirror_init(&ret[i], desc->pg_bsize);
> +		desc->pg_mirrors_dynamic = ret;
> +	}
> +	return ret;
> +}
> +
> /*
>  * nfs_pageio_setup_mirroring - determine if mirroring is to be used
>  *				by calling the pg_get_mirror_count op
>  */
> -static int nfs_pageio_setup_mirroring(struct nfs_pageio_descriptor *pgio,
> +static void nfs_pageio_setup_mirroring(struct nfs_pageio_descriptor *pgio,
> 				       struct nfs_page *req)
> {
> -	int mirror_count = 1;
> +	unsigned int mirror_count = 1;
> 
> -	if (!pgio->pg_ops->pg_get_mirror_count)
> -		return 0;
> -
> -	mirror_count = pgio->pg_ops->pg_get_mirror_count(pgio, req);
> -
> -	if (pgio->pg_error < 0)
> -		return pgio->pg_error;
> -
> -	if (!mirror_count || mirror_count > NFS_PAGEIO_DESCRIPTOR_MIRROR_MAX)
> -		return -EINVAL;
> +	if (pgio->pg_ops->pg_get_mirror_count)
> +		mirror_count = pgio->pg_ops->pg_get_mirror_count(pgio, req);
> +	if (mirror_count == pgio->pg_mirror_count || pgio->pg_error < 0)
> +		return;
> 
> -	if (WARN_ON_ONCE(!pgio->pg_mirrors_dynamic))
> -		return -EINVAL;
> +	if (!mirror_count || mirror_count > NFS_PAGEIO_DESCRIPTOR_MIRROR_MAX) {
> +		pgio->pg_error = -EINVAL;
> +		return;
> +	}
> 
> +	pgio->pg_mirrors = nfs_pageio_alloc_mirrors(pgio, mirror_count);
> +	if (pgio->pg_mirrors == NULL) {
> +		pgio->pg_error = -ENOMEM;
> +		pgio->pg_mirrors = pgio->pg_mirrors_static;
> +		mirror_count = 1;
> +	}
> 	pgio->pg_mirror_count = mirror_count;
> -
> -	return 0;
> }
> 
> /*
> -- 
> 2.13.5
> 
> --
> To unsubscribe from this list: send the line "unsubscribe linux-nfs" in
> the body of a message to majordomo@xxxxxxxxxxxxxxx
> More majordomo info at  http://vger.kernel.org/majordomo-info.html

--
To unsubscribe from this list: send the line "unsubscribe linux-nfs" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at  http://vger.kernel.org/majordomo-info.html




[Index of Archives]     [Linux Filesystem Development]     [Linux USB Development]     [Linux Media Development]     [Video for Linux]     [Linux NILFS]     [Linux Audio Users]     [Yosemite Info]     [Linux SCSI]

  Powered by Linux