diff options
Diffstat (limited to 'drivers/scsi/hosts.c')
-rw-r--r-- | drivers/scsi/hosts.c | 35 |
1 files changed, 30 insertions, 5 deletions
diff --git a/drivers/scsi/hosts.c b/drivers/scsi/hosts.c index 0632eee82620..6de80e352871 100644 --- a/drivers/scsi/hosts.c +++ b/drivers/scsi/hosts.c | |||
@@ -213,9 +213,24 @@ int scsi_add_host_with_dma(struct Scsi_Host *shost, struct device *dev, | |||
213 | goto fail; | 213 | goto fail; |
214 | } | 214 | } |
215 | 215 | ||
216 | if (shost_use_blk_mq(shost)) { | ||
217 | error = scsi_mq_setup_tags(shost); | ||
218 | if (error) | ||
219 | goto fail; | ||
220 | } | ||
221 | |||
222 | /* | ||
223 | * Note that we allocate the freelist even for the MQ case for now, | ||
224 | * as we need a command set aside for scsi_reset_provider. Having | ||
225 | * the full host freelist and one command available for that is a | ||
226 | * little heavy-handed, but avoids introducing a special allocator | ||
227 | * just for this. Eventually the structure of scsi_reset_provider | ||
228 | * will need a major overhaul. | ||
229 | */ | ||
216 | error = scsi_setup_command_freelist(shost); | 230 | error = scsi_setup_command_freelist(shost); |
217 | if (error) | 231 | if (error) |
218 | goto fail; | 232 | goto out_destroy_tags; |
233 | |||
219 | 234 | ||
220 | if (!shost->shost_gendev.parent) | 235 | if (!shost->shost_gendev.parent) |
221 | shost->shost_gendev.parent = dev ? dev : &platform_bus; | 236 | shost->shost_gendev.parent = dev ? dev : &platform_bus; |
@@ -226,7 +241,7 @@ int scsi_add_host_with_dma(struct Scsi_Host *shost, struct device *dev, | |||
226 | 241 | ||
227 | error = device_add(&shost->shost_gendev); | 242 | error = device_add(&shost->shost_gendev); |
228 | if (error) | 243 | if (error) |
229 | goto out; | 244 | goto out_destroy_freelist; |
230 | 245 | ||
231 | pm_runtime_set_active(&shost->shost_gendev); | 246 | pm_runtime_set_active(&shost->shost_gendev); |
232 | pm_runtime_enable(&shost->shost_gendev); | 247 | pm_runtime_enable(&shost->shost_gendev); |
@@ -279,8 +294,11 @@ int scsi_add_host_with_dma(struct Scsi_Host *shost, struct device *dev, | |||
279 | device_del(&shost->shost_dev); | 294 | device_del(&shost->shost_dev); |
280 | out_del_gendev: | 295 | out_del_gendev: |
281 | device_del(&shost->shost_gendev); | 296 | device_del(&shost->shost_gendev); |
282 | out: | 297 | out_destroy_freelist: |
283 | scsi_destroy_command_freelist(shost); | 298 | scsi_destroy_command_freelist(shost); |
299 | out_destroy_tags: | ||
300 | if (shost_use_blk_mq(shost)) | ||
301 | scsi_mq_destroy_tags(shost); | ||
284 | fail: | 302 | fail: |
285 | return error; | 303 | return error; |
286 | } | 304 | } |
@@ -309,8 +327,13 @@ static void scsi_host_dev_release(struct device *dev) | |||
309 | } | 327 | } |
310 | 328 | ||
311 | scsi_destroy_command_freelist(shost); | 329 | scsi_destroy_command_freelist(shost); |
312 | if (shost->bqt) | 330 | if (shost_use_blk_mq(shost)) { |
313 | blk_free_tags(shost->bqt); | 331 | if (shost->tag_set.tags) |
332 | scsi_mq_destroy_tags(shost); | ||
333 | } else { | ||
334 | if (shost->bqt) | ||
335 | blk_free_tags(shost->bqt); | ||
336 | } | ||
314 | 337 | ||
315 | kfree(shost->shost_data); | 338 | kfree(shost->shost_data); |
316 | 339 | ||
@@ -436,6 +459,8 @@ struct Scsi_Host *scsi_host_alloc(struct scsi_host_template *sht, int privsize) | |||
436 | else | 459 | else |
437 | shost->dma_boundary = 0xffffffff; | 460 | shost->dma_boundary = 0xffffffff; |
438 | 461 | ||
462 | shost->use_blk_mq = scsi_use_blk_mq && !shost->hostt->disable_blk_mq; | ||
463 | |||
439 | device_initialize(&shost->shost_gendev); | 464 | device_initialize(&shost->shost_gendev); |
440 | dev_set_name(&shost->shost_gendev, "host%d", shost->host_no); | 465 | dev_set_name(&shost->shost_gendev, "host%d", shost->host_no); |
441 | shost->shost_gendev.bus = &scsi_bus_type; | 466 | shost->shost_gendev.bus = &scsi_bus_type; |