Commit 6825b7bd authored by James Smart's avatar James Smart Committed by Martin K. Petersen

scsi: lpfc: Fix error in remote port address change

In a test with high nvme remote port counts connected via a multi-hop FC
switch config where switches were systematically reset (e.g. fabric
partitioning and re-establishment), the nvme remote ports would switch
addresses based on the switch reconfiguration events. The driver would get
into a situation where the nvme port changed address, PLOGI and PRLI would
succeed nvme transport registration occurred, but subsequent LS requests by
the nvme subsystem failed due to a bad ndlp state and connectivity to the
device failed.

The driver hit a race condition on multiple devices that address swapped
simultaneously. In cases where the driver notices the remote port structure
came back as the same value as previously (meaning a nvme_rport structure
was re-enabled and did not go through devloss_tmo/connect_tmo_failures on
all controllers) the driver would unconditionally exit assuming the ndlp
information was correct. But, if the ndlp's had been swapped, the ndlp had
stale port state information, which when used by the LS request commands,
would fail the commands.

Fix by checking whether a node swap had occurred, and only exit if no ndlp
swap had occurred.
Signed-off-by: default avatarDick Kennedy <dick.kennedy@broadcom.com>
Signed-off-by: default avatarJames Smart <jsmart2021@gmail.com>
Signed-off-by: default avatarMartin K. Petersen <martin.petersen@oracle.com>
parent a6d10f24
...@@ -2348,7 +2348,7 @@ lpfc_nvme_register_port(struct lpfc_vport *vport, struct lpfc_nodelist *ndlp) ...@@ -2348,7 +2348,7 @@ lpfc_nvme_register_port(struct lpfc_vport *vport, struct lpfc_nodelist *ndlp)
*/ */
lpfc_printf_vlog(ndlp->vport, KERN_INFO, lpfc_printf_vlog(ndlp->vport, KERN_INFO,
LOG_NVME_DISC, LOG_NVME_DISC,
"6014 Rebinding lport to " "6014 Rebind lport to current "
"remoteport %p wwpn 0x%llx, " "remoteport %p wwpn 0x%llx, "
"Data: x%x x%x %p %p x%x x%06x\n", "Data: x%x x%x %p %p x%x x%06x\n",
remote_port, remote_port,
...@@ -2359,7 +2359,16 @@ lpfc_nvme_register_port(struct lpfc_vport *vport, struct lpfc_nodelist *ndlp) ...@@ -2359,7 +2359,16 @@ lpfc_nvme_register_port(struct lpfc_vport *vport, struct lpfc_nodelist *ndlp)
ndlp, ndlp,
ndlp->nlp_type, ndlp->nlp_type,
ndlp->nlp_DID); ndlp->nlp_DID);
/* It's a complete rebind only if the driver
* is registering with the same ndlp. Otherwise
* the driver likely executed a node swap
* prior to this registration and the ndlp to
* remoteport binding needs to be redone.
*/
if (prev_ndlp == ndlp)
return 0; return 0;
} }
/* Sever the ndlp<->rport association /* Sever the ndlp<->rport association
...@@ -2393,8 +2402,8 @@ lpfc_nvme_register_port(struct lpfc_vport *vport, struct lpfc_nodelist *ndlp) ...@@ -2393,8 +2402,8 @@ lpfc_nvme_register_port(struct lpfc_vport *vport, struct lpfc_nodelist *ndlp)
spin_unlock_irq(&vport->phba->hbalock); spin_unlock_irq(&vport->phba->hbalock);
lpfc_printf_vlog(vport, KERN_INFO, lpfc_printf_vlog(vport, KERN_INFO,
LOG_NVME_DISC | LOG_NODE, LOG_NVME_DISC | LOG_NODE,
"6022 Binding new rport to " "6022 Bind lport x%px to remoteport x%px "
"lport %p Remoteport %p rport %p WWNN 0x%llx, " "rport x%px WWNN 0x%llx, "
"Rport WWPN 0x%llx DID " "Rport WWPN 0x%llx DID "
"x%06x Role x%x, ndlp %p prev_ndlp %p\n", "x%06x Role x%x, ndlp %p prev_ndlp %p\n",
lport, remote_port, rport, lport, remote_port, rport,
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment