scsi: qla2xxx: Unable to act on RSCN for port online

The device does not come online when the target port is online. There were
multiple RSCNs indicating multiple devices were affected. Driver is in the
process of finishing a fabric scan. A new RSCN (device up) arrived at the
tail end of the last fabric scan. Driver mistakenly thinks the new RSCN is
being taken care of by the previous fabric scan, where this notification is
cleared and not acted on. The laser needs to be blinked again to get the
device to show up.

To prevent driver from accidentally clearing the RSCN notification, each
RSCN is given a generation value.  A fabric scan will scan for that
generation(s).  Any new RSCN arrive after the scan start will have a new
generation value. This will trigger another scan to get latest data. The
RSCN notification flag will be cleared when the scan is associate to that
generation.

Reported-by: kernel test robot <lkp@intel.com>
Closes: https://lore.kernel.org/oe-kbuild-all/202406210538.w875N70K-lkp@intel.com/
Fixes: bb2ca6b3f09a ("scsi: qla2xxx: Relogin during fabric disturbance")
Cc: stable@vger.kernel.org
Signed-off-by: Quinn Tran <qutran@marvell.com>
Signed-off-by: Nilesh Javali <njavali@marvell.com>
Link: https://lore.kernel.org/r/20240710171057.35066-2-njavali@marvell.com
Reviewed-by: Himanshu Madhani <himanshu.madhani@oracle.com>
Signed-off-by: Martin K. Petersen <martin.petersen@oracle.com>
This commit is contained in:
Quinn Tran 2024-07-10 22:40:47 +05:30 committed by Martin K. Petersen
parent 1613e604df
commit c3d98b12ee
4 changed files with 60 additions and 8 deletions

View File

@ -3312,6 +3312,8 @@ struct fab_scan_rp {
struct fab_scan { struct fab_scan {
struct fab_scan_rp *l; struct fab_scan_rp *l;
u32 size; u32 size;
u32 rscn_gen_start;
u32 rscn_gen_end;
u16 scan_retry; u16 scan_retry;
#define MAX_SCAN_RETRIES 5 #define MAX_SCAN_RETRIES 5
enum scan_flags_t scan_flags; enum scan_flags_t scan_flags;
@ -5030,6 +5032,7 @@ typedef struct scsi_qla_host {
/* Counter to detect races between ELS and RSCN events */ /* Counter to detect races between ELS and RSCN events */
atomic_t generation_tick; atomic_t generation_tick;
atomic_t rscn_gen;
/* Time when global fcport update has been scheduled */ /* Time when global fcport update has been scheduled */
int total_fcport_update_gen; int total_fcport_update_gen;
/* List of pending LOGOs, protected by tgt_mutex */ /* List of pending LOGOs, protected by tgt_mutex */

View File

@ -3168,6 +3168,29 @@ static int qla2x00_is_a_vp(scsi_qla_host_t *vha, u64 wwn)
return rc; return rc;
} }
static bool qla_ok_to_clear_rscn(scsi_qla_host_t *vha, fc_port_t *fcport)
{
u32 rscn_gen;
rscn_gen = atomic_read(&vha->rscn_gen);
ql_dbg(ql_dbg_disc + ql_dbg_verbose, vha, 0x2017,
"%s %d %8phC rscn_gen %x start %x end %x current %x\n",
__func__, __LINE__, fcport->port_name, fcport->rscn_gen,
vha->scan.rscn_gen_start, vha->scan.rscn_gen_end, rscn_gen);
if (val_is_in_range(fcport->rscn_gen, vha->scan.rscn_gen_start,
vha->scan.rscn_gen_end))
/* rscn came in before fabric scan */
return true;
if (val_is_in_range(fcport->rscn_gen, vha->scan.rscn_gen_end, rscn_gen))
/* rscn came in after fabric scan */
return false;
/* rare: fcport's scan_needed + rscn_gen must be stale */
return true;
}
void qla24xx_async_gnnft_done(scsi_qla_host_t *vha, srb_t *sp) void qla24xx_async_gnnft_done(scsi_qla_host_t *vha, srb_t *sp)
{ {
fc_port_t *fcport; fc_port_t *fcport;
@ -3281,10 +3304,10 @@ void qla24xx_async_gnnft_done(scsi_qla_host_t *vha, srb_t *sp)
(fcport->scan_needed && (fcport->scan_needed &&
fcport->port_type != FCT_INITIATOR && fcport->port_type != FCT_INITIATOR &&
fcport->port_type != FCT_NVME_INITIATOR)) { fcport->port_type != FCT_NVME_INITIATOR)) {
fcport->scan_needed = 0;
qlt_schedule_sess_for_deletion(fcport); qlt_schedule_sess_for_deletion(fcport);
} }
fcport->d_id.b24 = rp->id.b24; fcport->d_id.b24 = rp->id.b24;
fcport->scan_needed = 0;
break; break;
} }
@ -3325,7 +3348,9 @@ login_logout:
do_delete = true; do_delete = true;
} }
fcport->scan_needed = 0; if (qla_ok_to_clear_rscn(vha, fcport))
fcport->scan_needed = 0;
if (((qla_dual_mode_enabled(vha) || if (((qla_dual_mode_enabled(vha) ||
qla_ini_mode_enabled(vha)) && qla_ini_mode_enabled(vha)) &&
atomic_read(&fcport->state) == FCS_ONLINE) || atomic_read(&fcport->state) == FCS_ONLINE) ||
@ -3355,7 +3380,9 @@ login_logout:
fcport->port_name, fcport->loop_id, fcport->port_name, fcport->loop_id,
fcport->login_retry); fcport->login_retry);
} }
fcport->scan_needed = 0;
if (qla_ok_to_clear_rscn(vha, fcport))
fcport->scan_needed = 0;
qla24xx_fcport_handle_login(vha, fcport); qla24xx_fcport_handle_login(vha, fcport);
} }
} }

