aboutsummaryrefslogtreecommitdiffstats
path: root/drivers
diff options
context:
space:
mode:
Diffstat (limited to 'drivers')
-rw-r--r--drivers/usb/serial/pl2303.c570
1 files changed, 269 insertions, 301 deletions
diff --git a/drivers/usb/serial/pl2303.c b/drivers/usb/serial/pl2303.c
index 0cd42bf4c6c8..e2d8f9b94eed 100644
--- a/drivers/usb/serial/pl2303.c
+++ b/drivers/usb/serial/pl2303.c
@@ -127,65 +127,6 @@ static struct usb_driver pl2303_driver = {
127#define UART_OVERRUN_ERROR 0x40 127#define UART_OVERRUN_ERROR 0x40
128#define UART_CTS 0x80 128#define UART_CTS 0x80
129 129
130/* function prototypes for a PL2303 serial converter */
131static int pl2303_open (struct usb_serial_port *port, struct file *filp);
132static void pl2303_close (struct usb_serial_port *port, struct file *filp);
133static void pl2303_set_termios (struct usb_serial_port *port,
134 struct termios *old);
135static int pl2303_ioctl (struct usb_serial_port *port, struct file *file,
136 unsigned int cmd, unsigned long arg);
137static void pl2303_read_int_callback (struct urb *urb, struct pt_regs *regs);
138static void pl2303_read_bulk_callback (struct urb *urb, struct pt_regs *regs);
139static void pl2303_write_bulk_callback (struct urb *urb, struct pt_regs *regs);
140static int pl2303_write (struct usb_serial_port *port,
141 const unsigned char *buf, int count);
142static void pl2303_send (struct usb_serial_port *port);
143static int pl2303_write_room(struct usb_serial_port *port);
144static int pl2303_chars_in_buffer(struct usb_serial_port *port);
145static void pl2303_break_ctl(struct usb_serial_port *port,int break_state);
146static int pl2303_tiocmget (struct usb_serial_port *port, struct file *file);
147static int pl2303_tiocmset (struct usb_serial_port *port, struct file *file,
148 unsigned int set, unsigned int clear);
149static int pl2303_startup (struct usb_serial *serial);
150static void pl2303_shutdown (struct usb_serial *serial);
151static struct pl2303_buf *pl2303_buf_alloc(unsigned int size);
152static void pl2303_buf_free(struct pl2303_buf *pb);
153static void pl2303_buf_clear(struct pl2303_buf *pb);
154static unsigned int pl2303_buf_data_avail(struct pl2303_buf *pb);
155static unsigned int pl2303_buf_space_avail(struct pl2303_buf *pb);
156static unsigned int pl2303_buf_put(struct pl2303_buf *pb, const char *buf,
157 unsigned int count);
158static unsigned int pl2303_buf_get(struct pl2303_buf *pb, char *buf,
159 unsigned int count);
160
161
162/* All of the device info needed for the PL2303 SIO serial converter */
163static struct usb_serial_driver pl2303_device = {
164 .driver = {
165 .owner = THIS_MODULE,
166 .name = "pl2303",
167 },
168 .id_table = id_table,
169 .num_interrupt_in = NUM_DONT_CARE,
170 .num_bulk_in = 1,
171 .num_bulk_out = 1,
172 .num_ports = 1,
173 .open = pl2303_open,
174 .close = pl2303_close,
175 .write = pl2303_write,
176 .ioctl = pl2303_ioctl,
177 .break_ctl = pl2303_break_ctl,
178 .set_termios = pl2303_set_termios,
179 .tiocmget = pl2303_tiocmget,
180 .tiocmset = pl2303_tiocmset,
181 .read_bulk_callback = pl2303_read_bulk_callback,
182 .read_int_callback = pl2303_read_int_callback,
183 .write_bulk_callback = pl2303_write_bulk_callback,
184 .write_room = pl2303_write_room,
185 .chars_in_buffer = pl2303_chars_in_buffer,
186 .attach = pl2303_startup,
187 .shutdown = pl2303_shutdown,
188};
189 130
190enum pl2303_type { 131enum pl2303_type {
191 type_0, /* don't know the difference between type 0 and */ 132 type_0, /* don't know the difference between type 0 and */
@@ -204,6 +145,164 @@ struct pl2303_private {
204 enum pl2303_type type; 145 enum pl2303_type type;
205}; 146};
206 147
148/*
149 * pl2303_buf_alloc
150 *
151 * Allocate a circular buffer and all associated memory.
152 */
153static struct pl2303_buf *pl2303_buf_alloc(unsigned int size)
154{
155 struct pl2303_buf *pb;
156
157 if (size == 0)
158 return NULL;
159
160 pb = (struct pl2303_buf *)kmalloc(sizeof(struct pl2303_buf), GFP_KERNEL);
161 if (pb == NULL)
162 return NULL;
163
164 pb->buf_buf = kmalloc(size, GFP_KERNEL);
165 if (pb->buf_buf == NULL) {
166 kfree(pb);
167 return NULL;
168 }
169
170 pb->buf_size = size;
171 pb->buf_get = pb->buf_put = pb->buf_buf;
172
173 return pb;
174}
175
176/*
177 * pl2303_buf_free
178 *
179 * Free the buffer and all associated memory.
180 */
181static void pl2303_buf_free(struct pl2303_buf *pb)
182{
183 if (pb) {
184 kfree(pb->buf_buf);
185 kfree(pb);
186 }
187}
188
189/*
190 * pl2303_buf_clear
191 *
192 * Clear out all data in the circular buffer.
193 */
194static void pl2303_buf_clear(struct pl2303_buf *pb)
195{
196 if (pb != NULL)
197 pb->buf_get = pb->buf_put;
198 /* equivalent to a get of all data available */
199}
200
201/*
202 * pl2303_buf_data_avail
203 *
204 * Return the number of bytes of data available in the circular
205 * buffer.
206 */
207static unsigned int pl2303_buf_data_avail(struct pl2303_buf *pb)
208{
209 if (pb == NULL)
210 return 0;
211
212 return ((pb->buf_size + pb->buf_put - pb->buf_get) % pb->buf_size);
213}
214
215/*
216 * pl2303_buf_space_avail
217 *
218 * Return the number of bytes of space available in the circular
219 * buffer.
220 */
221static unsigned int pl2303_buf_space_avail(struct pl2303_buf *pb)
222{
223 if (pb == NULL)
224 return 0;
225
226 return ((pb->buf_size + pb->buf_get - pb->buf_put - 1) % pb->buf_size);
227}
228
229/*
230 * pl2303_buf_put
231 *
232 * Copy data data from a user buffer and put it into the circular buffer.
233 * Restrict to the amount of space available.
234 *
235 * Return the number of bytes copied.
236 */
237static unsigned int pl2303_buf_put(struct pl2303_buf *pb, const char *buf,
238 unsigned int count)
239{
240 unsigned int len;
241
242 if (pb == NULL)
243 return 0;
244
245 len = pl2303_buf_space_avail(pb);
246 if (count > len)
247 count = len;
248
249 if (count == 0)
250 return 0;
251
252 len = pb->buf_buf + pb->buf_size - pb->buf_put;
253 if (count > len) {
254 memcpy(pb->buf_put, buf, len);
255 memcpy(pb->buf_buf, buf+len, count - len);
256 pb->buf_put = pb->buf_buf + count - len;
257 } else {
258 memcpy(pb->buf_put, buf, count);
259 if (count < len)
260 pb->buf_put += count;
261 else /* count == len */
262 pb->buf_put = pb->buf_buf;
263 }
264
265 return count;
266}
267
268/*
269 * pl2303_buf_get
270 *
271 * Get data from the circular buffer and copy to the given buffer.
272 * Restrict to the amount of data available.
273 *
274 * Return the number of bytes copied.
275 */
276static unsigned int pl2303_buf_get(struct pl2303_buf *pb, char *buf,
277 unsigned int count)
278{
279 unsigned int len;
280
281 if (pb == NULL)
282 return 0;
283
284 len = pl2303_buf_data_avail(pb);
285 if (count > len)
286 count = len;
287
288 if (count == 0)
289 return 0;
290
291 len = pb->buf_buf + pb->buf_size - pb->buf_get;
292 if (count > len) {
293 memcpy(buf, pb->buf_get, len);
294 memcpy(buf+len, pb->buf_buf, count - len);
295 pb->buf_get = pb->buf_buf + count - len;
296 } else {
297 memcpy(buf, pb->buf_get, count);
298 if (count < len)
299 pb->buf_get += count;
300 else /* count == len */
301 pb->buf_get = pb->buf_buf;
302 }
303
304 return count;
305}
207 306
208static int pl2303_startup(struct usb_serial *serial) 307static int pl2303_startup(struct usb_serial *serial)
209{ 308{
@@ -258,26 +357,6 @@ static int set_control_lines(struct usb_device *dev, u8 value)
258 return retval; 357 return retval;
259} 358}
260 359
261static int pl2303_write(struct usb_serial_port *port, const unsigned char *buf,
262 int count)
263{
264 struct pl2303_private *priv = usb_get_serial_port_data(port);
265 unsigned long flags;
266
267 dbg("%s - port %d, %d bytes", __FUNCTION__, port->number, count);
268
269 if (!count)
270 return count;
271
272 spin_lock_irqsave(&priv->lock, flags);
273 count = pl2303_buf_put(priv->buf, buf, count);
274 spin_unlock_irqrestore(&priv->lock, flags);
275
276 pl2303_send(port);
277
278 return count;
279}
280
281static void pl2303_send(struct usb_serial_port *port) 360static void pl2303_send(struct usb_serial_port *port)
282{ 361{
283 int count, result; 362 int count, result;
@@ -321,6 +400,26 @@ static void pl2303_send(struct usb_serial_port *port)
321 usb_serial_port_softint(port); 400 usb_serial_port_softint(port);
322} 401}
323 402
403static int pl2303_write(struct usb_serial_port *port, const unsigned char *buf,
404 int count)
405{
406 struct pl2303_private *priv = usb_get_serial_port_data(port);
407 unsigned long flags;
408
409 dbg("%s - port %d, %d bytes", __FUNCTION__, port->number, count);
410
411 if (!count)
412 return count;
413
414 spin_lock_irqsave(&priv->lock, flags);
415 count = pl2303_buf_put(priv->buf, buf, count);
416 spin_unlock_irqrestore(&priv->lock, flags);
417
418 pl2303_send(port);
419
420 return count;
421}
422
324static int pl2303_write_room(struct usb_serial_port *port) 423static int pl2303_write_room(struct usb_serial_port *port)
325{ 424{
326 struct pl2303_private *priv = usb_get_serial_port_data(port); 425 struct pl2303_private *priv = usb_get_serial_port_data(port);
@@ -520,6 +619,70 @@ static void pl2303_set_termios(struct usb_serial_port *port,
520 kfree(buf); 619 kfree(buf);
521} 620}
522 621
622static void pl2303_close(struct usb_serial_port *port, struct file *filp)
623{
624 struct pl2303_private *priv = usb_get_serial_port_data(port);
625 unsigned long flags;
626 unsigned int c_cflag;
627 int bps;
628 long timeout;
629 wait_queue_t wait;
630
631 dbg("%s - port %d", __FUNCTION__, port->number);
632
633 /* wait for data to drain from the buffer */
634 spin_lock_irqsave(&priv->lock, flags);
635 timeout = PL2303_CLOSING_WAIT;
636 init_waitqueue_entry(&wait, current);
637 add_wait_queue(&port->tty->write_wait, &wait);
638 for (;;) {
639 set_current_state(TASK_INTERRUPTIBLE);
640 if (pl2303_buf_data_avail(priv->buf) == 0 ||
641 timeout == 0 || signal_pending(current) ||
642 !usb_get_intfdata(port->serial->interface)) /* disconnect */
643 break;
644 spin_unlock_irqrestore(&priv->lock, flags);
645 timeout = schedule_timeout(timeout);
646 spin_lock_irqsave(&priv->lock, flags);
647 }
648 set_current_state(TASK_RUNNING);
649 remove_wait_queue(&port->tty->write_wait, &wait);
650 /* clear out any remaining data in the buffer */
651 pl2303_buf_clear(priv->buf);
652 spin_unlock_irqrestore(&priv->lock, flags);
653
654 /* wait for characters to drain from the device */
655 /* (this is long enough for the entire 256 byte */
656 /* pl2303 hardware buffer to drain with no flow */
657 /* control for data rates of 1200 bps or more, */
658 /* for lower rates we should really know how much */
659 /* data is in the buffer to compute a delay */
660 /* that is not unnecessarily long) */
661 bps = tty_get_baud_rate(port->tty);
662 if (bps > 1200)
663 timeout = max((HZ*2560)/bps,HZ/10);
664 else
665 timeout = 2*HZ;
666 schedule_timeout_interruptible(timeout);
667
668 /* shutdown our urbs */
669 dbg("%s - shutting down urbs", __FUNCTION__);
670 usb_kill_urb(port->write_urb);
671 usb_kill_urb(port->read_urb);
672 usb_kill_urb(port->interrupt_in_urb);
673
674 if (port->tty) {
675 c_cflag = port->tty->termios->c_cflag;
676 if (c_cflag & HUPCL) {
677 /* drop DTR and RTS */
678 spin_lock_irqsave(&priv->lock, flags);
679 priv->line_control = 0;
680 spin_unlock_irqrestore(&priv->lock, flags);
681 set_control_lines(port->serial->dev, 0);
682 }
683 }
684}
685
523static int pl2303_open(struct usb_serial_port *port, struct file *filp) 686static int pl2303_open(struct usb_serial_port *port, struct file *filp)
524{ 687{
525 struct termios tmp_termios; 688 struct termios tmp_termios;
@@ -601,70 +764,6 @@ static int pl2303_open(struct usb_serial_port *port, struct file *filp)
601 return 0; 764 return 0;
602} 765}
603 766
604static void pl2303_close(struct usb_serial_port *port, struct file *filp)
605{
606 struct pl2303_private *priv = usb_get_serial_port_data(port);
607 unsigned long flags;
608 unsigned int c_cflag;
609 int bps;
610 long timeout;
611 wait_queue_t wait;
612
613 dbg("%s - port %d", __FUNCTION__, port->number);
614
615 /* wait for data to drain from the buffer */
616 spin_lock_irqsave(&priv->lock, flags);
617 timeout = PL2303_CLOSING_WAIT;
618 init_waitqueue_entry(&wait, current);
619 add_wait_queue(&port->tty->write_wait, &wait);
620 for (;;) {
621 set_current_state(TASK_INTERRUPTIBLE);
622 if (pl2303_buf_data_avail(priv->buf) == 0 ||
623 timeout == 0 || signal_pending(current) ||
624 !usb_get_intfdata(port->serial->interface)) /* disconnect */
625 break;
626 spin_unlock_irqrestore(&priv->lock, flags);
627 timeout = schedule_timeout(timeout);
628 spin_lock_irqsave(&priv->lock, flags);
629 }
630 set_current_state(TASK_RUNNING);
631 remove_wait_queue(&port->tty->write_wait, &wait);
632 /* clear out any remaining data in the buffer */
633 pl2303_buf_clear(priv->buf);
634 spin_unlock_irqrestore(&priv->lock, flags);
635
636 /* wait for characters to drain from the device */
637 /* (this is long enough for the entire 256 byte */
638 /* pl2303 hardware buffer to drain with no flow */
639 /* control for data rates of 1200 bps or more, */
640 /* for lower rates we should really know how much */
641 /* data is in the buffer to compute a delay */
642 /* that is not unnecessarily long) */
643 bps = tty_get_baud_rate(port->tty);
644 if (bps > 1200)
645 timeout = max((HZ*2560)/bps,HZ/10);
646 else
647 timeout = 2*HZ;
648 schedule_timeout_interruptible(timeout);
649
650 /* shutdown our urbs */
651 dbg("%s - shutting down urbs", __FUNCTION__);
652 usb_kill_urb(port->write_urb);
653 usb_kill_urb(port->read_urb);
654 usb_kill_urb(port->interrupt_in_urb);
655
656 if (port->tty) {
657 c_cflag = port->tty->termios->c_cflag;
658 if (c_cflag & HUPCL) {
659 /* drop DTR and RTS */
660 spin_lock_irqsave(&priv->lock, flags);
661 priv->line_control = 0;
662 spin_unlock_irqrestore(&priv->lock, flags);
663 set_control_lines(port->serial->dev, 0);
664 }
665 }
666}
667
668static int pl2303_tiocmset(struct usb_serial_port *port, struct file *file, 767static int pl2303_tiocmset(struct usb_serial_port *port, struct file *file,
669 unsigned int set, unsigned int clear) 768 unsigned int set, unsigned int clear)
670{ 769{
@@ -1003,164 +1102,33 @@ static void pl2303_write_bulk_callback(struct urb *urb, struct pt_regs *regs)
1003 pl2303_send(port); 1102 pl2303_send(port);
1004} 1103}
1005 1104
1006/* 1105/* All of the device info needed for the PL2303 SIO serial converter */
1007 * pl2303_buf_alloc 1106static struct usb_serial_driver pl2303_device = {
1008 * 1107 .driver = {
1009 * Allocate a circular buffer and all associated memory. 1108 .owner = THIS_MODULE,
1010 */ 1109 .name = "pl2303",
1011static struct pl2303_buf *pl2303_buf_alloc(unsigned int size) 1110 },
1012{ 1111 .id_table = id_table,
1013 struct pl2303_buf *pb; 1112 .num_interrupt_in = NUM_DONT_CARE,
1014 1113 .num_bulk_in = 1,
1015 if (size == 0) 1114 .num_bulk_out = 1,
1016 return NULL; 1115 .num_ports = 1,
1017 1116 .open = pl2303_open,
1018 pb = (struct pl2303_buf *)kmalloc(sizeof(struct pl2303_buf), GFP_KERNEL); 1117 .close = pl2303_close,
1019 if (pb == NULL) 1118 .write = pl2303_write,
1020 return NULL; 1119 .ioctl = pl2303_ioctl,
1021 1120 .break_ctl = pl2303_break_ctl,
1022 pb->buf_buf = kmalloc(size, GFP_KERNEL); 1121 .set_termios = pl2303_set_termios,
1023 if (pb->buf_buf == NULL) { 1122 .tiocmget = pl2303_tiocmget,
1024 kfree(pb); 1123 .tiocmset = pl2303_tiocmset,
1025 return NULL; 1124 .read_bulk_callback = pl2303_read_bulk_callback,
1026 } 1125 .read_int_callback = pl2303_read_int_callback,
1027 1126 .write_bulk_callback = pl2303_write_bulk_callback,
1028 pb->buf_size = size; 1127 .write_room = pl2303_write_room,
1029 pb->buf_get = pb->buf_put = pb->buf_buf; 1128 .chars_in_buffer = pl2303_chars_in_buffer,
1030 1129 .attach = pl2303_startup,
1031 return pb; 1130 .shutdown = pl2303_shutdown,
1032} 1131};
1033
1034/*
1035 * pl2303_buf_free
1036 *
1037 * Free the buffer and all associated memory.
1038 */
1039static void pl2303_buf_free(struct pl2303_buf *pb)
1040{
1041 if (pb) {
1042 kfree(pb->buf_buf);
1043 kfree(pb);
1044 }
1045}
1046
1047/*
1048 * pl2303_buf_clear
1049 *
1050 * Clear out all data in the circular buffer.
1051 */
1052static void pl2303_buf_clear(struct pl2303_buf *pb)
1053{
1054 if (pb != NULL)
1055 pb->buf_get = pb->buf_put;
1056 /* equivalent to a get of all data available */
1057}
1058
1059/*
1060 * pl2303_buf_data_avail
1061 *
1062 * Return the number of bytes of data available in the circular
1063 * buffer.
1064 */
1065static unsigned int pl2303_buf_data_avail(struct pl2303_buf *pb)
1066{
1067 if (pb == NULL)
1068 return 0;
1069
1070 return ((pb->buf_size + pb->buf_put - pb->buf_get) % pb->buf_size);
1071}
1072
1073/*
1074 * pl2303_buf_space_avail
1075 *
1076 * Return the number of bytes of space available in the circular
1077 * buffer.
1078 */
1079static unsigned int pl2303_buf_space_avail(struct pl2303_buf *pb)
1080{
1081 if (pb == NULL)
1082 return 0;
1083
1084 return ((pb->buf_size + pb->buf_get - pb->buf_put - 1) % pb->buf_size);
1085}
1086
1087/*
1088 * pl2303_buf_put
1089 *
1090 * Copy data data from a user buffer and put it into the circular buffer.
1091 * Restrict to the amount of space available.
1092 *
1093 * Return the number of bytes copied.
1094 */
1095static unsigned int pl2303_buf_put(struct pl2303_buf *pb, const char *buf,
1096 unsigned int count)
1097{
1098 unsigned int len;
1099
1100 if (pb == NULL)
1101 return 0;
1102
1103 len = pl2303_buf_space_avail(pb);
1104 if (count > len)
1105 count = len;
1106
1107 if (count == 0)
1108 return 0;
1109
1110 len = pb->buf_buf + pb->buf_size - pb->buf_put;
1111 if (count > len) {
1112 memcpy(pb->buf_put, buf, len);
1113 memcpy(pb->buf_buf, buf+len, count - len);
1114 pb->buf_put = pb->buf_buf + count - len;
1115 } else {
1116 memcpy(pb->buf_put, buf, count);
1117 if (count < len)
1118 pb->buf_put += count;
1119 else /* count == len */
1120 pb->buf_put = pb->buf_buf;
1121 }
1122
1123 return count;
1124}
1125
1126/*
1127 * pl2303_buf_get
1128 *
1129 * Get data from the circular buffer and copy to the given buffer.
1130 * Restrict to the amount of data available.
1131 *
1132 * Return the number of bytes copied.
1133 */
1134static unsigned int pl2303_buf_get(struct pl2303_buf *pb, char *buf,
1135 unsigned int count)
1136{
1137 unsigned int len;
1138
1139 if (pb == NULL)
1140 return 0;
1141
1142 len = pl2303_buf_data_avail(pb);
1143 if (count > len)
1144 count = len;
1145
1146 if (count == 0)
1147 return 0;
1148
1149 len = pb->buf_buf + pb->buf_size - pb->buf_get;
1150 if (count > len) {
1151 memcpy(buf, pb->buf_get, len);
1152 memcpy(buf+len, pb->buf_buf, count - len);
1153 pb->buf_get = pb->buf_buf + count - len;
1154 } else {
1155 memcpy(buf, pb->buf_get, count);
1156 if (count < len)
1157 pb->buf_get += count;
1158 else /* count == len */
1159 pb->buf_get = pb->buf_buf;
1160 }
1161
1162 return count;
1163}
1164 1132
1165static int __init pl2303_init(void) 1133static int __init pl2303_init(void)
1166{ 1134{