提交 023b515e 编写于 作者: G Gerd Hoffmann 提交者: Greg Kroah-Hartman

uas: task mgmt & error handling

Add task management support, wind up in abort and device reset error
handlers.  Cancel all in-flight urbs in bus reset handler.
Signed-off-by: NGerd Hoffmann <kraxel@redhat.com>
Signed-off-by: NGreg Kroah-Hartman <gregkh@linuxfoundation.org>
上级 bdd000fb
...@@ -43,7 +43,8 @@ struct uas_dev_info { ...@@ -43,7 +43,8 @@ struct uas_dev_info {
struct usb_device *udev; struct usb_device *udev;
struct usb_anchor sense_urbs; struct usb_anchor sense_urbs;
struct usb_anchor data_urbs; struct usb_anchor data_urbs;
int qdepth; int qdepth, resetting;
struct response_ui response;
unsigned cmd_pipe, status_pipe, data_in_pipe, data_out_pipe; unsigned cmd_pipe, status_pipe, data_in_pipe, data_out_pipe;
unsigned use_streams:1; unsigned use_streams:1;
unsigned uas_sense_old:1; unsigned uas_sense_old:1;
...@@ -68,6 +69,7 @@ enum { ...@@ -68,6 +69,7 @@ enum {
struct uas_cmd_info { struct uas_cmd_info {
unsigned int state; unsigned int state;
unsigned int stream; unsigned int stream;
unsigned int aborted;
struct urb *cmd_urb; struct urb *cmd_urb;
struct urb *data_in_urb; struct urb *data_in_urb;
struct urb *data_out_urb; struct urb *data_out_urb;
...@@ -222,16 +224,24 @@ static void uas_stat_cmplt(struct urb *urb) ...@@ -222,16 +224,24 @@ static void uas_stat_cmplt(struct urb *urb)
return; return;
} }
if (devinfo->resetting) {
usb_free_urb(urb);
return;
}
tag = be16_to_cpup(&iu->tag) - 1; tag = be16_to_cpup(&iu->tag) - 1;
if (tag == 0) if (tag == 0)
cmnd = devinfo->cmnd; cmnd = devinfo->cmnd;
else else
cmnd = scsi_host_find_tag(shost, tag - 1); cmnd = scsi_host_find_tag(shost, tag - 1);
if (!cmnd) { if (!cmnd) {
if (iu->iu_id != IU_ID_RESPONSE) {
usb_free_urb(urb); usb_free_urb(urb);
return; return;
} }
} else {
cmdinfo = (void *)&cmnd->SCp; cmdinfo = (void *)&cmnd->SCp;
}
switch (iu->iu_id) { switch (iu->iu_id) {
case IU_ID_STATUS: case IU_ID_STATUS:
...@@ -260,6 +270,10 @@ static void uas_stat_cmplt(struct urb *urb) ...@@ -260,6 +270,10 @@ static void uas_stat_cmplt(struct urb *urb)
case IU_ID_WRITE_READY: case IU_ID_WRITE_READY:
uas_xfer_data(urb, cmnd, SUBMIT_DATA_OUT_URB); uas_xfer_data(urb, cmnd, SUBMIT_DATA_OUT_URB);
break; break;
case IU_ID_RESPONSE:
/* store results for uas_eh_task_mgmt() */
memcpy(&devinfo->response, iu, sizeof(devinfo->response));
break;
default: default:
scmd_printk(KERN_ERR, cmnd, scmd_printk(KERN_ERR, cmnd,
"Bogus IU (%d) received on status pipe\n", iu->iu_id); "Bogus IU (%d) received on status pipe\n", iu->iu_id);
...@@ -287,6 +301,9 @@ static void uas_data_cmplt(struct urb *urb) ...@@ -287,6 +301,9 @@ static void uas_data_cmplt(struct urb *urb)
} else { } else {
sdb->resid = sdb->length - urb->actual_length; sdb->resid = sdb->length - urb->actual_length;
} }
if (cmdinfo->aborted) {
return;
}
uas_try_complete(cmnd, __func__); uas_try_complete(cmnd, __func__);
} }
...@@ -377,6 +394,51 @@ static struct urb *uas_alloc_cmd_urb(struct uas_dev_info *devinfo, gfp_t gfp, ...@@ -377,6 +394,51 @@ static struct urb *uas_alloc_cmd_urb(struct uas_dev_info *devinfo, gfp_t gfp,
return NULL; return NULL;
} }
static int uas_submit_task_urb(struct scsi_cmnd *cmnd, gfp_t gfp,
u8 function, u16 stream_id)
{
struct uas_dev_info *devinfo = (void *)cmnd->device->hostdata;
struct usb_device *udev = devinfo->udev;
struct urb *urb = usb_alloc_urb(0, gfp);
struct task_mgmt_iu *iu;
int err = -ENOMEM;
if (!urb)
goto err;
iu = kzalloc(sizeof(*iu), gfp);
if (!iu)
goto err;
iu->iu_id = IU_ID_TASK_MGMT;
iu->tag = cpu_to_be16(stream_id);
int_to_scsilun(cmnd->device->lun, &iu->lun);
iu->function = function;
switch (function) {
case TMF_ABORT_TASK:
if (blk_rq_tagged(cmnd->request))
iu->task_tag = cpu_to_be16(cmnd->request->tag + 2);
else
iu->task_tag = cpu_to_be16(1);
break;
}
usb_fill_bulk_urb(urb, udev, devinfo->cmd_pipe, iu, sizeof(*iu),
usb_free_urb, NULL);
urb->transfer_flags |= URB_FREE_BUFFER;
err = usb_submit_urb(urb, gfp);
if (err)
goto err;
return 0;
err:
usb_free_urb(urb);
return err;
}
/* /*
* Why should I request the Status IU before sending the Command IU? Spec * Why should I request the Status IU before sending the Command IU? Spec
* says to, but also says the device may receive them in any order. Seems * says to, but also says the device may receive them in any order. Seems
...@@ -502,6 +564,7 @@ static int uas_queuecommand_lck(struct scsi_cmnd *cmnd, ...@@ -502,6 +564,7 @@ static int uas_queuecommand_lck(struct scsi_cmnd *cmnd,
cmdinfo->state = SUBMIT_STATUS_URB | cmdinfo->state = SUBMIT_STATUS_URB |
ALLOC_CMD_URB | SUBMIT_CMD_URB; ALLOC_CMD_URB | SUBMIT_CMD_URB;
cmdinfo->aborted = 0;
switch (cmnd->sc_data_direction) { switch (cmnd->sc_data_direction) {
case DMA_FROM_DEVICE: case DMA_FROM_DEVICE:
...@@ -537,34 +600,66 @@ static int uas_queuecommand_lck(struct scsi_cmnd *cmnd, ...@@ -537,34 +600,66 @@ static int uas_queuecommand_lck(struct scsi_cmnd *cmnd,
static DEF_SCSI_QCMD(uas_queuecommand) static DEF_SCSI_QCMD(uas_queuecommand)
static int uas_eh_abort_handler(struct scsi_cmnd *cmnd) static int uas_eh_task_mgmt(struct scsi_cmnd *cmnd,
const char *fname, u8 function)
{ {
uas_log_cmd_state(cmnd, __func__); struct Scsi_Host *shost = cmnd->device->host;
struct uas_dev_info *devinfo = (void *)shost->hostdata[0];
u16 tag = 9999; /* FIXME */
/* XXX: Send ABORT TASK Task Management command */ memset(&devinfo->response, 0, sizeof(devinfo->response));
if (uas_submit_sense_urb(shost, GFP_NOIO, tag)) {
shost_printk(KERN_INFO, shost,
"%s: %s: submit sense urb failed\n",
__func__, fname);
return FAILED;
}
if (uas_submit_task_urb(cmnd, GFP_NOIO, function, tag)) {
shost_printk(KERN_INFO, shost,
"%s: %s: submit task mgmt urb failed\n",
__func__, fname);
return FAILED;
}
if (0 == usb_wait_anchor_empty_timeout(&devinfo->sense_urbs, 3000)) {
shost_printk(KERN_INFO, shost,
"%s: %s timed out\n", __func__, fname);
return FAILED;
}
if (be16_to_cpu(devinfo->response.tag) != tag) {
shost_printk(KERN_INFO, shost,
"%s: %s failed (wrong tag %d/%d)\n", __func__,
fname, be16_to_cpu(devinfo->response.tag), tag);
return FAILED; return FAILED;
}
if (devinfo->response.response_code != RC_TMF_COMPLETE) {
shost_printk(KERN_INFO, shost,
"%s: %s failed (rc 0x%x)\n", __func__,
fname, devinfo->response.response_code);
return FAILED;
}
return SUCCESS;
} }
static int uas_eh_device_reset_handler(struct scsi_cmnd *cmnd) static int uas_eh_abort_handler(struct scsi_cmnd *cmnd)
{ {
struct scsi_device *sdev = cmnd->device; struct uas_cmd_info *cmdinfo = (void *)&cmnd->SCp;
sdev_printk(KERN_INFO, sdev, "%s tag %d\n", __func__, int ret;
cmnd->request->tag);
/* XXX: Send LOGICAL UNIT RESET Task Management command */ uas_log_cmd_state(cmnd, __func__);
return FAILED; cmdinfo->aborted = 1;
ret = uas_eh_task_mgmt(cmnd, "ABORT TASK", TMF_ABORT_TASK);
if (cmdinfo->state & DATA_IN_URB_INFLIGHT)
usb_kill_urb(cmdinfo->data_in_urb);
if (cmdinfo->state & DATA_OUT_URB_INFLIGHT)
usb_kill_urb(cmdinfo->data_out_urb);
return ret;
} }
static int uas_eh_target_reset_handler(struct scsi_cmnd *cmnd) static int uas_eh_device_reset_handler(struct scsi_cmnd *cmnd)
{ {
struct scsi_device *sdev = cmnd->device; sdev_printk(KERN_INFO, cmnd->device, "%s\n", __func__);
sdev_printk(KERN_INFO, sdev, "%s tag %d\n", __func__, return uas_eh_task_mgmt(cmnd, "LOGICAL UNIT RESET",
cmnd->request->tag); TMF_LOGICAL_UNIT_RESET);
/* XXX: Can we reset just the one USB interface?
* Would calling usb_set_interface() have the right effect?
*/
return FAILED;
} }
static int uas_eh_bus_reset_handler(struct scsi_cmnd *cmnd) static int uas_eh_bus_reset_handler(struct scsi_cmnd *cmnd)
...@@ -572,14 +667,21 @@ static int uas_eh_bus_reset_handler(struct scsi_cmnd *cmnd) ...@@ -572,14 +667,21 @@ static int uas_eh_bus_reset_handler(struct scsi_cmnd *cmnd)
struct scsi_device *sdev = cmnd->device; struct scsi_device *sdev = cmnd->device;
struct uas_dev_info *devinfo = sdev->hostdata; struct uas_dev_info *devinfo = sdev->hostdata;
struct usb_device *udev = devinfo->udev; struct usb_device *udev = devinfo->udev;
int err;
sdev_printk(KERN_INFO, sdev, "%s tag %d\n", __func__, devinfo->resetting = 1;
cmnd->request->tag); usb_kill_anchored_urbs(&devinfo->sense_urbs);
usb_kill_anchored_urbs(&devinfo->data_urbs);
if (usb_reset_device(udev)) err = usb_reset_device(udev);
return SUCCESS; devinfo->resetting = 0;
if (err) {
shost_printk(KERN_INFO, sdev->host, "%s FAILED\n", __func__);
return FAILED; return FAILED;
}
shost_printk(KERN_INFO, sdev->host, "%s success\n", __func__);
return SUCCESS;
} }
static int uas_slave_alloc(struct scsi_device *sdev) static int uas_slave_alloc(struct scsi_device *sdev)
...@@ -604,7 +706,6 @@ static struct scsi_host_template uas_host_template = { ...@@ -604,7 +706,6 @@ static struct scsi_host_template uas_host_template = {
.slave_configure = uas_slave_configure, .slave_configure = uas_slave_configure,
.eh_abort_handler = uas_eh_abort_handler, .eh_abort_handler = uas_eh_abort_handler,
.eh_device_reset_handler = uas_eh_device_reset_handler, .eh_device_reset_handler = uas_eh_device_reset_handler,
.eh_target_reset_handler = uas_eh_target_reset_handler,
.eh_bus_reset_handler = uas_eh_bus_reset_handler, .eh_bus_reset_handler = uas_eh_bus_reset_handler,
.can_queue = 65536, /* Is there a limit on the _host_ ? */ .can_queue = 65536, /* Is there a limit on the _host_ ? */
.this_id = -1, .this_id = -1,
...@@ -766,6 +867,7 @@ static int uas_probe(struct usb_interface *intf, const struct usb_device_id *id) ...@@ -766,6 +867,7 @@ static int uas_probe(struct usb_interface *intf, const struct usb_device_id *id)
devinfo->intf = intf; devinfo->intf = intf;
devinfo->udev = udev; devinfo->udev = udev;
devinfo->resetting = 0;
init_usb_anchor(&devinfo->sense_urbs); init_usb_anchor(&devinfo->sense_urbs);
init_usb_anchor(&devinfo->data_urbs); init_usb_anchor(&devinfo->data_urbs);
uas_configure_endpoints(devinfo); uas_configure_endpoints(devinfo);
......
...@@ -20,6 +20,28 @@ enum { ...@@ -20,6 +20,28 @@ enum {
IU_ID_WRITE_READY = 0x07, IU_ID_WRITE_READY = 0x07,
}; };
enum {
TMF_ABORT_TASK = 0x01,
TMF_ABORT_TASK_SET = 0x02,
TMF_CLEAR_TASK_SET = 0x04,
TMF_LOGICAL_UNIT_RESET = 0x08,
TMF_I_T_NEXUS_RESET = 0x10,
TMF_CLEAR_ACA = 0x40,
TMF_QUERY_TASK = 0x80,
TMF_QUERY_TASK_SET = 0x81,
TMF_QUERY_ASYNC_EVENT = 0x82,
};
enum {
RC_TMF_COMPLETE = 0x00,
RC_INVALID_INFO_UNIT = 0x02,
RC_TMF_NOT_SUPPORTED = 0x04,
RC_TMF_FAILED = 0x05,
RC_TMF_SUCCEEDED = 0x08,
RC_INCORRECT_LUN = 0x09,
RC_OVERLAPPED_TAG = 0x0a,
};
struct command_iu { struct command_iu {
__u8 iu_id; __u8 iu_id;
__u8 rsvd1; __u8 rsvd1;
...@@ -32,6 +54,16 @@ struct command_iu { ...@@ -32,6 +54,16 @@ struct command_iu {
__u8 cdb[16]; /* XXX: Overflow-checking tools may misunderstand */ __u8 cdb[16]; /* XXX: Overflow-checking tools may misunderstand */
}; };
struct task_mgmt_iu {
__u8 iu_id;
__u8 rsvd1;
__be16 tag;
__u8 function;
__u8 rsvd2;
__be16 task_tag;
struct scsi_lun lun;
};
/* /*
* Also used for the Read Ready and Write Ready IUs since they have the * Also used for the Read Ready and Write Ready IUs since they have the
* same first four bytes * same first four bytes
...@@ -47,6 +79,14 @@ struct sense_iu { ...@@ -47,6 +79,14 @@ struct sense_iu {
__u8 sense[SCSI_SENSE_BUFFERSIZE]; __u8 sense[SCSI_SENSE_BUFFERSIZE];
}; };
struct response_ui {
__u8 iu_id;
__u8 rsvd1;
__be16 tag;
__be16 add_response_info;
__u8 response_code;
};
struct usb_pipe_usage_descriptor { struct usb_pipe_usage_descriptor {
__u8 bLength; __u8 bLength;
__u8 bDescriptorType; __u8 bDescriptorType;
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册
新手
引导
客服 返回
顶部