Commit 093ea2d3 authored by Donald Lee's avatar Donald Lee Committed by Greg Kroah-Hartman

USB: serial: mos7840: Fixed MCS7820 device attach problem

A MCS7820 device supports two serial ports and a MCS7840 device supports
four serial ports. Both devices use the same driver, but the attach function
in driver was unable to correctly handle the port numbers for MCS7820
device. This problem has been fixed in this patch and this fix has been
verified on x86 Linux kernel 3.2.9 with both MCS7820 and MCS7840 devices.
Signed-off-by: default avatarDonald Lee <donald@asix.com.tw>
Cc: stable <stable@vger.kernel.org>
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@linuxfoundation.org>
parent 53c6bc24
...@@ -174,6 +174,7 @@ ...@@ -174,6 +174,7 @@
#define CLK_MULTI_REGISTER ((__u16)(0x02)) #define CLK_MULTI_REGISTER ((__u16)(0x02))
#define CLK_START_VALUE_REGISTER ((__u16)(0x03)) #define CLK_START_VALUE_REGISTER ((__u16)(0x03))
#define GPIO_REGISTER ((__u16)(0x07))
#define SERIAL_LCR_DLAB ((__u16)(0x0080)) #define SERIAL_LCR_DLAB ((__u16)(0x0080))
...@@ -1101,14 +1102,25 @@ static int mos7840_open(struct tty_struct *tty, struct usb_serial_port *port) ...@@ -1101,14 +1102,25 @@ static int mos7840_open(struct tty_struct *tty, struct usb_serial_port *port)
mos7840_port->read_urb = port->read_urb; mos7840_port->read_urb = port->read_urb;
/* set up our bulk in urb */ /* set up our bulk in urb */
if ((serial->num_ports == 2)
usb_fill_bulk_urb(mos7840_port->read_urb, && ((((__u16)port->number -
serial->dev, (__u16)(port->serial->minor)) % 2) != 0)) {
usb_rcvbulkpipe(serial->dev, usb_fill_bulk_urb(mos7840_port->read_urb,
port->bulk_in_endpointAddress), serial->dev,
port->bulk_in_buffer, usb_rcvbulkpipe(serial->dev,
mos7840_port->read_urb->transfer_buffer_length, (port->bulk_in_endpointAddress) + 2),
mos7840_bulk_in_callback, mos7840_port); port->bulk_in_buffer,
mos7840_port->read_urb->transfer_buffer_length,
mos7840_bulk_in_callback, mos7840_port);
} else {
usb_fill_bulk_urb(mos7840_port->read_urb,
serial->dev,
usb_rcvbulkpipe(serial->dev,
port->bulk_in_endpointAddress),
port->bulk_in_buffer,
mos7840_port->read_urb->transfer_buffer_length,
mos7840_bulk_in_callback, mos7840_port);
}
dbg("mos7840_open: bulkin endpoint is %d", dbg("mos7840_open: bulkin endpoint is %d",
port->bulk_in_endpointAddress); port->bulk_in_endpointAddress);
...@@ -1519,13 +1531,25 @@ static int mos7840_write(struct tty_struct *tty, struct usb_serial_port *port, ...@@ -1519,13 +1531,25 @@ static int mos7840_write(struct tty_struct *tty, struct usb_serial_port *port,
memcpy(urb->transfer_buffer, current_position, transfer_size); memcpy(urb->transfer_buffer, current_position, transfer_size);
/* fill urb with data and submit */ /* fill urb with data and submit */
usb_fill_bulk_urb(urb, if ((serial->num_ports == 2)
serial->dev, && ((((__u16)port->number -
usb_sndbulkpipe(serial->dev, (__u16)(port->serial->minor)) % 2) != 0)) {
port->bulk_out_endpointAddress), usb_fill_bulk_urb(urb,
urb->transfer_buffer, serial->dev,
transfer_size, usb_sndbulkpipe(serial->dev,
mos7840_bulk_out_data_callback, mos7840_port); (port->bulk_out_endpointAddress) + 2),
urb->transfer_buffer,
transfer_size,
mos7840_bulk_out_data_callback, mos7840_port);
} else {
usb_fill_bulk_urb(urb,
serial->dev,
usb_sndbulkpipe(serial->dev,
port->bulk_out_endpointAddress),
urb->transfer_buffer,
transfer_size,
mos7840_bulk_out_data_callback, mos7840_port);
}
data1 = urb->transfer_buffer; data1 = urb->transfer_buffer;
dbg("bulkout endpoint is %d", port->bulk_out_endpointAddress); dbg("bulkout endpoint is %d", port->bulk_out_endpointAddress);
...@@ -1838,7 +1862,7 @@ static int mos7840_send_cmd_write_baud_rate(struct moschip_port *mos7840_port, ...@@ -1838,7 +1862,7 @@ static int mos7840_send_cmd_write_baud_rate(struct moschip_port *mos7840_port,
} else { } else {
#ifdef HW_flow_control #ifdef HW_flow_control
/ *setting h/w flow control bit to 0 */ /* setting h/w flow control bit to 0 */
Data = 0xb; Data = 0xb;
mos7840_port->shadowMCR = Data; mos7840_port->shadowMCR = Data;
status = mos7840_set_uart_reg(port, MODEM_CONTROL_REGISTER, status = mos7840_set_uart_reg(port, MODEM_CONTROL_REGISTER,
...@@ -2305,19 +2329,26 @@ static int mos7840_ioctl(struct tty_struct *tty, ...@@ -2305,19 +2329,26 @@ static int mos7840_ioctl(struct tty_struct *tty,
static int mos7840_calc_num_ports(struct usb_serial *serial) static int mos7840_calc_num_ports(struct usb_serial *serial)
{ {
int mos7840_num_ports = 0; __u16 Data = 0x00;
int ret = 0;
dbg("numberofendpoints: cur %d, alt %d", int mos7840_num_ports;
(int)serial->interface->cur_altsetting->desc.bNumEndpoints,
(int)serial->interface->altsetting->desc.bNumEndpoints); ret = usb_control_msg(serial->dev, usb_rcvctrlpipe(serial->dev, 0),
if (serial->interface->cur_altsetting->desc.bNumEndpoints == 5) { MCS_RDREQ, MCS_RD_RTYPE, 0, GPIO_REGISTER, &Data,
mos7840_num_ports = serial->num_ports = 2; VENDOR_READ_LENGTH, MOS_WDR_TIMEOUT);
} else if (serial->interface->cur_altsetting->desc.bNumEndpoints == 9) {
if ((Data & 0x01) == 0) {
mos7840_num_ports = 2;
serial->num_bulk_in = 2;
serial->num_bulk_out = 2;
serial->num_ports = 2;
} else {
mos7840_num_ports = 4;
serial->num_bulk_in = 4; serial->num_bulk_in = 4;
serial->num_bulk_out = 4; serial->num_bulk_out = 4;
mos7840_num_ports = serial->num_ports = 4; serial->num_ports = 4;
} }
dbg ("mos7840_num_ports = %d", mos7840_num_ports);
return mos7840_num_ports; return mos7840_num_ports;
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment