aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/firewire/fw-sbp2.c
diff options
context:
space:
mode:
Diffstat (limited to 'drivers/firewire/fw-sbp2.c')
-rw-r--r--drivers/firewire/fw-sbp2.c49
1 files changed, 35 insertions, 14 deletions
diff --git a/drivers/firewire/fw-sbp2.c b/drivers/firewire/fw-sbp2.c
index 323b03bacd29..6d10934c58f1 100644
--- a/drivers/firewire/fw-sbp2.c
+++ b/drivers/firewire/fw-sbp2.c
@@ -762,22 +762,43 @@ static void sbp2_login(struct work_struct *work)
762 762
763 sdev = __scsi_add_device(shost, 0, 0, 763 sdev = __scsi_add_device(shost, 0, 0,
764 scsilun_to_int(&eight_bytes_lun), lu); 764 scsilun_to_int(&eight_bytes_lun), lu);
765 if (IS_ERR(sdev)) { 765 /*
766 smp_rmb(); /* generation may have changed */ 766 * FIXME: We are unable to perform reconnects while in sbp2_login().
767 generation = device->generation; 767 * Therefore __scsi_add_device() will get into trouble if a bus reset
768 smp_rmb(); /* node_id must not be older than generation */ 768 * happens in parallel. It will either fail or leave us with an
769 * unusable sdev. As a workaround we check for this and retry the
770 * whole login and SCSI probing.
771 */
769 772
770 sbp2_send_management_orb(lu, device->node_id, generation, 773 /* Reported error during __scsi_add_device() */
771 SBP2_LOGOUT_REQUEST, lu->login_id, NULL); 774 if (IS_ERR(sdev))
772 /* 775 goto out_logout_login;
773 * Set this back to sbp2_login so we fall back and 776
774 * retry login on bus reset. 777 scsi_device_put(sdev);
775 */ 778
776 PREPARE_DELAYED_WORK(&lu->work, sbp2_login); 779 /* Unreported error during __scsi_add_device() */
777 } else { 780 smp_rmb(); /* get current card generation */
778 lu->sdev = sdev; 781 if (generation != device->card->generation) {
779 scsi_device_put(sdev); 782 scsi_remove_device(sdev);
783 goto out_logout_login;
780 } 784 }
785
786 /* No error during __scsi_add_device() */
787 lu->sdev = sdev;
788 goto out;
789
790 out_logout_login:
791 smp_rmb(); /* generation may have changed */
792 generation = device->generation;
793 smp_rmb(); /* node_id must not be older than generation */
794
795 sbp2_send_management_orb(lu, device->node_id, generation,
796 SBP2_LOGOUT_REQUEST, lu->login_id, NULL);
797 /*
798 * If a bus reset happened, sbp2_update will have requeued
799 * lu->work already. Reset the work from reconnect to login.
800 */
801 PREPARE_DELAYED_WORK(&lu->work, sbp2_login);
781 out: 802 out:
782 sbp2_target_put(tgt); 803 sbp2_target_put(tgt);
783} 804}