|
|
#include <scsi/scsi_host.h> | #include <scsi/scsi_host.h> |
#include <scsi/scsi_ioctl.h> | #include <scsi/scsi_ioctl.h> |
#include <scsi/scsicam.h> | #include <scsi/scsicam.h> |
|
#include <scsi/sd.h> |
| |
#include "scsi_logging.h" | #include "scsi_logging.h" |
| |
/* |
|
* More than enough for everybody ;) The huge number of majors |
|
* is a leftover from 16bit dev_t days, we don't really need that |
|
* much numberspace. |
|
*/ |
|
#define SD_MAJORS 16 |
|
|
|
MODULE_AUTHOR("Eric Youngdale"); | MODULE_AUTHOR("Eric Youngdale"); |
MODULE_DESCRIPTION("SCSI disk (sd) driver"); | MODULE_DESCRIPTION("SCSI disk (sd) driver"); |
MODULE_LICENSE("GPL"); | MODULE_LICENSE("GPL"); |
|
Lines 89-133
MODULE_ALIAS_BLOCKDEV_MAJOR(SCSI_DISK13_MAJOR);
|
Link Here
|
|---|
|
MODULE_ALIAS_BLOCKDEV_MAJOR(SCSI_DISK14_MAJOR); | MODULE_ALIAS_BLOCKDEV_MAJOR(SCSI_DISK14_MAJOR); |
MODULE_ALIAS_BLOCKDEV_MAJOR(SCSI_DISK15_MAJOR); | MODULE_ALIAS_BLOCKDEV_MAJOR(SCSI_DISK15_MAJOR); |
| |
/* |
|
* This is limited by the naming scheme enforced in sd_probe, |
|
* add another character to it if you really need more disks. |
|
*/ |
|
#define SD_MAX_DISKS (((26 * 26) + 26 + 1) * 26) |
|
|
|
/* |
|
* Time out in seconds for disks and Magneto-opticals (which are slower). |
|
*/ |
|
#define SD_TIMEOUT (30 * HZ) |
|
#define SD_MOD_TIMEOUT (75 * HZ) |
|
|
|
/* |
|
* Number of allowed retries |
|
*/ |
|
#define SD_MAX_RETRIES 5 |
|
#define SD_PASSTHROUGH_RETRIES 1 |
|
|
|
/* |
|
* Size of the initial data buffer for mode and read capacity data |
|
*/ |
|
#define SD_BUF_SIZE 512 |
|
|
|
struct scsi_disk { |
|
struct scsi_driver *driver; /* always &sd_template */ |
|
struct scsi_device *device; |
|
struct class_device cdev; |
|
struct gendisk *disk; |
|
unsigned int openers; /* protected by BKL for now, yuck */ |
|
sector_t capacity; /* size in 512-byte sectors */ |
|
u32 index; |
|
u8 media_present; |
|
u8 write_prot; |
|
unsigned WCE : 1; /* state of disk WCE bit */ |
|
unsigned RCD : 1; /* state of disk RCD bit, unused */ |
|
unsigned DPOFUA : 1; /* state of disk DPOFUA bit */ |
|
}; |
|
#define to_scsi_disk(obj) container_of(obj,struct scsi_disk,cdev) |
|
|
|
static DEFINE_IDR(sd_index_idr); | static DEFINE_IDR(sd_index_idr); |
static DEFINE_SPINLOCK(sd_index_lock); | static DEFINE_SPINLOCK(sd_index_lock); |
| |
|
Lines 136-155
static DEFINE_SPINLOCK(sd_index_lock);
|
Link Here
|
|---|
|
* object after last put) */ | * object after last put) */ |
static DEFINE_MUTEX(sd_ref_mutex); | static DEFINE_MUTEX(sd_ref_mutex); |
| |
static int sd_revalidate_disk(struct gendisk *disk); |
|
static void sd_rw_intr(struct scsi_cmnd * SCpnt); |
|
|
|
static int sd_probe(struct device *); |
|
static int sd_remove(struct device *); |
|
static void sd_shutdown(struct device *dev); |
|
static void sd_rescan(struct device *); |
|
static int sd_init_command(struct scsi_cmnd *); |
|
static int sd_issue_flush(struct device *, sector_t *); |
|
static void sd_prepare_flush(request_queue_t *, struct request *); |
|
static void sd_read_capacity(struct scsi_disk *sdkp, char *diskname, |
|
unsigned char *buffer); |
|
static void scsi_disk_release(struct class_device *cdev); |
|
|
|
static const char *sd_cache_types[] = { | static const char *sd_cache_types[] = { |
"write through", "none", "write back", | "write through", "none", "write back", |
"write back, no read (daft)" | "write back, no read (daft)" |
|
Lines 199-205
static ssize_t sd_store_cache_type(struct class_device *cdev, const char *buf,
|
Link Here
|
|---|
|
if (scsi_mode_select(sdp, 1, sp, 8, buffer_data, len, SD_TIMEOUT, | if (scsi_mode_select(sdp, 1, sp, 8, buffer_data, len, SD_TIMEOUT, |
SD_MAX_RETRIES, &data, &sshdr)) { | SD_MAX_RETRIES, &data, &sshdr)) { |
if (scsi_sense_valid(&sshdr)) | if (scsi_sense_valid(&sshdr)) |
scsi_print_sense_hdr(sdkp->disk->disk_name, &sshdr); |
sd_print_sense_hdr(sdkp, &sshdr); |
return -EINVAL; | return -EINVAL; |
} | } |
sd_revalidate_disk(sdkp->disk); | sd_revalidate_disk(sdkp->disk); |
|
Lines 407-413
static int sd_init_command(struct scsi_cmnd * SCpnt)
|
Link Here
|
|---|
|
*/ | */ |
if (sdp->sector_size == 1024) { | if (sdp->sector_size == 1024) { |
if ((block & 1) || (rq->nr_sectors & 1)) { | if ((block & 1) || (rq->nr_sectors & 1)) { |
printk(KERN_ERR "sd: Bad block number requested"); |
scmd_printk(KERN_ERR, SCpnt, |
|
"Bad block number requested\n"); |
return 0; | return 0; |
} else { | } else { |
block = block >> 1; | block = block >> 1; |
|
Lines 416-422
static int sd_init_command(struct scsi_cmnd * SCpnt)
|
Link Here
|
|---|
|
} | } |
if (sdp->sector_size == 2048) { | if (sdp->sector_size == 2048) { |
if ((block & 3) || (rq->nr_sectors & 3)) { | if ((block & 3) || (rq->nr_sectors & 3)) { |
printk(KERN_ERR "sd: Bad block number requested"); |
scmd_printk(KERN_ERR, SCpnt, |
|
"Bad block number requested\n"); |
return 0; | return 0; |
} else { | } else { |
block = block >> 2; | block = block >> 2; |
|
Lines 425-431
static int sd_init_command(struct scsi_cmnd * SCpnt)
|
Link Here
|
|---|
|
} | } |
if (sdp->sector_size == 4096) { | if (sdp->sector_size == 4096) { |
if ((block & 7) || (rq->nr_sectors & 7)) { | if ((block & 7) || (rq->nr_sectors & 7)) { |
printk(KERN_ERR "sd: Bad block number requested"); |
scmd_printk(KERN_ERR, SCpnt, |
|
"Bad block number requested\n"); |
return 0; | return 0; |
} else { | } else { |
block = block >> 3; | block = block >> 3; |
|
Lines 442-448
static int sd_init_command(struct scsi_cmnd * SCpnt)
|
Link Here
|
|---|
|
SCpnt->cmnd[0] = READ_6; | SCpnt->cmnd[0] = READ_6; |
SCpnt->sc_data_direction = DMA_FROM_DEVICE; | SCpnt->sc_data_direction = DMA_FROM_DEVICE; |
} else { | } else { |
printk(KERN_ERR "sd: Unknown command %x\n", rq->cmd_flags); |
scmd_printk(KERN_ERR, SCpnt, "Unknown command %x\n", rq->cmd_flags); |
return 0; | return 0; |
} | } |
| |
|
Lines 490-496
static int sd_init_command(struct scsi_cmnd * SCpnt)
|
Link Here
|
|---|
|
* during operation and thus turned off | * during operation and thus turned off |
* use_10_for_rw. | * use_10_for_rw. |
*/ | */ |
printk(KERN_ERR "sd: FUA write on READ/WRITE(6) drive\n"); |
scmd_printk(KERN_ERR, SCpnt, |
|
"FUA write on READ/WRITE(6) drive\n"); |
return 0; | return 0; |
} | } |
| |
|
|
return 1; | return 1; |
} | } |
| |
static int sd_sync_cache(struct scsi_device *sdp) |
static int sd_sync_cache(struct scsi_disk *sdkp) |
{ | { |
int retries, res; | int retries, res; |
|
struct scsi_device *sdp = sdkp->device; |
struct scsi_sense_hdr sshdr; | struct scsi_sense_hdr sshdr; |
| |
if (!scsi_device_online(sdp)) | if (!scsi_device_online(sdp)) |
|
Lines 809-820
static int sd_sync_cache(struct scsi_device *sdp)
|
Link Here
|
|---|
|
break; | break; |
} | } |
| |
if (res) { printk(KERN_WARNING "FAILED\n status = %x, message = %02x, " |
if (res) { |
"host = %d, driver = %02x\n ", |
sd_print_result(sdkp, res); |
status_byte(res), msg_byte(res), |
if (driver_byte(res) & DRIVER_SENSE) |
host_byte(res), driver_byte(res)); |
sd_print_sense_hdr(sdkp, &sshdr); |
if (driver_byte(res) & DRIVER_SENSE) |
|
scsi_print_sense_hdr("sd", &sshdr); |
|
} | } |
| |
return res; | return res; |
|
Lines 823-836
static int sd_sync_cache(struct scsi_device *sdp)
|
Link Here
|
|---|
|
static int sd_issue_flush(struct device *dev, sector_t *error_sector) | static int sd_issue_flush(struct device *dev, sector_t *error_sector) |
{ | { |
int ret = 0; | int ret = 0; |
struct scsi_device *sdp = to_scsi_device(dev); |
|
struct scsi_disk *sdkp = scsi_disk_get_from_dev(dev); | struct scsi_disk *sdkp = scsi_disk_get_from_dev(dev); |
| |
if (!sdkp) | if (!sdkp) |
return -ENODEV; | return -ENODEV; |
| |
if (sdkp->WCE) | if (sdkp->WCE) |
ret = sd_sync_cache(sdp); |
ret = sd_sync_cache(sdkp); |
scsi_disk_put(sdkp); | scsi_disk_put(sdkp); |
return ret; | return ret; |
} | } |
|
Lines 1025-1031
static int media_not_present(struct scsi_disk *sdkp,
|
Link Here
|
|---|
|
* spinup disk - called only in sd_revalidate_disk() | * spinup disk - called only in sd_revalidate_disk() |
*/ | */ |
static void | static void |
sd_spinup_disk(struct scsi_disk *sdkp, char *diskname) |
sd_spinup_disk(struct scsi_disk *sdkp) |
{ | { |
unsigned char cmd[10]; | unsigned char cmd[10]; |
unsigned long spintime_expire = 0; | unsigned long spintime_expire = 0; |
|
Lines 1069-1077
sd_spinup_disk(struct scsi_disk *sdkp, char *diskname)
|
Link Here
|
|---|
|
if ((driver_byte(the_result) & DRIVER_SENSE) == 0) { | if ((driver_byte(the_result) & DRIVER_SENSE) == 0) { |
/* no sense, TUR either succeeded or failed | /* no sense, TUR either succeeded or failed |
* with a status error */ | * with a status error */ |
if(!spintime && !scsi_status_is_good(the_result)) |
if(!spintime && !scsi_status_is_good(the_result)) { |
printk(KERN_NOTICE "%s: Unit Not Ready, " |
sd_printk(KERN_NOTICE, sdkp, "Unit Not Ready\n"); |
"error = 0x%x\n", diskname, the_result); |
sd_print_result(sdkp, the_result); |
|
} |
break; | break; |
} | } |
| |
|
Lines 1096-1103
sd_spinup_disk(struct scsi_disk *sdkp, char *diskname)
|
Link Here
|
|---|
|
*/ | */ |
} else if (sense_valid && sshdr.sense_key == NOT_READY) { | } else if (sense_valid && sshdr.sense_key == NOT_READY) { |
if (!spintime) { | if (!spintime) { |
printk(KERN_NOTICE "%s: Spinning up disk...", |
sd_printk(KERN_NOTICE, sdkp, "Spinning up disk..."); |
diskname); |
|
cmd[0] = START_STOP; | cmd[0] = START_STOP; |
cmd[1] = 1; /* Return immediately */ | cmd[1] = 1; /* Return immediately */ |
memset((void *) &cmd[2], 0, 8); | memset((void *) &cmd[2], 0, 8); |
|
Lines 1130-1138
sd_spinup_disk(struct scsi_disk *sdkp, char *diskname)
|
Link Here
|
|---|
|
/* we don't understand the sense code, so it's | /* we don't understand the sense code, so it's |
* probably pointless to loop */ | * probably pointless to loop */ |
if(!spintime) { | if(!spintime) { |
printk(KERN_NOTICE "%s: Unit Not Ready, " |
sd_printk(KERN_NOTICE, sdkp, "Unit Not Ready\n"); |
"sense:\n", diskname); |
sd_print_sense_hdr(sdkp, &sshdr); |
scsi_print_sense_hdr("", &sshdr); |
|
} | } |
break; | break; |
} | } |
|
Lines 1151-1158
sd_spinup_disk(struct scsi_disk *sdkp, char *diskname)
|
Link Here
|
|---|
|
* read disk capacity | * read disk capacity |
*/ | */ |
static void | static void |
sd_read_capacity(struct scsi_disk *sdkp, char *diskname, |
sd_read_capacity(struct scsi_disk *sdkp, unsigned char *buffer) |
unsigned char *buffer) |
|
{ | { |
unsigned char cmd[16]; | unsigned char cmd[16]; |
int the_result, retries; | int the_result, retries; |
|
|
} while (the_result && retries); | } while (the_result && retries); |
| |
if (the_result && !longrc) { | if (the_result && !longrc) { |
printk(KERN_NOTICE "%s : READ CAPACITY failed.\n" |
sd_printk(KERN_NOTICE, sdkp, "READ CAPACITY failed\n"); |
"%s : status=%x, message=%02x, host=%d, driver=%02x \n", |
sd_print_result(sdkp, the_result); |
diskname, diskname, |
|
status_byte(the_result), |
|
msg_byte(the_result), |
|
host_byte(the_result), |
|
driver_byte(the_result)); |
|
|
|
if (driver_byte(the_result) & DRIVER_SENSE) | if (driver_byte(the_result) & DRIVER_SENSE) |
scsi_print_sense_hdr("sd", &sshdr); |
sd_print_sense_hdr(sdkp, &sshdr); |
else | else |
printk("%s : sense not available. \n", diskname); |
sd_printk(KERN_NOTICE, sdkp, "Sense not available.\n"); |
| |
/* Set dirty bit for removable devices if not ready - | /* Set dirty bit for removable devices if not ready - |
* sometimes drives will not report this properly. */ | * sometimes drives will not report this properly. */ |
|
|
return; | return; |
} else if (the_result && longrc) { | } else if (the_result && longrc) { |
/* READ CAPACITY(16) has been failed */ | /* READ CAPACITY(16) has been failed */ |
printk(KERN_NOTICE "%s : READ CAPACITY(16) failed.\n" |
sd_printk(KERN_NOTICE, sdkp, "READ CAPACITY(16) failed\n"); |
"%s : status=%x, message=%02x, host=%d, driver=%02x \n", |
sd_print_result(sdkp, the_result); |
diskname, diskname, |
sd_printk(KERN_NOTICE, sdkp, "Use 0xffffffff as device size\n"); |
status_byte(the_result), |
|
msg_byte(the_result), |
|
host_byte(the_result), |
|
driver_byte(the_result)); |
|
printk(KERN_NOTICE "%s : use 0xffffffff as device size\n", |
|
diskname); |
|
|
|
sdkp->capacity = 1 + (sector_t) 0xffffffff; | sdkp->capacity = 1 + (sector_t) 0xffffffff; |
goto got_data; | goto got_data; |
} | } |
|
|
if (buffer[0] == 0xff && buffer[1] == 0xff && | if (buffer[0] == 0xff && buffer[1] == 0xff && |
buffer[2] == 0xff && buffer[3] == 0xff) { | buffer[2] == 0xff && buffer[3] == 0xff) { |
if(sizeof(sdkp->capacity) > 4) { | if(sizeof(sdkp->capacity) > 4) { |
printk(KERN_NOTICE "%s : very big device. try to use" |
sd_printk(KERN_NOTICE, sdkp, "Very big device. " |
" READ CAPACITY(16).\n", diskname); |
"Trying to use READ CAPACITY(16).\n"); |
longrc = 1; | longrc = 1; |
goto repeat; | goto repeat; |
} | } |
printk(KERN_ERR "%s: too big for this kernel. Use a " |
sd_printk(KERN_ERR, sdkp, "Too big for this kernel. Use " |
"kernel compiled with support for large block " |
"a kernel compiled with support for large " |
"devices.\n", diskname); |
"block devices.\n"); |
sdkp->capacity = 0; | sdkp->capacity = 0; |
goto got_data; | goto got_data; |
} | } |
|
|
got_data: | got_data: |
if (sector_size == 0) { | if (sector_size == 0) { |
sector_size = 512; | sector_size = 512; |
printk(KERN_NOTICE "%s : sector size 0 reported, " |
sd_printk(KERN_NOTICE, sdkp, "Sector size 0 reported, " |
"assuming 512.\n", diskname); |
"assuming 512.\n"); |
} | } |
| |
if (sector_size != 512 && | if (sector_size != 512 && |
|
|
sector_size != 2048 && | sector_size != 2048 && |
sector_size != 4096 && | sector_size != 4096 && |
sector_size != 256) { | sector_size != 256) { |
printk(KERN_NOTICE "%s : unsupported sector size " |
sd_printk(KERN_NOTICE, sdkp, "Unsupported sector size %d.\n", |
"%d.\n", diskname, sector_size); |
sector_size); |
/* | /* |
* The user might want to re-format the drive with | * The user might want to re-format the drive with |
* a supported sectorsize. Once this happens, it | * a supported sectorsize. Once this happens, it |
|
|
mb -= sz - 974; | mb -= sz - 974; |
sector_div(mb, 1950); | sector_div(mb, 1950); |
| |
printk(KERN_NOTICE "SCSI device %s: " |
sd_printk(KERN_NOTICE, sdkp, |
"%llu %d-byte hdwr sectors (%llu MB)\n", |
"%llu %d-byte hardware sectors (%llu MB)\n", |
diskname, (unsigned long long)sdkp->capacity, |
(unsigned long long)sdkp->capacity, |
hard_sector, (unsigned long long)mb); |
hard_sector, (unsigned long long)mb); |
} | } |
| |
/* Rescale capacity to 512-byte units */ | /* Rescale capacity to 512-byte units */ |
|
Lines 1362-1369
sd_do_mode_sense(struct scsi_device *sdp, int dbd, int modepage,
|
Link Here
|
|---|
|
* called with buffer of length SD_BUF_SIZE | * called with buffer of length SD_BUF_SIZE |
*/ | */ |
static void | static void |
sd_read_write_protect_flag(struct scsi_disk *sdkp, char *diskname, |
sd_read_write_protect_flag(struct scsi_disk *sdkp, unsigned char *buffer) |
unsigned char *buffer) |
|
{ | { |
int res; | int res; |
struct scsi_device *sdp = sdkp->device; | struct scsi_device *sdp = sdkp->device; |
|
Lines 1371-1377
sd_read_write_protect_flag(struct scsi_disk *sdkp, char *diskname,
|
Link Here
|
|---|
|
| |
set_disk_ro(sdkp->disk, 0); | set_disk_ro(sdkp->disk, 0); |
if (sdp->skip_ms_page_3f) { | if (sdp->skip_ms_page_3f) { |
printk(KERN_NOTICE "%s: assuming Write Enabled\n", diskname); |
sd_printk(KERN_NOTICE, sdkp, "Assuming Write Enabled\n"); |
return; | return; |
} | } |
| |
|
Lines 1403-1417
sd_read_write_protect_flag(struct scsi_disk *sdkp, char *diskname,
|
Link Here
|
|---|
|
} | } |
| |
if (!scsi_status_is_good(res)) { | if (!scsi_status_is_good(res)) { |
printk(KERN_WARNING |
sd_printk(KERN_WARNING, sdkp, |
"%s: test WP failed, assume Write Enabled\n", diskname); |
"Test WP failed, assume Write Enabled\n"); |
} else { | } else { |
sdkp->write_prot = ((data.device_specific & 0x80) != 0); | sdkp->write_prot = ((data.device_specific & 0x80) != 0); |
set_disk_ro(sdkp->disk, sdkp->write_prot); | set_disk_ro(sdkp->disk, sdkp->write_prot); |
printk(KERN_NOTICE "%s: Write Protect is %s\n", diskname, |
sd_printk(KERN_NOTICE, sdkp, "Write Protect is %s\n", |
sdkp->write_prot ? "on" : "off"); |
sdkp->write_prot ? "on" : "off"); |
printk(KERN_DEBUG "%s: Mode Sense: %02x %02x %02x %02x\n", |
sd_printk(KERN_DEBUG, sdkp, |
diskname, buffer[0], buffer[1], buffer[2], buffer[3]); |
"Mode Sense: %02x %02x %02x %02x\n", |
|
buffer[0], buffer[1], buffer[2], buffer[3]); |
} | } |
} | } |
| |
|
Lines 1420-1427
sd_read_write_protect_flag(struct scsi_disk *sdkp, char *diskname,
|
Link Here
|
|---|
|
* called with buffer of length SD_BUF_SIZE | * called with buffer of length SD_BUF_SIZE |
*/ | */ |
static void | static void |
sd_read_cache_type(struct scsi_disk *sdkp, char *diskname, |
sd_read_cache_type(struct scsi_disk *sdkp, unsigned char *buffer) |
unsigned char *buffer) |
|
{ | { |
int len = 0, res; | int len = 0, res; |
struct scsi_device *sdp = sdkp->device; | struct scsi_device *sdp = sdkp->device; |
|
Lines 1450-1457
sd_read_cache_type(struct scsi_disk *sdkp, char *diskname,
|
Link Here
|
|---|
|
| |
if (!data.header_length) { | if (!data.header_length) { |
modepage = 6; | modepage = 6; |
printk(KERN_ERR "%s: missing header in MODE_SENSE response\n", |
sd_printk(KERN_ERR, sdkp, "Missing header in MODE_SENSE response\n"); |
diskname); |
|
} | } |
| |
/* that went OK, now ask for the proper length */ | /* that went OK, now ask for the proper length */ |
|
Lines 1478-1490
sd_read_cache_type(struct scsi_disk *sdkp, char *diskname,
|
Link Here
|
|---|
|
int offset = data.header_length + data.block_descriptor_length; | int offset = data.header_length + data.block_descriptor_length; |
| |
if (offset >= SD_BUF_SIZE - 2) { | if (offset >= SD_BUF_SIZE - 2) { |
printk(KERN_ERR "%s: malformed MODE SENSE response", |
sd_printk(KERN_ERR, sdkp, "Malformed MODE SENSE response\n"); |
diskname); |
|
goto defaults; | goto defaults; |
} | } |
| |
if ((buffer[offset] & 0x3f) != modepage) { | if ((buffer[offset] & 0x3f) != modepage) { |
printk(KERN_ERR "%s: got wrong page\n", diskname); |
sd_printk(KERN_ERR, sdkp, "Got wrong page\n"); |
goto defaults; | goto defaults; |
} | } |
| |
|
Lines 1498-1511
sd_read_cache_type(struct scsi_disk *sdkp, char *diskname,
|
Link Here
|
|---|
|
| |
sdkp->DPOFUA = (data.device_specific & 0x10) != 0; | sdkp->DPOFUA = (data.device_specific & 0x10) != 0; |
if (sdkp->DPOFUA && !sdkp->device->use_10_for_rw) { | if (sdkp->DPOFUA && !sdkp->device->use_10_for_rw) { |
printk(KERN_NOTICE "SCSI device %s: uses " |
sd_printk(KERN_NOTICE, sdkp, |
"READ/WRITE(6), disabling FUA\n", diskname); |
"Uses READ/WRITE(6), disabling FUA\n"); |
sdkp->DPOFUA = 0; | sdkp->DPOFUA = 0; |
} | } |
| |
printk(KERN_NOTICE "SCSI device %s: " |
sd_printk(KERN_NOTICE, sdkp, |
"write cache: %s, read cache: %s, %s\n", |
"Write cache: %s, read cache: %s, %s\n", |
diskname, |
|
sdkp->WCE ? "enabled" : "disabled", | sdkp->WCE ? "enabled" : "disabled", |
sdkp->RCD ? "disabled" : "enabled", | sdkp->RCD ? "disabled" : "enabled", |
sdkp->DPOFUA ? "supports DPO and FUA" | sdkp->DPOFUA ? "supports DPO and FUA" |
|
|
if (scsi_sense_valid(&sshdr) && | if (scsi_sense_valid(&sshdr) && |
sshdr.sense_key == ILLEGAL_REQUEST && | sshdr.sense_key == ILLEGAL_REQUEST && |
sshdr.asc == 0x24 && sshdr.ascq == 0x0) | sshdr.asc == 0x24 && sshdr.ascq == 0x0) |
printk(KERN_NOTICE "%s: cache data unavailable\n", |
/* Invalid field in CDB */ |
diskname); /* Invalid field in CDB */ |
sd_printk(KERN_NOTICE, sdkp, "Cache data unavailable\n"); |
else | else |
printk(KERN_ERR "%s: asking for cache data failed\n", |
sd_printk(KERN_ERR, sdkp, "Asking for cache data failed\n"); |
diskname); |
|
| |
defaults: | defaults: |
printk(KERN_ERR "%s: assuming drive cache: write through\n", |
sd_printk(KERN_ERR, sdkp, "Assuming drive cache: write through\n"); |
diskname); |
|
sdkp->WCE = 0; | sdkp->WCE = 0; |
sdkp->RCD = 0; | sdkp->RCD = 0; |
sdkp->DPOFUA = 0; | sdkp->DPOFUA = 0; |
|
Lines 1555-1562
static int sd_revalidate_disk(struct gendisk *disk)
|
Link Here
|
|---|
|
| |
buffer = kmalloc(SD_BUF_SIZE, GFP_KERNEL | __GFP_DMA); | buffer = kmalloc(SD_BUF_SIZE, GFP_KERNEL | __GFP_DMA); |
if (!buffer) { | if (!buffer) { |
printk(KERN_WARNING "(sd_revalidate_disk:) Memory allocation " |
sd_printk(KERN_WARNING, sdkp, "sd_revalidate_disk: Memory " |
"failure.\n"); |
"allocation failure.\n"); |
goto out; | goto out; |
} | } |
| |
|
Lines 1568-1583
static int sd_revalidate_disk(struct gendisk *disk)
|
Link Here
|
|---|
|
sdkp->WCE = 0; | sdkp->WCE = 0; |
sdkp->RCD = 0; | sdkp->RCD = 0; |
| |
sd_spinup_disk(sdkp, disk->disk_name); |
sd_spinup_disk(sdkp); |
| |
/* | /* |
* Without media there is no reason to ask; moreover, some devices | * Without media there is no reason to ask; moreover, some devices |
* react badly if we do. | * react badly if we do. |
*/ | */ |
if (sdkp->media_present) { | if (sdkp->media_present) { |
sd_read_capacity(sdkp, disk->disk_name, buffer); |
sd_read_capacity(sdkp, buffer); |
sd_read_write_protect_flag(sdkp, disk->disk_name, buffer); |
sd_read_write_protect_flag(sdkp, buffer); |
sd_read_cache_type(sdkp, disk->disk_name, buffer); |
sd_read_cache_type(sdkp, buffer); |
} | } |
| |
/* | /* |
|
Lines 1709-1716
static int sd_probe(struct device *dev)
|
Link Here
|
|---|
|
dev_set_drvdata(dev, sdkp); | dev_set_drvdata(dev, sdkp); |
add_disk(gd); | add_disk(gd); |
| |
sdev_printk(KERN_NOTICE, sdp, "Attached scsi %sdisk %s\n", |
sd_printk(KERN_NOTICE, sdkp, "Attached SCSI %sdisk\n", |
sdp->removable ? "removable " : "", gd->disk_name); |
sdp->removable ? "removable " : ""); |
| |
return 0; | return 0; |
| |
|
Lines 1781-1796
static void scsi_disk_release(struct class_device *cdev)
|
Link Here
|
|---|
|
*/ | */ |
static void sd_shutdown(struct device *dev) | static void sd_shutdown(struct device *dev) |
{ | { |
struct scsi_device *sdp = to_scsi_device(dev); |
|
struct scsi_disk *sdkp = scsi_disk_get_from_dev(dev); | struct scsi_disk *sdkp = scsi_disk_get_from_dev(dev); |
| |
if (!sdkp) | if (!sdkp) |
return; /* this can happen */ | return; /* this can happen */ |
| |
if (sdkp->WCE) { | if (sdkp->WCE) { |
printk(KERN_NOTICE "Synchronizing SCSI cache for disk %s: \n", |
sd_printk(KERN_NOTICE, sdkp, "Synchronizing SCSI cache\n"); |
sdkp->disk->disk_name); |
sd_sync_cache(sdkp); |
sd_sync_cache(sdp); |
|
} | } |
scsi_disk_put(sdkp); | scsi_disk_put(sdkp); |
} | } |
|
Lines 1852-1854
static void __exit exit_sd(void)
|
Link Here
|
|---|
|
| |
module_init(init_sd); | module_init(init_sd); |
module_exit(exit_sd); | module_exit(exit_sd); |
|
|
|
static void sd_print_sense_hdr(struct scsi_disk *sdkp, |
|
struct scsi_sense_hdr *sshdr) |
|
{ |
|
sd_printk(KERN_INFO, sdkp, ""); |
|
scsi_show_sense_hdr(sshdr); |
|
sd_printk(KERN_INFO, sdkp, ""); |
|
scsi_show_extd_sense(sshdr->asc, sshdr->ascq); |
|
} |
|
|
|
static void sd_print_result(struct scsi_disk *sdkp, int result) |
|
{ |
|
sd_printk(KERN_INFO, sdkp, ""); |
|
scsi_show_result(result); |
|
} |
|
|