162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0+ 262306a36Sopenharmony_ci/* 362306a36Sopenharmony_ci * Generic USB driver for report based interrupt in/out devices 462306a36Sopenharmony_ci * like LD Didactic's USB devices. LD Didactic's USB devices are 562306a36Sopenharmony_ci * HID devices which do not use HID report definitons (they use 662306a36Sopenharmony_ci * raw interrupt in and our reports only for communication). 762306a36Sopenharmony_ci * 862306a36Sopenharmony_ci * This driver uses a ring buffer for time critical reading of 962306a36Sopenharmony_ci * interrupt in reports and provides read and write methods for 1062306a36Sopenharmony_ci * raw interrupt reports (similar to the Windows HID driver). 1162306a36Sopenharmony_ci * Devices based on the book USB COMPLETE by Jan Axelson may need 1262306a36Sopenharmony_ci * such a compatibility to the Windows HID driver. 1362306a36Sopenharmony_ci * 1462306a36Sopenharmony_ci * Copyright (C) 2005 Michael Hund <mhund@ld-didactic.de> 1562306a36Sopenharmony_ci * 1662306a36Sopenharmony_ci * Derived from Lego USB Tower driver 1762306a36Sopenharmony_ci * Copyright (C) 2003 David Glance <advidgsf@sourceforge.net> 1862306a36Sopenharmony_ci * 2001-2004 Juergen Stuber <starblue@users.sourceforge.net> 1962306a36Sopenharmony_ci */ 2062306a36Sopenharmony_ci 2162306a36Sopenharmony_ci#include <linux/kernel.h> 2262306a36Sopenharmony_ci#include <linux/errno.h> 2362306a36Sopenharmony_ci#include <linux/slab.h> 2462306a36Sopenharmony_ci#include <linux/module.h> 2562306a36Sopenharmony_ci#include <linux/mutex.h> 2662306a36Sopenharmony_ci 2762306a36Sopenharmony_ci#include <linux/uaccess.h> 2862306a36Sopenharmony_ci#include <linux/input.h> 2962306a36Sopenharmony_ci#include <linux/usb.h> 3062306a36Sopenharmony_ci#include <linux/poll.h> 3162306a36Sopenharmony_ci 3262306a36Sopenharmony_ci/* Define these values to match your devices */ 3362306a36Sopenharmony_ci#define USB_VENDOR_ID_LD 0x0f11 /* USB Vendor ID of LD Didactic GmbH */ 3462306a36Sopenharmony_ci#define USB_DEVICE_ID_LD_CASSY 0x1000 /* USB Product ID of CASSY-S modules with 8 bytes endpoint size */ 3562306a36Sopenharmony_ci#define USB_DEVICE_ID_LD_CASSY2 0x1001 /* USB Product ID of CASSY-S modules with 64 bytes endpoint size */ 3662306a36Sopenharmony_ci#define USB_DEVICE_ID_LD_POCKETCASSY 0x1010 /* USB Product ID of Pocket-CASSY */ 3762306a36Sopenharmony_ci#define USB_DEVICE_ID_LD_POCKETCASSY2 0x1011 /* USB Product ID of Pocket-CASSY 2 (reserved) */ 3862306a36Sopenharmony_ci#define USB_DEVICE_ID_LD_MOBILECASSY 0x1020 /* USB Product ID of Mobile-CASSY */ 3962306a36Sopenharmony_ci#define USB_DEVICE_ID_LD_MOBILECASSY2 0x1021 /* USB Product ID of Mobile-CASSY 2 (reserved) */ 4062306a36Sopenharmony_ci#define USB_DEVICE_ID_LD_MICROCASSYVOLTAGE 0x1031 /* USB Product ID of Micro-CASSY Voltage */ 4162306a36Sopenharmony_ci#define USB_DEVICE_ID_LD_MICROCASSYCURRENT 0x1032 /* USB Product ID of Micro-CASSY Current */ 4262306a36Sopenharmony_ci#define USB_DEVICE_ID_LD_MICROCASSYTIME 0x1033 /* USB Product ID of Micro-CASSY Time (reserved) */ 4362306a36Sopenharmony_ci#define USB_DEVICE_ID_LD_MICROCASSYTEMPERATURE 0x1035 /* USB Product ID of Micro-CASSY Temperature */ 4462306a36Sopenharmony_ci#define USB_DEVICE_ID_LD_MICROCASSYPH 0x1038 /* USB Product ID of Micro-CASSY pH */ 4562306a36Sopenharmony_ci#define USB_DEVICE_ID_LD_POWERANALYSERCASSY 0x1040 /* USB Product ID of Power Analyser CASSY */ 4662306a36Sopenharmony_ci#define USB_DEVICE_ID_LD_CONVERTERCONTROLLERCASSY 0x1042 /* USB Product ID of Converter Controller CASSY */ 4762306a36Sopenharmony_ci#define USB_DEVICE_ID_LD_MACHINETESTCASSY 0x1043 /* USB Product ID of Machine Test CASSY */ 4862306a36Sopenharmony_ci#define USB_DEVICE_ID_LD_JWM 0x1080 /* USB Product ID of Joule and Wattmeter */ 4962306a36Sopenharmony_ci#define USB_DEVICE_ID_LD_DMMP 0x1081 /* USB Product ID of Digital Multimeter P (reserved) */ 5062306a36Sopenharmony_ci#define USB_DEVICE_ID_LD_UMIP 0x1090 /* USB Product ID of UMI P */ 5162306a36Sopenharmony_ci#define USB_DEVICE_ID_LD_UMIC 0x10A0 /* USB Product ID of UMI C */ 5262306a36Sopenharmony_ci#define USB_DEVICE_ID_LD_UMIB 0x10B0 /* USB Product ID of UMI B */ 5362306a36Sopenharmony_ci#define USB_DEVICE_ID_LD_XRAY 0x1100 /* USB Product ID of X-Ray Apparatus 55481 */ 5462306a36Sopenharmony_ci#define USB_DEVICE_ID_LD_XRAY2 0x1101 /* USB Product ID of X-Ray Apparatus 554800 */ 5562306a36Sopenharmony_ci#define USB_DEVICE_ID_LD_XRAYCT 0x1110 /* USB Product ID of X-Ray Apparatus CT 554821*/ 5662306a36Sopenharmony_ci#define USB_DEVICE_ID_LD_VIDEOCOM 0x1200 /* USB Product ID of VideoCom */ 5762306a36Sopenharmony_ci#define USB_DEVICE_ID_LD_MOTOR 0x1210 /* USB Product ID of Motor (reserved) */ 5862306a36Sopenharmony_ci#define USB_DEVICE_ID_LD_COM3LAB 0x2000 /* USB Product ID of COM3LAB */ 5962306a36Sopenharmony_ci#define USB_DEVICE_ID_LD_TELEPORT 0x2010 /* USB Product ID of Terminal Adapter */ 6062306a36Sopenharmony_ci#define USB_DEVICE_ID_LD_NETWORKANALYSER 0x2020 /* USB Product ID of Network Analyser */ 6162306a36Sopenharmony_ci#define USB_DEVICE_ID_LD_POWERCONTROL 0x2030 /* USB Product ID of Converter Control Unit */ 6262306a36Sopenharmony_ci#define USB_DEVICE_ID_LD_MACHINETEST 0x2040 /* USB Product ID of Machine Test System */ 6362306a36Sopenharmony_ci#define USB_DEVICE_ID_LD_MOSTANALYSER 0x2050 /* USB Product ID of MOST Protocol Analyser */ 6462306a36Sopenharmony_ci#define USB_DEVICE_ID_LD_MOSTANALYSER2 0x2051 /* USB Product ID of MOST Protocol Analyser 2 */ 6562306a36Sopenharmony_ci#define USB_DEVICE_ID_LD_ABSESP 0x2060 /* USB Product ID of ABS ESP */ 6662306a36Sopenharmony_ci#define USB_DEVICE_ID_LD_AUTODATABUS 0x2070 /* USB Product ID of Automotive Data Buses */ 6762306a36Sopenharmony_ci#define USB_DEVICE_ID_LD_MCT 0x2080 /* USB Product ID of Microcontroller technique */ 6862306a36Sopenharmony_ci#define USB_DEVICE_ID_LD_HYBRID 0x2090 /* USB Product ID of Automotive Hybrid */ 6962306a36Sopenharmony_ci#define USB_DEVICE_ID_LD_HEATCONTROL 0x20A0 /* USB Product ID of Heat control */ 7062306a36Sopenharmony_ci 7162306a36Sopenharmony_ci#ifdef CONFIG_USB_DYNAMIC_MINORS 7262306a36Sopenharmony_ci#define USB_LD_MINOR_BASE 0 7362306a36Sopenharmony_ci#else 7462306a36Sopenharmony_ci#define USB_LD_MINOR_BASE 176 7562306a36Sopenharmony_ci#endif 7662306a36Sopenharmony_ci 7762306a36Sopenharmony_ci/* table of devices that work with this driver */ 7862306a36Sopenharmony_cistatic const struct usb_device_id ld_usb_table[] = { 7962306a36Sopenharmony_ci { USB_DEVICE(USB_VENDOR_ID_LD, USB_DEVICE_ID_LD_CASSY) }, 8062306a36Sopenharmony_ci { USB_DEVICE(USB_VENDOR_ID_LD, USB_DEVICE_ID_LD_CASSY2) }, 8162306a36Sopenharmony_ci { USB_DEVICE(USB_VENDOR_ID_LD, USB_DEVICE_ID_LD_POCKETCASSY) }, 8262306a36Sopenharmony_ci { USB_DEVICE(USB_VENDOR_ID_LD, USB_DEVICE_ID_LD_POCKETCASSY2) }, 8362306a36Sopenharmony_ci { USB_DEVICE(USB_VENDOR_ID_LD, USB_DEVICE_ID_LD_MOBILECASSY) }, 8462306a36Sopenharmony_ci { USB_DEVICE(USB_VENDOR_ID_LD, USB_DEVICE_ID_LD_MOBILECASSY2) }, 8562306a36Sopenharmony_ci { USB_DEVICE(USB_VENDOR_ID_LD, USB_DEVICE_ID_LD_MICROCASSYVOLTAGE) }, 8662306a36Sopenharmony_ci { USB_DEVICE(USB_VENDOR_ID_LD, USB_DEVICE_ID_LD_MICROCASSYCURRENT) }, 8762306a36Sopenharmony_ci { USB_DEVICE(USB_VENDOR_ID_LD, USB_DEVICE_ID_LD_MICROCASSYTIME) }, 8862306a36Sopenharmony_ci { USB_DEVICE(USB_VENDOR_ID_LD, USB_DEVICE_ID_LD_MICROCASSYTEMPERATURE) }, 8962306a36Sopenharmony_ci { USB_DEVICE(USB_VENDOR_ID_LD, USB_DEVICE_ID_LD_MICROCASSYPH) }, 9062306a36Sopenharmony_ci { USB_DEVICE(USB_VENDOR_ID_LD, USB_DEVICE_ID_LD_POWERANALYSERCASSY) }, 9162306a36Sopenharmony_ci { USB_DEVICE(USB_VENDOR_ID_LD, USB_DEVICE_ID_LD_CONVERTERCONTROLLERCASSY) }, 9262306a36Sopenharmony_ci { USB_DEVICE(USB_VENDOR_ID_LD, USB_DEVICE_ID_LD_MACHINETESTCASSY) }, 9362306a36Sopenharmony_ci { USB_DEVICE(USB_VENDOR_ID_LD, USB_DEVICE_ID_LD_JWM) }, 9462306a36Sopenharmony_ci { USB_DEVICE(USB_VENDOR_ID_LD, USB_DEVICE_ID_LD_DMMP) }, 9562306a36Sopenharmony_ci { USB_DEVICE(USB_VENDOR_ID_LD, USB_DEVICE_ID_LD_UMIP) }, 9662306a36Sopenharmony_ci { USB_DEVICE(USB_VENDOR_ID_LD, USB_DEVICE_ID_LD_UMIC) }, 9762306a36Sopenharmony_ci { USB_DEVICE(USB_VENDOR_ID_LD, USB_DEVICE_ID_LD_UMIB) }, 9862306a36Sopenharmony_ci { USB_DEVICE(USB_VENDOR_ID_LD, USB_DEVICE_ID_LD_XRAY) }, 9962306a36Sopenharmony_ci { USB_DEVICE(USB_VENDOR_ID_LD, USB_DEVICE_ID_LD_XRAY2) }, 10062306a36Sopenharmony_ci { USB_DEVICE(USB_VENDOR_ID_LD, USB_DEVICE_ID_LD_VIDEOCOM) }, 10162306a36Sopenharmony_ci { USB_DEVICE(USB_VENDOR_ID_LD, USB_DEVICE_ID_LD_MOTOR) }, 10262306a36Sopenharmony_ci { USB_DEVICE(USB_VENDOR_ID_LD, USB_DEVICE_ID_LD_COM3LAB) }, 10362306a36Sopenharmony_ci { USB_DEVICE(USB_VENDOR_ID_LD, USB_DEVICE_ID_LD_TELEPORT) }, 10462306a36Sopenharmony_ci { USB_DEVICE(USB_VENDOR_ID_LD, USB_DEVICE_ID_LD_NETWORKANALYSER) }, 10562306a36Sopenharmony_ci { USB_DEVICE(USB_VENDOR_ID_LD, USB_DEVICE_ID_LD_POWERCONTROL) }, 10662306a36Sopenharmony_ci { USB_DEVICE(USB_VENDOR_ID_LD, USB_DEVICE_ID_LD_MACHINETEST) }, 10762306a36Sopenharmony_ci { USB_DEVICE(USB_VENDOR_ID_LD, USB_DEVICE_ID_LD_MOSTANALYSER) }, 10862306a36Sopenharmony_ci { USB_DEVICE(USB_VENDOR_ID_LD, USB_DEVICE_ID_LD_MOSTANALYSER2) }, 10962306a36Sopenharmony_ci { USB_DEVICE(USB_VENDOR_ID_LD, USB_DEVICE_ID_LD_ABSESP) }, 11062306a36Sopenharmony_ci { USB_DEVICE(USB_VENDOR_ID_LD, USB_DEVICE_ID_LD_AUTODATABUS) }, 11162306a36Sopenharmony_ci { USB_DEVICE(USB_VENDOR_ID_LD, USB_DEVICE_ID_LD_MCT) }, 11262306a36Sopenharmony_ci { USB_DEVICE(USB_VENDOR_ID_LD, USB_DEVICE_ID_LD_HYBRID) }, 11362306a36Sopenharmony_ci { USB_DEVICE(USB_VENDOR_ID_LD, USB_DEVICE_ID_LD_HEATCONTROL) }, 11462306a36Sopenharmony_ci { } /* Terminating entry */ 11562306a36Sopenharmony_ci}; 11662306a36Sopenharmony_ciMODULE_DEVICE_TABLE(usb, ld_usb_table); 11762306a36Sopenharmony_ciMODULE_AUTHOR("Michael Hund <mhund@ld-didactic.de>"); 11862306a36Sopenharmony_ciMODULE_DESCRIPTION("LD USB Driver"); 11962306a36Sopenharmony_ciMODULE_LICENSE("GPL"); 12062306a36Sopenharmony_ci 12162306a36Sopenharmony_ci/* All interrupt in transfers are collected in a ring buffer to 12262306a36Sopenharmony_ci * avoid racing conditions and get better performance of the driver. 12362306a36Sopenharmony_ci */ 12462306a36Sopenharmony_cistatic int ring_buffer_size = 128; 12562306a36Sopenharmony_cimodule_param(ring_buffer_size, int, 0000); 12662306a36Sopenharmony_ciMODULE_PARM_DESC(ring_buffer_size, "Read ring buffer size in reports"); 12762306a36Sopenharmony_ci 12862306a36Sopenharmony_ci/* The write_buffer can contain more than one interrupt out transfer. 12962306a36Sopenharmony_ci */ 13062306a36Sopenharmony_cistatic int write_buffer_size = 10; 13162306a36Sopenharmony_cimodule_param(write_buffer_size, int, 0000); 13262306a36Sopenharmony_ciMODULE_PARM_DESC(write_buffer_size, "Write buffer size in reports"); 13362306a36Sopenharmony_ci 13462306a36Sopenharmony_ci/* As of kernel version 2.6.4 ehci-hcd uses an 13562306a36Sopenharmony_ci * "only one interrupt transfer per frame" shortcut 13662306a36Sopenharmony_ci * to simplify the scheduling of periodic transfers. 13762306a36Sopenharmony_ci * This conflicts with our standard 1ms intervals for in and out URBs. 13862306a36Sopenharmony_ci * We use default intervals of 2ms for in and 2ms for out transfers, 13962306a36Sopenharmony_ci * which should be fast enough. 14062306a36Sopenharmony_ci * Increase the interval to allow more devices that do interrupt transfers, 14162306a36Sopenharmony_ci * or set to 1 to use the standard interval from the endpoint descriptors. 14262306a36Sopenharmony_ci */ 14362306a36Sopenharmony_cistatic int min_interrupt_in_interval = 2; 14462306a36Sopenharmony_cimodule_param(min_interrupt_in_interval, int, 0000); 14562306a36Sopenharmony_ciMODULE_PARM_DESC(min_interrupt_in_interval, "Minimum interrupt in interval in ms"); 14662306a36Sopenharmony_ci 14762306a36Sopenharmony_cistatic int min_interrupt_out_interval = 2; 14862306a36Sopenharmony_cimodule_param(min_interrupt_out_interval, int, 0000); 14962306a36Sopenharmony_ciMODULE_PARM_DESC(min_interrupt_out_interval, "Minimum interrupt out interval in ms"); 15062306a36Sopenharmony_ci 15162306a36Sopenharmony_ci/* Structure to hold all of our device specific stuff */ 15262306a36Sopenharmony_cistruct ld_usb { 15362306a36Sopenharmony_ci struct mutex mutex; /* locks this structure */ 15462306a36Sopenharmony_ci struct usb_interface *intf; /* save off the usb interface pointer */ 15562306a36Sopenharmony_ci unsigned long disconnected:1; 15662306a36Sopenharmony_ci 15762306a36Sopenharmony_ci int open_count; /* number of times this port has been opened */ 15862306a36Sopenharmony_ci 15962306a36Sopenharmony_ci char *ring_buffer; 16062306a36Sopenharmony_ci unsigned int ring_head; 16162306a36Sopenharmony_ci unsigned int ring_tail; 16262306a36Sopenharmony_ci 16362306a36Sopenharmony_ci wait_queue_head_t read_wait; 16462306a36Sopenharmony_ci wait_queue_head_t write_wait; 16562306a36Sopenharmony_ci 16662306a36Sopenharmony_ci char *interrupt_in_buffer; 16762306a36Sopenharmony_ci struct usb_endpoint_descriptor *interrupt_in_endpoint; 16862306a36Sopenharmony_ci struct urb *interrupt_in_urb; 16962306a36Sopenharmony_ci int interrupt_in_interval; 17062306a36Sopenharmony_ci size_t interrupt_in_endpoint_size; 17162306a36Sopenharmony_ci int interrupt_in_running; 17262306a36Sopenharmony_ci int interrupt_in_done; 17362306a36Sopenharmony_ci int buffer_overflow; 17462306a36Sopenharmony_ci spinlock_t rbsl; 17562306a36Sopenharmony_ci 17662306a36Sopenharmony_ci char *interrupt_out_buffer; 17762306a36Sopenharmony_ci struct usb_endpoint_descriptor *interrupt_out_endpoint; 17862306a36Sopenharmony_ci struct urb *interrupt_out_urb; 17962306a36Sopenharmony_ci int interrupt_out_interval; 18062306a36Sopenharmony_ci size_t interrupt_out_endpoint_size; 18162306a36Sopenharmony_ci int interrupt_out_busy; 18262306a36Sopenharmony_ci}; 18362306a36Sopenharmony_ci 18462306a36Sopenharmony_cistatic struct usb_driver ld_usb_driver; 18562306a36Sopenharmony_ci 18662306a36Sopenharmony_ci/* 18762306a36Sopenharmony_ci * ld_usb_abort_transfers 18862306a36Sopenharmony_ci * aborts transfers and frees associated data structures 18962306a36Sopenharmony_ci */ 19062306a36Sopenharmony_cistatic void ld_usb_abort_transfers(struct ld_usb *dev) 19162306a36Sopenharmony_ci{ 19262306a36Sopenharmony_ci /* shutdown transfer */ 19362306a36Sopenharmony_ci if (dev->interrupt_in_running) { 19462306a36Sopenharmony_ci dev->interrupt_in_running = 0; 19562306a36Sopenharmony_ci usb_kill_urb(dev->interrupt_in_urb); 19662306a36Sopenharmony_ci } 19762306a36Sopenharmony_ci if (dev->interrupt_out_busy) 19862306a36Sopenharmony_ci usb_kill_urb(dev->interrupt_out_urb); 19962306a36Sopenharmony_ci} 20062306a36Sopenharmony_ci 20162306a36Sopenharmony_ci/* 20262306a36Sopenharmony_ci * ld_usb_delete 20362306a36Sopenharmony_ci */ 20462306a36Sopenharmony_cistatic void ld_usb_delete(struct ld_usb *dev) 20562306a36Sopenharmony_ci{ 20662306a36Sopenharmony_ci /* free data structures */ 20762306a36Sopenharmony_ci usb_free_urb(dev->interrupt_in_urb); 20862306a36Sopenharmony_ci usb_free_urb(dev->interrupt_out_urb); 20962306a36Sopenharmony_ci kfree(dev->ring_buffer); 21062306a36Sopenharmony_ci kfree(dev->interrupt_in_buffer); 21162306a36Sopenharmony_ci kfree(dev->interrupt_out_buffer); 21262306a36Sopenharmony_ci kfree(dev); 21362306a36Sopenharmony_ci} 21462306a36Sopenharmony_ci 21562306a36Sopenharmony_ci/* 21662306a36Sopenharmony_ci * ld_usb_interrupt_in_callback 21762306a36Sopenharmony_ci */ 21862306a36Sopenharmony_cistatic void ld_usb_interrupt_in_callback(struct urb *urb) 21962306a36Sopenharmony_ci{ 22062306a36Sopenharmony_ci struct ld_usb *dev = urb->context; 22162306a36Sopenharmony_ci size_t *actual_buffer; 22262306a36Sopenharmony_ci unsigned int next_ring_head; 22362306a36Sopenharmony_ci int status = urb->status; 22462306a36Sopenharmony_ci unsigned long flags; 22562306a36Sopenharmony_ci int retval; 22662306a36Sopenharmony_ci 22762306a36Sopenharmony_ci if (status) { 22862306a36Sopenharmony_ci if (status == -ENOENT || 22962306a36Sopenharmony_ci status == -ECONNRESET || 23062306a36Sopenharmony_ci status == -ESHUTDOWN) { 23162306a36Sopenharmony_ci goto exit; 23262306a36Sopenharmony_ci } else { 23362306a36Sopenharmony_ci dev_dbg(&dev->intf->dev, 23462306a36Sopenharmony_ci "%s: nonzero status received: %d\n", __func__, 23562306a36Sopenharmony_ci status); 23662306a36Sopenharmony_ci spin_lock_irqsave(&dev->rbsl, flags); 23762306a36Sopenharmony_ci goto resubmit; /* maybe we can recover */ 23862306a36Sopenharmony_ci } 23962306a36Sopenharmony_ci } 24062306a36Sopenharmony_ci 24162306a36Sopenharmony_ci spin_lock_irqsave(&dev->rbsl, flags); 24262306a36Sopenharmony_ci if (urb->actual_length > 0) { 24362306a36Sopenharmony_ci next_ring_head = (dev->ring_head+1) % ring_buffer_size; 24462306a36Sopenharmony_ci if (next_ring_head != dev->ring_tail) { 24562306a36Sopenharmony_ci actual_buffer = (size_t *)(dev->ring_buffer + dev->ring_head * (sizeof(size_t)+dev->interrupt_in_endpoint_size)); 24662306a36Sopenharmony_ci /* actual_buffer gets urb->actual_length + interrupt_in_buffer */ 24762306a36Sopenharmony_ci *actual_buffer = urb->actual_length; 24862306a36Sopenharmony_ci memcpy(actual_buffer+1, dev->interrupt_in_buffer, urb->actual_length); 24962306a36Sopenharmony_ci dev->ring_head = next_ring_head; 25062306a36Sopenharmony_ci dev_dbg(&dev->intf->dev, "%s: received %d bytes\n", 25162306a36Sopenharmony_ci __func__, urb->actual_length); 25262306a36Sopenharmony_ci } else { 25362306a36Sopenharmony_ci dev_warn(&dev->intf->dev, 25462306a36Sopenharmony_ci "Ring buffer overflow, %d bytes dropped\n", 25562306a36Sopenharmony_ci urb->actual_length); 25662306a36Sopenharmony_ci dev->buffer_overflow = 1; 25762306a36Sopenharmony_ci } 25862306a36Sopenharmony_ci } 25962306a36Sopenharmony_ci 26062306a36Sopenharmony_ciresubmit: 26162306a36Sopenharmony_ci /* resubmit if we're still running */ 26262306a36Sopenharmony_ci if (dev->interrupt_in_running && !dev->buffer_overflow) { 26362306a36Sopenharmony_ci retval = usb_submit_urb(dev->interrupt_in_urb, GFP_ATOMIC); 26462306a36Sopenharmony_ci if (retval) { 26562306a36Sopenharmony_ci dev_err(&dev->intf->dev, 26662306a36Sopenharmony_ci "usb_submit_urb failed (%d)\n", retval); 26762306a36Sopenharmony_ci dev->buffer_overflow = 1; 26862306a36Sopenharmony_ci } 26962306a36Sopenharmony_ci } 27062306a36Sopenharmony_ci spin_unlock_irqrestore(&dev->rbsl, flags); 27162306a36Sopenharmony_ciexit: 27262306a36Sopenharmony_ci dev->interrupt_in_done = 1; 27362306a36Sopenharmony_ci wake_up_interruptible(&dev->read_wait); 27462306a36Sopenharmony_ci} 27562306a36Sopenharmony_ci 27662306a36Sopenharmony_ci/* 27762306a36Sopenharmony_ci * ld_usb_interrupt_out_callback 27862306a36Sopenharmony_ci */ 27962306a36Sopenharmony_cistatic void ld_usb_interrupt_out_callback(struct urb *urb) 28062306a36Sopenharmony_ci{ 28162306a36Sopenharmony_ci struct ld_usb *dev = urb->context; 28262306a36Sopenharmony_ci int status = urb->status; 28362306a36Sopenharmony_ci 28462306a36Sopenharmony_ci /* sync/async unlink faults aren't errors */ 28562306a36Sopenharmony_ci if (status && !(status == -ENOENT || 28662306a36Sopenharmony_ci status == -ECONNRESET || 28762306a36Sopenharmony_ci status == -ESHUTDOWN)) 28862306a36Sopenharmony_ci dev_dbg(&dev->intf->dev, 28962306a36Sopenharmony_ci "%s - nonzero write interrupt status received: %d\n", 29062306a36Sopenharmony_ci __func__, status); 29162306a36Sopenharmony_ci 29262306a36Sopenharmony_ci dev->interrupt_out_busy = 0; 29362306a36Sopenharmony_ci wake_up_interruptible(&dev->write_wait); 29462306a36Sopenharmony_ci} 29562306a36Sopenharmony_ci 29662306a36Sopenharmony_ci/* 29762306a36Sopenharmony_ci * ld_usb_open 29862306a36Sopenharmony_ci */ 29962306a36Sopenharmony_cistatic int ld_usb_open(struct inode *inode, struct file *file) 30062306a36Sopenharmony_ci{ 30162306a36Sopenharmony_ci struct ld_usb *dev; 30262306a36Sopenharmony_ci int subminor; 30362306a36Sopenharmony_ci int retval; 30462306a36Sopenharmony_ci struct usb_interface *interface; 30562306a36Sopenharmony_ci 30662306a36Sopenharmony_ci stream_open(inode, file); 30762306a36Sopenharmony_ci subminor = iminor(inode); 30862306a36Sopenharmony_ci 30962306a36Sopenharmony_ci interface = usb_find_interface(&ld_usb_driver, subminor); 31062306a36Sopenharmony_ci 31162306a36Sopenharmony_ci if (!interface) { 31262306a36Sopenharmony_ci printk(KERN_ERR "%s - error, can't find device for minor %d\n", 31362306a36Sopenharmony_ci __func__, subminor); 31462306a36Sopenharmony_ci return -ENODEV; 31562306a36Sopenharmony_ci } 31662306a36Sopenharmony_ci 31762306a36Sopenharmony_ci dev = usb_get_intfdata(interface); 31862306a36Sopenharmony_ci 31962306a36Sopenharmony_ci if (!dev) 32062306a36Sopenharmony_ci return -ENODEV; 32162306a36Sopenharmony_ci 32262306a36Sopenharmony_ci /* lock this device */ 32362306a36Sopenharmony_ci if (mutex_lock_interruptible(&dev->mutex)) 32462306a36Sopenharmony_ci return -ERESTARTSYS; 32562306a36Sopenharmony_ci 32662306a36Sopenharmony_ci /* allow opening only once */ 32762306a36Sopenharmony_ci if (dev->open_count) { 32862306a36Sopenharmony_ci retval = -EBUSY; 32962306a36Sopenharmony_ci goto unlock_exit; 33062306a36Sopenharmony_ci } 33162306a36Sopenharmony_ci dev->open_count = 1; 33262306a36Sopenharmony_ci 33362306a36Sopenharmony_ci /* initialize in direction */ 33462306a36Sopenharmony_ci dev->ring_head = 0; 33562306a36Sopenharmony_ci dev->ring_tail = 0; 33662306a36Sopenharmony_ci dev->buffer_overflow = 0; 33762306a36Sopenharmony_ci usb_fill_int_urb(dev->interrupt_in_urb, 33862306a36Sopenharmony_ci interface_to_usbdev(interface), 33962306a36Sopenharmony_ci usb_rcvintpipe(interface_to_usbdev(interface), 34062306a36Sopenharmony_ci dev->interrupt_in_endpoint->bEndpointAddress), 34162306a36Sopenharmony_ci dev->interrupt_in_buffer, 34262306a36Sopenharmony_ci dev->interrupt_in_endpoint_size, 34362306a36Sopenharmony_ci ld_usb_interrupt_in_callback, 34462306a36Sopenharmony_ci dev, 34562306a36Sopenharmony_ci dev->interrupt_in_interval); 34662306a36Sopenharmony_ci 34762306a36Sopenharmony_ci dev->interrupt_in_running = 1; 34862306a36Sopenharmony_ci dev->interrupt_in_done = 0; 34962306a36Sopenharmony_ci 35062306a36Sopenharmony_ci retval = usb_submit_urb(dev->interrupt_in_urb, GFP_KERNEL); 35162306a36Sopenharmony_ci if (retval) { 35262306a36Sopenharmony_ci dev_err(&interface->dev, "Couldn't submit interrupt_in_urb %d\n", retval); 35362306a36Sopenharmony_ci dev->interrupt_in_running = 0; 35462306a36Sopenharmony_ci dev->open_count = 0; 35562306a36Sopenharmony_ci goto unlock_exit; 35662306a36Sopenharmony_ci } 35762306a36Sopenharmony_ci 35862306a36Sopenharmony_ci /* save device in the file's private structure */ 35962306a36Sopenharmony_ci file->private_data = dev; 36062306a36Sopenharmony_ci 36162306a36Sopenharmony_ciunlock_exit: 36262306a36Sopenharmony_ci mutex_unlock(&dev->mutex); 36362306a36Sopenharmony_ci 36462306a36Sopenharmony_ci return retval; 36562306a36Sopenharmony_ci} 36662306a36Sopenharmony_ci 36762306a36Sopenharmony_ci/* 36862306a36Sopenharmony_ci * ld_usb_release 36962306a36Sopenharmony_ci */ 37062306a36Sopenharmony_cistatic int ld_usb_release(struct inode *inode, struct file *file) 37162306a36Sopenharmony_ci{ 37262306a36Sopenharmony_ci struct ld_usb *dev; 37362306a36Sopenharmony_ci int retval = 0; 37462306a36Sopenharmony_ci 37562306a36Sopenharmony_ci dev = file->private_data; 37662306a36Sopenharmony_ci 37762306a36Sopenharmony_ci if (dev == NULL) { 37862306a36Sopenharmony_ci retval = -ENODEV; 37962306a36Sopenharmony_ci goto exit; 38062306a36Sopenharmony_ci } 38162306a36Sopenharmony_ci 38262306a36Sopenharmony_ci mutex_lock(&dev->mutex); 38362306a36Sopenharmony_ci 38462306a36Sopenharmony_ci if (dev->open_count != 1) { 38562306a36Sopenharmony_ci retval = -ENODEV; 38662306a36Sopenharmony_ci goto unlock_exit; 38762306a36Sopenharmony_ci } 38862306a36Sopenharmony_ci if (dev->disconnected) { 38962306a36Sopenharmony_ci /* the device was unplugged before the file was released */ 39062306a36Sopenharmony_ci mutex_unlock(&dev->mutex); 39162306a36Sopenharmony_ci /* unlock here as ld_usb_delete frees dev */ 39262306a36Sopenharmony_ci ld_usb_delete(dev); 39362306a36Sopenharmony_ci goto exit; 39462306a36Sopenharmony_ci } 39562306a36Sopenharmony_ci 39662306a36Sopenharmony_ci /* wait until write transfer is finished */ 39762306a36Sopenharmony_ci if (dev->interrupt_out_busy) 39862306a36Sopenharmony_ci wait_event_interruptible_timeout(dev->write_wait, !dev->interrupt_out_busy, 2 * HZ); 39962306a36Sopenharmony_ci ld_usb_abort_transfers(dev); 40062306a36Sopenharmony_ci dev->open_count = 0; 40162306a36Sopenharmony_ci 40262306a36Sopenharmony_ciunlock_exit: 40362306a36Sopenharmony_ci mutex_unlock(&dev->mutex); 40462306a36Sopenharmony_ci 40562306a36Sopenharmony_ciexit: 40662306a36Sopenharmony_ci return retval; 40762306a36Sopenharmony_ci} 40862306a36Sopenharmony_ci 40962306a36Sopenharmony_ci/* 41062306a36Sopenharmony_ci * ld_usb_poll 41162306a36Sopenharmony_ci */ 41262306a36Sopenharmony_cistatic __poll_t ld_usb_poll(struct file *file, poll_table *wait) 41362306a36Sopenharmony_ci{ 41462306a36Sopenharmony_ci struct ld_usb *dev; 41562306a36Sopenharmony_ci __poll_t mask = 0; 41662306a36Sopenharmony_ci 41762306a36Sopenharmony_ci dev = file->private_data; 41862306a36Sopenharmony_ci 41962306a36Sopenharmony_ci if (dev->disconnected) 42062306a36Sopenharmony_ci return EPOLLERR | EPOLLHUP; 42162306a36Sopenharmony_ci 42262306a36Sopenharmony_ci poll_wait(file, &dev->read_wait, wait); 42362306a36Sopenharmony_ci poll_wait(file, &dev->write_wait, wait); 42462306a36Sopenharmony_ci 42562306a36Sopenharmony_ci if (dev->ring_head != dev->ring_tail) 42662306a36Sopenharmony_ci mask |= EPOLLIN | EPOLLRDNORM; 42762306a36Sopenharmony_ci if (!dev->interrupt_out_busy) 42862306a36Sopenharmony_ci mask |= EPOLLOUT | EPOLLWRNORM; 42962306a36Sopenharmony_ci 43062306a36Sopenharmony_ci return mask; 43162306a36Sopenharmony_ci} 43262306a36Sopenharmony_ci 43362306a36Sopenharmony_ci/* 43462306a36Sopenharmony_ci * ld_usb_read 43562306a36Sopenharmony_ci */ 43662306a36Sopenharmony_cistatic ssize_t ld_usb_read(struct file *file, char __user *buffer, size_t count, 43762306a36Sopenharmony_ci loff_t *ppos) 43862306a36Sopenharmony_ci{ 43962306a36Sopenharmony_ci struct ld_usb *dev; 44062306a36Sopenharmony_ci size_t *actual_buffer; 44162306a36Sopenharmony_ci size_t bytes_to_read; 44262306a36Sopenharmony_ci int retval = 0; 44362306a36Sopenharmony_ci int rv; 44462306a36Sopenharmony_ci 44562306a36Sopenharmony_ci dev = file->private_data; 44662306a36Sopenharmony_ci 44762306a36Sopenharmony_ci /* verify that we actually have some data to read */ 44862306a36Sopenharmony_ci if (count == 0) 44962306a36Sopenharmony_ci goto exit; 45062306a36Sopenharmony_ci 45162306a36Sopenharmony_ci /* lock this object */ 45262306a36Sopenharmony_ci if (mutex_lock_interruptible(&dev->mutex)) { 45362306a36Sopenharmony_ci retval = -ERESTARTSYS; 45462306a36Sopenharmony_ci goto exit; 45562306a36Sopenharmony_ci } 45662306a36Sopenharmony_ci 45762306a36Sopenharmony_ci /* verify that the device wasn't unplugged */ 45862306a36Sopenharmony_ci if (dev->disconnected) { 45962306a36Sopenharmony_ci retval = -ENODEV; 46062306a36Sopenharmony_ci printk(KERN_ERR "ldusb: No device or device unplugged %d\n", retval); 46162306a36Sopenharmony_ci goto unlock_exit; 46262306a36Sopenharmony_ci } 46362306a36Sopenharmony_ci 46462306a36Sopenharmony_ci /* wait for data */ 46562306a36Sopenharmony_ci spin_lock_irq(&dev->rbsl); 46662306a36Sopenharmony_ci while (dev->ring_head == dev->ring_tail) { 46762306a36Sopenharmony_ci dev->interrupt_in_done = 0; 46862306a36Sopenharmony_ci spin_unlock_irq(&dev->rbsl); 46962306a36Sopenharmony_ci if (file->f_flags & O_NONBLOCK) { 47062306a36Sopenharmony_ci retval = -EAGAIN; 47162306a36Sopenharmony_ci goto unlock_exit; 47262306a36Sopenharmony_ci } 47362306a36Sopenharmony_ci retval = wait_event_interruptible(dev->read_wait, dev->interrupt_in_done); 47462306a36Sopenharmony_ci if (retval < 0) 47562306a36Sopenharmony_ci goto unlock_exit; 47662306a36Sopenharmony_ci 47762306a36Sopenharmony_ci spin_lock_irq(&dev->rbsl); 47862306a36Sopenharmony_ci } 47962306a36Sopenharmony_ci spin_unlock_irq(&dev->rbsl); 48062306a36Sopenharmony_ci 48162306a36Sopenharmony_ci /* actual_buffer contains actual_length + interrupt_in_buffer */ 48262306a36Sopenharmony_ci actual_buffer = (size_t *)(dev->ring_buffer + dev->ring_tail * (sizeof(size_t)+dev->interrupt_in_endpoint_size)); 48362306a36Sopenharmony_ci if (*actual_buffer > dev->interrupt_in_endpoint_size) { 48462306a36Sopenharmony_ci retval = -EIO; 48562306a36Sopenharmony_ci goto unlock_exit; 48662306a36Sopenharmony_ci } 48762306a36Sopenharmony_ci bytes_to_read = min(count, *actual_buffer); 48862306a36Sopenharmony_ci if (bytes_to_read < *actual_buffer) 48962306a36Sopenharmony_ci dev_warn(&dev->intf->dev, "Read buffer overflow, %zu bytes dropped\n", 49062306a36Sopenharmony_ci *actual_buffer-bytes_to_read); 49162306a36Sopenharmony_ci 49262306a36Sopenharmony_ci /* copy one interrupt_in_buffer from ring_buffer into userspace */ 49362306a36Sopenharmony_ci if (copy_to_user(buffer, actual_buffer+1, bytes_to_read)) { 49462306a36Sopenharmony_ci retval = -EFAULT; 49562306a36Sopenharmony_ci goto unlock_exit; 49662306a36Sopenharmony_ci } 49762306a36Sopenharmony_ci retval = bytes_to_read; 49862306a36Sopenharmony_ci 49962306a36Sopenharmony_ci spin_lock_irq(&dev->rbsl); 50062306a36Sopenharmony_ci dev->ring_tail = (dev->ring_tail + 1) % ring_buffer_size; 50162306a36Sopenharmony_ci 50262306a36Sopenharmony_ci if (dev->buffer_overflow) { 50362306a36Sopenharmony_ci dev->buffer_overflow = 0; 50462306a36Sopenharmony_ci spin_unlock_irq(&dev->rbsl); 50562306a36Sopenharmony_ci rv = usb_submit_urb(dev->interrupt_in_urb, GFP_KERNEL); 50662306a36Sopenharmony_ci if (rv < 0) 50762306a36Sopenharmony_ci dev->buffer_overflow = 1; 50862306a36Sopenharmony_ci } else { 50962306a36Sopenharmony_ci spin_unlock_irq(&dev->rbsl); 51062306a36Sopenharmony_ci } 51162306a36Sopenharmony_ci 51262306a36Sopenharmony_ciunlock_exit: 51362306a36Sopenharmony_ci /* unlock the device */ 51462306a36Sopenharmony_ci mutex_unlock(&dev->mutex); 51562306a36Sopenharmony_ci 51662306a36Sopenharmony_ciexit: 51762306a36Sopenharmony_ci return retval; 51862306a36Sopenharmony_ci} 51962306a36Sopenharmony_ci 52062306a36Sopenharmony_ci/* 52162306a36Sopenharmony_ci * ld_usb_write 52262306a36Sopenharmony_ci */ 52362306a36Sopenharmony_cistatic ssize_t ld_usb_write(struct file *file, const char __user *buffer, 52462306a36Sopenharmony_ci size_t count, loff_t *ppos) 52562306a36Sopenharmony_ci{ 52662306a36Sopenharmony_ci struct ld_usb *dev; 52762306a36Sopenharmony_ci size_t bytes_to_write; 52862306a36Sopenharmony_ci int retval = 0; 52962306a36Sopenharmony_ci 53062306a36Sopenharmony_ci dev = file->private_data; 53162306a36Sopenharmony_ci 53262306a36Sopenharmony_ci /* verify that we actually have some data to write */ 53362306a36Sopenharmony_ci if (count == 0) 53462306a36Sopenharmony_ci goto exit; 53562306a36Sopenharmony_ci 53662306a36Sopenharmony_ci /* lock this object */ 53762306a36Sopenharmony_ci if (mutex_lock_interruptible(&dev->mutex)) { 53862306a36Sopenharmony_ci retval = -ERESTARTSYS; 53962306a36Sopenharmony_ci goto exit; 54062306a36Sopenharmony_ci } 54162306a36Sopenharmony_ci 54262306a36Sopenharmony_ci /* verify that the device wasn't unplugged */ 54362306a36Sopenharmony_ci if (dev->disconnected) { 54462306a36Sopenharmony_ci retval = -ENODEV; 54562306a36Sopenharmony_ci printk(KERN_ERR "ldusb: No device or device unplugged %d\n", retval); 54662306a36Sopenharmony_ci goto unlock_exit; 54762306a36Sopenharmony_ci } 54862306a36Sopenharmony_ci 54962306a36Sopenharmony_ci /* wait until previous transfer is finished */ 55062306a36Sopenharmony_ci if (dev->interrupt_out_busy) { 55162306a36Sopenharmony_ci if (file->f_flags & O_NONBLOCK) { 55262306a36Sopenharmony_ci retval = -EAGAIN; 55362306a36Sopenharmony_ci goto unlock_exit; 55462306a36Sopenharmony_ci } 55562306a36Sopenharmony_ci retval = wait_event_interruptible(dev->write_wait, !dev->interrupt_out_busy); 55662306a36Sopenharmony_ci if (retval < 0) { 55762306a36Sopenharmony_ci goto unlock_exit; 55862306a36Sopenharmony_ci } 55962306a36Sopenharmony_ci } 56062306a36Sopenharmony_ci 56162306a36Sopenharmony_ci /* write the data into interrupt_out_buffer from userspace */ 56262306a36Sopenharmony_ci bytes_to_write = min(count, write_buffer_size*dev->interrupt_out_endpoint_size); 56362306a36Sopenharmony_ci if (bytes_to_write < count) 56462306a36Sopenharmony_ci dev_warn(&dev->intf->dev, "Write buffer overflow, %zu bytes dropped\n", 56562306a36Sopenharmony_ci count - bytes_to_write); 56662306a36Sopenharmony_ci dev_dbg(&dev->intf->dev, "%s: count = %zu, bytes_to_write = %zu\n", 56762306a36Sopenharmony_ci __func__, count, bytes_to_write); 56862306a36Sopenharmony_ci 56962306a36Sopenharmony_ci if (copy_from_user(dev->interrupt_out_buffer, buffer, bytes_to_write)) { 57062306a36Sopenharmony_ci retval = -EFAULT; 57162306a36Sopenharmony_ci goto unlock_exit; 57262306a36Sopenharmony_ci } 57362306a36Sopenharmony_ci 57462306a36Sopenharmony_ci if (dev->interrupt_out_endpoint == NULL) { 57562306a36Sopenharmony_ci /* try HID_REQ_SET_REPORT=9 on control_endpoint instead of interrupt_out_endpoint */ 57662306a36Sopenharmony_ci retval = usb_control_msg(interface_to_usbdev(dev->intf), 57762306a36Sopenharmony_ci usb_sndctrlpipe(interface_to_usbdev(dev->intf), 0), 57862306a36Sopenharmony_ci 9, 57962306a36Sopenharmony_ci USB_TYPE_CLASS | USB_RECIP_INTERFACE | USB_DIR_OUT, 58062306a36Sopenharmony_ci 1 << 8, 0, 58162306a36Sopenharmony_ci dev->interrupt_out_buffer, 58262306a36Sopenharmony_ci bytes_to_write, 58362306a36Sopenharmony_ci USB_CTRL_SET_TIMEOUT); 58462306a36Sopenharmony_ci if (retval < 0) 58562306a36Sopenharmony_ci dev_err(&dev->intf->dev, 58662306a36Sopenharmony_ci "Couldn't submit HID_REQ_SET_REPORT %d\n", 58762306a36Sopenharmony_ci retval); 58862306a36Sopenharmony_ci goto unlock_exit; 58962306a36Sopenharmony_ci } 59062306a36Sopenharmony_ci 59162306a36Sopenharmony_ci /* send off the urb */ 59262306a36Sopenharmony_ci usb_fill_int_urb(dev->interrupt_out_urb, 59362306a36Sopenharmony_ci interface_to_usbdev(dev->intf), 59462306a36Sopenharmony_ci usb_sndintpipe(interface_to_usbdev(dev->intf), 59562306a36Sopenharmony_ci dev->interrupt_out_endpoint->bEndpointAddress), 59662306a36Sopenharmony_ci dev->interrupt_out_buffer, 59762306a36Sopenharmony_ci bytes_to_write, 59862306a36Sopenharmony_ci ld_usb_interrupt_out_callback, 59962306a36Sopenharmony_ci dev, 60062306a36Sopenharmony_ci dev->interrupt_out_interval); 60162306a36Sopenharmony_ci 60262306a36Sopenharmony_ci dev->interrupt_out_busy = 1; 60362306a36Sopenharmony_ci wmb(); 60462306a36Sopenharmony_ci 60562306a36Sopenharmony_ci retval = usb_submit_urb(dev->interrupt_out_urb, GFP_KERNEL); 60662306a36Sopenharmony_ci if (retval) { 60762306a36Sopenharmony_ci dev->interrupt_out_busy = 0; 60862306a36Sopenharmony_ci dev_err(&dev->intf->dev, 60962306a36Sopenharmony_ci "Couldn't submit interrupt_out_urb %d\n", retval); 61062306a36Sopenharmony_ci goto unlock_exit; 61162306a36Sopenharmony_ci } 61262306a36Sopenharmony_ci retval = bytes_to_write; 61362306a36Sopenharmony_ci 61462306a36Sopenharmony_ciunlock_exit: 61562306a36Sopenharmony_ci /* unlock the device */ 61662306a36Sopenharmony_ci mutex_unlock(&dev->mutex); 61762306a36Sopenharmony_ci 61862306a36Sopenharmony_ciexit: 61962306a36Sopenharmony_ci return retval; 62062306a36Sopenharmony_ci} 62162306a36Sopenharmony_ci 62262306a36Sopenharmony_ci/* file operations needed when we register this driver */ 62362306a36Sopenharmony_cistatic const struct file_operations ld_usb_fops = { 62462306a36Sopenharmony_ci .owner = THIS_MODULE, 62562306a36Sopenharmony_ci .read = ld_usb_read, 62662306a36Sopenharmony_ci .write = ld_usb_write, 62762306a36Sopenharmony_ci .open = ld_usb_open, 62862306a36Sopenharmony_ci .release = ld_usb_release, 62962306a36Sopenharmony_ci .poll = ld_usb_poll, 63062306a36Sopenharmony_ci .llseek = no_llseek, 63162306a36Sopenharmony_ci}; 63262306a36Sopenharmony_ci 63362306a36Sopenharmony_ci/* 63462306a36Sopenharmony_ci * usb class driver info in order to get a minor number from the usb core, 63562306a36Sopenharmony_ci * and to have the device registered with the driver core 63662306a36Sopenharmony_ci */ 63762306a36Sopenharmony_cistatic struct usb_class_driver ld_usb_class = { 63862306a36Sopenharmony_ci .name = "ldusb%d", 63962306a36Sopenharmony_ci .fops = &ld_usb_fops, 64062306a36Sopenharmony_ci .minor_base = USB_LD_MINOR_BASE, 64162306a36Sopenharmony_ci}; 64262306a36Sopenharmony_ci 64362306a36Sopenharmony_ci/* 64462306a36Sopenharmony_ci * ld_usb_probe 64562306a36Sopenharmony_ci * 64662306a36Sopenharmony_ci * Called by the usb core when a new device is connected that it thinks 64762306a36Sopenharmony_ci * this driver might be interested in. 64862306a36Sopenharmony_ci */ 64962306a36Sopenharmony_cistatic int ld_usb_probe(struct usb_interface *intf, const struct usb_device_id *id) 65062306a36Sopenharmony_ci{ 65162306a36Sopenharmony_ci struct usb_device *udev = interface_to_usbdev(intf); 65262306a36Sopenharmony_ci struct ld_usb *dev = NULL; 65362306a36Sopenharmony_ci struct usb_host_interface *iface_desc; 65462306a36Sopenharmony_ci char *buffer; 65562306a36Sopenharmony_ci int retval = -ENOMEM; 65662306a36Sopenharmony_ci int res; 65762306a36Sopenharmony_ci 65862306a36Sopenharmony_ci /* allocate memory for our device state and initialize it */ 65962306a36Sopenharmony_ci 66062306a36Sopenharmony_ci dev = kzalloc(sizeof(*dev), GFP_KERNEL); 66162306a36Sopenharmony_ci if (!dev) 66262306a36Sopenharmony_ci goto exit; 66362306a36Sopenharmony_ci mutex_init(&dev->mutex); 66462306a36Sopenharmony_ci spin_lock_init(&dev->rbsl); 66562306a36Sopenharmony_ci dev->intf = intf; 66662306a36Sopenharmony_ci init_waitqueue_head(&dev->read_wait); 66762306a36Sopenharmony_ci init_waitqueue_head(&dev->write_wait); 66862306a36Sopenharmony_ci 66962306a36Sopenharmony_ci /* workaround for early firmware versions on fast computers */ 67062306a36Sopenharmony_ci if ((le16_to_cpu(udev->descriptor.idVendor) == USB_VENDOR_ID_LD) && 67162306a36Sopenharmony_ci ((le16_to_cpu(udev->descriptor.idProduct) == USB_DEVICE_ID_LD_CASSY) || 67262306a36Sopenharmony_ci (le16_to_cpu(udev->descriptor.idProduct) == USB_DEVICE_ID_LD_COM3LAB)) && 67362306a36Sopenharmony_ci (le16_to_cpu(udev->descriptor.bcdDevice) <= 0x103)) { 67462306a36Sopenharmony_ci buffer = kmalloc(256, GFP_KERNEL); 67562306a36Sopenharmony_ci if (!buffer) 67662306a36Sopenharmony_ci goto error; 67762306a36Sopenharmony_ci /* usb_string makes SETUP+STALL to leave always ControlReadLoop */ 67862306a36Sopenharmony_ci usb_string(udev, 255, buffer, 256); 67962306a36Sopenharmony_ci kfree(buffer); 68062306a36Sopenharmony_ci } 68162306a36Sopenharmony_ci 68262306a36Sopenharmony_ci iface_desc = intf->cur_altsetting; 68362306a36Sopenharmony_ci 68462306a36Sopenharmony_ci res = usb_find_last_int_in_endpoint(iface_desc, 68562306a36Sopenharmony_ci &dev->interrupt_in_endpoint); 68662306a36Sopenharmony_ci if (res) { 68762306a36Sopenharmony_ci dev_err(&intf->dev, "Interrupt in endpoint not found\n"); 68862306a36Sopenharmony_ci retval = res; 68962306a36Sopenharmony_ci goto error; 69062306a36Sopenharmony_ci } 69162306a36Sopenharmony_ci 69262306a36Sopenharmony_ci res = usb_find_last_int_out_endpoint(iface_desc, 69362306a36Sopenharmony_ci &dev->interrupt_out_endpoint); 69462306a36Sopenharmony_ci if (res) 69562306a36Sopenharmony_ci dev_warn(&intf->dev, "Interrupt out endpoint not found (using control endpoint instead)\n"); 69662306a36Sopenharmony_ci 69762306a36Sopenharmony_ci dev->interrupt_in_endpoint_size = usb_endpoint_maxp(dev->interrupt_in_endpoint); 69862306a36Sopenharmony_ci dev->ring_buffer = kcalloc(ring_buffer_size, 69962306a36Sopenharmony_ci sizeof(size_t) + dev->interrupt_in_endpoint_size, 70062306a36Sopenharmony_ci GFP_KERNEL); 70162306a36Sopenharmony_ci if (!dev->ring_buffer) 70262306a36Sopenharmony_ci goto error; 70362306a36Sopenharmony_ci dev->interrupt_in_buffer = kmalloc(dev->interrupt_in_endpoint_size, GFP_KERNEL); 70462306a36Sopenharmony_ci if (!dev->interrupt_in_buffer) 70562306a36Sopenharmony_ci goto error; 70662306a36Sopenharmony_ci dev->interrupt_in_urb = usb_alloc_urb(0, GFP_KERNEL); 70762306a36Sopenharmony_ci if (!dev->interrupt_in_urb) 70862306a36Sopenharmony_ci goto error; 70962306a36Sopenharmony_ci dev->interrupt_out_endpoint_size = dev->interrupt_out_endpoint ? usb_endpoint_maxp(dev->interrupt_out_endpoint) : 71062306a36Sopenharmony_ci udev->descriptor.bMaxPacketSize0; 71162306a36Sopenharmony_ci dev->interrupt_out_buffer = 71262306a36Sopenharmony_ci kmalloc_array(write_buffer_size, 71362306a36Sopenharmony_ci dev->interrupt_out_endpoint_size, GFP_KERNEL); 71462306a36Sopenharmony_ci if (!dev->interrupt_out_buffer) 71562306a36Sopenharmony_ci goto error; 71662306a36Sopenharmony_ci dev->interrupt_out_urb = usb_alloc_urb(0, GFP_KERNEL); 71762306a36Sopenharmony_ci if (!dev->interrupt_out_urb) 71862306a36Sopenharmony_ci goto error; 71962306a36Sopenharmony_ci dev->interrupt_in_interval = max_t(int, min_interrupt_in_interval, 72062306a36Sopenharmony_ci dev->interrupt_in_endpoint->bInterval); 72162306a36Sopenharmony_ci if (dev->interrupt_out_endpoint) 72262306a36Sopenharmony_ci dev->interrupt_out_interval = max_t(int, min_interrupt_out_interval, 72362306a36Sopenharmony_ci dev->interrupt_out_endpoint->bInterval); 72462306a36Sopenharmony_ci 72562306a36Sopenharmony_ci /* we can register the device now, as it is ready */ 72662306a36Sopenharmony_ci usb_set_intfdata(intf, dev); 72762306a36Sopenharmony_ci 72862306a36Sopenharmony_ci retval = usb_register_dev(intf, &ld_usb_class); 72962306a36Sopenharmony_ci if (retval) { 73062306a36Sopenharmony_ci /* something prevented us from registering this driver */ 73162306a36Sopenharmony_ci dev_err(&intf->dev, "Not able to get a minor for this device.\n"); 73262306a36Sopenharmony_ci usb_set_intfdata(intf, NULL); 73362306a36Sopenharmony_ci goto error; 73462306a36Sopenharmony_ci } 73562306a36Sopenharmony_ci 73662306a36Sopenharmony_ci /* let the user know what node this device is now attached to */ 73762306a36Sopenharmony_ci dev_info(&intf->dev, "LD USB Device #%d now attached to major %d minor %d\n", 73862306a36Sopenharmony_ci (intf->minor - USB_LD_MINOR_BASE), USB_MAJOR, intf->minor); 73962306a36Sopenharmony_ci 74062306a36Sopenharmony_ciexit: 74162306a36Sopenharmony_ci return retval; 74262306a36Sopenharmony_ci 74362306a36Sopenharmony_cierror: 74462306a36Sopenharmony_ci ld_usb_delete(dev); 74562306a36Sopenharmony_ci 74662306a36Sopenharmony_ci return retval; 74762306a36Sopenharmony_ci} 74862306a36Sopenharmony_ci 74962306a36Sopenharmony_ci/* 75062306a36Sopenharmony_ci * ld_usb_disconnect 75162306a36Sopenharmony_ci * 75262306a36Sopenharmony_ci * Called by the usb core when the device is removed from the system. 75362306a36Sopenharmony_ci */ 75462306a36Sopenharmony_cistatic void ld_usb_disconnect(struct usb_interface *intf) 75562306a36Sopenharmony_ci{ 75662306a36Sopenharmony_ci struct ld_usb *dev; 75762306a36Sopenharmony_ci int minor; 75862306a36Sopenharmony_ci 75962306a36Sopenharmony_ci dev = usb_get_intfdata(intf); 76062306a36Sopenharmony_ci usb_set_intfdata(intf, NULL); 76162306a36Sopenharmony_ci 76262306a36Sopenharmony_ci minor = intf->minor; 76362306a36Sopenharmony_ci 76462306a36Sopenharmony_ci /* give back our minor */ 76562306a36Sopenharmony_ci usb_deregister_dev(intf, &ld_usb_class); 76662306a36Sopenharmony_ci 76762306a36Sopenharmony_ci usb_poison_urb(dev->interrupt_in_urb); 76862306a36Sopenharmony_ci usb_poison_urb(dev->interrupt_out_urb); 76962306a36Sopenharmony_ci 77062306a36Sopenharmony_ci mutex_lock(&dev->mutex); 77162306a36Sopenharmony_ci 77262306a36Sopenharmony_ci /* if the device is not opened, then we clean up right now */ 77362306a36Sopenharmony_ci if (!dev->open_count) { 77462306a36Sopenharmony_ci mutex_unlock(&dev->mutex); 77562306a36Sopenharmony_ci ld_usb_delete(dev); 77662306a36Sopenharmony_ci } else { 77762306a36Sopenharmony_ci dev->disconnected = 1; 77862306a36Sopenharmony_ci /* wake up pollers */ 77962306a36Sopenharmony_ci wake_up_interruptible_all(&dev->read_wait); 78062306a36Sopenharmony_ci wake_up_interruptible_all(&dev->write_wait); 78162306a36Sopenharmony_ci mutex_unlock(&dev->mutex); 78262306a36Sopenharmony_ci } 78362306a36Sopenharmony_ci 78462306a36Sopenharmony_ci dev_info(&intf->dev, "LD USB Device #%d now disconnected\n", 78562306a36Sopenharmony_ci (minor - USB_LD_MINOR_BASE)); 78662306a36Sopenharmony_ci} 78762306a36Sopenharmony_ci 78862306a36Sopenharmony_ci/* usb specific object needed to register this driver with the usb subsystem */ 78962306a36Sopenharmony_cistatic struct usb_driver ld_usb_driver = { 79062306a36Sopenharmony_ci .name = "ldusb", 79162306a36Sopenharmony_ci .probe = ld_usb_probe, 79262306a36Sopenharmony_ci .disconnect = ld_usb_disconnect, 79362306a36Sopenharmony_ci .id_table = ld_usb_table, 79462306a36Sopenharmony_ci}; 79562306a36Sopenharmony_ci 79662306a36Sopenharmony_cimodule_usb_driver(ld_usb_driver); 79762306a36Sopenharmony_ci 798