提交 32782557 编写于 作者: H Hannes Reinecke 提交者: Jens Axboe

scsi_dh_rdac: switch to scsi_execute_req_flags()

Switch to scsi_execute_req_flags() and scsi_get_vpd_page() instead of
open-coding it.  Using scsi_execute_req_flags() will set REQ_QUIET and
REQ_PREEMPT, but this is okay as we're evaluating the errors anyway and
should be able to send the command even if the device is quiesced.
Signed-off-by: NHannes Reinecke <hare@suse.com>
Signed-off-by: NChristoph Hellwig <hch@lst.de>
Acked-by: NMartin K. Petersen <martin.petersen@oracle.com>
Signed-off-by: NJens Axboe <axboe@fb.com>
上级 eb8db831
...@@ -205,7 +205,6 @@ struct rdac_dh_data { ...@@ -205,7 +205,6 @@ struct rdac_dh_data {
#define RDAC_NON_PREFERRED 1 #define RDAC_NON_PREFERRED 1
char preferred; char preferred;
unsigned char sense[SCSI_SENSE_BUFFERSIZE];
union { union {
struct c2_inquiry c2; struct c2_inquiry c2;
struct c4_inquiry c4; struct c4_inquiry c4;
...@@ -262,40 +261,12 @@ do { \ ...@@ -262,40 +261,12 @@ do { \
sdev_printk(KERN_INFO, sdev, RDAC_NAME ": " f "\n", ## arg); \ sdev_printk(KERN_INFO, sdev, RDAC_NAME ": " f "\n", ## arg); \
} while (0); } while (0);
static struct request *get_rdac_req(struct scsi_device *sdev, static unsigned int rdac_failover_get(struct rdac_controller *ctlr,
void *buffer, unsigned buflen, int rw) struct list_head *list,
unsigned char *cdb)
{ {
struct request *rq; struct scsi_device *sdev = ctlr->ms_sdev;
struct request_queue *q = sdev->request_queue; struct rdac_dh_data *h = sdev->handler_data;
rq = blk_get_request(q, rw, GFP_NOIO);
if (IS_ERR(rq)) {
sdev_printk(KERN_INFO, sdev,
"get_rdac_req: blk_get_request failed.\n");
return NULL;
}
blk_rq_set_block_pc(rq);
if (buflen && blk_rq_map_kern(q, rq, buffer, buflen, GFP_NOIO)) {
blk_put_request(rq);
sdev_printk(KERN_INFO, sdev,
"get_rdac_req: blk_rq_map_kern failed.\n");
return NULL;
}
rq->cmd_flags |= REQ_FAILFAST_DEV | REQ_FAILFAST_TRANSPORT |
REQ_FAILFAST_DRIVER;
rq->retries = RDAC_RETRIES;
rq->timeout = RDAC_TIMEOUT;
return rq;
}
static struct request *rdac_failover_get(struct scsi_device *sdev,
struct rdac_dh_data *h, struct list_head *list)
{
struct request *rq;
struct rdac_mode_common *common; struct rdac_mode_common *common;
unsigned data_size; unsigned data_size;
struct rdac_queue_data *qdata; struct rdac_queue_data *qdata;
...@@ -332,27 +303,17 @@ static struct request *rdac_failover_get(struct scsi_device *sdev, ...@@ -332,27 +303,17 @@ static struct request *rdac_failover_get(struct scsi_device *sdev,
lun_table[qdata->h->lun] = 0x81; lun_table[qdata->h->lun] = 0x81;
} }
/* get request for block layer packet command */
rq = get_rdac_req(sdev, &h->ctlr->mode_select, data_size, WRITE);
if (!rq)
return NULL;
/* Prepare the command. */ /* Prepare the command. */
if (h->ctlr->use_ms10) { if (h->ctlr->use_ms10) {
rq->cmd[0] = MODE_SELECT_10; cdb[0] = MODE_SELECT_10;
rq->cmd[7] = data_size >> 8; cdb[7] = data_size >> 8;
rq->cmd[8] = data_size & 0xff; cdb[8] = data_size & 0xff;
} else { } else {
rq->cmd[0] = MODE_SELECT; cdb[0] = MODE_SELECT;
rq->cmd[4] = data_size; cdb[4] = data_size;
} }
rq->cmd_len = COMMAND_SIZE(rq->cmd[0]);
rq->sense = h->sense; return data_size;
memset(rq->sense, 0, SCSI_SENSE_BUFFERSIZE);
rq->sense_len = 0;
return rq;
} }
static void release_controller(struct kref *kref) static void release_controller(struct kref *kref)
...@@ -400,46 +361,14 @@ static struct rdac_controller *get_controller(int index, char *array_name, ...@@ -400,46 +361,14 @@ static struct rdac_controller *get_controller(int index, char *array_name,
return ctlr; return ctlr;
} }
static int submit_inquiry(struct scsi_device *sdev, int page_code,
unsigned int len, struct rdac_dh_data *h)
{
struct request *rq;
struct request_queue *q = sdev->request_queue;
int err = SCSI_DH_RES_TEMP_UNAVAIL;
rq = get_rdac_req(sdev, &h->inq, len, READ);
if (!rq)
goto done;
/* Prepare the command. */
rq->cmd[0] = INQUIRY;
rq->cmd[1] = 1;
rq->cmd[2] = page_code;
rq->cmd[4] = len;
rq->cmd_len = COMMAND_SIZE(INQUIRY);
rq->sense = h->sense;
memset(rq->sense, 0, SCSI_SENSE_BUFFERSIZE);
rq->sense_len = 0;
err = blk_execute_rq(q, NULL, rq, 1);
if (err == -EIO)
err = SCSI_DH_IO;
blk_put_request(rq);
done:
return err;
}
static int get_lun_info(struct scsi_device *sdev, struct rdac_dh_data *h, static int get_lun_info(struct scsi_device *sdev, struct rdac_dh_data *h,
char *array_name, u8 *array_id) char *array_name, u8 *array_id)
{ {
int err, i; int err = SCSI_DH_IO, i;
struct c8_inquiry *inqp; struct c8_inquiry *inqp = &h->inq.c8;
err = submit_inquiry(sdev, 0xC8, sizeof(struct c8_inquiry), h); if (!scsi_get_vpd_page(sdev, 0xC8, (unsigned char *)inqp,
if (err == SCSI_DH_OK) { sizeof(struct c8_inquiry))) {
inqp = &h->inq.c8;
if (inqp->page_code != 0xc8) if (inqp->page_code != 0xc8)
return SCSI_DH_NOSYS; return SCSI_DH_NOSYS;
if (inqp->page_id[0] != 'e' || inqp->page_id[1] != 'd' || if (inqp->page_id[0] != 'e' || inqp->page_id[1] != 'd' ||
...@@ -453,20 +382,20 @@ static int get_lun_info(struct scsi_device *sdev, struct rdac_dh_data *h, ...@@ -453,20 +382,20 @@ static int get_lun_info(struct scsi_device *sdev, struct rdac_dh_data *h,
*(array_name+ARRAY_LABEL_LEN-1) = '\0'; *(array_name+ARRAY_LABEL_LEN-1) = '\0';
memset(array_id, 0, UNIQUE_ID_LEN); memset(array_id, 0, UNIQUE_ID_LEN);
memcpy(array_id, inqp->array_unique_id, inqp->array_uniq_id_len); memcpy(array_id, inqp->array_unique_id, inqp->array_uniq_id_len);
err = SCSI_DH_OK;
} }
return err; return err;
} }
static int check_ownership(struct scsi_device *sdev, struct rdac_dh_data *h) static int check_ownership(struct scsi_device *sdev, struct rdac_dh_data *h)
{ {
int err, access_state; int err = SCSI_DH_IO, access_state;
struct rdac_dh_data *tmp; struct rdac_dh_data *tmp;
struct c9_inquiry *inqp; struct c9_inquiry *inqp = &h->inq.c9;
h->state = RDAC_STATE_ACTIVE; h->state = RDAC_STATE_ACTIVE;
err = submit_inquiry(sdev, 0xC9, sizeof(struct c9_inquiry), h); if (!scsi_get_vpd_page(sdev, 0xC9, (unsigned char *)inqp,
if (err == SCSI_DH_OK) { sizeof(struct c9_inquiry))) {
inqp = &h->inq.c9;
/* detect the operating mode */ /* detect the operating mode */
if ((inqp->avte_cvp >> 5) & 0x1) if ((inqp->avte_cvp >> 5) & 0x1)
h->mode = RDAC_MODE_IOSHIP; /* LUN in IOSHIP mode */ h->mode = RDAC_MODE_IOSHIP; /* LUN in IOSHIP mode */
...@@ -501,6 +430,7 @@ static int check_ownership(struct scsi_device *sdev, struct rdac_dh_data *h) ...@@ -501,6 +430,7 @@ static int check_ownership(struct scsi_device *sdev, struct rdac_dh_data *h)
tmp->sdev->access_state = access_state; tmp->sdev->access_state = access_state;
} }
rcu_read_unlock(); rcu_read_unlock();
err = SCSI_DH_OK;
} }
return err; return err;
...@@ -509,12 +439,11 @@ static int check_ownership(struct scsi_device *sdev, struct rdac_dh_data *h) ...@@ -509,12 +439,11 @@ static int check_ownership(struct scsi_device *sdev, struct rdac_dh_data *h)
static int initialize_controller(struct scsi_device *sdev, static int initialize_controller(struct scsi_device *sdev,
struct rdac_dh_data *h, char *array_name, u8 *array_id) struct rdac_dh_data *h, char *array_name, u8 *array_id)
{ {
int err, index; int err = SCSI_DH_IO, index;
struct c4_inquiry *inqp; struct c4_inquiry *inqp = &h->inq.c4;
err = submit_inquiry(sdev, 0xC4, sizeof(struct c4_inquiry), h); if (!scsi_get_vpd_page(sdev, 0xC4, (unsigned char *)inqp,
if (err == SCSI_DH_OK) { sizeof(struct c4_inquiry))) {
inqp = &h->inq.c4;
/* get the controller index */ /* get the controller index */
if (inqp->slot_id[1] == 0x31) if (inqp->slot_id[1] == 0x31)
index = 0; index = 0;
...@@ -530,18 +459,18 @@ static int initialize_controller(struct scsi_device *sdev, ...@@ -530,18 +459,18 @@ static int initialize_controller(struct scsi_device *sdev,
h->sdev = sdev; h->sdev = sdev;
} }
spin_unlock(&list_lock); spin_unlock(&list_lock);
err = SCSI_DH_OK;
} }
return err; return err;
} }
static int set_mode_select(struct scsi_device *sdev, struct rdac_dh_data *h) static int set_mode_select(struct scsi_device *sdev, struct rdac_dh_data *h)
{ {
int err; int err = SCSI_DH_IO;
struct c2_inquiry *inqp; struct c2_inquiry *inqp = &h->inq.c2;
err = submit_inquiry(sdev, 0xC2, sizeof(struct c2_inquiry), h); if (!scsi_get_vpd_page(sdev, 0xC2, (unsigned char *)inqp,
if (err == SCSI_DH_OK) { sizeof(struct c2_inquiry))) {
inqp = &h->inq.c2;
/* /*
* If more than MODE6_MAX_LUN luns are supported, use * If more than MODE6_MAX_LUN luns are supported, use
* mode select 10 * mode select 10
...@@ -550,36 +479,35 @@ static int set_mode_select(struct scsi_device *sdev, struct rdac_dh_data *h) ...@@ -550,36 +479,35 @@ static int set_mode_select(struct scsi_device *sdev, struct rdac_dh_data *h)
h->ctlr->use_ms10 = 1; h->ctlr->use_ms10 = 1;
else else
h->ctlr->use_ms10 = 0; h->ctlr->use_ms10 = 0;
err = SCSI_DH_OK;
} }
return err; return err;
} }
static int mode_select_handle_sense(struct scsi_device *sdev, static int mode_select_handle_sense(struct scsi_device *sdev,
unsigned char *sensebuf) struct scsi_sense_hdr *sense_hdr)
{ {
struct scsi_sense_hdr sense_hdr; int err = SCSI_DH_IO;
int err = SCSI_DH_IO, ret;
struct rdac_dh_data *h = sdev->handler_data; struct rdac_dh_data *h = sdev->handler_data;
ret = scsi_normalize_sense(sensebuf, SCSI_SENSE_BUFFERSIZE, &sense_hdr); if (!scsi_sense_valid(sense_hdr))
if (!ret)
goto done; goto done;
switch (sense_hdr.sense_key) { switch (sense_hdr->sense_key) {
case NO_SENSE: case NO_SENSE:
case ABORTED_COMMAND: case ABORTED_COMMAND:
case UNIT_ATTENTION: case UNIT_ATTENTION:
err = SCSI_DH_RETRY; err = SCSI_DH_RETRY;
break; break;
case NOT_READY: case NOT_READY:
if (sense_hdr.asc == 0x04 && sense_hdr.ascq == 0x01) if (sense_hdr->asc == 0x04 && sense_hdr->ascq == 0x01)
/* LUN Not Ready and is in the Process of Becoming /* LUN Not Ready and is in the Process of Becoming
* Ready * Ready
*/ */
err = SCSI_DH_RETRY; err = SCSI_DH_RETRY;
break; break;
case ILLEGAL_REQUEST: case ILLEGAL_REQUEST:
if (sense_hdr.asc == 0x91 && sense_hdr.ascq == 0x36) if (sense_hdr->asc == 0x91 && sense_hdr->ascq == 0x36)
/* /*
* Command Lock contention * Command Lock contention
*/ */
...@@ -592,7 +520,7 @@ static int mode_select_handle_sense(struct scsi_device *sdev, ...@@ -592,7 +520,7 @@ static int mode_select_handle_sense(struct scsi_device *sdev,
RDAC_LOG(RDAC_LOG_FAILOVER, sdev, "array %s, ctlr %d, " RDAC_LOG(RDAC_LOG_FAILOVER, sdev, "array %s, ctlr %d, "
"MODE_SELECT returned with sense %02x/%02x/%02x", "MODE_SELECT returned with sense %02x/%02x/%02x",
(char *) h->ctlr->array_name, h->ctlr->index, (char *) h->ctlr->array_name, h->ctlr->index,
sense_hdr.sense_key, sense_hdr.asc, sense_hdr.ascq); sense_hdr->sense_key, sense_hdr->asc, sense_hdr->ascq);
done: done:
return err; return err;
...@@ -602,13 +530,16 @@ static void send_mode_select(struct work_struct *work) ...@@ -602,13 +530,16 @@ static void send_mode_select(struct work_struct *work)
{ {
struct rdac_controller *ctlr = struct rdac_controller *ctlr =
container_of(work, struct rdac_controller, ms_work); container_of(work, struct rdac_controller, ms_work);
struct request *rq;
struct scsi_device *sdev = ctlr->ms_sdev; struct scsi_device *sdev = ctlr->ms_sdev;
struct rdac_dh_data *h = sdev->handler_data; struct rdac_dh_data *h = sdev->handler_data;
struct request_queue *q = sdev->request_queue; int err = SCSI_DH_OK, retry_cnt = RDAC_RETRY_COUNT;
int err, retry_cnt = RDAC_RETRY_COUNT;
struct rdac_queue_data *tmp, *qdata; struct rdac_queue_data *tmp, *qdata;
LIST_HEAD(list); LIST_HEAD(list);
unsigned char cdb[COMMAND_SIZE(MODE_SELECT_10)];
struct scsi_sense_hdr sshdr;
unsigned int data_size;
u64 req_flags = REQ_FAILFAST_DEV | REQ_FAILFAST_TRANSPORT |
REQ_FAILFAST_DRIVER;
spin_lock(&ctlr->ms_lock); spin_lock(&ctlr->ms_lock);
list_splice_init(&ctlr->ms_head, &list); list_splice_init(&ctlr->ms_head, &list);
...@@ -616,21 +547,19 @@ static void send_mode_select(struct work_struct *work) ...@@ -616,21 +547,19 @@ static void send_mode_select(struct work_struct *work)
ctlr->ms_sdev = NULL; ctlr->ms_sdev = NULL;
spin_unlock(&ctlr->ms_lock); spin_unlock(&ctlr->ms_lock);
retry: retry:
err = SCSI_DH_RES_TEMP_UNAVAIL; data_size = rdac_failover_get(ctlr, &list, cdb);
rq = rdac_failover_get(sdev, h, &list);
if (!rq)
goto done;
RDAC_LOG(RDAC_LOG_FAILOVER, sdev, "array %s, ctlr %d, " RDAC_LOG(RDAC_LOG_FAILOVER, sdev, "array %s, ctlr %d, "
"%s MODE_SELECT command", "%s MODE_SELECT command",
(char *) h->ctlr->array_name, h->ctlr->index, (char *) h->ctlr->array_name, h->ctlr->index,
(retry_cnt == RDAC_RETRY_COUNT) ? "queueing" : "retrying"); (retry_cnt == RDAC_RETRY_COUNT) ? "queueing" : "retrying");
err = blk_execute_rq(q, NULL, rq, 1); if (scsi_execute_req_flags(sdev, cdb, DMA_TO_DEVICE,
blk_put_request(rq); &h->ctlr->mode_select, data_size, &sshdr,
if (err != SCSI_DH_OK) { RDAC_TIMEOUT * HZ,
err = mode_select_handle_sense(sdev, h->sense); RDAC_RETRIES, NULL, req_flags, 0)) {
err = mode_select_handle_sense(sdev, &sshdr);
if (err == SCSI_DH_RETRY && retry_cnt--) if (err == SCSI_DH_RETRY && retry_cnt--)
goto retry; goto retry;
if (err == SCSI_DH_IMM_RETRY) if (err == SCSI_DH_IMM_RETRY)
...@@ -643,7 +572,6 @@ static void send_mode_select(struct work_struct *work) ...@@ -643,7 +572,6 @@ static void send_mode_select(struct work_struct *work)
(char *) h->ctlr->array_name, h->ctlr->index); (char *) h->ctlr->array_name, h->ctlr->index);
} }
done:
list_for_each_entry_safe(qdata, tmp, &list, entry) { list_for_each_entry_safe(qdata, tmp, &list, entry) {
list_del(&qdata->entry); list_del(&qdata->entry);
if (err == SCSI_DH_OK) if (err == SCSI_DH_OK)
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册