|  | // SPDX-License-Identifier: GPL-2.0 | 
|  | /* | 
|  | * USB Debug cable driver | 
|  | * | 
|  | * Copyright (C) 2006 Greg Kroah-Hartman <greg@kroah.com> | 
|  | */ | 
|  |  | 
|  | #include <linux/gfp.h> | 
|  | #include <linux/kernel.h> | 
|  | #include <linux/tty.h> | 
|  | #include <linux/module.h> | 
|  | #include <linux/usb.h> | 
|  | #include <linux/usb/serial.h> | 
|  |  | 
|  | #define USB_DEBUG_MAX_PACKET_SIZE	8 | 
|  | #define USB_DEBUG_BRK_SIZE		8 | 
|  | static const char USB_DEBUG_BRK[USB_DEBUG_BRK_SIZE] = { | 
|  | 0x00, | 
|  | 0xff, | 
|  | 0x01, | 
|  | 0xfe, | 
|  | 0x00, | 
|  | 0xfe, | 
|  | 0x01, | 
|  | 0xff, | 
|  | }; | 
|  |  | 
|  | static const struct usb_device_id id_table[] = { | 
|  | { USB_DEVICE(0x0525, 0x127a) }, | 
|  | { }, | 
|  | }; | 
|  |  | 
|  | static const struct usb_device_id dbc_id_table[] = { | 
|  | { USB_DEVICE(0x1d6b, 0x0010) }, | 
|  | { USB_DEVICE(0x1d6b, 0x0011) }, | 
|  | { }, | 
|  | }; | 
|  |  | 
|  | static const struct usb_device_id id_table_combined[] = { | 
|  | { USB_DEVICE(0x0525, 0x127a) }, | 
|  | { USB_DEVICE(0x1d6b, 0x0010) }, | 
|  | { USB_DEVICE(0x1d6b, 0x0011) }, | 
|  | { }, | 
|  | }; | 
|  | MODULE_DEVICE_TABLE(usb, id_table_combined); | 
|  |  | 
|  | /* This HW really does not support a serial break, so one will be | 
|  | * emulated when ever the break state is set to true. | 
|  | */ | 
|  | static int usb_debug_break_ctl(struct tty_struct *tty, int break_state) | 
|  | { | 
|  | struct usb_serial_port *port = tty->driver_data; | 
|  | int ret; | 
|  |  | 
|  | if (!break_state) | 
|  | return 0; | 
|  |  | 
|  | ret = usb_serial_generic_write(tty, port, USB_DEBUG_BRK, USB_DEBUG_BRK_SIZE); | 
|  | if (ret < 0) | 
|  | return ret; | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static void usb_debug_process_read_urb(struct urb *urb) | 
|  | { | 
|  | struct usb_serial_port *port = urb->context; | 
|  |  | 
|  | if (urb->actual_length == USB_DEBUG_BRK_SIZE && | 
|  | memcmp(urb->transfer_buffer, USB_DEBUG_BRK, | 
|  | USB_DEBUG_BRK_SIZE) == 0) { | 
|  | usb_serial_handle_break(port); | 
|  | return; | 
|  | } | 
|  |  | 
|  | usb_serial_generic_process_read_urb(urb); | 
|  | } | 
|  |  | 
|  | static void usb_debug_init_termios(struct tty_struct *tty) | 
|  | { | 
|  | tty->termios.c_lflag &= ~(ECHO | ECHONL); | 
|  | } | 
|  |  | 
|  | static struct usb_serial_driver debug_device = { | 
|  | .driver = { | 
|  | .name =		"debug", | 
|  | }, | 
|  | .id_table =		id_table, | 
|  | .num_ports =		1, | 
|  | .bulk_out_size =	USB_DEBUG_MAX_PACKET_SIZE, | 
|  | .break_ctl =		usb_debug_break_ctl, | 
|  | .init_termios =		usb_debug_init_termios, | 
|  | .process_read_urb =	usb_debug_process_read_urb, | 
|  | }; | 
|  |  | 
|  | static struct usb_serial_driver dbc_device = { | 
|  | .driver = { | 
|  | .name =		"xhci_dbc", | 
|  | }, | 
|  | .id_table =		dbc_id_table, | 
|  | .num_ports =		1, | 
|  | .break_ctl =		usb_debug_break_ctl, | 
|  | .init_termios =		usb_debug_init_termios, | 
|  | .process_read_urb =	usb_debug_process_read_urb, | 
|  | }; | 
|  |  | 
|  | static struct usb_serial_driver * const serial_drivers[] = { | 
|  | &debug_device, &dbc_device, NULL | 
|  | }; | 
|  |  | 
|  | module_usb_serial_driver(serial_drivers, id_table_combined); | 
|  | MODULE_DESCRIPTION("USB Debug cable driver"); | 
|  | MODULE_LICENSE("GPL v2"); |