diff options
Diffstat (limited to 'drivers/scsi/scsi_scan.c')
-rw-r--r-- | drivers/scsi/scsi_scan.c | 72 |
1 files changed, 34 insertions, 38 deletions
diff --git a/drivers/scsi/scsi_scan.c b/drivers/scsi/scsi_scan.c index e1644b270cdc..fcd7455ffc39 100644 --- a/drivers/scsi/scsi_scan.c +++ b/drivers/scsi/scsi_scan.c | |||
@@ -322,6 +322,21 @@ out: | |||
322 | return NULL; | 322 | return NULL; |
323 | } | 323 | } |
324 | 324 | ||
325 | static void scsi_target_destroy(struct scsi_target *starget) | ||
326 | { | ||
327 | struct device *dev = &starget->dev; | ||
328 | struct Scsi_Host *shost = dev_to_shost(dev->parent); | ||
329 | unsigned long flags; | ||
330 | |||
331 | transport_destroy_device(dev); | ||
332 | spin_lock_irqsave(shost->host_lock, flags); | ||
333 | if (shost->hostt->target_destroy) | ||
334 | shost->hostt->target_destroy(starget); | ||
335 | list_del_init(&starget->siblings); | ||
336 | spin_unlock_irqrestore(shost->host_lock, flags); | ||
337 | put_device(dev); | ||
338 | } | ||
339 | |||
325 | static void scsi_target_dev_release(struct device *dev) | 340 | static void scsi_target_dev_release(struct device *dev) |
326 | { | 341 | { |
327 | struct device *parent = dev->parent; | 342 | struct device *parent = dev->parent; |
@@ -406,7 +421,7 @@ static struct scsi_target *scsi_alloc_target(struct device *parent, | |||
406 | starget->channel = channel; | 421 | starget->channel = channel; |
407 | INIT_LIST_HEAD(&starget->siblings); | 422 | INIT_LIST_HEAD(&starget->siblings); |
408 | INIT_LIST_HEAD(&starget->devices); | 423 | INIT_LIST_HEAD(&starget->devices); |
409 | starget->state = STARGET_RUNNING; | 424 | starget->state = STARGET_CREATED; |
410 | starget->scsi_level = SCSI_2; | 425 | starget->scsi_level = SCSI_2; |
411 | retry: | 426 | retry: |
412 | spin_lock_irqsave(shost->host_lock, flags); | 427 | spin_lock_irqsave(shost->host_lock, flags); |
@@ -419,18 +434,6 @@ static struct scsi_target *scsi_alloc_target(struct device *parent, | |||
419 | spin_unlock_irqrestore(shost->host_lock, flags); | 434 | spin_unlock_irqrestore(shost->host_lock, flags); |
420 | /* allocate and add */ | 435 | /* allocate and add */ |
421 | transport_setup_device(dev); | 436 | transport_setup_device(dev); |
422 | error = device_add(dev); | ||
423 | if (error) { | ||
424 | dev_err(dev, "target device_add failed, error %d\n", error); | ||
425 | spin_lock_irqsave(shost->host_lock, flags); | ||
426 | list_del_init(&starget->siblings); | ||
427 | spin_unlock_irqrestore(shost->host_lock, flags); | ||
428 | transport_destroy_device(dev); | ||
429 | put_device(parent); | ||
430 | kfree(starget); | ||
431 | return NULL; | ||
432 | } | ||
433 | transport_add_device(dev); | ||
434 | if (shost->hostt->target_alloc) { | 437 | if (shost->hostt->target_alloc) { |
435 | error = shost->hostt->target_alloc(starget); | 438 | error = shost->hostt->target_alloc(starget); |
436 | 439 | ||
@@ -438,9 +441,7 @@ static struct scsi_target *scsi_alloc_target(struct device *parent, | |||
438 | dev_printk(KERN_ERR, dev, "target allocation failed, error %d\n", error); | 441 | dev_printk(KERN_ERR, dev, "target allocation failed, error %d\n", error); |
439 | /* don't want scsi_target_reap to do the final | 442 | /* don't want scsi_target_reap to do the final |
440 | * put because it will be under the host lock */ | 443 | * put because it will be under the host lock */ |
441 | get_device(dev); | 444 | scsi_target_destroy(starget); |
442 | scsi_target_reap(starget); | ||
443 | put_device(dev); | ||
444 | return NULL; | 445 | return NULL; |
445 | } | 446 | } |
446 | } | 447 | } |
@@ -467,18 +468,10 @@ static void scsi_target_reap_usercontext(struct work_struct *work) | |||
467 | { | 468 | { |
468 | struct scsi_target *starget = | 469 | struct scsi_target *starget = |
469 | container_of(work, struct scsi_target, ew.work); | 470 | container_of(work, struct scsi_target, ew.work); |
470 | struct Scsi_Host *shost = dev_to_shost(starget->dev.parent); | ||
471 | unsigned long flags; | ||
472 | 471 | ||
473 | transport_remove_device(&starget->dev); | 472 | transport_remove_device(&starget->dev); |
474 | device_del(&starget->dev); | 473 | device_del(&starget->dev); |
475 | transport_destroy_device(&starget->dev); | 474 | scsi_target_destroy(starget); |
476 | spin_lock_irqsave(shost->host_lock, flags); | ||
477 | if (shost->hostt->target_destroy) | ||
478 | shost->hostt->target_destroy(starget); | ||
479 | list_del_init(&starget->siblings); | ||
480 | spin_unlock_irqrestore(shost->host_lock, flags); | ||
481 | put_device(&starget->dev); | ||
482 | } | 475 | } |
483 | 476 | ||
484 | /** | 477 | /** |
@@ -493,21 +486,25 @@ void scsi_target_reap(struct scsi_target *starget) | |||
493 | { | 486 | { |
494 | struct Scsi_Host *shost = dev_to_shost(starget->dev.parent); | 487 | struct Scsi_Host *shost = dev_to_shost(starget->dev.parent); |
495 | unsigned long flags; | 488 | unsigned long flags; |
489 | enum scsi_target_state state; | ||
490 | int empty; | ||
496 | 491 | ||
497 | spin_lock_irqsave(shost->host_lock, flags); | 492 | spin_lock_irqsave(shost->host_lock, flags); |
493 | state = starget->state; | ||
494 | empty = --starget->reap_ref == 0 && | ||
495 | list_empty(&starget->devices) ? 1 : 0; | ||
496 | spin_unlock_irqrestore(shost->host_lock, flags); | ||
498 | 497 | ||
499 | if (--starget->reap_ref == 0 && list_empty(&starget->devices)) { | 498 | if (!empty) |
500 | BUG_ON(starget->state == STARGET_DEL); | ||
501 | starget->state = STARGET_DEL; | ||
502 | spin_unlock_irqrestore(shost->host_lock, flags); | ||
503 | execute_in_process_context(scsi_target_reap_usercontext, | ||
504 | &starget->ew); | ||
505 | return; | 499 | return; |
506 | 500 | ||
507 | } | 501 | BUG_ON(state == STARGET_DEL); |
508 | spin_unlock_irqrestore(shost->host_lock, flags); | 502 | starget->state = STARGET_DEL; |
509 | 503 | if (state == STARGET_CREATED) | |
510 | return; | 504 | scsi_target_destroy(starget); |
505 | else | ||
506 | execute_in_process_context(scsi_target_reap_usercontext, | ||
507 | &starget->ew); | ||
511 | } | 508 | } |
512 | 509 | ||
513 | /** | 510 | /** |
@@ -1056,8 +1053,9 @@ static int scsi_probe_and_add_lun(struct scsi_target *starget, | |||
1056 | scsi_inq_str(vend, result, 8, 16), | 1053 | scsi_inq_str(vend, result, 8, 16), |
1057 | scsi_inq_str(mod, result, 16, 32)); | 1054 | scsi_inq_str(mod, result, 16, 32)); |
1058 | }); | 1055 | }); |
1056 | |||
1059 | } | 1057 | } |
1060 | 1058 | ||
1061 | res = SCSI_SCAN_TARGET_PRESENT; | 1059 | res = SCSI_SCAN_TARGET_PRESENT; |
1062 | goto out_free_result; | 1060 | goto out_free_result; |
1063 | } | 1061 | } |
@@ -1497,7 +1495,6 @@ struct scsi_device *__scsi_add_device(struct Scsi_Host *shost, uint channel, | |||
1497 | if (scsi_host_scan_allowed(shost)) | 1495 | if (scsi_host_scan_allowed(shost)) |
1498 | scsi_probe_and_add_lun(starget, lun, NULL, &sdev, 1, hostdata); | 1496 | scsi_probe_and_add_lun(starget, lun, NULL, &sdev, 1, hostdata); |
1499 | mutex_unlock(&shost->scan_mutex); | 1497 | mutex_unlock(&shost->scan_mutex); |
1500 | transport_configure_device(&starget->dev); | ||
1501 | scsi_target_reap(starget); | 1498 | scsi_target_reap(starget); |
1502 | put_device(&starget->dev); | 1499 | put_device(&starget->dev); |
1503 | 1500 | ||
@@ -1578,7 +1575,6 @@ static void __scsi_scan_target(struct device *parent, unsigned int channel, | |||
1578 | out_reap: | 1575 | out_reap: |
1579 | /* now determine if the target has any children at all | 1576 | /* now determine if the target has any children at all |
1580 | * and if not, nuke it */ | 1577 | * and if not, nuke it */ |
1581 | transport_configure_device(&starget->dev); | ||
1582 | scsi_target_reap(starget); | 1578 | scsi_target_reap(starget); |
1583 | 1579 | ||
1584 | put_device(&starget->dev); | 1580 | put_device(&starget->dev); |