diff options
Diffstat (limited to 'drivers/scsi/scsi_lib.c')
| -rw-r--r-- | drivers/scsi/scsi_lib.c | 42 |
1 files changed, 29 insertions, 13 deletions
diff --git a/drivers/scsi/scsi_lib.c b/drivers/scsi/scsi_lib.c index 4afef5cdcb17..dc249cb970ea 100644 --- a/drivers/scsi/scsi_lib.c +++ b/drivers/scsi/scsi_lib.c | |||
| @@ -542,10 +542,17 @@ static void scsi_requeue_command(struct request_queue *q, struct scsi_cmnd *cmd) | |||
| 542 | 542 | ||
| 543 | void scsi_next_command(struct scsi_cmnd *cmd) | 543 | void scsi_next_command(struct scsi_cmnd *cmd) |
| 544 | { | 544 | { |
| 545 | struct request_queue *q = cmd->device->request_queue; | 545 | struct scsi_device *sdev = cmd->device; |
| 546 | struct request_queue *q = sdev->request_queue; | ||
| 547 | |||
| 548 | /* need to hold a reference on the device before we let go of the cmd */ | ||
| 549 | get_device(&sdev->sdev_gendev); | ||
| 546 | 550 | ||
| 547 | scsi_put_command(cmd); | 551 | scsi_put_command(cmd); |
| 548 | scsi_run_queue(q); | 552 | scsi_run_queue(q); |
| 553 | |||
| 554 | /* ok to remove device now */ | ||
| 555 | put_device(&sdev->sdev_gendev); | ||
| 549 | } | 556 | } |
| 550 | 557 | ||
| 551 | void scsi_run_host_queues(struct Scsi_Host *shost) | 558 | void scsi_run_host_queues(struct Scsi_Host *shost) |
| @@ -1078,6 +1085,26 @@ static void scsi_generic_done(struct scsi_cmnd *cmd) | |||
| 1078 | scsi_io_completion(cmd, cmd->result == 0 ? cmd->bufflen : 0, 0); | 1085 | scsi_io_completion(cmd, cmd->result == 0 ? cmd->bufflen : 0, 0); |
| 1079 | } | 1086 | } |
| 1080 | 1087 | ||
| 1088 | void scsi_setup_blk_pc_cmnd(struct scsi_cmnd *cmd, int retries) | ||
| 1089 | { | ||
| 1090 | struct request *req = cmd->request; | ||
| 1091 | |||
| 1092 | BUG_ON(sizeof(req->cmd) > sizeof(cmd->cmnd)); | ||
| 1093 | memcpy(cmd->cmnd, req->cmd, sizeof(cmd->cmnd)); | ||
| 1094 | cmd->cmd_len = req->cmd_len; | ||
| 1095 | if (!req->data_len) | ||
| 1096 | cmd->sc_data_direction = DMA_NONE; | ||
| 1097 | else if (rq_data_dir(req) == WRITE) | ||
| 1098 | cmd->sc_data_direction = DMA_TO_DEVICE; | ||
| 1099 | else | ||
| 1100 | cmd->sc_data_direction = DMA_FROM_DEVICE; | ||
| 1101 | |||
| 1102 | cmd->transfersize = req->data_len; | ||
| 1103 | cmd->allowed = retries; | ||
| 1104 | cmd->timeout_per_command = req->timeout; | ||
| 1105 | } | ||
| 1106 | EXPORT_SYMBOL_GPL(scsi_setup_blk_pc_cmnd); | ||
| 1107 | |||
| 1081 | static int scsi_prep_fn(struct request_queue *q, struct request *req) | 1108 | static int scsi_prep_fn(struct request_queue *q, struct request *req) |
| 1082 | { | 1109 | { |
| 1083 | struct scsi_device *sdev = q->queuedata; | 1110 | struct scsi_device *sdev = q->queuedata; |
| @@ -1213,18 +1240,7 @@ static int scsi_prep_fn(struct request_queue *q, struct request *req) | |||
| 1213 | goto kill; | 1240 | goto kill; |
| 1214 | } | 1241 | } |
| 1215 | } else { | 1242 | } else { |
| 1216 | memcpy(cmd->cmnd, req->cmd, sizeof(cmd->cmnd)); | 1243 | scsi_setup_blk_pc_cmnd(cmd, 3); |
| 1217 | cmd->cmd_len = req->cmd_len; | ||
| 1218 | if (rq_data_dir(req) == WRITE) | ||
| 1219 | cmd->sc_data_direction = DMA_TO_DEVICE; | ||
| 1220 | else if (req->data_len) | ||
| 1221 | cmd->sc_data_direction = DMA_FROM_DEVICE; | ||
| 1222 | else | ||
| 1223 | cmd->sc_data_direction = DMA_NONE; | ||
| 1224 | |||
| 1225 | cmd->transfersize = req->data_len; | ||
| 1226 | cmd->allowed = 3; | ||
| 1227 | cmd->timeout_per_command = req->timeout; | ||
| 1228 | cmd->done = scsi_generic_done; | 1244 | cmd->done = scsi_generic_done; |
| 1229 | } | 1245 | } |
| 1230 | } | 1246 | } |
