Re: allocation under spinlock

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

 



On Wed, Feb 24, 2010 at 06:31:49PM -0800, Trond Myklebust wrote:
> On Wed, 2010-02-24 at 16:23 -0500, J. Bruce Fields wrote: 
> > nfs_dns_ent_init() is trying to kstrndup() something, but it's called under a
> > spinlock.
> > 
> > Could refernce count that string instead, but other code I believe actually
> > just does something like:
> > 
> > 	new->hostname = key->hostname;
> > 	key->hostname = NULL;
> > 
> > --b.
> > 
> > BUG: sleeping function called from invalid context at mm/slab.c:3034
> > in_atomic(): 1, irqs_disabled(): 0, pid: 3300, name: nfs_cache_geten
> > 2 locks held by nfs_cache_geten/3300:
> >  #0:  (&sb->s_type->i_mutex_key#4){+.+.+.}, at: [<c1795506>] cache_write+0x46/0x100
> >  #1:  (&cd->hash_lock){+++++.}, at: [<c17964a5>] sunrpc_cache_update+0x25/0x170
> > Pid: 3300, comm: nfs_cache_geten Not tainted 2.6.33-rc8-13669-g724e6d3 #156
> 
> Does the following fix it for you?

No more warning; thanks!

--b.

> 
> Cheers
>   Trond
> ---------------------------------------------------------------------------- 
> NFS: Fix an allocation-under-spinlock bug
> 
> From: Trond Myklebust <Trond.Myklebust@xxxxxxxxxx>
> 
> sunrpc_cache_update() will always call detail->update() from inside the
> detail->hash_lock, so it cannot allocate memory.
> 
> Signed-off-by: Trond Myklebust <Trond.Myklebust@xxxxxxxxxx>
> Cc: stable@xxxxxxxxxx
> ---
> 
>  fs/nfs/dns_resolve.c |   18 +++++++++++++++---
>  1 files changed, 15 insertions(+), 3 deletions(-)
> 
> 
> diff --git a/fs/nfs/dns_resolve.c b/fs/nfs/dns_resolve.c
> index 95e1ca7..3f0cd4d 100644
> --- a/fs/nfs/dns_resolve.c
> +++ b/fs/nfs/dns_resolve.c
> @@ -36,6 +36,19 @@ struct nfs_dns_ent {
>  };
>  
> 
> +static void nfs_dns_ent_update(struct cache_head *cnew,
> +		struct cache_head *ckey)
> +{
> +	struct nfs_dns_ent *new;
> +	struct nfs_dns_ent *key;
> +
> +	new = container_of(cnew, struct nfs_dns_ent, h);
> +	key = container_of(ckey, struct nfs_dns_ent, h);
> +
> +	memcpy(&new->addr, &key->addr, key->addrlen);
> +	new->addrlen = key->addrlen;
> +}
> +
>  static void nfs_dns_ent_init(struct cache_head *cnew,
>  		struct cache_head *ckey)
>  {
> @@ -49,8 +62,7 @@ static void nfs_dns_ent_init(struct cache_head *cnew,
>  	new->hostname = kstrndup(key->hostname, key->namelen, GFP_KERNEL);
>  	if (new->hostname) {
>  		new->namelen = key->namelen;
> -		memcpy(&new->addr, &key->addr, key->addrlen);
> -		new->addrlen = key->addrlen;
> +		nfs_dns_ent_update(cnew, ckey);
>  	} else {
>  		new->namelen = 0;
>  		new->addrlen = 0;
> @@ -234,7 +246,7 @@ static struct cache_detail nfs_dns_resolve = {
>  	.cache_show = nfs_dns_show,
>  	.match = nfs_dns_match,
>  	.init = nfs_dns_ent_init,
> -	.update = nfs_dns_ent_init,
> +	.update = nfs_dns_ent_update,
>  	.alloc = nfs_dns_ent_alloc,
>  };
>  
> 
--
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