Browse Source

usb: qcserial avoid pointing to freed memory

Rework the qcprobe logic such that serial->private is not set when
qcprobe exits with -ENODEV, otherwise serial->private will point to freed
memory on -ENODEV

Signed-off-by: Steven Hardy <shardy@redhat.com>
Cc: stable <stable@kernel.org>
Signed-off-by: Greg Kroah-Hartman <gregkh@suse.de>
Steven Hardy 14 years ago
parent
commit
99ab3f9e4e
1 changed files with 9 additions and 6 deletions
  1. 9 6
      drivers/usb/serial/qcserial.c

+ 9 - 6
drivers/usb/serial/qcserial.c

@@ -111,7 +111,7 @@ static int qcprobe(struct usb_serial *serial, const struct usb_device_id *id)
 	ifnum = intf->desc.bInterfaceNumber;
 	ifnum = intf->desc.bInterfaceNumber;
 	dbg("This Interface = %d", ifnum);
 	dbg("This Interface = %d", ifnum);
 
 
-	data = serial->private = kzalloc(sizeof(struct usb_wwan_intf_private),
+	data = kzalloc(sizeof(struct usb_wwan_intf_private),
 					 GFP_KERNEL);
 					 GFP_KERNEL);
 	if (!data)
 	if (!data)
 		return -ENOMEM;
 		return -ENOMEM;
@@ -134,8 +134,10 @@ static int qcprobe(struct usb_serial *serial, const struct usb_device_id *id)
 		    usb_endpoint_is_bulk_out(&intf->endpoint[1].desc)) {
 		    usb_endpoint_is_bulk_out(&intf->endpoint[1].desc)) {
 			dbg("QDL port found");
 			dbg("QDL port found");
 
 
-			if (serial->interface->num_altsetting == 1)
-				return 0;
+			if (serial->interface->num_altsetting == 1) {
+				retval = 0; /* Success */
+				break;
+			}
 
 
 			retval = usb_set_interface(serial->dev, ifnum, 1);
 			retval = usb_set_interface(serial->dev, ifnum, 1);
 			if (retval < 0) {
 			if (retval < 0) {
@@ -145,7 +147,6 @@ static int qcprobe(struct usb_serial *serial, const struct usb_device_id *id)
 				retval = -ENODEV;
 				retval = -ENODEV;
 				kfree(data);
 				kfree(data);
 			}
 			}
-			return retval;
 		}
 		}
 		break;
 		break;
 
 
@@ -177,7 +178,6 @@ static int qcprobe(struct usb_serial *serial, const struct usb_device_id *id)
 				retval = -ENODEV;
 				retval = -ENODEV;
 				kfree(data);
 				kfree(data);
 			}
 			}
-			return retval;
 		} else if (ifnum==3) {
 		} else if (ifnum==3) {
 			/*
 			/*
 			 * NMEA (serial line 9600 8N1)
 			 * NMEA (serial line 9600 8N1)
@@ -199,9 +199,12 @@ static int qcprobe(struct usb_serial *serial, const struct usb_device_id *id)
 		dev_err(&serial->dev->dev,
 		dev_err(&serial->dev->dev,
 			"unknown number of interfaces: %d\n", nintf);
 			"unknown number of interfaces: %d\n", nintf);
 		kfree(data);
 		kfree(data);
-		return -ENODEV;
+		retval = -ENODEV;
 	}
 	}
 
 
+	/* Set serial->private if not returning -ENODEV */
+	if (retval != -ENODEV)
+		usb_set_serial_data(serial, data);
 	return retval;
 	return retval;
 }
 }