aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/scsi/hosts.c
diff options
context:
space:
mode:
Diffstat (limited to 'drivers/scsi/hosts.c')
-rw-r--r--drivers/scsi/hosts.c35
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;