| From: Johan Hovold <jhovold@gmail.com> |
| Date: Mon, 1 Jul 2013 14:03:33 +0200 |
| Subject: USB: mos7840: fix memory leak in open |
| |
| commit 5f8a2e68b679b41cc8e9b642f2f5aa45dd678641 upstream. |
| |
| Allocated urbs and buffers were never freed on errors in open. |
| |
| Signed-off-by: Johan Hovold <jhovold@gmail.com> |
| Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org> |
| [bwh: Backported to 3.2: adjust context] |
| Signed-off-by: Ben Hutchings <ben@decadent.org.uk> |
| --- |
| drivers/usb/serial/mos7840.c | 25 +++++++++++++++++-------- |
| 1 file changed, 17 insertions(+), 8 deletions(-) |
| |
| --- a/drivers/usb/serial/mos7840.c |
| +++ b/drivers/usb/serial/mos7840.c |
| @@ -925,20 +925,20 @@ static int mos7840_open(struct tty_struc |
| status = mos7840_get_reg_sync(port, mos7840_port->SpRegOffset, &Data); |
| if (status < 0) { |
| dbg("Reading Spreg failed"); |
| - return -1; |
| + goto err; |
| } |
| Data |= 0x80; |
| status = mos7840_set_reg_sync(port, mos7840_port->SpRegOffset, Data); |
| if (status < 0) { |
| dbg("writing Spreg failed"); |
| - return -1; |
| + goto err; |
| } |
| |
| Data &= ~0x80; |
| status = mos7840_set_reg_sync(port, mos7840_port->SpRegOffset, Data); |
| if (status < 0) { |
| dbg("writing Spreg failed"); |
| - return -1; |
| + goto err; |
| } |
| /* End of block to be checked */ |
| |
| @@ -947,7 +947,7 @@ static int mos7840_open(struct tty_struc |
| &Data); |
| if (status < 0) { |
| dbg("Reading Controlreg failed"); |
| - return -1; |
| + goto err; |
| } |
| Data |= 0x08; /* Driver done bit */ |
| Data |= 0x20; /* rx_disable */ |
| @@ -955,7 +955,7 @@ static int mos7840_open(struct tty_struc |
| mos7840_port->ControlRegOffset, Data); |
| if (status < 0) { |
| dbg("writing Controlreg failed"); |
| - return -1; |
| + goto err; |
| } |
| /* do register settings here */ |
| /* Set all regs to the device default values. */ |
| @@ -966,21 +966,21 @@ static int mos7840_open(struct tty_struc |
| status = mos7840_set_uart_reg(port, INTERRUPT_ENABLE_REGISTER, Data); |
| if (status < 0) { |
| dbg("disabling interrupts failed"); |
| - return -1; |
| + goto err; |
| } |
| /* Set FIFO_CONTROL_REGISTER to the default value */ |
| Data = 0x00; |
| status = mos7840_set_uart_reg(port, FIFO_CONTROL_REGISTER, Data); |
| if (status < 0) { |
| dbg("Writing FIFO_CONTROL_REGISTER failed"); |
| - return -1; |
| + goto err; |
| } |
| |
| Data = 0xcf; |
| status = mos7840_set_uart_reg(port, FIFO_CONTROL_REGISTER, Data); |
| if (status < 0) { |
| dbg("Writing FIFO_CONTROL_REGISTER failed"); |
| - return -1; |
| + goto err; |
| } |
| |
| Data = 0x03; |
| @@ -1136,7 +1136,15 @@ static int mos7840_open(struct tty_struc |
| dbg ("%s leave", __func__); |
| |
| return 0; |
| - |
| +err: |
| + for (j = 0; j < NUM_URBS; ++j) { |
| + urb = mos7840_port->write_urb_pool[j]; |
| + if (!urb) |
| + continue; |
| + kfree(urb->transfer_buffer); |
| + usb_free_urb(urb); |
| + } |
| + return status; |
| } |
| |
| /***************************************************************************** |