diff options
Diffstat (limited to 'drivers/usb/storage/transport.c')
| -rw-r--r-- | drivers/usb/storage/transport.c | 59 | 
1 files changed, 49 insertions, 10 deletions
| diff --git a/drivers/usb/storage/transport.c b/drivers/usb/storage/transport.c index e20dc525d177..cc313d16d727 100644 --- a/drivers/usb/storage/transport.c +++ b/drivers/usb/storage/transport.c @@ -666,10 +666,11 @@ void usb_stor_invoke_transport(struct scsi_cmnd *srb, struct us_data *us)  	 * to wait for at least one CHECK_CONDITION to determine  	 * SANE_SENSE support  	 */ -	if ((srb->cmnd[0] == ATA_16 || srb->cmnd[0] == ATA_12) && +	if (unlikely((srb->cmnd[0] == ATA_16 || srb->cmnd[0] == ATA_12) &&  	    result == USB_STOR_TRANSPORT_GOOD &&  	    !(us->fflags & US_FL_SANE_SENSE) && -	    !(srb->cmnd[2] & 0x20)) { +	    !(us->fflags & US_FL_BAD_SENSE) && +	    !(srb->cmnd[2] & 0x20))) {  		US_DEBUGP("-- SAT supported, increasing auto-sense\n");  		us->fflags |= US_FL_SANE_SENSE;  	} @@ -696,7 +697,7 @@ void usb_stor_invoke_transport(struct scsi_cmnd *srb, struct us_data *us)  		/* device supports and needs bigger sense buffer */  		if (us->fflags & US_FL_SANE_SENSE)  			sense_size = ~0; - +Retry_Sense:  		US_DEBUGP("Issuing auto-REQUEST_SENSE\n");  		scsi_eh_prep_cmnd(srb, &ses, NULL, 0, sense_size); @@ -718,8 +719,30 @@ void usb_stor_invoke_transport(struct scsi_cmnd *srb, struct us_data *us)  		if (test_bit(US_FLIDX_TIMED_OUT, &us->dflags)) {  			US_DEBUGP("-- auto-sense aborted\n");  			srb->result = DID_ABORT << 16; + +			/* If SANE_SENSE caused this problem, disable it */ +			if (sense_size != US_SENSE_SIZE) { +				us->fflags &= ~US_FL_SANE_SENSE; +				us->fflags |= US_FL_BAD_SENSE; +			}  			goto Handle_Errors;  		} + +		/* Some devices claim to support larger sense but fail when +		 * trying to request it. When a transport failure happens +		 * using US_FS_SANE_SENSE, we always retry with a standard +		 * (small) sense request. This fixes some USB GSM modems +		 */ +		if (temp_result == USB_STOR_TRANSPORT_FAILED && +				sense_size != US_SENSE_SIZE) { +			US_DEBUGP("-- auto-sense failure, retry small sense\n"); +			sense_size = US_SENSE_SIZE; +			us->fflags &= ~US_FL_SANE_SENSE; +			us->fflags |= US_FL_BAD_SENSE; +			goto Retry_Sense; +		} + +		/* Other failures */  		if (temp_result != USB_STOR_TRANSPORT_GOOD) {  			US_DEBUGP("-- auto-sense failure\n"); @@ -739,6 +762,7 @@ void usb_stor_invoke_transport(struct scsi_cmnd *srb, struct us_data *us)  		 */  		if (srb->sense_buffer[7] > (US_SENSE_SIZE - 8) &&  		    !(us->fflags & US_FL_SANE_SENSE) && +		    !(us->fflags & US_FL_BAD_SENSE) &&  		    (srb->sense_buffer[0] & 0x7C) == 0x70) {  			US_DEBUGP("-- SANE_SENSE support enabled\n");  			us->fflags |= US_FL_SANE_SENSE; @@ -768,17 +792,32 @@ void usb_stor_invoke_transport(struct scsi_cmnd *srb, struct us_data *us)  		/* set the result so the higher layers expect this data */  		srb->result = SAM_STAT_CHECK_CONDITION; -		/* If things are really okay, then let's show that.  Zero -		 * out the sense buffer so the higher layers won't realize -		 * we did an unsolicited auto-sense. */ -		if (result == USB_STOR_TRANSPORT_GOOD && -			/* Filemark 0, ignore EOM, ILI 0, no sense */ +		/* We often get empty sense data.  This could indicate that +		 * everything worked or that there was an unspecified +		 * problem.  We have to decide which. +		 */ +		if (	/* Filemark 0, ignore EOM, ILI 0, no sense */  				(srb->sense_buffer[2] & 0xaf) == 0 &&  			/* No ASC or ASCQ */  				srb->sense_buffer[12] == 0 &&  				srb->sense_buffer[13] == 0) { -			srb->result = SAM_STAT_GOOD; -			srb->sense_buffer[0] = 0x0; + +			/* If things are really okay, then let's show that. +			 * Zero out the sense buffer so the higher layers +			 * won't realize we did an unsolicited auto-sense. +			 */ +			if (result == USB_STOR_TRANSPORT_GOOD) { +				srb->result = SAM_STAT_GOOD; +				srb->sense_buffer[0] = 0x0; + +			/* If there was a problem, report an unspecified +			 * hardware error to prevent the higher layers from +			 * entering an infinite retry loop. +			 */ +			} else { +				srb->result = DID_ERROR << 16; +				srb->sense_buffer[2] = HARDWARE_ERROR; +			}  		}  	} | 
