USB: usbfs: properly clean up the as structure on error paths

I notice that the processcompl_compat() function seems to be leaking the
'struct async *as' in the error paths. 

I think that the calling convention is fundamentally buggered. The
caller is the one that did the "reap_as()" to get the as thing, the
caller should be the one to free it too. 

Freeing it in the caller also means that it very clearly always gets
freed, and avoids the need for any "free in the error case too".

From: Linus Torvalds <torvalds@linux-foundation.org>
Cc: Alan Stern <stern@rowland.harvard.edu>
Cc: Marcus Meissner <meissner@suse.de>
Cc: stable <stable@kernel.org>
Signed-off-by: Greg Kroah-Hartman <gregkh@suse.de>
This commit is contained in:
Linus Torvalds 2010-02-16 12:35:07 -08:00 committed by Greg Kroah-Hartman
parent d4a4683ca0
commit ddeee0b2ee
1 changed files with 26 additions and 14 deletions

View File

@ -1334,14 +1334,11 @@ static int processcompl(struct async *as, void __user * __user *arg)
} }
} }
free_async(as);
if (put_user(addr, (void __user * __user *)arg)) if (put_user(addr, (void __user * __user *)arg))
return -EFAULT; return -EFAULT;
return 0; return 0;
err_out: err_out:
free_async(as);
return -EFAULT; return -EFAULT;
} }
@ -1371,8 +1368,11 @@ static struct async *reap_as(struct dev_state *ps)
static int proc_reapurb(struct dev_state *ps, void __user *arg) static int proc_reapurb(struct dev_state *ps, void __user *arg)
{ {
struct async *as = reap_as(ps); struct async *as = reap_as(ps);
if (as) if (as) {
return processcompl(as, (void __user * __user *)arg); int retval = processcompl(as, (void __user * __user *)arg);
free_async(as);
return retval;
}
if (signal_pending(current)) if (signal_pending(current))
return -EINTR; return -EINTR;
return -EIO; return -EIO;
@ -1380,11 +1380,16 @@ static int proc_reapurb(struct dev_state *ps, void __user *arg)
static int proc_reapurbnonblock(struct dev_state *ps, void __user *arg) static int proc_reapurbnonblock(struct dev_state *ps, void __user *arg)
{ {
int retval;
struct async *as; struct async *as;
if (!(as = async_getcompleted(ps))) as = async_getcompleted(ps);
return -EAGAIN; retval = -EAGAIN;
return processcompl(as, (void __user * __user *)arg); if (as) {
retval = processcompl(as, (void __user * __user *)arg);
free_async(as);
}
return retval;
} }
#ifdef CONFIG_COMPAT #ifdef CONFIG_COMPAT
@ -1497,7 +1502,6 @@ static int processcompl_compat(struct async *as, void __user * __user *arg)
} }
} }
free_async(as);
if (put_user(ptr_to_compat(addr), (u32 __user *)arg)) if (put_user(ptr_to_compat(addr), (u32 __user *)arg))
return -EFAULT; return -EFAULT;
return 0; return 0;
@ -1506,8 +1510,11 @@ static int processcompl_compat(struct async *as, void __user * __user *arg)
static int proc_reapurb_compat(struct dev_state *ps, void __user *arg) static int proc_reapurb_compat(struct dev_state *ps, void __user *arg)
{ {
struct async *as = reap_as(ps); struct async *as = reap_as(ps);
if (as) if (as) {
return processcompl_compat(as, (void __user * __user *)arg); int retval = processcompl_compat(as, (void __user * __user *)arg);
free_async(as);
return retval;
}
if (signal_pending(current)) if (signal_pending(current))
return -EINTR; return -EINTR;
return -EIO; return -EIO;
@ -1515,11 +1522,16 @@ static int proc_reapurb_compat(struct dev_state *ps, void __user *arg)
static int proc_reapurbnonblock_compat(struct dev_state *ps, void __user *arg) static int proc_reapurbnonblock_compat(struct dev_state *ps, void __user *arg)
{ {
int retval;
struct async *as; struct async *as;
if (!(as = async_getcompleted(ps))) retval = -EAGAIN;
return -EAGAIN; as = async_getcompleted(ps);
return processcompl_compat(as, (void __user * __user *)arg); if (as) {
retval = processcompl_compat(as, (void __user * __user *)arg);
free_async(as);
}
return retval;
} }