View File

@ -1842,10 +1842,18 @@ int qla24xx_post_newsess_work(struct scsi_qla_host *vha, port_id_t *id,
return qla2x00_post_work(vha, e); return qla2x00_post_work(vha, e);
} }
static void qla_rscn_gen_tick(scsi_qla_host_t *vha, u32 *ret_rscn_gen)
{
*ret_rscn_gen = atomic_inc_return(&vha->rscn_gen);
/* memory barrier */
wmb();
}
void qla2x00_handle_rscn(scsi_qla_host_t *vha, struct event_arg *ea) void qla2x00_handle_rscn(scsi_qla_host_t *vha, struct event_arg *ea)
{ {
fc_port_t *fcport; fc_port_t *fcport;
unsigned long flags; unsigned long flags;
u32 rscn_gen;
switch (ea->id.b.rsvd_1) { switch (ea->id.b.rsvd_1) {
case RSCN_PORT_ADDR: case RSCN_PORT_ADDR:
@ -1875,15 +1883,16 @@ void qla2x00_handle_rscn(scsi_qla_host_t *vha, struct event_arg *ea)
* Otherwise we're already in the middle of a relogin * Otherwise we're already in the middle of a relogin
*/ */
fcport->scan_needed = 1; fcport->scan_needed = 1;
fcport->rscn_gen++; qla_rscn_gen_tick(vha, &fcport->rscn_gen);
} }
} else { } else {
fcport->scan_needed = 1; fcport->scan_needed = 1;
fcport->rscn_gen++; qla_rscn_gen_tick(vha, &fcport->rscn_gen);
} }
} }
break; break;
case RSCN_AREA_ADDR: case RSCN_AREA_ADDR:
qla_rscn_gen_tick(vha, &rscn_gen);
list_for_each_entry(fcport, &vha->vp_fcports, list) { list_for_each_entry(fcport, &vha->vp_fcports, list) {
if (fcport->flags & FCF_FCP2_DEVICE && if (fcport->flags & FCF_FCP2_DEVICE &&
atomic_read(&fcport->state) == FCS_ONLINE) atomic_read(&fcport->state) == FCS_ONLINE)
@ -1891,11 +1900,12 @@ void qla2x00_handle_rscn(scsi_qla_host_t *vha, struct event_arg *ea)
if ((ea->id.b24 & 0xffff00) == (fcport->d_id.b24 & 0xffff00)) { if ((ea->id.b24 & 0xffff00) == (fcport->d_id.b24 & 0xffff00)) {
fcport->scan_needed = 1; fcport->scan_needed = 1;
fcport->rscn_gen++; fcport->rscn_gen = rscn_gen;
} }
} }
break; break;
case RSCN_DOM_ADDR: case RSCN_DOM_ADDR:
qla_rscn_gen_tick(vha, &rscn_gen);
list_for_each_entry(fcport, &vha->vp_fcports, list) { list_for_each_entry(fcport, &vha->vp_fcports, list) {
if (fcport->flags & FCF_FCP2_DEVICE && if (fcport->flags & FCF_FCP2_DEVICE &&
atomic_read(&fcport->state) == FCS_ONLINE) atomic_read(&fcport->state) == FCS_ONLINE)
@ -1903,19 +1913,20 @@ void qla2x00_handle_rscn(scsi_qla_host_t *vha, struct event_arg *ea)
if ((ea->id.b24 & 0xff0000) == (fcport->d_id.b24 & 0xff0000)) { if ((ea->id.b24 & 0xff0000) == (fcport->d_id.b24 & 0xff0000)) {
fcport->scan_needed = 1; fcport->scan_needed = 1;
fcport->rscn_gen++; fcport->rscn_gen = rscn_gen;
} }
} }
break; break;
case RSCN_FAB_ADDR: case RSCN_FAB_ADDR:
default: default:
qla_rscn_gen_tick(vha, &rscn_gen);
list_for_each_entry(fcport, &vha->vp_fcports, list) { list_for_each_entry(fcport, &vha->vp_fcports, list) {
if (fcport->flags & FCF_FCP2_DEVICE && if (fcport->flags & FCF_FCP2_DEVICE &&
atomic_read(&fcport->state) == FCS_ONLINE) atomic_read(&fcport->state) == FCS_ONLINE)
continue; continue;
fcport->scan_needed = 1; fcport->scan_needed = 1;
fcport->rscn_gen++; fcport->rscn_gen = rscn_gen;
} }
break; break;
} }
@ -1924,6 +1935,7 @@ void qla2x00_handle_rscn(scsi_qla_host_t *vha, struct event_arg *ea)
if (vha->scan.scan_flags == 0) { if (vha->scan.scan_flags == 0) {
ql_dbg(ql_dbg_disc, vha, 0xffff, "%s: schedule\n", __func__); ql_dbg(ql_dbg_disc, vha, 0xffff, "%s: schedule\n", __func__);
vha->scan.scan_flags |= SF_QUEUED; vha->scan.scan_flags |= SF_QUEUED;
vha->scan.rscn_gen_start = atomic_read(&vha->rscn_gen);
schedule_delayed_work(&vha->scan.scan_work, 5); schedule_delayed_work(&vha->scan.scan_work, 5);
} }
spin_unlock_irqrestore(&vha->work_lock, flags); spin_unlock_irqrestore(&vha->work_lock, flags);
@ -6393,6 +6405,8 @@ qla2x00_configure_fabric(scsi_qla_host_t *vha)
qlt_do_generation_tick(vha, &discovery_gen); qlt_do_generation_tick(vha, &discovery_gen);
if (USE_ASYNC_SCAN(ha)) { if (USE_ASYNC_SCAN(ha)) {
/* start of scan begins here */
vha->scan.rscn_gen_end = atomic_read(&vha->rscn_gen);
rval = qla24xx_async_gpnft(vha, FC4_TYPE_FCP_SCSI, rval = qla24xx_async_gpnft(vha, FC4_TYPE_FCP_SCSI,
NULL); NULL);
if (rval) if (rval)

View File

@ -631,3 +631,11 @@ static inline int qla_mapq_alloc_qp_cpu_map(struct qla_hw_data *ha)
} }
return 0; return 0;
} }
static inline bool val_is_in_range(u32 val, u32 start, u32 end)
{
if (val >= start && val <= end)
return true;
else
return false;
}