| /* Driver for USB Mass Storage compliant devices |
| * SCSI layer glue code |
| * |
| * $Id: scsiglue.c,v 1.26 2002/04/22 03:39:43 mdharm Exp $ |
| * |
| * Current development and maintenance by: |
| * (c) 1999-2002 Matthew Dharm (mdharm-usb@one-eyed-alien.net) |
| * |
| * Developed with the assistance of: |
| * (c) 2000 David L. Brown, Jr. (usb-storage@davidb.org) |
| * (c) 2000 Stephen J. Gowdy (SGowdy@lbl.gov) |
| * |
| * Initial work by: |
| * (c) 1999 Michael Gee (michael@linuxspecific.com) |
| * |
| * This driver is based on the 'USB Mass Storage Class' document. This |
| * describes in detail the protocol used to communicate with such |
| * devices. Clearly, the designers had SCSI and ATAPI commands in |
| * mind when they created this document. The commands are all very |
| * similar to commands in the SCSI-II and ATAPI specifications. |
| * |
| * It is important to note that in a number of cases this class |
| * exhibits class-specific exemptions from the USB specification. |
| * Notably the usage of NAK, STALL and ACK differs from the norm, in |
| * that they are used to communicate wait, failed and OK on commands. |
| * |
| * Also, for certain devices, the interrupt endpoint is used to convey |
| * status of a command. |
| * |
| * Please see http://www.one-eyed-alien.net/~mdharm/linux-usb for more |
| * information about this driver. |
| * |
| * This program is free software; you can redistribute it and/or modify it |
| * under the terms of the GNU General Public License as published by the |
| * Free Software Foundation; either version 2, or (at your option) any |
| * later version. |
| * |
| * This program is distributed in the hope that it will be useful, but |
| * WITHOUT ANY WARRANTY; without even the implied warranty of |
| * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
| * General Public License for more details. |
| * |
| * You should have received a copy of the GNU General Public License along |
| * with this program; if not, write to the Free Software Foundation, Inc., |
| * 675 Mass Ave, Cambridge, MA 02139, USA. |
| */ |
| |
| #include <linux/slab.h> |
| #include <linux/module.h> |
| |
| #include <scsi/scsi.h> |
| #include <scsi/scsi_cmnd.h> |
| #include <scsi/scsi_devinfo.h> |
| #include <scsi/scsi_device.h> |
| #include <scsi/scsi_eh.h> |
| |
| #include "usb.h" |
| #include "scsiglue.h" |
| #include "debug.h" |
| #include "transport.h" |
| #include "protocol.h" |
| |
| /*********************************************************************** |
| * Host functions |
| ***********************************************************************/ |
| |
| static const char* host_info(struct Scsi_Host *host) |
| { |
| return "SCSI emulation for USB Mass Storage devices"; |
| } |
| |
| static int slave_alloc (struct scsi_device *sdev) |
| { |
| /* |
| * Set the INQUIRY transfer length to 36. We don't use any of |
| * the extra data and many devices choke if asked for more or |
| * less than 36 bytes. |
| */ |
| sdev->inquiry_len = 36; |
| return 0; |
| } |
| |
| static int slave_configure(struct scsi_device *sdev) |
| { |
| struct us_data *us = host_to_us(sdev->host); |
| |
| /* Scatter-gather buffers (all but the last) must have a length |
| * divisible by the bulk maxpacket size. Otherwise a data packet |
| * would end up being short, causing a premature end to the data |
| * transfer. Since high-speed bulk pipes have a maxpacket size |
| * of 512, we'll use that as the scsi device queue's DMA alignment |
| * mask. Guaranteeing proper alignment of the first buffer will |
| * have the desired effect because, except at the beginning and |
| * the end, scatter-gather buffers follow page boundaries. */ |
| blk_queue_dma_alignment(sdev->request_queue, (512 - 1)); |
| |
| /* Set the SCSI level to at least 2. We'll leave it at 3 if that's |
| * what is originally reported. We need this to avoid confusing |
| * the SCSI layer with devices that report 0 or 1, but need 10-byte |
| * commands (ala ATAPI devices behind certain bridges, or devices |
| * which simply have broken INQUIRY data). |
| * |
| * NOTE: This means /dev/sg programs (ala cdrecord) will get the |
| * actual information. This seems to be the preference for |
| * programs like that. |
| * |
| * NOTE: This also means that /proc/scsi/scsi and sysfs may report |
| * the actual value or the modified one, depending on where the |
| * data comes from. |
| */ |
| if (sdev->scsi_level < SCSI_2) |
| sdev->scsi_level = SCSI_2; |
| |
| /* According to the technical support people at Genesys Logic, |
| * devices using their chips have problems transferring more than |
| * 32 KB at a time. In practice people have found that 64 KB |
| * works okay and that's what Windows does. But we'll be |
| * conservative; people can always use the sysfs interface to |
| * increase max_sectors. */ |
| if (le16_to_cpu(us->pusb_dev->descriptor.idVendor) == USB_VENDOR_ID_GENESYS && |
| sdev->request_queue->max_sectors > 64) |
| blk_queue_max_sectors(sdev->request_queue, 64); |
| |
| /* We can't put these settings in slave_alloc() because that gets |
| * called before the device type is known. Consequently these |
| * settings can't be overridden via the scsi devinfo mechanism. */ |
| if (sdev->type == TYPE_DISK) { |
| |
| /* Disk-type devices use MODE SENSE(6) if the protocol |
| * (SubClass) is Transparent SCSI, otherwise they use |
| * MODE SENSE(10). */ |
| if (us->subclass != US_SC_SCSI) |
| sdev->use_10_for_ms = 1; |
| |
| /* Many disks only accept MODE SENSE transfer lengths of |
| * 192 bytes (that's what Windows uses). */ |
| sdev->use_192_bytes_for_3f = 1; |
| |
| /* Some devices don't like MODE SENSE with page=0x3f, |
| * which is the command used for checking if a device |
| * is write-protected. Now that we tell the sd driver |
| * to do a 192-byte transfer with this command the |
| * majority of devices work fine, but a few still can't |
| * handle it. The sd driver will simply assume those |
| * devices are write-enabled. */ |
| if (us->flags & US_FL_NO_WP_DETECT) |
| sdev->skip_ms_page_3f = 1; |
| |
| /* A number of devices have problems with MODE SENSE for |
| * page x08, so we will skip it. */ |
| sdev->skip_ms_page_8 = 1; |
| |
| /* Some disks return the total number of blocks in response |
| * to READ CAPACITY rather than the highest block number. |
| * If this device makes that mistake, tell the sd driver. */ |
| if (us->flags & US_FL_FIX_CAPACITY) |
| sdev->fix_capacity = 1; |
| |
| /* Some devices report a SCSI revision level above 2 but are |
| * unable to handle the REPORT LUNS command (for which |
| * support is mandatory at level 3). Since we already have |
| * a Get-Max-LUN request, we won't lose much by setting the |
| * revision level down to 2. The only devices that would be |
| * affected are those with sparse LUNs. */ |
| sdev->scsi_level = SCSI_2; |
| |
| /* USB-IDE bridges tend to report SK = 0x04 (Non-recoverable |
| * Hardware Error) when any low-level error occurs, |
| * recoverable or not. Setting this flag tells the SCSI |
| * midlayer to retry such commands, which frequently will |
| * succeed and fix the error. The worst this can lead to |
| * is an occasional series of retries that will all fail. */ |
| sdev->retry_hwerror = 1; |
| |
| } else { |
| |
| /* Non-disk-type devices don't need to blacklist any pages |
| * or to force 192-byte transfer lengths for MODE SENSE. |
| * But they do need to use MODE SENSE(10). */ |
| sdev->use_10_for_ms = 1; |
| } |
| |
| /* Some devices choke when they receive a PREVENT-ALLOW MEDIUM |
| * REMOVAL command, so suppress those commands. */ |
| if (us->flags & US_FL_NOT_LOCKABLE) |
| sdev->lockable = 0; |
| |
| /* this is to satisfy the compiler, tho I don't think the |
| * return code is ever checked anywhere. */ |
| return 0; |
| } |
| |
| /* queue a command */ |
| /* This is always called with scsi_lock(host) held */ |
| static int queuecommand(struct scsi_cmnd *srb, |
| void (*done)(struct scsi_cmnd *)) |
| { |
| struct us_data *us = host_to_us(srb->device->host); |
| |
| US_DEBUGP("%s called\n", __FUNCTION__); |
| |
| /* check for state-transition errors */ |
| if (us->srb != NULL) { |
| printk(KERN_ERR USB_STORAGE "Error in %s: us->srb = %p\n", |
| __FUNCTION__, us->srb); |
| return SCSI_MLQUEUE_HOST_BUSY; |
| } |
| |
| /* fail the command if we are disconnecting */ |
| if (test_bit(US_FLIDX_DISCONNECTING, &us->flags)) { |
| US_DEBUGP("Fail command during disconnect\n"); |
| srb->result = DID_NO_CONNECT << 16; |
| done(srb); |
| return 0; |
| } |
| |
| /* enqueue the command and wake up the control thread */ |
| srb->scsi_done = done; |
| us->srb = srb; |
| up(&(us->sema)); |
| |
| return 0; |
| } |
| |
| /*********************************************************************** |
| * Error handling functions |
| ***********************************************************************/ |
| |
| /* Command timeout and abort */ |
| static int command_abort(struct scsi_cmnd *srb) |
| { |
| struct us_data *us = host_to_us(srb->device->host); |
| |
| US_DEBUGP("%s called\n", __FUNCTION__); |
| |
| /* us->srb together with the TIMED_OUT, RESETTING, and ABORTING |
| * bits are protected by the host lock. */ |
| scsi_lock(us_to_host(us)); |
| |
| /* Is this command still active? */ |
| if (us->srb != srb) { |
| scsi_unlock(us_to_host(us)); |
| US_DEBUGP ("-- nothing to abort\n"); |
| return FAILED; |
| } |
| |
| /* Set the TIMED_OUT bit. Also set the ABORTING bit, but only if |
| * a device reset isn't already in progress (to avoid interfering |
| * with the reset). Note that we must retain the host lock while |
| * calling usb_stor_stop_transport(); otherwise it might interfere |
| * with an auto-reset that begins as soon as we release the lock. */ |
| set_bit(US_FLIDX_TIMED_OUT, &us->flags); |
| if (!test_bit(US_FLIDX_RESETTING, &us->flags)) { |
| set_bit(US_FLIDX_ABORTING, &us->flags); |
| usb_stor_stop_transport(us); |
| } |
| scsi_unlock(us_to_host(us)); |
| |
| /* Wait for the aborted command to finish */ |
| wait_for_completion(&us->notify); |
| return SUCCESS; |
| } |
| |
| /* This invokes the transport reset mechanism to reset the state of the |
| * device */ |
| static int device_reset(struct scsi_cmnd *srb) |
| { |
| struct us_data *us = host_to_us(srb->device->host); |
| int result; |
| |
| US_DEBUGP("%s called\n", __FUNCTION__); |
| |
| /* lock the device pointers and do the reset */ |
| down(&(us->dev_semaphore)); |
| result = us->transport_reset(us); |
| up(&(us->dev_semaphore)); |
| |
| return result < 0 ? FAILED : SUCCESS; |
| } |
| |
| /* Simulate a SCSI bus reset by resetting the device's USB port. */ |
| static int bus_reset(struct scsi_cmnd *srb) |
| { |
| struct us_data *us = host_to_us(srb->device->host); |
| int result; |
| |
| US_DEBUGP("%s called\n", __FUNCTION__); |
| |
| down(&(us->dev_semaphore)); |
| result = usb_stor_port_reset(us); |
| up(&(us->dev_semaphore)); |
| |
| return result < 0 ? FAILED : SUCCESS; |
| } |
| |
| /* Report a driver-initiated device reset to the SCSI layer. |
| * Calling this for a SCSI-initiated reset is unnecessary but harmless. |
| * The caller must own the SCSI host lock. */ |
| void usb_stor_report_device_reset(struct us_data *us) |
| { |
| int i; |
| struct Scsi_Host *host = us_to_host(us); |
| |
| scsi_report_device_reset(host, 0, 0); |
| if (us->flags & US_FL_SCM_MULT_TARG) { |
| for (i = 1; i < host->max_id; ++i) |
| scsi_report_device_reset(host, 0, i); |
| } |
| } |
| |
| /* Report a driver-initiated bus reset to the SCSI layer. |
| * Calling this for a SCSI-initiated reset is unnecessary but harmless. |
| * The caller must own the SCSI host lock. */ |
| void usb_stor_report_bus_reset(struct us_data *us) |
| { |
| scsi_report_bus_reset(us_to_host(us), 0); |
| } |
| |
| /*********************************************************************** |
| * /proc/scsi/ functions |
| ***********************************************************************/ |
| |
| /* we use this macro to help us write into the buffer */ |
| #undef SPRINTF |
| #define SPRINTF(args...) \ |
| do { if (pos < buffer+length) pos += sprintf(pos, ## args); } while (0) |
| |
| static int proc_info (struct Scsi_Host *host, char *buffer, |
| char **start, off_t offset, int length, int inout) |
| { |
| struct us_data *us = host_to_us(host); |
| char *pos = buffer; |
| const char *string; |
| |
| /* if someone is sending us data, just throw it away */ |
| if (inout) |
| return length; |
| |
| /* print the controller name */ |
| SPRINTF(" Host scsi%d: usb-storage\n", host->host_no); |
| |
| /* print product, vendor, and serial number strings */ |
| if (us->pusb_dev->manufacturer) |
| string = us->pusb_dev->manufacturer; |
| else if (us->unusual_dev->vendorName) |
| string = us->unusual_dev->vendorName; |
| else |
| string = "Unknown"; |
| SPRINTF(" Vendor: %s\n", string); |
| if (us->pusb_dev->product) |
| string = us->pusb_dev->product; |
| else if (us->unusual_dev->productName) |
| string = us->unusual_dev->productName; |
| else |
| string = "Unknown"; |
| SPRINTF(" Product: %s\n", string); |
| if (us->pusb_dev->serial) |
| string = us->pusb_dev->serial; |
| else |
| string = "None"; |
| SPRINTF("Serial Number: %s\n", string); |
| |
| /* show the protocol and transport */ |
| SPRINTF(" Protocol: %s\n", us->protocol_name); |
| SPRINTF(" Transport: %s\n", us->transport_name); |
| |
| /* show the device flags */ |
| if (pos < buffer + length) { |
| pos += sprintf(pos, " Quirks:"); |
| |
| #define US_FLAG(name, value) \ |
| if (us->flags & value) pos += sprintf(pos, " " #name); |
| US_DO_ALL_FLAGS |
| #undef US_FLAG |
| |
| *(pos++) = '\n'; |
| } |
| |
| /* |
| * Calculate start of next buffer, and return value. |
| */ |
| *start = buffer + offset; |
| |
| if ((pos - buffer) < offset) |
| return (0); |
| else if ((pos - buffer - offset) < length) |
| return (pos - buffer - offset); |
| else |
| return (length); |
| } |
| |
| /*********************************************************************** |
| * Sysfs interface |
| ***********************************************************************/ |
| |
| /* Output routine for the sysfs max_sectors file */ |
| static ssize_t show_max_sectors(struct device *dev, struct device_attribute *attr, char *buf) |
| { |
| struct scsi_device *sdev = to_scsi_device(dev); |
| |
| return sprintf(buf, "%u\n", sdev->request_queue->max_sectors); |
| } |
| |
| /* Input routine for the sysfs max_sectors file */ |
| static ssize_t store_max_sectors(struct device *dev, struct device_attribute *attr, const char *buf, |
| size_t count) |
| { |
| struct scsi_device *sdev = to_scsi_device(dev); |
| unsigned short ms; |
| |
| if (sscanf(buf, "%hu", &ms) > 0 && ms <= SCSI_DEFAULT_MAX_SECTORS) { |
| blk_queue_max_sectors(sdev->request_queue, ms); |
| return strlen(buf); |
| } |
| return -EINVAL; |
| } |
| |
| static DEVICE_ATTR(max_sectors, S_IRUGO | S_IWUSR, show_max_sectors, |
| store_max_sectors); |
| |
| static struct device_attribute *sysfs_device_attr_list[] = { |
| &dev_attr_max_sectors, |
| NULL, |
| }; |
| |
| /* |
| * this defines our host template, with which we'll allocate hosts |
| */ |
| |
| struct scsi_host_template usb_stor_host_template = { |
| /* basic userland interface stuff */ |
| .name = "usb-storage", |
| .proc_name = "usb-storage", |
| .proc_info = proc_info, |
| .info = host_info, |
| |
| /* command interface -- queued only */ |
| .queuecommand = queuecommand, |
| |
| /* error and abort handlers */ |
| .eh_abort_handler = command_abort, |
| .eh_device_reset_handler = device_reset, |
| .eh_bus_reset_handler = bus_reset, |
| |
| /* queue commands only, only one command per LUN */ |
| .can_queue = 1, |
| .cmd_per_lun = 1, |
| |
| /* unknown initiator id */ |
| .this_id = -1, |
| |
| .slave_alloc = slave_alloc, |
| .slave_configure = slave_configure, |
| |
| /* lots of sg segments can be handled */ |
| .sg_tablesize = SG_ALL, |
| |
| /* limit the total size of a transfer to 120 KB */ |
| .max_sectors = 240, |
| |
| /* merge commands... this seems to help performance, but |
| * periodically someone should test to see which setting is more |
| * optimal. |
| */ |
| .use_clustering = 1, |
| |
| /* emulated HBA */ |
| .emulated = 1, |
| |
| /* we do our own delay after a device or bus reset */ |
| .skip_settle_delay = 1, |
| |
| /* sysfs device attributes */ |
| .sdev_attrs = sysfs_device_attr_list, |
| |
| /* module management */ |
| .module = THIS_MODULE |
| }; |
| |
| /* To Report "Illegal Request: Invalid Field in CDB */ |
| unsigned char usb_stor_sense_invalidCDB[18] = { |
| [0] = 0x70, /* current error */ |
| [2] = ILLEGAL_REQUEST, /* Illegal Request = 0x05 */ |
| [7] = 0x0a, /* additional length */ |
| [12] = 0x24 /* Invalid Field in CDB */ |
| }; |
| |