162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0 262306a36Sopenharmony_ci/* Driver for Microtek Scanmaker X6 USB scanner, and possibly others. 362306a36Sopenharmony_ci * 462306a36Sopenharmony_ci * (C) Copyright 2000 John Fremlin <vii@penguinpowered.com> 562306a36Sopenharmony_ci * (C) Copyright 2000 Oliver Neukum <Oliver.Neukum@lrz.uni-muenchen.de> 662306a36Sopenharmony_ci * 762306a36Sopenharmony_ci * Parts shamelessly stolen from usb-storage and copyright by their 862306a36Sopenharmony_ci * authors. Thanks to Matt Dharm for giving us permission! 962306a36Sopenharmony_ci * 1062306a36Sopenharmony_ci * This driver implements a SCSI host controller driver and a USB 1162306a36Sopenharmony_ci * device driver. To avoid confusion, all the USB related stuff is 1262306a36Sopenharmony_ci * prefixed by mts_usb_ and all the SCSI stuff by mts_scsi_. 1362306a36Sopenharmony_ci * 1462306a36Sopenharmony_ci * Microtek (www.microtek.com) did not release the specifications for 1562306a36Sopenharmony_ci * their USB protocol to us, so we had to reverse engineer them. We 1662306a36Sopenharmony_ci * don't know for which models they are valid. 1762306a36Sopenharmony_ci * 1862306a36Sopenharmony_ci * The X6 USB has three bulk endpoints, one output (0x1) down which 1962306a36Sopenharmony_ci * commands and outgoing data are sent, and two input: 0x82 from which 2062306a36Sopenharmony_ci * normal data is read from the scanner (in packets of maximum 32 2162306a36Sopenharmony_ci * bytes) and from which the status byte is read, and 0x83 from which 2262306a36Sopenharmony_ci * the results of a scan (or preview) are read in up to 64 * 1024 byte 2362306a36Sopenharmony_ci * chunks by the Windows driver. We don't know how much it is possible 2462306a36Sopenharmony_ci * to read at a time from 0x83. 2562306a36Sopenharmony_ci * 2662306a36Sopenharmony_ci * It seems possible to read (with URB transfers) everything from 0x82 2762306a36Sopenharmony_ci * in one go, without bothering to read in 32 byte chunks. 2862306a36Sopenharmony_ci * 2962306a36Sopenharmony_ci * There seems to be an optimisation of a further READ implicit if 3062306a36Sopenharmony_ci * you simply read from 0x83. 3162306a36Sopenharmony_ci * 3262306a36Sopenharmony_ci * Guessed protocol: 3362306a36Sopenharmony_ci * 3462306a36Sopenharmony_ci * Send raw SCSI command to EP 0x1 3562306a36Sopenharmony_ci * 3662306a36Sopenharmony_ci * If there is data to receive: 3762306a36Sopenharmony_ci * If the command was READ datatype=image: 3862306a36Sopenharmony_ci * Read a lot of data from EP 0x83 3962306a36Sopenharmony_ci * Else: 4062306a36Sopenharmony_ci * Read data from EP 0x82 4162306a36Sopenharmony_ci * Else: 4262306a36Sopenharmony_ci * If there is data to transmit: 4362306a36Sopenharmony_ci * Write it to EP 0x1 4462306a36Sopenharmony_ci * 4562306a36Sopenharmony_ci * Read status byte from EP 0x82 4662306a36Sopenharmony_ci * 4762306a36Sopenharmony_ci * References: 4862306a36Sopenharmony_ci * 4962306a36Sopenharmony_ci * The SCSI command set for the scanner is available from 5062306a36Sopenharmony_ci * ftp://ftp.microtek.com/microtek/devpack/ 5162306a36Sopenharmony_ci * 5262306a36Sopenharmony_ci * Microtek NV sent us a more up to date version of the document. If 5362306a36Sopenharmony_ci * you want it, just send mail. 5462306a36Sopenharmony_ci * 5562306a36Sopenharmony_ci * Status: 5662306a36Sopenharmony_ci * 5762306a36Sopenharmony_ci * Untested with multiple scanners. 5862306a36Sopenharmony_ci * Untested on SMP. 5962306a36Sopenharmony_ci * Untested on a bigendian machine. 6062306a36Sopenharmony_ci * 6162306a36Sopenharmony_ci * History: 6262306a36Sopenharmony_ci * 6362306a36Sopenharmony_ci * 20000417 starting history 6462306a36Sopenharmony_ci * 20000417 fixed load oops 6562306a36Sopenharmony_ci * 20000417 fixed unload oops 6662306a36Sopenharmony_ci * 20000419 fixed READ IMAGE detection 6762306a36Sopenharmony_ci * 20000424 started conversion to use URBs 6862306a36Sopenharmony_ci * 20000502 handled short transfers as errors 6962306a36Sopenharmony_ci * 20000513 rename and organisation of functions (john) 7062306a36Sopenharmony_ci * 20000513 added IDs for all products supported by Windows driver (john) 7162306a36Sopenharmony_ci * 20000514 Rewrote mts_scsi_queuecommand to use URBs (john) 7262306a36Sopenharmony_ci * 20000514 Version 0.0.8j 7362306a36Sopenharmony_ci * 20000514 Fix reporting of non-existent devices to SCSI layer (john) 7462306a36Sopenharmony_ci * 20000514 Added MTS_DEBUG_INT (john) 7562306a36Sopenharmony_ci * 20000514 Changed "usb-microtek" to "microtek" for consistency (john) 7662306a36Sopenharmony_ci * 20000514 Stupid bug fixes (john) 7762306a36Sopenharmony_ci * 20000514 Version 0.0.9j 7862306a36Sopenharmony_ci * 20000515 Put transfer context and URB in mts_desc (john) 7962306a36Sopenharmony_ci * 20000515 Added prelim turn off debugging support (john) 8062306a36Sopenharmony_ci * 20000515 Version 0.0.10j 8162306a36Sopenharmony_ci * 20000515 Fixed up URB allocation (clear URB on alloc) (john) 8262306a36Sopenharmony_ci * 20000515 Version 0.0.11j 8362306a36Sopenharmony_ci * 20000516 Removed unnecessary spinlock in mts_transfer_context (john) 8462306a36Sopenharmony_ci * 20000516 Removed unnecessary up on instance lock in mts_remove_nolock (john) 8562306a36Sopenharmony_ci * 20000516 Implemented (badly) scsi_abort (john) 8662306a36Sopenharmony_ci * 20000516 Version 0.0.12j 8762306a36Sopenharmony_ci * 20000517 Hopefully removed mts_remove_nolock quasideadlock (john) 8862306a36Sopenharmony_ci * 20000517 Added mts_debug_dump to print ll USB info (john) 8962306a36Sopenharmony_ci * 20000518 Tweaks and documentation updates (john) 9062306a36Sopenharmony_ci * 20000518 Version 0.0.13j 9162306a36Sopenharmony_ci * 20000518 Cleaned up abort handling (john) 9262306a36Sopenharmony_ci * 20000523 Removed scsi_command and various scsi_..._resets (john) 9362306a36Sopenharmony_ci * 20000523 Added unlink URB on scsi_abort, now OHCI supports it (john) 9462306a36Sopenharmony_ci * 20000523 Fixed last tiresome compile warning (john) 9562306a36Sopenharmony_ci * 20000523 Version 0.0.14j (though version 0.1 has come out?) 9662306a36Sopenharmony_ci * 20000602 Added primitive reset 9762306a36Sopenharmony_ci * 20000602 Version 0.2.0 9862306a36Sopenharmony_ci * 20000603 various cosmetic changes 9962306a36Sopenharmony_ci * 20000603 Version 0.2.1 10062306a36Sopenharmony_ci * 20000620 minor cosmetic changes 10162306a36Sopenharmony_ci * 20000620 Version 0.2.2 10262306a36Sopenharmony_ci * 20000822 Hopefully fixed deadlock in mts_remove_nolock() 10362306a36Sopenharmony_ci * 20000822 Fixed minor race in mts_transfer_cleanup() 10462306a36Sopenharmony_ci * 20000822 Fixed deadlock on submission error in queuecommand 10562306a36Sopenharmony_ci * 20000822 Version 0.2.3 10662306a36Sopenharmony_ci * 20000913 Reduced module size if debugging is off 10762306a36Sopenharmony_ci * 20000913 Version 0.2.4 10862306a36Sopenharmony_ci * 20010210 New abort logic 10962306a36Sopenharmony_ci * 20010210 Version 0.3.0 11062306a36Sopenharmony_ci * 20010217 Merged scatter/gather 11162306a36Sopenharmony_ci * 20010218 Version 0.4.0 11262306a36Sopenharmony_ci * 20010218 Cosmetic fixes 11362306a36Sopenharmony_ci * 20010218 Version 0.4.1 11462306a36Sopenharmony_ci * 20010306 Abort while using scatter/gather 11562306a36Sopenharmony_ci * 20010306 Version 0.4.2 11662306a36Sopenharmony_ci * 20010311 Remove all timeouts and tidy up generally (john) 11762306a36Sopenharmony_ci * 20010320 check return value of scsi_register() 11862306a36Sopenharmony_ci * 20010320 Version 0.4.3 11962306a36Sopenharmony_ci * 20010408 Identify version on module load. 12062306a36Sopenharmony_ci * 20011003 Fix multiple requests 12162306a36Sopenharmony_ci */ 12262306a36Sopenharmony_ci 12362306a36Sopenharmony_ci#include <linux/module.h> 12462306a36Sopenharmony_ci#include <linux/kernel.h> 12562306a36Sopenharmony_ci#include <linux/signal.h> 12662306a36Sopenharmony_ci#include <linux/errno.h> 12762306a36Sopenharmony_ci#include <linux/random.h> 12862306a36Sopenharmony_ci#include <linux/poll.h> 12962306a36Sopenharmony_ci#include <linux/slab.h> 13062306a36Sopenharmony_ci#include <linux/spinlock.h> 13162306a36Sopenharmony_ci#include <linux/usb.h> 13262306a36Sopenharmony_ci#include <linux/proc_fs.h> 13362306a36Sopenharmony_ci#include <linux/atomic.h> 13462306a36Sopenharmony_ci#include <linux/blkdev.h> 13562306a36Sopenharmony_ci 13662306a36Sopenharmony_ci#include <scsi/scsi.h> 13762306a36Sopenharmony_ci#include <scsi/scsi_cmnd.h> 13862306a36Sopenharmony_ci#include <scsi/scsi_device.h> 13962306a36Sopenharmony_ci#include <scsi/scsi_eh.h> 14062306a36Sopenharmony_ci#include <scsi/scsi_host.h> 14162306a36Sopenharmony_ci#include <scsi/scsi_tcq.h> 14262306a36Sopenharmony_ci 14362306a36Sopenharmony_ci#include "microtek.h" 14462306a36Sopenharmony_ci 14562306a36Sopenharmony_ci#define DRIVER_AUTHOR "John Fremlin <vii@penguinpowered.com>, Oliver Neukum <Oliver.Neukum@lrz.uni-muenchen.de>" 14662306a36Sopenharmony_ci#define DRIVER_DESC "Microtek Scanmaker X6 USB scanner driver" 14762306a36Sopenharmony_ci 14862306a36Sopenharmony_ci/* Should we do debugging? */ 14962306a36Sopenharmony_ci 15062306a36Sopenharmony_ci//#define MTS_DO_DEBUG 15162306a36Sopenharmony_ci 15262306a36Sopenharmony_ci/* USB layer driver interface */ 15362306a36Sopenharmony_ci 15462306a36Sopenharmony_cistatic int mts_usb_probe(struct usb_interface *intf, 15562306a36Sopenharmony_ci const struct usb_device_id *id); 15662306a36Sopenharmony_cistatic void mts_usb_disconnect(struct usb_interface *intf); 15762306a36Sopenharmony_ci 15862306a36Sopenharmony_cistatic const struct usb_device_id mts_usb_ids[]; 15962306a36Sopenharmony_ci 16062306a36Sopenharmony_cistatic struct usb_driver mts_usb_driver = { 16162306a36Sopenharmony_ci .name = "microtekX6", 16262306a36Sopenharmony_ci .probe = mts_usb_probe, 16362306a36Sopenharmony_ci .disconnect = mts_usb_disconnect, 16462306a36Sopenharmony_ci .id_table = mts_usb_ids, 16562306a36Sopenharmony_ci}; 16662306a36Sopenharmony_ci 16762306a36Sopenharmony_ci 16862306a36Sopenharmony_ci/* Internal driver stuff */ 16962306a36Sopenharmony_ci 17062306a36Sopenharmony_ci#define MTS_VERSION "0.4.3" 17162306a36Sopenharmony_ci#define MTS_NAME "microtek usb (rev " MTS_VERSION "): " 17262306a36Sopenharmony_ci 17362306a36Sopenharmony_ci#define MTS_WARNING(x...) \ 17462306a36Sopenharmony_ci printk( KERN_WARNING MTS_NAME x ) 17562306a36Sopenharmony_ci#define MTS_ERROR(x...) \ 17662306a36Sopenharmony_ci printk( KERN_ERR MTS_NAME x ) 17762306a36Sopenharmony_ci#define MTS_INT_ERROR(x...) \ 17862306a36Sopenharmony_ci MTS_ERROR(x) 17962306a36Sopenharmony_ci#define MTS_MESSAGE(x...) \ 18062306a36Sopenharmony_ci printk( KERN_INFO MTS_NAME x ) 18162306a36Sopenharmony_ci 18262306a36Sopenharmony_ci#if defined MTS_DO_DEBUG 18362306a36Sopenharmony_ci 18462306a36Sopenharmony_ci#define MTS_DEBUG(x...) \ 18562306a36Sopenharmony_ci printk( KERN_DEBUG MTS_NAME x ) 18662306a36Sopenharmony_ci 18762306a36Sopenharmony_ci#define MTS_DEBUG_GOT_HERE() \ 18862306a36Sopenharmony_ci MTS_DEBUG("got to %s:%d (%s)\n", __FILE__, (int)__LINE__, __func__ ) 18962306a36Sopenharmony_ci#define MTS_DEBUG_INT() \ 19062306a36Sopenharmony_ci do { MTS_DEBUG_GOT_HERE(); \ 19162306a36Sopenharmony_ci MTS_DEBUG("transfer = 0x%x context = 0x%x\n",(int)transfer,(int)context ); \ 19262306a36Sopenharmony_ci MTS_DEBUG("status = 0x%x data-length = 0x%x sent = 0x%x\n",transfer->status,(int)context->data_length, (int)transfer->actual_length ); \ 19362306a36Sopenharmony_ci mts_debug_dump(context->instance);\ 19462306a36Sopenharmony_ci } while(0) 19562306a36Sopenharmony_ci#else 19662306a36Sopenharmony_ci 19762306a36Sopenharmony_ci#define MTS_NUL_STATEMENT do { } while(0) 19862306a36Sopenharmony_ci 19962306a36Sopenharmony_ci#define MTS_DEBUG(x...) MTS_NUL_STATEMENT 20062306a36Sopenharmony_ci#define MTS_DEBUG_GOT_HERE() MTS_NUL_STATEMENT 20162306a36Sopenharmony_ci#define MTS_DEBUG_INT() MTS_NUL_STATEMENT 20262306a36Sopenharmony_ci 20362306a36Sopenharmony_ci#endif 20462306a36Sopenharmony_ci 20562306a36Sopenharmony_ci 20662306a36Sopenharmony_ci 20762306a36Sopenharmony_ci#define MTS_INT_INIT()\ 20862306a36Sopenharmony_ci struct mts_transfer_context* context = (struct mts_transfer_context*)transfer->context; \ 20962306a36Sopenharmony_ci MTS_DEBUG_INT();\ 21062306a36Sopenharmony_ci 21162306a36Sopenharmony_ci#ifdef MTS_DO_DEBUG 21262306a36Sopenharmony_ci 21362306a36Sopenharmony_cistatic inline void mts_debug_dump(struct mts_desc* desc) { 21462306a36Sopenharmony_ci MTS_DEBUG("desc at 0x%x: toggle = %02x%02x\n", 21562306a36Sopenharmony_ci (int)desc, 21662306a36Sopenharmony_ci (int)desc->usb_dev->toggle[1],(int)desc->usb_dev->toggle[0] 21762306a36Sopenharmony_ci ); 21862306a36Sopenharmony_ci MTS_DEBUG("ep_out=%x ep_response=%x ep_image=%x\n", 21962306a36Sopenharmony_ci usb_sndbulkpipe(desc->usb_dev,desc->ep_out), 22062306a36Sopenharmony_ci usb_rcvbulkpipe(desc->usb_dev,desc->ep_response), 22162306a36Sopenharmony_ci usb_rcvbulkpipe(desc->usb_dev,desc->ep_image) 22262306a36Sopenharmony_ci ); 22362306a36Sopenharmony_ci} 22462306a36Sopenharmony_ci 22562306a36Sopenharmony_ci 22662306a36Sopenharmony_cistatic inline void mts_show_command(struct scsi_cmnd *srb) 22762306a36Sopenharmony_ci{ 22862306a36Sopenharmony_ci char *what = NULL; 22962306a36Sopenharmony_ci 23062306a36Sopenharmony_ci switch (srb->cmnd[0]) { 23162306a36Sopenharmony_ci case TEST_UNIT_READY: what = "TEST_UNIT_READY"; break; 23262306a36Sopenharmony_ci case REZERO_UNIT: what = "REZERO_UNIT"; break; 23362306a36Sopenharmony_ci case REQUEST_SENSE: what = "REQUEST_SENSE"; break; 23462306a36Sopenharmony_ci case FORMAT_UNIT: what = "FORMAT_UNIT"; break; 23562306a36Sopenharmony_ci case READ_BLOCK_LIMITS: what = "READ_BLOCK_LIMITS"; break; 23662306a36Sopenharmony_ci case REASSIGN_BLOCKS: what = "REASSIGN_BLOCKS"; break; 23762306a36Sopenharmony_ci case READ_6: what = "READ_6"; break; 23862306a36Sopenharmony_ci case WRITE_6: what = "WRITE_6"; break; 23962306a36Sopenharmony_ci case SEEK_6: what = "SEEK_6"; break; 24062306a36Sopenharmony_ci case READ_REVERSE: what = "READ_REVERSE"; break; 24162306a36Sopenharmony_ci case WRITE_FILEMARKS: what = "WRITE_FILEMARKS"; break; 24262306a36Sopenharmony_ci case SPACE: what = "SPACE"; break; 24362306a36Sopenharmony_ci case INQUIRY: what = "INQUIRY"; break; 24462306a36Sopenharmony_ci case RECOVER_BUFFERED_DATA: what = "RECOVER_BUFFERED_DATA"; break; 24562306a36Sopenharmony_ci case MODE_SELECT: what = "MODE_SELECT"; break; 24662306a36Sopenharmony_ci case RESERVE: what = "RESERVE"; break; 24762306a36Sopenharmony_ci case RELEASE: what = "RELEASE"; break; 24862306a36Sopenharmony_ci case COPY: what = "COPY"; break; 24962306a36Sopenharmony_ci case ERASE: what = "ERASE"; break; 25062306a36Sopenharmony_ci case MODE_SENSE: what = "MODE_SENSE"; break; 25162306a36Sopenharmony_ci case START_STOP: what = "START_STOP"; break; 25262306a36Sopenharmony_ci case RECEIVE_DIAGNOSTIC: what = "RECEIVE_DIAGNOSTIC"; break; 25362306a36Sopenharmony_ci case SEND_DIAGNOSTIC: what = "SEND_DIAGNOSTIC"; break; 25462306a36Sopenharmony_ci case ALLOW_MEDIUM_REMOVAL: what = "ALLOW_MEDIUM_REMOVAL"; break; 25562306a36Sopenharmony_ci case SET_WINDOW: what = "SET_WINDOW"; break; 25662306a36Sopenharmony_ci case READ_CAPACITY: what = "READ_CAPACITY"; break; 25762306a36Sopenharmony_ci case READ_10: what = "READ_10"; break; 25862306a36Sopenharmony_ci case WRITE_10: what = "WRITE_10"; break; 25962306a36Sopenharmony_ci case SEEK_10: what = "SEEK_10"; break; 26062306a36Sopenharmony_ci case WRITE_VERIFY: what = "WRITE_VERIFY"; break; 26162306a36Sopenharmony_ci case VERIFY: what = "VERIFY"; break; 26262306a36Sopenharmony_ci case SEARCH_HIGH: what = "SEARCH_HIGH"; break; 26362306a36Sopenharmony_ci case SEARCH_EQUAL: what = "SEARCH_EQUAL"; break; 26462306a36Sopenharmony_ci case SEARCH_LOW: what = "SEARCH_LOW"; break; 26562306a36Sopenharmony_ci case SET_LIMITS: what = "SET_LIMITS"; break; 26662306a36Sopenharmony_ci case READ_POSITION: what = "READ_POSITION"; break; 26762306a36Sopenharmony_ci case SYNCHRONIZE_CACHE: what = "SYNCHRONIZE_CACHE"; break; 26862306a36Sopenharmony_ci case LOCK_UNLOCK_CACHE: what = "LOCK_UNLOCK_CACHE"; break; 26962306a36Sopenharmony_ci case READ_DEFECT_DATA: what = "READ_DEFECT_DATA"; break; 27062306a36Sopenharmony_ci case MEDIUM_SCAN: what = "MEDIUM_SCAN"; break; 27162306a36Sopenharmony_ci case COMPARE: what = "COMPARE"; break; 27262306a36Sopenharmony_ci case COPY_VERIFY: what = "COPY_VERIFY"; break; 27362306a36Sopenharmony_ci case WRITE_BUFFER: what = "WRITE_BUFFER"; break; 27462306a36Sopenharmony_ci case READ_BUFFER: what = "READ_BUFFER"; break; 27562306a36Sopenharmony_ci case UPDATE_BLOCK: what = "UPDATE_BLOCK"; break; 27662306a36Sopenharmony_ci case READ_LONG: what = "READ_LONG"; break; 27762306a36Sopenharmony_ci case WRITE_LONG: what = "WRITE_LONG"; break; 27862306a36Sopenharmony_ci case CHANGE_DEFINITION: what = "CHANGE_DEFINITION"; break; 27962306a36Sopenharmony_ci case WRITE_SAME: what = "WRITE_SAME"; break; 28062306a36Sopenharmony_ci case READ_TOC: what = "READ_TOC"; break; 28162306a36Sopenharmony_ci case LOG_SELECT: what = "LOG_SELECT"; break; 28262306a36Sopenharmony_ci case LOG_SENSE: what = "LOG_SENSE"; break; 28362306a36Sopenharmony_ci case MODE_SELECT_10: what = "MODE_SELECT_10"; break; 28462306a36Sopenharmony_ci case MODE_SENSE_10: what = "MODE_SENSE_10"; break; 28562306a36Sopenharmony_ci case MOVE_MEDIUM: what = "MOVE_MEDIUM"; break; 28662306a36Sopenharmony_ci case READ_12: what = "READ_12"; break; 28762306a36Sopenharmony_ci case WRITE_12: what = "WRITE_12"; break; 28862306a36Sopenharmony_ci case WRITE_VERIFY_12: what = "WRITE_VERIFY_12"; break; 28962306a36Sopenharmony_ci case SEARCH_HIGH_12: what = "SEARCH_HIGH_12"; break; 29062306a36Sopenharmony_ci case SEARCH_EQUAL_12: what = "SEARCH_EQUAL_12"; break; 29162306a36Sopenharmony_ci case SEARCH_LOW_12: what = "SEARCH_LOW_12"; break; 29262306a36Sopenharmony_ci case READ_ELEMENT_STATUS: what = "READ_ELEMENT_STATUS"; break; 29362306a36Sopenharmony_ci case SEND_VOLUME_TAG: what = "SEND_VOLUME_TAG"; break; 29462306a36Sopenharmony_ci case WRITE_LONG_2: what = "WRITE_LONG_2"; break; 29562306a36Sopenharmony_ci default: 29662306a36Sopenharmony_ci MTS_DEBUG("can't decode command\n"); 29762306a36Sopenharmony_ci goto out; 29862306a36Sopenharmony_ci break; 29962306a36Sopenharmony_ci } 30062306a36Sopenharmony_ci MTS_DEBUG( "Command %s (%d bytes)\n", what, srb->cmd_len); 30162306a36Sopenharmony_ci 30262306a36Sopenharmony_ci out: 30362306a36Sopenharmony_ci MTS_DEBUG( " %10ph\n", srb->cmnd); 30462306a36Sopenharmony_ci} 30562306a36Sopenharmony_ci 30662306a36Sopenharmony_ci#else 30762306a36Sopenharmony_ci 30862306a36Sopenharmony_cistatic inline void mts_show_command(struct scsi_cmnd * dummy) 30962306a36Sopenharmony_ci{ 31062306a36Sopenharmony_ci} 31162306a36Sopenharmony_ci 31262306a36Sopenharmony_cistatic inline void mts_debug_dump(struct mts_desc* dummy) 31362306a36Sopenharmony_ci{ 31462306a36Sopenharmony_ci} 31562306a36Sopenharmony_ci 31662306a36Sopenharmony_ci#endif 31762306a36Sopenharmony_ci 31862306a36Sopenharmony_cistatic inline void mts_urb_abort(struct mts_desc* desc) { 31962306a36Sopenharmony_ci MTS_DEBUG_GOT_HERE(); 32062306a36Sopenharmony_ci mts_debug_dump(desc); 32162306a36Sopenharmony_ci 32262306a36Sopenharmony_ci usb_kill_urb( desc->urb ); 32362306a36Sopenharmony_ci} 32462306a36Sopenharmony_ci 32562306a36Sopenharmony_cistatic int mts_slave_alloc (struct scsi_device *s) 32662306a36Sopenharmony_ci{ 32762306a36Sopenharmony_ci s->inquiry_len = 0x24; 32862306a36Sopenharmony_ci return 0; 32962306a36Sopenharmony_ci} 33062306a36Sopenharmony_ci 33162306a36Sopenharmony_cistatic int mts_slave_configure (struct scsi_device *s) 33262306a36Sopenharmony_ci{ 33362306a36Sopenharmony_ci blk_queue_dma_alignment(s->request_queue, (512 - 1)); 33462306a36Sopenharmony_ci return 0; 33562306a36Sopenharmony_ci} 33662306a36Sopenharmony_ci 33762306a36Sopenharmony_cistatic int mts_scsi_abort(struct scsi_cmnd *srb) 33862306a36Sopenharmony_ci{ 33962306a36Sopenharmony_ci struct mts_desc* desc = (struct mts_desc*)(srb->device->host->hostdata[0]); 34062306a36Sopenharmony_ci 34162306a36Sopenharmony_ci MTS_DEBUG_GOT_HERE(); 34262306a36Sopenharmony_ci 34362306a36Sopenharmony_ci mts_urb_abort(desc); 34462306a36Sopenharmony_ci 34562306a36Sopenharmony_ci return FAILED; 34662306a36Sopenharmony_ci} 34762306a36Sopenharmony_ci 34862306a36Sopenharmony_cistatic int mts_scsi_host_reset(struct scsi_cmnd *srb) 34962306a36Sopenharmony_ci{ 35062306a36Sopenharmony_ci struct mts_desc* desc = (struct mts_desc*)(srb->device->host->hostdata[0]); 35162306a36Sopenharmony_ci int result; 35262306a36Sopenharmony_ci 35362306a36Sopenharmony_ci MTS_DEBUG_GOT_HERE(); 35462306a36Sopenharmony_ci mts_debug_dump(desc); 35562306a36Sopenharmony_ci 35662306a36Sopenharmony_ci result = usb_lock_device_for_reset(desc->usb_dev, desc->usb_intf); 35762306a36Sopenharmony_ci if (result == 0) { 35862306a36Sopenharmony_ci result = usb_reset_device(desc->usb_dev); 35962306a36Sopenharmony_ci usb_unlock_device(desc->usb_dev); 36062306a36Sopenharmony_ci } 36162306a36Sopenharmony_ci return result ? FAILED : SUCCESS; 36262306a36Sopenharmony_ci} 36362306a36Sopenharmony_ci 36462306a36Sopenharmony_cistatic int 36562306a36Sopenharmony_cimts_scsi_queuecommand(struct Scsi_Host *shost, struct scsi_cmnd *srb); 36662306a36Sopenharmony_ci 36762306a36Sopenharmony_cistatic void mts_transfer_cleanup( struct urb *transfer ); 36862306a36Sopenharmony_cistatic void mts_do_sg(struct urb * transfer); 36962306a36Sopenharmony_ci 37062306a36Sopenharmony_cistatic inline 37162306a36Sopenharmony_civoid mts_int_submit_urb (struct urb* transfer, 37262306a36Sopenharmony_ci int pipe, 37362306a36Sopenharmony_ci void* data, 37462306a36Sopenharmony_ci unsigned length, 37562306a36Sopenharmony_ci usb_complete_t callback ) 37662306a36Sopenharmony_ci/* Interrupt context! */ 37762306a36Sopenharmony_ci 37862306a36Sopenharmony_ci/* Holding transfer->context->lock! */ 37962306a36Sopenharmony_ci{ 38062306a36Sopenharmony_ci int res; 38162306a36Sopenharmony_ci 38262306a36Sopenharmony_ci MTS_INT_INIT(); 38362306a36Sopenharmony_ci 38462306a36Sopenharmony_ci usb_fill_bulk_urb(transfer, 38562306a36Sopenharmony_ci context->instance->usb_dev, 38662306a36Sopenharmony_ci pipe, 38762306a36Sopenharmony_ci data, 38862306a36Sopenharmony_ci length, 38962306a36Sopenharmony_ci callback, 39062306a36Sopenharmony_ci context 39162306a36Sopenharmony_ci ); 39262306a36Sopenharmony_ci 39362306a36Sopenharmony_ci res = usb_submit_urb( transfer, GFP_ATOMIC ); 39462306a36Sopenharmony_ci if ( unlikely(res) ) { 39562306a36Sopenharmony_ci MTS_INT_ERROR( "could not submit URB! Error was %d\n",(int)res ); 39662306a36Sopenharmony_ci set_host_byte(context->srb, DID_ERROR); 39762306a36Sopenharmony_ci mts_transfer_cleanup(transfer); 39862306a36Sopenharmony_ci } 39962306a36Sopenharmony_ci} 40062306a36Sopenharmony_ci 40162306a36Sopenharmony_ci 40262306a36Sopenharmony_cistatic void mts_transfer_cleanup( struct urb *transfer ) 40362306a36Sopenharmony_ci/* Interrupt context! */ 40462306a36Sopenharmony_ci{ 40562306a36Sopenharmony_ci MTS_INT_INIT(); 40662306a36Sopenharmony_ci 40762306a36Sopenharmony_ci if ( likely(context->final_callback != NULL) ) 40862306a36Sopenharmony_ci context->final_callback(context->srb); 40962306a36Sopenharmony_ci} 41062306a36Sopenharmony_ci 41162306a36Sopenharmony_cistatic void mts_transfer_done( struct urb *transfer ) 41262306a36Sopenharmony_ci{ 41362306a36Sopenharmony_ci MTS_INT_INIT(); 41462306a36Sopenharmony_ci 41562306a36Sopenharmony_ci context->srb->result &= MTS_SCSI_ERR_MASK; 41662306a36Sopenharmony_ci context->srb->result |= (unsigned)(*context->scsi_status)<<1; 41762306a36Sopenharmony_ci 41862306a36Sopenharmony_ci mts_transfer_cleanup(transfer); 41962306a36Sopenharmony_ci} 42062306a36Sopenharmony_ci 42162306a36Sopenharmony_ci 42262306a36Sopenharmony_cistatic void mts_get_status( struct urb *transfer ) 42362306a36Sopenharmony_ci/* Interrupt context! */ 42462306a36Sopenharmony_ci{ 42562306a36Sopenharmony_ci MTS_INT_INIT(); 42662306a36Sopenharmony_ci 42762306a36Sopenharmony_ci mts_int_submit_urb(transfer, 42862306a36Sopenharmony_ci usb_rcvbulkpipe(context->instance->usb_dev, 42962306a36Sopenharmony_ci context->instance->ep_response), 43062306a36Sopenharmony_ci context->scsi_status, 43162306a36Sopenharmony_ci 1, 43262306a36Sopenharmony_ci mts_transfer_done ); 43362306a36Sopenharmony_ci} 43462306a36Sopenharmony_ci 43562306a36Sopenharmony_cistatic void mts_data_done( struct urb* transfer ) 43662306a36Sopenharmony_ci/* Interrupt context! */ 43762306a36Sopenharmony_ci{ 43862306a36Sopenharmony_ci int status = transfer->status; 43962306a36Sopenharmony_ci MTS_INT_INIT(); 44062306a36Sopenharmony_ci 44162306a36Sopenharmony_ci if ( context->data_length != transfer->actual_length ) { 44262306a36Sopenharmony_ci scsi_set_resid(context->srb, context->data_length - 44362306a36Sopenharmony_ci transfer->actual_length); 44462306a36Sopenharmony_ci } else if ( unlikely(status) ) { 44562306a36Sopenharmony_ci set_host_byte(context->srb, (status == -ENOENT ? DID_ABORT : DID_ERROR)); 44662306a36Sopenharmony_ci } 44762306a36Sopenharmony_ci 44862306a36Sopenharmony_ci mts_get_status(transfer); 44962306a36Sopenharmony_ci} 45062306a36Sopenharmony_ci 45162306a36Sopenharmony_ci 45262306a36Sopenharmony_cistatic void mts_command_done( struct urb *transfer ) 45362306a36Sopenharmony_ci/* Interrupt context! */ 45462306a36Sopenharmony_ci{ 45562306a36Sopenharmony_ci int status = transfer->status; 45662306a36Sopenharmony_ci MTS_INT_INIT(); 45762306a36Sopenharmony_ci 45862306a36Sopenharmony_ci if ( unlikely(status) ) { 45962306a36Sopenharmony_ci if (status == -ENOENT) { 46062306a36Sopenharmony_ci /* We are being killed */ 46162306a36Sopenharmony_ci MTS_DEBUG_GOT_HERE(); 46262306a36Sopenharmony_ci set_host_byte(context->srb, DID_ABORT); 46362306a36Sopenharmony_ci } else { 46462306a36Sopenharmony_ci /* A genuine error has occurred */ 46562306a36Sopenharmony_ci MTS_DEBUG_GOT_HERE(); 46662306a36Sopenharmony_ci 46762306a36Sopenharmony_ci set_host_byte(context->srb, DID_ERROR); 46862306a36Sopenharmony_ci } 46962306a36Sopenharmony_ci mts_transfer_cleanup(transfer); 47062306a36Sopenharmony_ci 47162306a36Sopenharmony_ci return; 47262306a36Sopenharmony_ci } 47362306a36Sopenharmony_ci 47462306a36Sopenharmony_ci if (context->srb->cmnd[0] == REQUEST_SENSE) { 47562306a36Sopenharmony_ci mts_int_submit_urb(transfer, 47662306a36Sopenharmony_ci context->data_pipe, 47762306a36Sopenharmony_ci context->srb->sense_buffer, 47862306a36Sopenharmony_ci context->data_length, 47962306a36Sopenharmony_ci mts_data_done); 48062306a36Sopenharmony_ci } else { if ( context->data ) { 48162306a36Sopenharmony_ci mts_int_submit_urb(transfer, 48262306a36Sopenharmony_ci context->data_pipe, 48362306a36Sopenharmony_ci context->data, 48462306a36Sopenharmony_ci context->data_length, 48562306a36Sopenharmony_ci scsi_sg_count(context->srb) > 1 ? 48662306a36Sopenharmony_ci mts_do_sg : mts_data_done); 48762306a36Sopenharmony_ci } else { 48862306a36Sopenharmony_ci mts_get_status(transfer); 48962306a36Sopenharmony_ci } 49062306a36Sopenharmony_ci } 49162306a36Sopenharmony_ci} 49262306a36Sopenharmony_ci 49362306a36Sopenharmony_cistatic void mts_do_sg (struct urb* transfer) 49462306a36Sopenharmony_ci{ 49562306a36Sopenharmony_ci int status = transfer->status; 49662306a36Sopenharmony_ci MTS_INT_INIT(); 49762306a36Sopenharmony_ci 49862306a36Sopenharmony_ci MTS_DEBUG("Processing fragment %d of %d\n", context->fragment, 49962306a36Sopenharmony_ci scsi_sg_count(context->srb)); 50062306a36Sopenharmony_ci 50162306a36Sopenharmony_ci if (unlikely(status)) { 50262306a36Sopenharmony_ci set_host_byte(context->srb, (status == -ENOENT ? DID_ABORT : DID_ERROR)); 50362306a36Sopenharmony_ci mts_transfer_cleanup(transfer); 50462306a36Sopenharmony_ci } 50562306a36Sopenharmony_ci 50662306a36Sopenharmony_ci context->curr_sg = sg_next(context->curr_sg); 50762306a36Sopenharmony_ci mts_int_submit_urb(transfer, 50862306a36Sopenharmony_ci context->data_pipe, 50962306a36Sopenharmony_ci sg_virt(context->curr_sg), 51062306a36Sopenharmony_ci context->curr_sg->length, 51162306a36Sopenharmony_ci sg_is_last(context->curr_sg) ? 51262306a36Sopenharmony_ci mts_data_done : mts_do_sg); 51362306a36Sopenharmony_ci} 51462306a36Sopenharmony_ci 51562306a36Sopenharmony_cistatic const u8 mts_read_image_sig[] = { 0x28, 00, 00, 00 }; 51662306a36Sopenharmony_cistatic const u8 mts_read_image_sig_len = 4; 51762306a36Sopenharmony_cistatic const unsigned char mts_direction[256/8] = { 51862306a36Sopenharmony_ci 0x28, 0x81, 0x14, 0x14, 0x20, 0x01, 0x90, 0x77, 51962306a36Sopenharmony_ci 0x0C, 0x20, 0x00, 0x04, 0x00, 0x00, 0x00, 0x00, 52062306a36Sopenharmony_ci 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x00, 0x01, 52162306a36Sopenharmony_ci 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 52262306a36Sopenharmony_ci}; 52362306a36Sopenharmony_ci 52462306a36Sopenharmony_ci 52562306a36Sopenharmony_ci#define MTS_DIRECTION_IS_IN(x) ((mts_direction[x>>3] >> (x & 7)) & 1) 52662306a36Sopenharmony_ci 52762306a36Sopenharmony_cistatic void 52862306a36Sopenharmony_cimts_build_transfer_context(struct scsi_cmnd *srb, struct mts_desc* desc) 52962306a36Sopenharmony_ci{ 53062306a36Sopenharmony_ci int pipe; 53162306a36Sopenharmony_ci 53262306a36Sopenharmony_ci MTS_DEBUG_GOT_HERE(); 53362306a36Sopenharmony_ci 53462306a36Sopenharmony_ci desc->context.instance = desc; 53562306a36Sopenharmony_ci desc->context.srb = srb; 53662306a36Sopenharmony_ci 53762306a36Sopenharmony_ci if (!scsi_bufflen(srb)) { 53862306a36Sopenharmony_ci desc->context.data = NULL; 53962306a36Sopenharmony_ci desc->context.data_length = 0; 54062306a36Sopenharmony_ci return; 54162306a36Sopenharmony_ci } else { 54262306a36Sopenharmony_ci desc->context.curr_sg = scsi_sglist(srb); 54362306a36Sopenharmony_ci desc->context.data = sg_virt(desc->context.curr_sg); 54462306a36Sopenharmony_ci desc->context.data_length = desc->context.curr_sg->length; 54562306a36Sopenharmony_ci } 54662306a36Sopenharmony_ci 54762306a36Sopenharmony_ci 54862306a36Sopenharmony_ci /* can't rely on srb->sc_data_direction */ 54962306a36Sopenharmony_ci 55062306a36Sopenharmony_ci /* Brutally ripped from usb-storage */ 55162306a36Sopenharmony_ci 55262306a36Sopenharmony_ci if ( !memcmp( srb->cmnd, mts_read_image_sig, mts_read_image_sig_len ) 55362306a36Sopenharmony_ci) { pipe = usb_rcvbulkpipe(desc->usb_dev,desc->ep_image); 55462306a36Sopenharmony_ci MTS_DEBUG( "transferring from desc->ep_image == %d\n", 55562306a36Sopenharmony_ci (int)desc->ep_image ); 55662306a36Sopenharmony_ci } else if ( MTS_DIRECTION_IS_IN(srb->cmnd[0]) ) { 55762306a36Sopenharmony_ci pipe = usb_rcvbulkpipe(desc->usb_dev,desc->ep_response); 55862306a36Sopenharmony_ci MTS_DEBUG( "transferring from desc->ep_response == %d\n", 55962306a36Sopenharmony_ci (int)desc->ep_response); 56062306a36Sopenharmony_ci } else { 56162306a36Sopenharmony_ci MTS_DEBUG("transferring to desc->ep_out == %d\n", 56262306a36Sopenharmony_ci (int)desc->ep_out); 56362306a36Sopenharmony_ci pipe = usb_sndbulkpipe(desc->usb_dev,desc->ep_out); 56462306a36Sopenharmony_ci } 56562306a36Sopenharmony_ci desc->context.data_pipe = pipe; 56662306a36Sopenharmony_ci} 56762306a36Sopenharmony_ci 56862306a36Sopenharmony_cistatic int mts_scsi_queuecommand_lck(struct scsi_cmnd *srb) 56962306a36Sopenharmony_ci{ 57062306a36Sopenharmony_ci mts_scsi_cmnd_callback callback = scsi_done; 57162306a36Sopenharmony_ci struct mts_desc* desc = (struct mts_desc*)(srb->device->host->hostdata[0]); 57262306a36Sopenharmony_ci int res; 57362306a36Sopenharmony_ci 57462306a36Sopenharmony_ci MTS_DEBUG_GOT_HERE(); 57562306a36Sopenharmony_ci mts_show_command(srb); 57662306a36Sopenharmony_ci mts_debug_dump(desc); 57762306a36Sopenharmony_ci 57862306a36Sopenharmony_ci if ( srb->device->lun || srb->device->id || srb->device->channel ) { 57962306a36Sopenharmony_ci 58062306a36Sopenharmony_ci MTS_DEBUG("Command to LUN=%d ID=%d CHANNEL=%d from SCSI layer\n",(int)srb->device->lun,(int)srb->device->id, (int)srb->device->channel ); 58162306a36Sopenharmony_ci 58262306a36Sopenharmony_ci MTS_DEBUG("this device doesn't exist\n"); 58362306a36Sopenharmony_ci 58462306a36Sopenharmony_ci set_host_byte(srb, DID_BAD_TARGET); 58562306a36Sopenharmony_ci 58662306a36Sopenharmony_ci if(likely(callback != NULL)) 58762306a36Sopenharmony_ci callback(srb); 58862306a36Sopenharmony_ci 58962306a36Sopenharmony_ci goto out; 59062306a36Sopenharmony_ci } 59162306a36Sopenharmony_ci 59262306a36Sopenharmony_ci 59362306a36Sopenharmony_ci usb_fill_bulk_urb(desc->urb, 59462306a36Sopenharmony_ci desc->usb_dev, 59562306a36Sopenharmony_ci usb_sndbulkpipe(desc->usb_dev,desc->ep_out), 59662306a36Sopenharmony_ci srb->cmnd, 59762306a36Sopenharmony_ci srb->cmd_len, 59862306a36Sopenharmony_ci mts_command_done, 59962306a36Sopenharmony_ci &desc->context 60062306a36Sopenharmony_ci ); 60162306a36Sopenharmony_ci 60262306a36Sopenharmony_ci 60362306a36Sopenharmony_ci mts_build_transfer_context( srb, desc ); 60462306a36Sopenharmony_ci desc->context.final_callback = callback; 60562306a36Sopenharmony_ci 60662306a36Sopenharmony_ci /* here we need ATOMIC as we are called with the iolock */ 60762306a36Sopenharmony_ci res=usb_submit_urb(desc->urb, GFP_ATOMIC); 60862306a36Sopenharmony_ci 60962306a36Sopenharmony_ci if(unlikely(res)){ 61062306a36Sopenharmony_ci MTS_ERROR("error %d submitting URB\n",(int)res); 61162306a36Sopenharmony_ci set_host_byte(srb, DID_ERROR); 61262306a36Sopenharmony_ci 61362306a36Sopenharmony_ci if(likely(callback != NULL)) 61462306a36Sopenharmony_ci callback(srb); 61562306a36Sopenharmony_ci 61662306a36Sopenharmony_ci } 61762306a36Sopenharmony_ciout: 61862306a36Sopenharmony_ci return 0; 61962306a36Sopenharmony_ci} 62062306a36Sopenharmony_ci 62162306a36Sopenharmony_cistatic DEF_SCSI_QCMD(mts_scsi_queuecommand) 62262306a36Sopenharmony_ci 62362306a36Sopenharmony_cistatic const struct scsi_host_template mts_scsi_host_template = { 62462306a36Sopenharmony_ci .module = THIS_MODULE, 62562306a36Sopenharmony_ci .name = "microtekX6", 62662306a36Sopenharmony_ci .proc_name = "microtekX6", 62762306a36Sopenharmony_ci .queuecommand = mts_scsi_queuecommand, 62862306a36Sopenharmony_ci .eh_abort_handler = mts_scsi_abort, 62962306a36Sopenharmony_ci .eh_host_reset_handler = mts_scsi_host_reset, 63062306a36Sopenharmony_ci .sg_tablesize = SG_ALL, 63162306a36Sopenharmony_ci .can_queue = 1, 63262306a36Sopenharmony_ci .this_id = -1, 63362306a36Sopenharmony_ci .emulated = 1, 63462306a36Sopenharmony_ci .slave_alloc = mts_slave_alloc, 63562306a36Sopenharmony_ci .slave_configure = mts_slave_configure, 63662306a36Sopenharmony_ci .max_sectors= 256, /* 128 K */ 63762306a36Sopenharmony_ci}; 63862306a36Sopenharmony_ci 63962306a36Sopenharmony_ci/* The entries of microtek_table must correspond, line-by-line to 64062306a36Sopenharmony_ci the entries of mts_supported_products[]. */ 64162306a36Sopenharmony_ci 64262306a36Sopenharmony_cistatic const struct usb_device_id mts_usb_ids[] = 64362306a36Sopenharmony_ci{ 64462306a36Sopenharmony_ci { USB_DEVICE(0x4ce, 0x0300) }, 64562306a36Sopenharmony_ci { USB_DEVICE(0x5da, 0x0094) }, 64662306a36Sopenharmony_ci { USB_DEVICE(0x5da, 0x0099) }, 64762306a36Sopenharmony_ci { USB_DEVICE(0x5da, 0x009a) }, 64862306a36Sopenharmony_ci { USB_DEVICE(0x5da, 0x00a0) }, 64962306a36Sopenharmony_ci { USB_DEVICE(0x5da, 0x00a3) }, 65062306a36Sopenharmony_ci { USB_DEVICE(0x5da, 0x80a3) }, 65162306a36Sopenharmony_ci { USB_DEVICE(0x5da, 0x80ac) }, 65262306a36Sopenharmony_ci { USB_DEVICE(0x5da, 0x00b6) }, 65362306a36Sopenharmony_ci { } /* Terminating entry */ 65462306a36Sopenharmony_ci}; 65562306a36Sopenharmony_ci 65662306a36Sopenharmony_ciMODULE_DEVICE_TABLE (usb, mts_usb_ids); 65762306a36Sopenharmony_ci 65862306a36Sopenharmony_ci 65962306a36Sopenharmony_cistatic int mts_usb_probe(struct usb_interface *intf, 66062306a36Sopenharmony_ci const struct usb_device_id *id) 66162306a36Sopenharmony_ci{ 66262306a36Sopenharmony_ci int i; 66362306a36Sopenharmony_ci int ep_out = -1; 66462306a36Sopenharmony_ci int ep_in_set[3]; /* this will break if we have more than three endpoints 66562306a36Sopenharmony_ci which is why we check */ 66662306a36Sopenharmony_ci int *ep_in_current = ep_in_set; 66762306a36Sopenharmony_ci int err_retval = -ENOMEM; 66862306a36Sopenharmony_ci 66962306a36Sopenharmony_ci struct mts_desc * new_desc; 67062306a36Sopenharmony_ci struct usb_device *dev = interface_to_usbdev (intf); 67162306a36Sopenharmony_ci 67262306a36Sopenharmony_ci /* the current altsetting on the interface we're probing */ 67362306a36Sopenharmony_ci struct usb_host_interface *altsetting; 67462306a36Sopenharmony_ci 67562306a36Sopenharmony_ci MTS_DEBUG_GOT_HERE(); 67662306a36Sopenharmony_ci MTS_DEBUG( "usb-device descriptor at %x\n", (int)dev ); 67762306a36Sopenharmony_ci 67862306a36Sopenharmony_ci MTS_DEBUG( "product id = 0x%x, vendor id = 0x%x\n", 67962306a36Sopenharmony_ci le16_to_cpu(dev->descriptor.idProduct), 68062306a36Sopenharmony_ci le16_to_cpu(dev->descriptor.idVendor) ); 68162306a36Sopenharmony_ci 68262306a36Sopenharmony_ci MTS_DEBUG_GOT_HERE(); 68362306a36Sopenharmony_ci 68462306a36Sopenharmony_ci /* the current altsetting on the interface we're probing */ 68562306a36Sopenharmony_ci altsetting = intf->cur_altsetting; 68662306a36Sopenharmony_ci 68762306a36Sopenharmony_ci 68862306a36Sopenharmony_ci /* Check if the config is sane */ 68962306a36Sopenharmony_ci 69062306a36Sopenharmony_ci if ( altsetting->desc.bNumEndpoints != MTS_EP_TOTAL ) { 69162306a36Sopenharmony_ci MTS_WARNING( "expecting %d got %d endpoints! Bailing out.\n", 69262306a36Sopenharmony_ci (int)MTS_EP_TOTAL, (int)altsetting->desc.bNumEndpoints ); 69362306a36Sopenharmony_ci return -ENODEV; 69462306a36Sopenharmony_ci } 69562306a36Sopenharmony_ci 69662306a36Sopenharmony_ci for( i = 0; i < altsetting->desc.bNumEndpoints; i++ ) { 69762306a36Sopenharmony_ci if ((altsetting->endpoint[i].desc.bmAttributes & 69862306a36Sopenharmony_ci USB_ENDPOINT_XFERTYPE_MASK) != USB_ENDPOINT_XFER_BULK) { 69962306a36Sopenharmony_ci 70062306a36Sopenharmony_ci MTS_WARNING( "can only deal with bulk endpoints; endpoint %d is not bulk.\n", 70162306a36Sopenharmony_ci (int)altsetting->endpoint[i].desc.bEndpointAddress ); 70262306a36Sopenharmony_ci } else { 70362306a36Sopenharmony_ci if (altsetting->endpoint[i].desc.bEndpointAddress & 70462306a36Sopenharmony_ci USB_DIR_IN) 70562306a36Sopenharmony_ci *ep_in_current++ 70662306a36Sopenharmony_ci = altsetting->endpoint[i].desc.bEndpointAddress & 70762306a36Sopenharmony_ci USB_ENDPOINT_NUMBER_MASK; 70862306a36Sopenharmony_ci else { 70962306a36Sopenharmony_ci if ( ep_out != -1 ) { 71062306a36Sopenharmony_ci MTS_WARNING( "can only deal with one output endpoints. Bailing out." ); 71162306a36Sopenharmony_ci return -ENODEV; 71262306a36Sopenharmony_ci } 71362306a36Sopenharmony_ci 71462306a36Sopenharmony_ci ep_out = altsetting->endpoint[i].desc.bEndpointAddress & 71562306a36Sopenharmony_ci USB_ENDPOINT_NUMBER_MASK; 71662306a36Sopenharmony_ci } 71762306a36Sopenharmony_ci } 71862306a36Sopenharmony_ci 71962306a36Sopenharmony_ci } 72062306a36Sopenharmony_ci 72162306a36Sopenharmony_ci if (ep_in_current != &ep_in_set[2]) { 72262306a36Sopenharmony_ci MTS_WARNING("couldn't find two input bulk endpoints. Bailing out.\n"); 72362306a36Sopenharmony_ci return -ENODEV; 72462306a36Sopenharmony_ci } 72562306a36Sopenharmony_ci 72662306a36Sopenharmony_ci if ( ep_out == -1 ) { 72762306a36Sopenharmony_ci MTS_WARNING( "couldn't find an output bulk endpoint. Bailing out.\n" ); 72862306a36Sopenharmony_ci return -ENODEV; 72962306a36Sopenharmony_ci } 73062306a36Sopenharmony_ci 73162306a36Sopenharmony_ci 73262306a36Sopenharmony_ci new_desc = kzalloc(sizeof(struct mts_desc), GFP_KERNEL); 73362306a36Sopenharmony_ci if (!new_desc) 73462306a36Sopenharmony_ci goto out; 73562306a36Sopenharmony_ci 73662306a36Sopenharmony_ci new_desc->urb = usb_alloc_urb(0, GFP_KERNEL); 73762306a36Sopenharmony_ci if (!new_desc->urb) 73862306a36Sopenharmony_ci goto out_kfree; 73962306a36Sopenharmony_ci 74062306a36Sopenharmony_ci new_desc->context.scsi_status = kmalloc(1, GFP_KERNEL); 74162306a36Sopenharmony_ci if (!new_desc->context.scsi_status) 74262306a36Sopenharmony_ci goto out_free_urb; 74362306a36Sopenharmony_ci 74462306a36Sopenharmony_ci new_desc->usb_dev = dev; 74562306a36Sopenharmony_ci new_desc->usb_intf = intf; 74662306a36Sopenharmony_ci 74762306a36Sopenharmony_ci /* endpoints */ 74862306a36Sopenharmony_ci new_desc->ep_out = ep_out; 74962306a36Sopenharmony_ci new_desc->ep_response = ep_in_set[0]; 75062306a36Sopenharmony_ci new_desc->ep_image = ep_in_set[1]; 75162306a36Sopenharmony_ci 75262306a36Sopenharmony_ci if ( new_desc->ep_out != MTS_EP_OUT ) 75362306a36Sopenharmony_ci MTS_WARNING( "will this work? Command EP is not usually %d\n", 75462306a36Sopenharmony_ci (int)new_desc->ep_out ); 75562306a36Sopenharmony_ci 75662306a36Sopenharmony_ci if ( new_desc->ep_response != MTS_EP_RESPONSE ) 75762306a36Sopenharmony_ci MTS_WARNING( "will this work? Response EP is not usually %d\n", 75862306a36Sopenharmony_ci (int)new_desc->ep_response ); 75962306a36Sopenharmony_ci 76062306a36Sopenharmony_ci if ( new_desc->ep_image != MTS_EP_IMAGE ) 76162306a36Sopenharmony_ci MTS_WARNING( "will this work? Image data EP is not usually %d\n", 76262306a36Sopenharmony_ci (int)new_desc->ep_image ); 76362306a36Sopenharmony_ci 76462306a36Sopenharmony_ci new_desc->host = scsi_host_alloc(&mts_scsi_host_template, 76562306a36Sopenharmony_ci sizeof(new_desc)); 76662306a36Sopenharmony_ci if (!new_desc->host) 76762306a36Sopenharmony_ci goto out_kfree2; 76862306a36Sopenharmony_ci 76962306a36Sopenharmony_ci new_desc->host->hostdata[0] = (unsigned long)new_desc; 77062306a36Sopenharmony_ci if (scsi_add_host(new_desc->host, &dev->dev)) { 77162306a36Sopenharmony_ci err_retval = -EIO; 77262306a36Sopenharmony_ci goto out_host_put; 77362306a36Sopenharmony_ci } 77462306a36Sopenharmony_ci scsi_scan_host(new_desc->host); 77562306a36Sopenharmony_ci 77662306a36Sopenharmony_ci usb_set_intfdata(intf, new_desc); 77762306a36Sopenharmony_ci return 0; 77862306a36Sopenharmony_ci 77962306a36Sopenharmony_ci out_host_put: 78062306a36Sopenharmony_ci scsi_host_put(new_desc->host); 78162306a36Sopenharmony_ci out_kfree2: 78262306a36Sopenharmony_ci kfree(new_desc->context.scsi_status); 78362306a36Sopenharmony_ci out_free_urb: 78462306a36Sopenharmony_ci usb_free_urb(new_desc->urb); 78562306a36Sopenharmony_ci out_kfree: 78662306a36Sopenharmony_ci kfree(new_desc); 78762306a36Sopenharmony_ci out: 78862306a36Sopenharmony_ci return err_retval; 78962306a36Sopenharmony_ci} 79062306a36Sopenharmony_ci 79162306a36Sopenharmony_cistatic void mts_usb_disconnect (struct usb_interface *intf) 79262306a36Sopenharmony_ci{ 79362306a36Sopenharmony_ci struct mts_desc *desc = usb_get_intfdata(intf); 79462306a36Sopenharmony_ci 79562306a36Sopenharmony_ci usb_set_intfdata(intf, NULL); 79662306a36Sopenharmony_ci 79762306a36Sopenharmony_ci usb_kill_urb(desc->urb); 79862306a36Sopenharmony_ci scsi_remove_host(desc->host); 79962306a36Sopenharmony_ci 80062306a36Sopenharmony_ci scsi_host_put(desc->host); 80162306a36Sopenharmony_ci usb_free_urb(desc->urb); 80262306a36Sopenharmony_ci kfree(desc->context.scsi_status); 80362306a36Sopenharmony_ci kfree(desc); 80462306a36Sopenharmony_ci} 80562306a36Sopenharmony_ci 80662306a36Sopenharmony_cimodule_usb_driver(mts_usb_driver); 80762306a36Sopenharmony_ci 80862306a36Sopenharmony_ciMODULE_AUTHOR( DRIVER_AUTHOR ); 80962306a36Sopenharmony_ciMODULE_DESCRIPTION( DRIVER_DESC ); 81062306a36Sopenharmony_ciMODULE_LICENSE("GPL"); 811