1958e8741SGreg Kroah-Hartman /* 2958e8741SGreg Kroah-Hartman * USB Debug cable driver 3958e8741SGreg Kroah-Hartman * 4958e8741SGreg Kroah-Hartman * Copyright (C) 2006 Greg Kroah-Hartman <greg@kroah.com> 5958e8741SGreg Kroah-Hartman * 6958e8741SGreg Kroah-Hartman * This program is free software; you can redistribute it and/or 7958e8741SGreg Kroah-Hartman * modify it under the terms of the GNU General Public License version 8958e8741SGreg Kroah-Hartman * 2 as published by the Free Software Foundation. 9958e8741SGreg Kroah-Hartman */ 10958e8741SGreg Kroah-Hartman 11958e8741SGreg Kroah-Hartman #include <linux/kernel.h> 12958e8741SGreg Kroah-Hartman #include <linux/init.h> 13958e8741SGreg Kroah-Hartman #include <linux/tty.h> 14958e8741SGreg Kroah-Hartman #include <linux/module.h> 15958e8741SGreg Kroah-Hartman #include <linux/usb.h> 16958e8741SGreg Kroah-Hartman #include <linux/usb/serial.h> 17958e8741SGreg Kroah-Hartman 18715b1dc0SJason Wessel #define URB_DEBUG_MAX_IN_FLIGHT_URBS 4000 1971be4f81SAleksey Gorelov #define USB_DEBUG_MAX_PACKET_SIZE 8 2098fcb5f7SJason Wessel #define USB_DEBUG_BRK_SIZE 8 2198fcb5f7SJason Wessel static char USB_DEBUG_BRK[USB_DEBUG_BRK_SIZE] = { 2298fcb5f7SJason Wessel 0x00, 2398fcb5f7SJason Wessel 0xff, 2498fcb5f7SJason Wessel 0x01, 2598fcb5f7SJason Wessel 0xfe, 2698fcb5f7SJason Wessel 0x00, 2798fcb5f7SJason Wessel 0xfe, 2898fcb5f7SJason Wessel 0x01, 2998fcb5f7SJason Wessel 0xff, 3098fcb5f7SJason Wessel }; 3171be4f81SAleksey Gorelov 32*7d40d7e8SNémeth Márton static const struct usb_device_id id_table[] = { 33958e8741SGreg Kroah-Hartman { USB_DEVICE(0x0525, 0x127a) }, 34958e8741SGreg Kroah-Hartman { }, 35958e8741SGreg Kroah-Hartman }; 36958e8741SGreg Kroah-Hartman MODULE_DEVICE_TABLE(usb, id_table); 37958e8741SGreg Kroah-Hartman 38958e8741SGreg Kroah-Hartman static struct usb_driver debug_driver = { 39958e8741SGreg Kroah-Hartman .name = "debug", 40958e8741SGreg Kroah-Hartman .probe = usb_serial_probe, 41958e8741SGreg Kroah-Hartman .disconnect = usb_serial_disconnect, 42958e8741SGreg Kroah-Hartman .id_table = id_table, 43958e8741SGreg Kroah-Hartman .no_dynamic_id = 1, 44958e8741SGreg Kroah-Hartman }; 45958e8741SGreg Kroah-Hartman 46a509a7e4SAlan Cox static int usb_debug_open(struct tty_struct *tty, struct usb_serial_port *port) 4771be4f81SAleksey Gorelov { 4871be4f81SAleksey Gorelov port->bulk_out_size = USB_DEBUG_MAX_PACKET_SIZE; 49a509a7e4SAlan Cox return usb_serial_generic_open(tty, port); 5071be4f81SAleksey Gorelov } 5171be4f81SAleksey Gorelov 5298fcb5f7SJason Wessel /* This HW really does not support a serial break, so one will be 5398fcb5f7SJason Wessel * emulated when ever the break state is set to true. 5498fcb5f7SJason Wessel */ 5598fcb5f7SJason Wessel static void usb_debug_break_ctl(struct tty_struct *tty, int break_state) 5698fcb5f7SJason Wessel { 5798fcb5f7SJason Wessel struct usb_serial_port *port = tty->driver_data; 5898fcb5f7SJason Wessel if (!break_state) 5998fcb5f7SJason Wessel return; 6098fcb5f7SJason Wessel usb_serial_generic_write(tty, port, USB_DEBUG_BRK, USB_DEBUG_BRK_SIZE); 6198fcb5f7SJason Wessel } 6298fcb5f7SJason Wessel 6398fcb5f7SJason Wessel static void usb_debug_read_bulk_callback(struct urb *urb) 6498fcb5f7SJason Wessel { 6598fcb5f7SJason Wessel struct usb_serial_port *port = urb->context; 6698fcb5f7SJason Wessel 6798fcb5f7SJason Wessel if (urb->actual_length == USB_DEBUG_BRK_SIZE && 6898fcb5f7SJason Wessel memcmp(urb->transfer_buffer, USB_DEBUG_BRK, 6998fcb5f7SJason Wessel USB_DEBUG_BRK_SIZE) == 0) { 7098fcb5f7SJason Wessel usb_serial_handle_break(port); 7198fcb5f7SJason Wessel usb_serial_generic_resubmit_read_urb(port, GFP_ATOMIC); 7298fcb5f7SJason Wessel return; 7398fcb5f7SJason Wessel } 7498fcb5f7SJason Wessel 7598fcb5f7SJason Wessel usb_serial_generic_read_bulk_callback(urb); 7698fcb5f7SJason Wessel } 7798fcb5f7SJason Wessel 78958e8741SGreg Kroah-Hartman static struct usb_serial_driver debug_device = { 79958e8741SGreg Kroah-Hartman .driver = { 80958e8741SGreg Kroah-Hartman .owner = THIS_MODULE, 81958e8741SGreg Kroah-Hartman .name = "debug", 82958e8741SGreg Kroah-Hartman }, 83958e8741SGreg Kroah-Hartman .id_table = id_table, 84958e8741SGreg Kroah-Hartman .num_ports = 1, 8571be4f81SAleksey Gorelov .open = usb_debug_open, 86715b1dc0SJason Wessel .max_in_flight_urbs = URB_DEBUG_MAX_IN_FLIGHT_URBS, 8798fcb5f7SJason Wessel .break_ctl = usb_debug_break_ctl, 8898fcb5f7SJason Wessel .read_bulk_callback = usb_debug_read_bulk_callback, 89958e8741SGreg Kroah-Hartman }; 90958e8741SGreg Kroah-Hartman 91958e8741SGreg Kroah-Hartman static int __init debug_init(void) 92958e8741SGreg Kroah-Hartman { 93958e8741SGreg Kroah-Hartman int retval; 94958e8741SGreg Kroah-Hartman 95958e8741SGreg Kroah-Hartman retval = usb_serial_register(&debug_device); 96958e8741SGreg Kroah-Hartman if (retval) 97958e8741SGreg Kroah-Hartman return retval; 98958e8741SGreg Kroah-Hartman retval = usb_register(&debug_driver); 99958e8741SGreg Kroah-Hartman if (retval) 100958e8741SGreg Kroah-Hartman usb_serial_deregister(&debug_device); 101958e8741SGreg Kroah-Hartman return retval; 102958e8741SGreg Kroah-Hartman } 103958e8741SGreg Kroah-Hartman 104958e8741SGreg Kroah-Hartman static void __exit debug_exit(void) 105958e8741SGreg Kroah-Hartman { 106958e8741SGreg Kroah-Hartman usb_deregister(&debug_driver); 107958e8741SGreg Kroah-Hartman usb_serial_deregister(&debug_device); 108958e8741SGreg Kroah-Hartman } 109958e8741SGreg Kroah-Hartman 110958e8741SGreg Kroah-Hartman module_init(debug_init); 111958e8741SGreg Kroah-Hartman module_exit(debug_exit); 112958e8741SGreg Kroah-Hartman MODULE_LICENSE("GPL"); 113