diff options
author | Alan Stern <stern@rowland.harvard.edu> | 2012-01-10 13:43:30 -0500 |
---|---|---|
committer | Greg Kroah-Hartman <gregkh@linuxfoundation.org> | 2012-02-08 20:36:41 -0500 |
commit | 09b6b51b0b6c1b9bb61815baf205e4d74c89ff04 (patch) | |
tree | 03b7c55b764dc555b1d89528fac457a740f897c3 /drivers/usb/storage/scsiglue.c | |
parent | de8c46bfc032fbdf490cfb67f534d2a0188ebeb0 (diff) |
SCSI & usb-storage: add flags for VPD pages and REPORT LUNS
This patch (as1507) adds a skip_vpd_pages flag to struct scsi_device
and a no_report_luns flag to struct scsi_target. The first is used to
control whether sd will look at VPD pages for information on block
provisioning, limits, and characteristics. The second prevents
scsi_report_lun_scan() from issuing a REPORT LUNS command.
The patch also modifies usb-storage to set the new flag bits for all
USB devices and targets, and to stop adjusting the scsi_level value.
Historically we have seen that USB mass-storage devices often don't
support VPD pages or REPORT LUNS properly. Until now we have avoided
these things by setting the scsi_level to SCSI_2 for all USB devices.
But this has the side effect of storing the LUN bits into the second
byte of each CDB, and now we have a report of a device which doesn't
like that. The best solution is to stop abusing scsi_level and
instead have separate flags for VPD pages and REPORT LUNS.
Signed-off-by: Alan Stern <stern@rowland.harvard.edu>
Reported-by: Perry Wagle <wagle@mac.com>
CC: Matthew Dharm <mdharm-usb@one-eyed-alien.net>
Cc: James Bottomley <James.Bottomley@HansenPartnership.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
Diffstat (limited to 'drivers/usb/storage/scsiglue.c')
-rw-r--r-- | drivers/usb/storage/scsiglue.c | 26 |
1 files changed, 16 insertions, 10 deletions
diff --git a/drivers/usb/storage/scsiglue.c b/drivers/usb/storage/scsiglue.c index 13b8bcdf3dba..dc68cc9fef5d 100644 --- a/drivers/usb/storage/scsiglue.c +++ b/drivers/usb/storage/scsiglue.c | |||
@@ -197,6 +197,9 @@ static int slave_configure(struct scsi_device *sdev) | |||
197 | * page x08, so we will skip it. */ | 197 | * page x08, so we will skip it. */ |
198 | sdev->skip_ms_page_8 = 1; | 198 | sdev->skip_ms_page_8 = 1; |
199 | 199 | ||
200 | /* Some devices don't handle VPD pages correctly */ | ||
201 | sdev->skip_vpd_pages = 1; | ||
202 | |||
200 | /* Some disks return the total number of blocks in response | 203 | /* Some disks return the total number of blocks in response |
201 | * to READ CAPACITY rather than the highest block number. | 204 | * to READ CAPACITY rather than the highest block number. |
202 | * If this device makes that mistake, tell the sd driver. */ | 205 | * If this device makes that mistake, tell the sd driver. */ |
@@ -217,16 +220,6 @@ static int slave_configure(struct scsi_device *sdev) | |||
217 | if (sdev->scsi_level > SCSI_SPC_2) | 220 | if (sdev->scsi_level > SCSI_SPC_2) |
218 | us->fflags |= US_FL_SANE_SENSE; | 221 | us->fflags |= US_FL_SANE_SENSE; |
219 | 222 | ||
220 | /* Some devices report a SCSI revision level above 2 but are | ||
221 | * unable to handle the REPORT LUNS command (for which | ||
222 | * support is mandatory at level 3). Since we already have | ||
223 | * a Get-Max-LUN request, we won't lose much by setting the | ||
224 | * revision level down to 2. The only devices that would be | ||
225 | * affected are those with sparse LUNs. */ | ||
226 | if (sdev->scsi_level > SCSI_2) | ||
227 | sdev->sdev_target->scsi_level = | ||
228 | sdev->scsi_level = SCSI_2; | ||
229 | |||
230 | /* USB-IDE bridges tend to report SK = 0x04 (Non-recoverable | 223 | /* USB-IDE bridges tend to report SK = 0x04 (Non-recoverable |
231 | * Hardware Error) when any low-level error occurs, | 224 | * Hardware Error) when any low-level error occurs, |
232 | * recoverable or not. Setting this flag tells the SCSI | 225 | * recoverable or not. Setting this flag tells the SCSI |
@@ -283,6 +276,18 @@ static int slave_configure(struct scsi_device *sdev) | |||
283 | return 0; | 276 | return 0; |
284 | } | 277 | } |
285 | 278 | ||
279 | static int target_alloc(struct scsi_target *starget) | ||
280 | { | ||
281 | /* | ||
282 | * Some USB drives don't support REPORT LUNS, even though they | ||
283 | * report a SCSI revision level above 2. Tell the SCSI layer | ||
284 | * not to issue that command; it will perform a normal sequential | ||
285 | * scan instead. | ||
286 | */ | ||
287 | starget->no_report_luns = 1; | ||
288 | return 0; | ||
289 | } | ||
290 | |||
286 | /* queue a command */ | 291 | /* queue a command */ |
287 | /* This is always called with scsi_lock(host) held */ | 292 | /* This is always called with scsi_lock(host) held */ |
288 | static int queuecommand_lck(struct scsi_cmnd *srb, | 293 | static int queuecommand_lck(struct scsi_cmnd *srb, |
@@ -546,6 +551,7 @@ struct scsi_host_template usb_stor_host_template = { | |||
546 | 551 | ||
547 | .slave_alloc = slave_alloc, | 552 | .slave_alloc = slave_alloc, |
548 | .slave_configure = slave_configure, | 553 | .slave_configure = slave_configure, |
554 | .target_alloc = target_alloc, | ||
549 | 555 | ||
550 | /* lots of sg segments can be handled */ | 556 | /* lots of sg segments can be handled */ |
551 | .sg_tablesize = SCSI_MAX_SG_CHAIN_SEGMENTS, | 557 | .sg_tablesize = SCSI_MAX_SG_CHAIN_SEGMENTS, |