diff options
author | James Smart <James.Smart@Emulex.Com> | 2009-10-02 15:16:39 -0400 |
---|---|---|
committer | James Bottomley <James.Bottomley@suse.de> | 2009-12-04 13:01:39 -0500 |
commit | 4d9ab994e214d35107017c342aca42477b137316 (patch) | |
tree | 0ee7dd76ce9938eceeac20e4dab287194dc42c41 /drivers/scsi/lpfc | |
parent | 1796e72291b2b6aafaec5954e666d0b5a95da935 (diff) |
[SCSI] lpfc 8.3.5: fix reset path, ELS ordering and discovery issues
This patch includes the following fixes:
- Fixed panic during HBA reset.
- Fixed FCoE event tag passed in resume_rpi.
- Fix out of order ELS commands
- Fixed discovery issues found during VLAN testing.
- Fix UNREG_VPI failure on extended link pull
- Fixed crash while processing unsolicited FC frames.
- Clear retry count in the delayed ELS handler
- Fixed discovery failure during quick link bounce.
Signed-off-by: James Smart <james.smart@emulex.com>
Signed-off-by: James Bottomley <James.Bottomley@suse.de>
Diffstat (limited to 'drivers/scsi/lpfc')
-rw-r--r-- | drivers/scsi/lpfc/lpfc.h | 3 | ||||
-rw-r--r-- | drivers/scsi/lpfc/lpfc_attr.c | 11 | ||||
-rw-r--r-- | drivers/scsi/lpfc/lpfc_crtn.h | 2 | ||||
-rw-r--r-- | drivers/scsi/lpfc/lpfc_els.c | 1 | ||||
-rw-r--r-- | drivers/scsi/lpfc/lpfc_hbadisc.c | 67 | ||||
-rw-r--r-- | drivers/scsi/lpfc/lpfc_init.c | 51 | ||||
-rw-r--r-- | drivers/scsi/lpfc/lpfc_sli.c | 293 | ||||
-rw-r--r-- | drivers/scsi/lpfc/lpfc_sli.h | 21 | ||||
-rw-r--r-- | drivers/scsi/lpfc/lpfc_sli4.h | 13 |
9 files changed, 242 insertions, 220 deletions
diff --git a/drivers/scsi/lpfc/lpfc.h b/drivers/scsi/lpfc/lpfc.h index aa10f7951634..c618eaf3c0c8 100644 --- a/drivers/scsi/lpfc/lpfc.h +++ b/drivers/scsi/lpfc/lpfc.h | |||
@@ -109,7 +109,7 @@ struct hbq_dmabuf { | |||
109 | struct lpfc_dmabuf dbuf; | 109 | struct lpfc_dmabuf dbuf; |
110 | uint32_t size; | 110 | uint32_t size; |
111 | uint32_t tag; | 111 | uint32_t tag; |
112 | struct lpfc_rcqe rcqe; | 112 | struct lpfc_cq_event cq_event; |
113 | }; | 113 | }; |
114 | 114 | ||
115 | /* Priority bit. Set value to exceed low water mark in lpfc_mem. */ | 115 | /* Priority bit. Set value to exceed low water mark in lpfc_mem. */ |
@@ -551,6 +551,7 @@ struct lpfc_hba { | |||
551 | uint8_t fc_linkspeed; /* Link speed after last READ_LA */ | 551 | uint8_t fc_linkspeed; /* Link speed after last READ_LA */ |
552 | 552 | ||
553 | uint32_t fc_eventTag; /* event tag for link attention */ | 553 | uint32_t fc_eventTag; /* event tag for link attention */ |
554 | uint32_t link_events; | ||
554 | 555 | ||
555 | /* These fields used to be binfo */ | 556 | /* These fields used to be binfo */ |
556 | uint32_t fc_pref_DID; /* preferred D_ID */ | 557 | uint32_t fc_pref_DID; /* preferred D_ID */ |
diff --git a/drivers/scsi/lpfc/lpfc_attr.c b/drivers/scsi/lpfc/lpfc_attr.c index e1a30a16a9fa..07f0172674c9 100644 --- a/drivers/scsi/lpfc/lpfc_attr.c +++ b/drivers/scsi/lpfc/lpfc_attr.c | |||
@@ -3815,7 +3815,11 @@ lpfc_get_stats(struct Scsi_Host *shost) | |||
3815 | hs->invalid_crc_count -= lso->invalid_crc_count; | 3815 | hs->invalid_crc_count -= lso->invalid_crc_count; |
3816 | hs->error_frames -= lso->error_frames; | 3816 | hs->error_frames -= lso->error_frames; |
3817 | 3817 | ||
3818 | if (phba->fc_topology == TOPOLOGY_LOOP) { | 3818 | if (phba->hba_flag & HBA_FCOE_SUPPORT) { |
3819 | hs->lip_count = -1; | ||
3820 | hs->nos_count = (phba->link_events >> 1); | ||
3821 | hs->nos_count -= lso->link_events; | ||
3822 | } else if (phba->fc_topology == TOPOLOGY_LOOP) { | ||
3819 | hs->lip_count = (phba->fc_eventTag >> 1); | 3823 | hs->lip_count = (phba->fc_eventTag >> 1); |
3820 | hs->lip_count -= lso->link_events; | 3824 | hs->lip_count -= lso->link_events; |
3821 | hs->nos_count = -1; | 3825 | hs->nos_count = -1; |
@@ -3906,7 +3910,10 @@ lpfc_reset_stats(struct Scsi_Host *shost) | |||
3906 | lso->invalid_tx_word_count = pmb->un.varRdLnk.invalidXmitWord; | 3910 | lso->invalid_tx_word_count = pmb->un.varRdLnk.invalidXmitWord; |
3907 | lso->invalid_crc_count = pmb->un.varRdLnk.crcCnt; | 3911 | lso->invalid_crc_count = pmb->un.varRdLnk.crcCnt; |
3908 | lso->error_frames = pmb->un.varRdLnk.crcCnt; | 3912 | lso->error_frames = pmb->un.varRdLnk.crcCnt; |
3909 | lso->link_events = (phba->fc_eventTag >> 1); | 3913 | if (phba->hba_flag & HBA_FCOE_SUPPORT) |
3914 | lso->link_events = (phba->link_events >> 1); | ||
3915 | else | ||
3916 | lso->link_events = (phba->fc_eventTag >> 1); | ||
3910 | 3917 | ||
3911 | psli->stats_start = get_seconds(); | 3918 | psli->stats_start = get_seconds(); |
3912 | 3919 | ||
diff --git a/drivers/scsi/lpfc/lpfc_crtn.h b/drivers/scsi/lpfc/lpfc_crtn.h index 0830f37409a3..4438f8665a4a 100644 --- a/drivers/scsi/lpfc/lpfc_crtn.h +++ b/drivers/scsi/lpfc/lpfc_crtn.h | |||
@@ -235,7 +235,7 @@ void lpfc_sli_mbox_sys_shutdown(struct lpfc_hba *); | |||
235 | int lpfc_sli_check_eratt(struct lpfc_hba *); | 235 | int lpfc_sli_check_eratt(struct lpfc_hba *); |
236 | void lpfc_sli_handle_slow_ring_event(struct lpfc_hba *, | 236 | void lpfc_sli_handle_slow_ring_event(struct lpfc_hba *, |
237 | struct lpfc_sli_ring *, uint32_t); | 237 | struct lpfc_sli_ring *, uint32_t); |
238 | int lpfc_sli4_handle_received_buffer(struct lpfc_hba *); | 238 | void lpfc_sli4_handle_received_buffer(struct lpfc_hba *, struct hbq_dmabuf *); |
239 | void lpfc_sli_def_mbox_cmpl(struct lpfc_hba *, LPFC_MBOXQ_t *); | 239 | void lpfc_sli_def_mbox_cmpl(struct lpfc_hba *, LPFC_MBOXQ_t *); |
240 | int lpfc_sli_issue_iocb(struct lpfc_hba *, uint32_t, | 240 | int lpfc_sli_issue_iocb(struct lpfc_hba *, uint32_t, |
241 | struct lpfc_iocbq *, uint32_t); | 241 | struct lpfc_iocbq *, uint32_t); |
diff --git a/drivers/scsi/lpfc/lpfc_els.c b/drivers/scsi/lpfc/lpfc_els.c index 45337cd23feb..4ea863f50650 100644 --- a/drivers/scsi/lpfc/lpfc_els.c +++ b/drivers/scsi/lpfc/lpfc_els.c | |||
@@ -2452,6 +2452,7 @@ lpfc_els_retry_delay_handler(struct lpfc_nodelist *ndlp) | |||
2452 | */ | 2452 | */ |
2453 | del_timer_sync(&ndlp->nlp_delayfunc); | 2453 | del_timer_sync(&ndlp->nlp_delayfunc); |
2454 | retry = ndlp->nlp_retry; | 2454 | retry = ndlp->nlp_retry; |
2455 | ndlp->nlp_retry = 0; | ||
2455 | 2456 | ||
2456 | switch (cmd) { | 2457 | switch (cmd) { |
2457 | case ELS_CMD_FLOGI: | 2458 | case ELS_CMD_FLOGI: |
diff --git a/drivers/scsi/lpfc/lpfc_hbadisc.c b/drivers/scsi/lpfc/lpfc_hbadisc.c index e6a47e25b218..5073c127bfe1 100644 --- a/drivers/scsi/lpfc/lpfc_hbadisc.c +++ b/drivers/scsi/lpfc/lpfc_hbadisc.c | |||
@@ -525,8 +525,6 @@ lpfc_work_done(struct lpfc_hba *phba) | |||
525 | spin_unlock_irq(&phba->hbalock); | 525 | spin_unlock_irq(&phba->hbalock); |
526 | lpfc_sli_hbqbuf_add_hbqs(phba, LPFC_ELS_HBQ); | 526 | lpfc_sli_hbqbuf_add_hbqs(phba, LPFC_ELS_HBQ); |
527 | } | 527 | } |
528 | if (phba->hba_flag & HBA_RECEIVE_BUFFER) | ||
529 | lpfc_sli4_handle_received_buffer(phba); | ||
530 | } | 528 | } |
531 | 529 | ||
532 | vports = lpfc_create_vport_work_array(phba); | 530 | vports = lpfc_create_vport_work_array(phba); |
@@ -568,8 +566,9 @@ lpfc_work_done(struct lpfc_hba *phba) | |||
568 | pring = &phba->sli.ring[LPFC_ELS_RING]; | 566 | pring = &phba->sli.ring[LPFC_ELS_RING]; |
569 | status = (ha_copy & (HA_RXMASK << (4*LPFC_ELS_RING))); | 567 | status = (ha_copy & (HA_RXMASK << (4*LPFC_ELS_RING))); |
570 | status >>= (4*LPFC_ELS_RING); | 568 | status >>= (4*LPFC_ELS_RING); |
571 | if ((status & HA_RXMASK) | 569 | if ((status & HA_RXMASK) || |
572 | || (pring->flag & LPFC_DEFERRED_RING_EVENT)) { | 570 | (pring->flag & LPFC_DEFERRED_RING_EVENT) || |
571 | (phba->hba_flag & HBA_RECEIVE_BUFFER)) { | ||
573 | if (pring->flag & LPFC_STOP_IOCB_EVENT) { | 572 | if (pring->flag & LPFC_STOP_IOCB_EVENT) { |
574 | pring->flag |= LPFC_DEFERRED_RING_EVENT; | 573 | pring->flag |= LPFC_DEFERRED_RING_EVENT; |
575 | /* Set the lpfc data pending flag */ | 574 | /* Set the lpfc data pending flag */ |
@@ -688,7 +687,8 @@ lpfc_cleanup_rpis(struct lpfc_vport *vport, int remove) | |||
688 | lpfc_unreg_rpi(vport, ndlp); | 687 | lpfc_unreg_rpi(vport, ndlp); |
689 | 688 | ||
690 | /* Leave Fabric nodes alone on link down */ | 689 | /* Leave Fabric nodes alone on link down */ |
691 | if (!remove && ndlp->nlp_type & NLP_FABRIC) | 690 | if ((phba->sli_rev < LPFC_SLI_REV4) && |
691 | (!remove && ndlp->nlp_type & NLP_FABRIC)) | ||
692 | continue; | 692 | continue; |
693 | rc = lpfc_disc_state_machine(vport, ndlp, NULL, | 693 | rc = lpfc_disc_state_machine(vport, ndlp, NULL, |
694 | remove | 694 | remove |
@@ -1015,10 +1015,10 @@ lpfc_mbx_cmpl_reg_fcfi(struct lpfc_hba *phba, LPFC_MBOXQ_t *mboxq) | |||
1015 | mempool_free(mboxq, phba->mbox_mem_pool); | 1015 | mempool_free(mboxq, phba->mbox_mem_pool); |
1016 | return; | 1016 | return; |
1017 | } | 1017 | } |
1018 | phba->fcf.fcf_flag |= (FCF_DISCOVERED | FCF_IN_USE); | ||
1019 | phba->hba_flag &= ~FCF_DISC_INPROGRESS; | ||
1018 | if (vport->port_state != LPFC_FLOGI) { | 1020 | if (vport->port_state != LPFC_FLOGI) { |
1019 | spin_lock_irqsave(&phba->hbalock, flags); | 1021 | spin_lock_irqsave(&phba->hbalock, flags); |
1020 | phba->fcf.fcf_flag |= (FCF_DISCOVERED | FCF_IN_USE); | ||
1021 | phba->hba_flag &= ~FCF_DISC_INPROGRESS; | ||
1022 | spin_unlock_irqrestore(&phba->hbalock, flags); | 1022 | spin_unlock_irqrestore(&phba->hbalock, flags); |
1023 | lpfc_initial_flogi(vport); | 1023 | lpfc_initial_flogi(vport); |
1024 | } | 1024 | } |
@@ -1199,6 +1199,7 @@ lpfc_register_fcf(struct lpfc_hba *phba) | |||
1199 | 1199 | ||
1200 | /* If the FCF is not availabe do nothing. */ | 1200 | /* If the FCF is not availabe do nothing. */ |
1201 | if (!(phba->fcf.fcf_flag & FCF_AVAILABLE)) { | 1201 | if (!(phba->fcf.fcf_flag & FCF_AVAILABLE)) { |
1202 | phba->hba_flag &= ~FCF_DISC_INPROGRESS; | ||
1202 | spin_unlock_irqrestore(&phba->hbalock, flags); | 1203 | spin_unlock_irqrestore(&phba->hbalock, flags); |
1203 | return; | 1204 | return; |
1204 | } | 1205 | } |
@@ -1216,15 +1217,23 @@ lpfc_register_fcf(struct lpfc_hba *phba) | |||
1216 | 1217 | ||
1217 | fcf_mbxq = mempool_alloc(phba->mbox_mem_pool, | 1218 | fcf_mbxq = mempool_alloc(phba->mbox_mem_pool, |
1218 | GFP_KERNEL); | 1219 | GFP_KERNEL); |
1219 | if (!fcf_mbxq) | 1220 | if (!fcf_mbxq) { |
1221 | spin_lock_irqsave(&phba->hbalock, flags); | ||
1222 | phba->hba_flag &= ~FCF_DISC_INPROGRESS; | ||
1223 | spin_unlock_irqrestore(&phba->hbalock, flags); | ||
1220 | return; | 1224 | return; |
1225 | } | ||
1221 | 1226 | ||
1222 | lpfc_reg_fcfi(phba, fcf_mbxq); | 1227 | lpfc_reg_fcfi(phba, fcf_mbxq); |
1223 | fcf_mbxq->vport = phba->pport; | 1228 | fcf_mbxq->vport = phba->pport; |
1224 | fcf_mbxq->mbox_cmpl = lpfc_mbx_cmpl_reg_fcfi; | 1229 | fcf_mbxq->mbox_cmpl = lpfc_mbx_cmpl_reg_fcfi; |
1225 | rc = lpfc_sli_issue_mbox(phba, fcf_mbxq, MBX_NOWAIT); | 1230 | rc = lpfc_sli_issue_mbox(phba, fcf_mbxq, MBX_NOWAIT); |
1226 | if (rc == MBX_NOT_FINISHED) | 1231 | if (rc == MBX_NOT_FINISHED) { |
1232 | spin_lock_irqsave(&phba->hbalock, flags); | ||
1233 | phba->hba_flag &= ~FCF_DISC_INPROGRESS; | ||
1234 | spin_unlock_irqrestore(&phba->hbalock, flags); | ||
1227 | mempool_free(fcf_mbxq, phba->mbox_mem_pool); | 1235 | mempool_free(fcf_mbxq, phba->mbox_mem_pool); |
1236 | } | ||
1228 | 1237 | ||
1229 | return; | 1238 | return; |
1230 | } | 1239 | } |
@@ -1253,6 +1262,20 @@ lpfc_match_fcf_conn_list(struct lpfc_hba *phba, | |||
1253 | uint16_t *vlan_id) | 1262 | uint16_t *vlan_id) |
1254 | { | 1263 | { |
1255 | struct lpfc_fcf_conn_entry *conn_entry; | 1264 | struct lpfc_fcf_conn_entry *conn_entry; |
1265 | int i, j, fcf_vlan_id = 0; | ||
1266 | |||
1267 | /* Find the lowest VLAN id in the FCF record */ | ||
1268 | for (i = 0; i < 512; i++) { | ||
1269 | if (new_fcf_record->vlan_bitmap[i]) { | ||
1270 | fcf_vlan_id = i * 8; | ||
1271 | j = 0; | ||
1272 | while (!((new_fcf_record->vlan_bitmap[i] >> j) & 1)) { | ||
1273 | j++; | ||
1274 | fcf_vlan_id++; | ||
1275 | } | ||
1276 | break; | ||
1277 | } | ||
1278 | } | ||
1256 | 1279 | ||
1257 | /* If FCF not available return 0 */ | 1280 | /* If FCF not available return 0 */ |
1258 | if (!bf_get(lpfc_fcf_record_fcf_avail, new_fcf_record) || | 1281 | if (!bf_get(lpfc_fcf_record_fcf_avail, new_fcf_record) || |
@@ -1286,7 +1309,11 @@ lpfc_match_fcf_conn_list(struct lpfc_hba *phba, | |||
1286 | if (*addr_mode & LPFC_FCF_FPMA) | 1309 | if (*addr_mode & LPFC_FCF_FPMA) |
1287 | *addr_mode = LPFC_FCF_FPMA; | 1310 | *addr_mode = LPFC_FCF_FPMA; |
1288 | 1311 | ||
1289 | *vlan_id = 0xFFFF; | 1312 | /* If FCF record report a vlan id use that vlan id */ |
1313 | if (fcf_vlan_id) | ||
1314 | *vlan_id = fcf_vlan_id; | ||
1315 | else | ||
1316 | *vlan_id = 0xFFFF; | ||
1290 | return 1; | 1317 | return 1; |
1291 | } | 1318 | } |
1292 | 1319 | ||
@@ -1384,8 +1411,15 @@ lpfc_match_fcf_conn_list(struct lpfc_hba *phba, | |||
1384 | (*addr_mode & LPFC_FCF_FPMA)) | 1411 | (*addr_mode & LPFC_FCF_FPMA)) |
1385 | *addr_mode = LPFC_FCF_FPMA; | 1412 | *addr_mode = LPFC_FCF_FPMA; |
1386 | 1413 | ||
1414 | /* If matching connect list has a vlan id, use it */ | ||
1387 | if (conn_entry->conn_rec.flags & FCFCNCT_VLAN_VALID) | 1415 | if (conn_entry->conn_rec.flags & FCFCNCT_VLAN_VALID) |
1388 | *vlan_id = conn_entry->conn_rec.vlan_tag; | 1416 | *vlan_id = conn_entry->conn_rec.vlan_tag; |
1417 | /* | ||
1418 | * If no vlan id is specified in connect list, use the vlan id | ||
1419 | * in the FCF record | ||
1420 | */ | ||
1421 | else if (fcf_vlan_id) | ||
1422 | *vlan_id = fcf_vlan_id; | ||
1389 | else | 1423 | else |
1390 | *vlan_id = 0xFFFF; | 1424 | *vlan_id = 0xFFFF; |
1391 | 1425 | ||
@@ -1423,6 +1457,12 @@ lpfc_check_pending_fcoe_event(struct lpfc_hba *phba, uint8_t unreg_fcf) | |||
1423 | 1457 | ||
1424 | if (phba->link_state >= LPFC_LINK_UP) | 1458 | if (phba->link_state >= LPFC_LINK_UP) |
1425 | lpfc_sli4_read_fcf_record(phba, LPFC_FCOE_FCF_GET_FIRST); | 1459 | lpfc_sli4_read_fcf_record(phba, LPFC_FCOE_FCF_GET_FIRST); |
1460 | else | ||
1461 | /* | ||
1462 | * Do not continue FCF discovery and clear FCF_DISC_INPROGRESS | ||
1463 | * flag | ||
1464 | */ | ||
1465 | phba->hba_flag &= ~FCF_DISC_INPROGRESS; | ||
1426 | 1466 | ||
1427 | if (unreg_fcf) { | 1467 | if (unreg_fcf) { |
1428 | spin_lock_irq(&phba->hbalock); | 1468 | spin_lock_irq(&phba->hbalock); |
@@ -2085,6 +2125,7 @@ lpfc_mbx_cmpl_read_la(struct lpfc_hba *phba, LPFC_MBOXQ_t *pmb) | |||
2085 | else | 2125 | else |
2086 | phba->sli.sli_flag &= ~LPFC_MENLO_MAINT; | 2126 | phba->sli.sli_flag &= ~LPFC_MENLO_MAINT; |
2087 | 2127 | ||
2128 | phba->link_events++; | ||
2088 | if (la->attType == AT_LINK_UP && (!la->mm)) { | 2129 | if (la->attType == AT_LINK_UP && (!la->mm)) { |
2089 | phba->fc_stat.LinkUp++; | 2130 | phba->fc_stat.LinkUp++; |
2090 | if (phba->link_flag & LS_LOOPBACK_MODE) { | 2131 | if (phba->link_flag & LS_LOOPBACK_MODE) { |
@@ -4409,6 +4450,8 @@ lpfc_unregister_unused_fcf(struct lpfc_hba *phba) | |||
4409 | if (lpfc_fcf_inuse(phba)) | 4450 | if (lpfc_fcf_inuse(phba)) |
4410 | return; | 4451 | return; |
4411 | 4452 | ||
4453 | /* At this point, all discovery is aborted */ | ||
4454 | phba->pport->port_state = LPFC_VPORT_UNKNOWN; | ||
4412 | 4455 | ||
4413 | /* Unregister VPIs */ | 4456 | /* Unregister VPIs */ |
4414 | vports = lpfc_create_vport_work_array(phba); | 4457 | vports = lpfc_create_vport_work_array(phba); |
@@ -4512,8 +4555,10 @@ lpfc_read_fcf_conn_tbl(struct lpfc_hba *phba, | |||
4512 | 4555 | ||
4513 | /* Free the current connect table */ | 4556 | /* Free the current connect table */ |
4514 | list_for_each_entry_safe(conn_entry, next_conn_entry, | 4557 | list_for_each_entry_safe(conn_entry, next_conn_entry, |
4515 | &phba->fcf_conn_rec_list, list) | 4558 | &phba->fcf_conn_rec_list, list) { |
4559 | list_del_init(&conn_entry->list); | ||
4516 | kfree(conn_entry); | 4560 | kfree(conn_entry); |
4561 | } | ||
4517 | 4562 | ||
4518 | conn_hdr = (struct lpfc_fcf_conn_hdr *) buff; | 4563 | conn_hdr = (struct lpfc_fcf_conn_hdr *) buff; |
4519 | record_count = conn_hdr->length * sizeof(uint32_t)/ | 4564 | record_count = conn_hdr->length * sizeof(uint32_t)/ |
diff --git a/drivers/scsi/lpfc/lpfc_init.c b/drivers/scsi/lpfc/lpfc_init.c index f913f1e93635..d654c0e3db4d 100644 --- a/drivers/scsi/lpfc/lpfc_init.c +++ b/drivers/scsi/lpfc/lpfc_init.c | |||
@@ -2919,6 +2919,7 @@ lpfc_sli4_async_fcoe_evt(struct lpfc_hba *phba, | |||
2919 | uint8_t event_type = bf_get(lpfc_acqe_fcoe_event_type, acqe_fcoe); | 2919 | uint8_t event_type = bf_get(lpfc_acqe_fcoe_event_type, acqe_fcoe); |
2920 | int rc; | 2920 | int rc; |
2921 | 2921 | ||
2922 | phba->fc_eventTag = acqe_fcoe->event_tag; | ||
2922 | phba->fcoe_eventtag = acqe_fcoe->event_tag; | 2923 | phba->fcoe_eventtag = acqe_fcoe->event_tag; |
2923 | switch (event_type) { | 2924 | switch (event_type) { |
2924 | case LPFC_FCOE_EVENT_TYPE_NEW_FCF: | 2925 | case LPFC_FCOE_EVENT_TYPE_NEW_FCF: |
@@ -2990,6 +2991,7 @@ static void | |||
2990 | lpfc_sli4_async_dcbx_evt(struct lpfc_hba *phba, | 2991 | lpfc_sli4_async_dcbx_evt(struct lpfc_hba *phba, |
2991 | struct lpfc_acqe_dcbx *acqe_dcbx) | 2992 | struct lpfc_acqe_dcbx *acqe_dcbx) |
2992 | { | 2993 | { |
2994 | phba->fc_eventTag = acqe_dcbx->event_tag; | ||
2993 | lpfc_printf_log(phba, KERN_ERR, LOG_SLI, | 2995 | lpfc_printf_log(phba, KERN_ERR, LOG_SLI, |
2994 | "0290 The SLI4 DCBX asynchronous event is not " | 2996 | "0290 The SLI4 DCBX asynchronous event is not " |
2995 | "handled yet\n"); | 2997 | "handled yet\n"); |
@@ -3594,8 +3596,10 @@ lpfc_sli4_driver_resource_unset(struct lpfc_hba *phba) | |||
3594 | 3596 | ||
3595 | /* Free the current connect table */ | 3597 | /* Free the current connect table */ |
3596 | list_for_each_entry_safe(conn_entry, next_conn_entry, | 3598 | list_for_each_entry_safe(conn_entry, next_conn_entry, |
3597 | &phba->fcf_conn_rec_list, list) | 3599 | &phba->fcf_conn_rec_list, list) { |
3600 | list_del_init(&conn_entry->list); | ||
3598 | kfree(conn_entry); | 3601 | kfree(conn_entry); |
3602 | } | ||
3599 | 3603 | ||
3600 | return; | 3604 | return; |
3601 | } | 3605 | } |
@@ -5058,15 +5062,6 @@ lpfc_sli4_queue_create(struct lpfc_hba *phba) | |||
5058 | } | 5062 | } |
5059 | phba->sli4_hba.els_cq = qdesc; | 5063 | phba->sli4_hba.els_cq = qdesc; |
5060 | 5064 | ||
5061 | /* Create slow-path Unsolicited Receive Complete Queue */ | ||
5062 | qdesc = lpfc_sli4_queue_alloc(phba, phba->sli4_hba.cq_esize, | ||
5063 | phba->sli4_hba.cq_ecount); | ||
5064 | if (!qdesc) { | ||
5065 | lpfc_printf_log(phba, KERN_ERR, LOG_INIT, | ||
5066 | "0502 Failed allocate slow-path USOL RX CQ\n"); | ||
5067 | goto out_free_els_cq; | ||
5068 | } | ||
5069 | phba->sli4_hba.rxq_cq = qdesc; | ||
5070 | 5065 | ||
5071 | /* Create fast-path FCP Completion Queue(s), one-to-one with EQs */ | 5066 | /* Create fast-path FCP Completion Queue(s), one-to-one with EQs */ |
5072 | phba->sli4_hba.fcp_cq = kzalloc((sizeof(struct lpfc_queue *) * | 5067 | phba->sli4_hba.fcp_cq = kzalloc((sizeof(struct lpfc_queue *) * |
@@ -5075,7 +5070,7 @@ lpfc_sli4_queue_create(struct lpfc_hba *phba) | |||
5075 | lpfc_printf_log(phba, KERN_ERR, LOG_INIT, | 5070 | lpfc_printf_log(phba, KERN_ERR, LOG_INIT, |
5076 | "2577 Failed allocate memory for fast-path " | 5071 | "2577 Failed allocate memory for fast-path " |
5077 | "CQ record array\n"); | 5072 | "CQ record array\n"); |
5078 | goto out_free_rxq_cq; | 5073 | goto out_free_els_cq; |
5079 | } | 5074 | } |
5080 | for (fcp_cqidx = 0; fcp_cqidx < phba->cfg_fcp_eq_count; fcp_cqidx++) { | 5075 | for (fcp_cqidx = 0; fcp_cqidx < phba->cfg_fcp_eq_count; fcp_cqidx++) { |
5081 | qdesc = lpfc_sli4_queue_alloc(phba, phba->sli4_hba.cq_esize, | 5076 | qdesc = lpfc_sli4_queue_alloc(phba, phba->sli4_hba.cq_esize, |
@@ -5188,9 +5183,6 @@ out_free_fcp_cq: | |||
5188 | phba->sli4_hba.fcp_cq[fcp_cqidx] = NULL; | 5183 | phba->sli4_hba.fcp_cq[fcp_cqidx] = NULL; |
5189 | } | 5184 | } |
5190 | kfree(phba->sli4_hba.fcp_cq); | 5185 | kfree(phba->sli4_hba.fcp_cq); |
5191 | out_free_rxq_cq: | ||
5192 | lpfc_sli4_queue_free(phba->sli4_hba.rxq_cq); | ||
5193 | phba->sli4_hba.rxq_cq = NULL; | ||
5194 | out_free_els_cq: | 5186 | out_free_els_cq: |
5195 | lpfc_sli4_queue_free(phba->sli4_hba.els_cq); | 5187 | lpfc_sli4_queue_free(phba->sli4_hba.els_cq); |
5196 | phba->sli4_hba.els_cq = NULL; | 5188 | phba->sli4_hba.els_cq = NULL; |
@@ -5247,10 +5239,6 @@ lpfc_sli4_queue_destroy(struct lpfc_hba *phba) | |||
5247 | lpfc_sli4_queue_free(phba->sli4_hba.dat_rq); | 5239 | lpfc_sli4_queue_free(phba->sli4_hba.dat_rq); |
5248 | phba->sli4_hba.dat_rq = NULL; | 5240 | phba->sli4_hba.dat_rq = NULL; |
5249 | 5241 | ||
5250 | /* Release unsolicited receive complete queue */ | ||
5251 | lpfc_sli4_queue_free(phba->sli4_hba.rxq_cq); | ||
5252 | phba->sli4_hba.rxq_cq = NULL; | ||
5253 | |||
5254 | /* Release ELS complete queue */ | 5242 | /* Release ELS complete queue */ |
5255 | lpfc_sli4_queue_free(phba->sli4_hba.els_cq); | 5243 | lpfc_sli4_queue_free(phba->sli4_hba.els_cq); |
5256 | phba->sli4_hba.els_cq = NULL; | 5244 | phba->sli4_hba.els_cq = NULL; |
@@ -5383,25 +5371,6 @@ lpfc_sli4_queue_setup(struct lpfc_hba *phba) | |||
5383 | phba->sli4_hba.els_cq->queue_id, | 5371 | phba->sli4_hba.els_cq->queue_id, |
5384 | phba->sli4_hba.sp_eq->queue_id); | 5372 | phba->sli4_hba.sp_eq->queue_id); |
5385 | 5373 | ||
5386 | /* Set up slow-path Unsolicited Receive Complete Queue */ | ||
5387 | if (!phba->sli4_hba.rxq_cq) { | ||
5388 | lpfc_printf_log(phba, KERN_ERR, LOG_INIT, | ||
5389 | "0532 USOL RX CQ not allocated\n"); | ||
5390 | goto out_destroy_els_cq; | ||
5391 | } | ||
5392 | rc = lpfc_cq_create(phba, phba->sli4_hba.rxq_cq, phba->sli4_hba.sp_eq, | ||
5393 | LPFC_RCQ, LPFC_USOL); | ||
5394 | if (rc) { | ||
5395 | lpfc_printf_log(phba, KERN_ERR, LOG_INIT, | ||
5396 | "0533 Failed setup of slow-path USOL RX CQ: " | ||
5397 | "rc = 0x%x\n", rc); | ||
5398 | goto out_destroy_els_cq; | ||
5399 | } | ||
5400 | lpfc_printf_log(phba, KERN_INFO, LOG_INIT, | ||
5401 | "2587 USL CQ setup: cq-id=%d, parent eq-id=%d\n", | ||
5402 | phba->sli4_hba.rxq_cq->queue_id, | ||
5403 | phba->sli4_hba.sp_eq->queue_id); | ||
5404 | |||
5405 | /* Set up fast-path FCP Response Complete Queue */ | 5374 | /* Set up fast-path FCP Response Complete Queue */ |
5406 | for (fcp_cqidx = 0; fcp_cqidx < phba->cfg_fcp_eq_count; fcp_cqidx++) { | 5375 | for (fcp_cqidx = 0; fcp_cqidx < phba->cfg_fcp_eq_count; fcp_cqidx++) { |
5407 | if (!phba->sli4_hba.fcp_cq[fcp_cqidx]) { | 5376 | if (!phba->sli4_hba.fcp_cq[fcp_cqidx]) { |
@@ -5507,7 +5476,7 @@ lpfc_sli4_queue_setup(struct lpfc_hba *phba) | |||
5507 | goto out_destroy_fcp_wq; | 5476 | goto out_destroy_fcp_wq; |
5508 | } | 5477 | } |
5509 | rc = lpfc_rq_create(phba, phba->sli4_hba.hdr_rq, phba->sli4_hba.dat_rq, | 5478 | rc = lpfc_rq_create(phba, phba->sli4_hba.hdr_rq, phba->sli4_hba.dat_rq, |
5510 | phba->sli4_hba.rxq_cq, LPFC_USOL); | 5479 | phba->sli4_hba.els_cq, LPFC_USOL); |
5511 | if (rc) { | 5480 | if (rc) { |
5512 | lpfc_printf_log(phba, KERN_ERR, LOG_INIT, | 5481 | lpfc_printf_log(phba, KERN_ERR, LOG_INIT, |
5513 | "0541 Failed setup of Receive Queue: " | 5482 | "0541 Failed setup of Receive Queue: " |
@@ -5519,7 +5488,7 @@ lpfc_sli4_queue_setup(struct lpfc_hba *phba) | |||
5519 | "parent cq-id=%d\n", | 5488 | "parent cq-id=%d\n", |
5520 | phba->sli4_hba.hdr_rq->queue_id, | 5489 | phba->sli4_hba.hdr_rq->queue_id, |
5521 | phba->sli4_hba.dat_rq->queue_id, | 5490 | phba->sli4_hba.dat_rq->queue_id, |
5522 | phba->sli4_hba.rxq_cq->queue_id); | 5491 | phba->sli4_hba.els_cq->queue_id); |
5523 | return 0; | 5492 | return 0; |
5524 | 5493 | ||
5525 | out_destroy_fcp_wq: | 5494 | out_destroy_fcp_wq: |
@@ -5531,8 +5500,6 @@ out_destroy_mbx_wq: | |||
5531 | out_destroy_fcp_cq: | 5500 | out_destroy_fcp_cq: |
5532 | for (--fcp_cqidx; fcp_cqidx >= 0; fcp_cqidx--) | 5501 | for (--fcp_cqidx; fcp_cqidx >= 0; fcp_cqidx--) |
5533 | lpfc_cq_destroy(phba, phba->sli4_hba.fcp_cq[fcp_cqidx]); | 5502 | lpfc_cq_destroy(phba, phba->sli4_hba.fcp_cq[fcp_cqidx]); |
5534 | lpfc_cq_destroy(phba, phba->sli4_hba.rxq_cq); | ||
5535 | out_destroy_els_cq: | ||
5536 | lpfc_cq_destroy(phba, phba->sli4_hba.els_cq); | 5503 | lpfc_cq_destroy(phba, phba->sli4_hba.els_cq); |
5537 | out_destroy_mbx_cq: | 5504 | out_destroy_mbx_cq: |
5538 | lpfc_cq_destroy(phba, phba->sli4_hba.mbx_cq); | 5505 | lpfc_cq_destroy(phba, phba->sli4_hba.mbx_cq); |
@@ -5574,8 +5541,6 @@ lpfc_sli4_queue_unset(struct lpfc_hba *phba) | |||
5574 | lpfc_cq_destroy(phba, phba->sli4_hba.mbx_cq); | 5541 | lpfc_cq_destroy(phba, phba->sli4_hba.mbx_cq); |
5575 | /* Unset ELS complete queue */ | 5542 | /* Unset ELS complete queue */ |
5576 | lpfc_cq_destroy(phba, phba->sli4_hba.els_cq); | 5543 | lpfc_cq_destroy(phba, phba->sli4_hba.els_cq); |
5577 | /* Unset unsolicited receive complete queue */ | ||
5578 | lpfc_cq_destroy(phba, phba->sli4_hba.rxq_cq); | ||
5579 | /* Unset FCP response complete queue */ | 5544 | /* Unset FCP response complete queue */ |
5580 | for (fcp_qidx = 0; fcp_qidx < phba->cfg_fcp_eq_count; fcp_qidx++) | 5545 | for (fcp_qidx = 0; fcp_qidx < phba->cfg_fcp_eq_count; fcp_qidx++) |
5581 | lpfc_cq_destroy(phba, phba->sli4_hba.fcp_cq[fcp_qidx]); | 5546 | lpfc_cq_destroy(phba, phba->sli4_hba.fcp_cq[fcp_qidx]); |
diff --git a/drivers/scsi/lpfc/lpfc_sli.c b/drivers/scsi/lpfc/lpfc_sli.c index 43cbe336f1f8..8d884d8e18be 100644 --- a/drivers/scsi/lpfc/lpfc_sli.c +++ b/drivers/scsi/lpfc/lpfc_sli.c | |||
@@ -3018,16 +3018,31 @@ lpfc_sli_handle_slow_ring_event_s4(struct lpfc_hba *phba, | |||
3018 | struct lpfc_sli_ring *pring, uint32_t mask) | 3018 | struct lpfc_sli_ring *pring, uint32_t mask) |
3019 | { | 3019 | { |
3020 | struct lpfc_iocbq *irspiocbq; | 3020 | struct lpfc_iocbq *irspiocbq; |
3021 | struct hbq_dmabuf *dmabuf; | ||
3022 | struct lpfc_cq_event *cq_event; | ||
3021 | unsigned long iflag; | 3023 | unsigned long iflag; |
3022 | 3024 | ||
3023 | while (!list_empty(&phba->sli4_hba.sp_rspiocb_work_queue)) { | 3025 | while (!list_empty(&phba->sli4_hba.sp_rspiocb_work_queue)) { |
3024 | /* Get the response iocb from the head of work queue */ | 3026 | /* Get the response iocb from the head of work queue */ |
3025 | spin_lock_irqsave(&phba->hbalock, iflag); | 3027 | spin_lock_irqsave(&phba->hbalock, iflag); |
3026 | list_remove_head(&phba->sli4_hba.sp_rspiocb_work_queue, | 3028 | list_remove_head(&phba->sli4_hba.sp_rspiocb_work_queue, |
3027 | irspiocbq, struct lpfc_iocbq, list); | 3029 | cq_event, struct lpfc_cq_event, list); |
3028 | spin_unlock_irqrestore(&phba->hbalock, iflag); | 3030 | spin_unlock_irqrestore(&phba->hbalock, iflag); |
3029 | /* Process the response iocb */ | 3031 | |
3030 | lpfc_sli_sp_handle_rspiocb(phba, pring, irspiocbq); | 3032 | switch (bf_get(lpfc_wcqe_c_code, &cq_event->cqe.wcqe_cmpl)) { |
3033 | case CQE_CODE_COMPL_WQE: | ||
3034 | irspiocbq = container_of(cq_event, struct lpfc_iocbq, | ||
3035 | cq_event); | ||
3036 | lpfc_sli_sp_handle_rspiocb(phba, pring, irspiocbq); | ||
3037 | break; | ||
3038 | case CQE_CODE_RECEIVE: | ||
3039 | dmabuf = container_of(cq_event, struct hbq_dmabuf, | ||
3040 | cq_event); | ||
3041 | lpfc_sli4_handle_received_buffer(phba, dmabuf); | ||
3042 | break; | ||
3043 | default: | ||
3044 | break; | ||
3045 | } | ||
3031 | } | 3046 | } |
3032 | } | 3047 | } |
3033 | 3048 | ||
@@ -3416,6 +3431,7 @@ lpfc_sli_brdreset(struct lpfc_hba *phba) | |||
3416 | 3431 | ||
3417 | /* perform board reset */ | 3432 | /* perform board reset */ |
3418 | phba->fc_eventTag = 0; | 3433 | phba->fc_eventTag = 0; |
3434 | phba->link_events = 0; | ||
3419 | phba->pport->fc_myDID = 0; | 3435 | phba->pport->fc_myDID = 0; |
3420 | phba->pport->fc_prevDID = 0; | 3436 | phba->pport->fc_prevDID = 0; |
3421 | 3437 | ||
@@ -3476,6 +3492,7 @@ lpfc_sli4_brdreset(struct lpfc_hba *phba) | |||
3476 | 3492 | ||
3477 | /* perform board reset */ | 3493 | /* perform board reset */ |
3478 | phba->fc_eventTag = 0; | 3494 | phba->fc_eventTag = 0; |
3495 | phba->link_events = 0; | ||
3479 | phba->pport->fc_myDID = 0; | 3496 | phba->pport->fc_myDID = 0; |
3480 | phba->pport->fc_prevDID = 0; | 3497 | phba->pport->fc_prevDID = 0; |
3481 | 3498 | ||
@@ -3495,7 +3512,6 @@ lpfc_sli4_brdreset(struct lpfc_hba *phba) | |||
3495 | list_del_init(&phba->sli4_hba.dat_rq->list); | 3512 | list_del_init(&phba->sli4_hba.dat_rq->list); |
3496 | list_del_init(&phba->sli4_hba.mbx_cq->list); | 3513 | list_del_init(&phba->sli4_hba.mbx_cq->list); |
3497 | list_del_init(&phba->sli4_hba.els_cq->list); | 3514 | list_del_init(&phba->sli4_hba.els_cq->list); |
3498 | list_del_init(&phba->sli4_hba.rxq_cq->list); | ||
3499 | for (qindx = 0; qindx < phba->cfg_fcp_wq_count; qindx++) | 3515 | for (qindx = 0; qindx < phba->cfg_fcp_wq_count; qindx++) |
3500 | list_del_init(&phba->sli4_hba.fcp_wq[qindx]->list); | 3516 | list_del_init(&phba->sli4_hba.fcp_wq[qindx]->list); |
3501 | for (qindx = 0; qindx < phba->cfg_fcp_eq_count; qindx++) | 3517 | for (qindx = 0; qindx < phba->cfg_fcp_eq_count; qindx++) |
@@ -4243,7 +4259,6 @@ lpfc_sli4_arm_cqeq_intr(struct lpfc_hba *phba) | |||
4243 | 4259 | ||
4244 | lpfc_sli4_cq_release(phba->sli4_hba.mbx_cq, LPFC_QUEUE_REARM); | 4260 | lpfc_sli4_cq_release(phba->sli4_hba.mbx_cq, LPFC_QUEUE_REARM); |
4245 | lpfc_sli4_cq_release(phba->sli4_hba.els_cq, LPFC_QUEUE_REARM); | 4261 | lpfc_sli4_cq_release(phba->sli4_hba.els_cq, LPFC_QUEUE_REARM); |
4246 | lpfc_sli4_cq_release(phba->sli4_hba.rxq_cq, LPFC_QUEUE_REARM); | ||
4247 | for (fcp_eqidx = 0; fcp_eqidx < phba->cfg_fcp_eq_count; fcp_eqidx++) | 4262 | for (fcp_eqidx = 0; fcp_eqidx < phba->cfg_fcp_eq_count; fcp_eqidx++) |
4248 | lpfc_sli4_cq_release(phba->sli4_hba.fcp_cq[fcp_eqidx], | 4263 | lpfc_sli4_cq_release(phba->sli4_hba.fcp_cq[fcp_eqidx], |
4249 | LPFC_QUEUE_REARM); | 4264 | LPFC_QUEUE_REARM); |
@@ -8351,8 +8366,7 @@ lpfc_sli4_iocb_param_transfer(struct lpfc_iocbq *pIocbIn, | |||
8351 | 8366 | ||
8352 | memcpy((char *)pIocbIn + offset, (char *)pIocbOut + offset, | 8367 | memcpy((char *)pIocbIn + offset, (char *)pIocbOut + offset, |
8353 | sizeof(struct lpfc_iocbq) - offset); | 8368 | sizeof(struct lpfc_iocbq) - offset); |
8354 | memset(&pIocbIn->sli4_info, 0, | 8369 | pIocbIn->cq_event.cqe.wcqe_cmpl = *wcqe; |
8355 | sizeof(struct lpfc_sli4_rspiocb_info)); | ||
8356 | /* Map WCQE parameters into irspiocb parameters */ | 8370 | /* Map WCQE parameters into irspiocb parameters */ |
8357 | pIocbIn->iocb.ulpStatus = bf_get(lpfc_wcqe_c_status, wcqe); | 8371 | pIocbIn->iocb.ulpStatus = bf_get(lpfc_wcqe_c_status, wcqe); |
8358 | if (pIocbOut->iocb_flag & LPFC_IO_FCP) | 8372 | if (pIocbOut->iocb_flag & LPFC_IO_FCP) |
@@ -8364,16 +8378,6 @@ lpfc_sli4_iocb_param_transfer(struct lpfc_iocbq *pIocbIn, | |||
8364 | pIocbIn->iocb.un.ulpWord[4] = wcqe->parameter; | 8378 | pIocbIn->iocb.un.ulpWord[4] = wcqe->parameter; |
8365 | else | 8379 | else |
8366 | pIocbIn->iocb.un.ulpWord[4] = wcqe->parameter; | 8380 | pIocbIn->iocb.un.ulpWord[4] = wcqe->parameter; |
8367 | /* Load in additional WCQE parameters */ | ||
8368 | pIocbIn->sli4_info.hw_status = bf_get(lpfc_wcqe_c_hw_status, wcqe); | ||
8369 | pIocbIn->sli4_info.bfield = 0; | ||
8370 | if (bf_get(lpfc_wcqe_c_xb, wcqe)) | ||
8371 | pIocbIn->sli4_info.bfield |= LPFC_XB; | ||
8372 | if (bf_get(lpfc_wcqe_c_pv, wcqe)) { | ||
8373 | pIocbIn->sli4_info.bfield |= LPFC_PV; | ||
8374 | pIocbIn->sli4_info.priority = | ||
8375 | bf_get(lpfc_wcqe_c_priority, wcqe); | ||
8376 | } | ||
8377 | } | 8381 | } |
8378 | 8382 | ||
8379 | /** | 8383 | /** |
@@ -8598,7 +8602,8 @@ lpfc_sli4_sp_handle_els_wcqe(struct lpfc_hba *phba, | |||
8598 | 8602 | ||
8599 | /* Add the irspiocb to the response IOCB work list */ | 8603 | /* Add the irspiocb to the response IOCB work list */ |
8600 | spin_lock_irqsave(&phba->hbalock, iflags); | 8604 | spin_lock_irqsave(&phba->hbalock, iflags); |
8601 | list_add_tail(&irspiocbq->list, &phba->sli4_hba.sp_rspiocb_work_queue); | 8605 | list_add_tail(&irspiocbq->cq_event.list, |
8606 | &phba->sli4_hba.sp_rspiocb_work_queue); | ||
8602 | /* Indicate ELS ring attention */ | 8607 | /* Indicate ELS ring attention */ |
8603 | phba->work_ha |= (HA_R0ATT << (4*LPFC_ELS_RING)); | 8608 | phba->work_ha |= (HA_R0ATT << (4*LPFC_ELS_RING)); |
8604 | spin_unlock_irqrestore(&phba->hbalock, iflags); | 8609 | spin_unlock_irqrestore(&phba->hbalock, iflags); |
@@ -8690,52 +8695,6 @@ lpfc_sli4_sp_handle_abort_xri_wcqe(struct lpfc_hba *phba, | |||
8690 | } | 8695 | } |
8691 | 8696 | ||
8692 | /** | 8697 | /** |
8693 | * lpfc_sli4_sp_handle_wcqe - Process a work-queue completion queue entry | ||
8694 | * @phba: Pointer to HBA context object. | ||
8695 | * @cq: Pointer to the completion queue. | ||
8696 | * @wcqe: Pointer to a completion queue entry. | ||
8697 | * | ||
8698 | * This routine process a slow-path work-queue completion queue entry. | ||
8699 | * | ||
8700 | * Return: true if work posted to worker thread, otherwise false. | ||
8701 | **/ | ||
8702 | static bool | ||
8703 | lpfc_sli4_sp_handle_wcqe(struct lpfc_hba *phba, struct lpfc_queue *cq, | ||
8704 | struct lpfc_cqe *cqe) | ||
8705 | { | ||
8706 | struct lpfc_wcqe_complete wcqe; | ||
8707 | bool workposted = false; | ||
8708 | |||
8709 | /* Copy the work queue CQE and convert endian order if needed */ | ||
8710 | lpfc_sli_pcimem_bcopy(cqe, &wcqe, sizeof(struct lpfc_cqe)); | ||
8711 | |||
8712 | /* Check and process for different type of WCQE and dispatch */ | ||
8713 | switch (bf_get(lpfc_wcqe_c_code, &wcqe)) { | ||
8714 | case CQE_CODE_COMPL_WQE: | ||
8715 | /* Process the WQ complete event */ | ||
8716 | workposted = lpfc_sli4_sp_handle_els_wcqe(phba, | ||
8717 | (struct lpfc_wcqe_complete *)&wcqe); | ||
8718 | break; | ||
8719 | case CQE_CODE_RELEASE_WQE: | ||
8720 | /* Process the WQ release event */ | ||
8721 | lpfc_sli4_sp_handle_rel_wcqe(phba, | ||
8722 | (struct lpfc_wcqe_release *)&wcqe); | ||
8723 | break; | ||
8724 | case CQE_CODE_XRI_ABORTED: | ||
8725 | /* Process the WQ XRI abort event */ | ||
8726 | workposted = lpfc_sli4_sp_handle_abort_xri_wcqe(phba, cq, | ||
8727 | (struct sli4_wcqe_xri_aborted *)&wcqe); | ||
8728 | break; | ||
8729 | default: | ||
8730 | lpfc_printf_log(phba, KERN_ERR, LOG_SLI, | ||
8731 | "0388 Not a valid WCQE code: x%x\n", | ||
8732 | bf_get(lpfc_wcqe_c_code, &wcqe)); | ||
8733 | break; | ||
8734 | } | ||
8735 | return workposted; | ||
8736 | } | ||
8737 | |||
8738 | /** | ||
8739 | * lpfc_sli4_sp_handle_rcqe - Process a receive-queue completion queue entry | 8698 | * lpfc_sli4_sp_handle_rcqe - Process a receive-queue completion queue entry |
8740 | * @phba: Pointer to HBA context object. | 8699 | * @phba: Pointer to HBA context object. |
8741 | * @rcqe: Pointer to receive-queue completion queue entry. | 8700 | * @rcqe: Pointer to receive-queue completion queue entry. |
@@ -8745,9 +8704,8 @@ lpfc_sli4_sp_handle_wcqe(struct lpfc_hba *phba, struct lpfc_queue *cq, | |||
8745 | * Return: true if work posted to worker thread, otherwise false. | 8704 | * Return: true if work posted to worker thread, otherwise false. |
8746 | **/ | 8705 | **/ |
8747 | static bool | 8706 | static bool |
8748 | lpfc_sli4_sp_handle_rcqe(struct lpfc_hba *phba, struct lpfc_cqe *cqe) | 8707 | lpfc_sli4_sp_handle_rcqe(struct lpfc_hba *phba, struct lpfc_rcqe *rcqe) |
8749 | { | 8708 | { |
8750 | struct lpfc_rcqe rcqe; | ||
8751 | bool workposted = false; | 8709 | bool workposted = false; |
8752 | struct lpfc_queue *hrq = phba->sli4_hba.hdr_rq; | 8710 | struct lpfc_queue *hrq = phba->sli4_hba.hdr_rq; |
8753 | struct lpfc_queue *drq = phba->sli4_hba.dat_rq; | 8711 | struct lpfc_queue *drq = phba->sli4_hba.dat_rq; |
@@ -8755,15 +8713,13 @@ lpfc_sli4_sp_handle_rcqe(struct lpfc_hba *phba, struct lpfc_cqe *cqe) | |||
8755 | uint32_t status; | 8713 | uint32_t status; |
8756 | unsigned long iflags; | 8714 | unsigned long iflags; |
8757 | 8715 | ||
8758 | /* Copy the receive queue CQE and convert endian order if needed */ | ||
8759 | lpfc_sli_pcimem_bcopy(cqe, &rcqe, sizeof(struct lpfc_rcqe)); | ||
8760 | lpfc_sli4_rq_release(hrq, drq); | 8716 | lpfc_sli4_rq_release(hrq, drq); |
8761 | if (bf_get(lpfc_rcqe_code, &rcqe) != CQE_CODE_RECEIVE) | 8717 | if (bf_get(lpfc_rcqe_code, rcqe) != CQE_CODE_RECEIVE) |
8762 | goto out; | 8718 | goto out; |
8763 | if (bf_get(lpfc_rcqe_rq_id, &rcqe) != hrq->queue_id) | 8719 | if (bf_get(lpfc_rcqe_rq_id, rcqe) != hrq->queue_id) |
8764 | goto out; | 8720 | goto out; |
8765 | 8721 | ||
8766 | status = bf_get(lpfc_rcqe_status, &rcqe); | 8722 | status = bf_get(lpfc_rcqe_status, rcqe); |
8767 | switch (status) { | 8723 | switch (status) { |
8768 | case FC_STATUS_RQ_BUF_LEN_EXCEEDED: | 8724 | case FC_STATUS_RQ_BUF_LEN_EXCEEDED: |
8769 | lpfc_printf_log(phba, KERN_ERR, LOG_SLI, | 8725 | lpfc_printf_log(phba, KERN_ERR, LOG_SLI, |
@@ -8775,9 +8731,10 @@ lpfc_sli4_sp_handle_rcqe(struct lpfc_hba *phba, struct lpfc_cqe *cqe) | |||
8775 | spin_unlock_irqrestore(&phba->hbalock, iflags); | 8731 | spin_unlock_irqrestore(&phba->hbalock, iflags); |
8776 | goto out; | 8732 | goto out; |
8777 | } | 8733 | } |
8778 | memcpy(&dma_buf->rcqe, &rcqe, sizeof(rcqe)); | 8734 | memcpy(&dma_buf->cq_event.cqe.rcqe_cmpl, rcqe, sizeof(*rcqe)); |
8779 | /* save off the frame for the word thread to process */ | 8735 | /* save off the frame for the word thread to process */ |
8780 | list_add_tail(&dma_buf->dbuf.list, &phba->rb_pend_list); | 8736 | list_add_tail(&dma_buf->cq_event.list, |
8737 | &phba->sli4_hba.sp_rspiocb_work_queue); | ||
8781 | /* Frame received */ | 8738 | /* Frame received */ |
8782 | phba->hba_flag |= HBA_RECEIVE_BUFFER; | 8739 | phba->hba_flag |= HBA_RECEIVE_BUFFER; |
8783 | spin_unlock_irqrestore(&phba->hbalock, iflags); | 8740 | spin_unlock_irqrestore(&phba->hbalock, iflags); |
@@ -8798,6 +8755,58 @@ out: | |||
8798 | } | 8755 | } |
8799 | 8756 | ||
8800 | /** | 8757 | /** |
8758 | * lpfc_sli4_sp_handle_cqe - Process a slow path completion queue entry | ||
8759 | * @phba: Pointer to HBA context object. | ||
8760 | * @cq: Pointer to the completion queue. | ||
8761 | * @wcqe: Pointer to a completion queue entry. | ||
8762 | * | ||
8763 | * This routine process a slow-path work-queue or recieve queue completion queue | ||
8764 | * entry. | ||
8765 | * | ||
8766 | * Return: true if work posted to worker thread, otherwise false. | ||
8767 | **/ | ||
8768 | static bool | ||
8769 | lpfc_sli4_sp_handle_cqe(struct lpfc_hba *phba, struct lpfc_queue *cq, | ||
8770 | struct lpfc_cqe *cqe) | ||
8771 | { | ||
8772 | struct lpfc_wcqe_complete wcqe; | ||
8773 | bool workposted = false; | ||
8774 | |||
8775 | /* Copy the work queue CQE and convert endian order if needed */ | ||
8776 | lpfc_sli_pcimem_bcopy(cqe, &wcqe, sizeof(struct lpfc_cqe)); | ||
8777 | |||
8778 | /* Check and process for different type of WCQE and dispatch */ | ||
8779 | switch (bf_get(lpfc_wcqe_c_code, &wcqe)) { | ||
8780 | case CQE_CODE_COMPL_WQE: | ||
8781 | /* Process the WQ complete event */ | ||
8782 | workposted = lpfc_sli4_sp_handle_els_wcqe(phba, | ||
8783 | (struct lpfc_wcqe_complete *)&wcqe); | ||
8784 | break; | ||
8785 | case CQE_CODE_RELEASE_WQE: | ||
8786 | /* Process the WQ release event */ | ||
8787 | lpfc_sli4_sp_handle_rel_wcqe(phba, | ||
8788 | (struct lpfc_wcqe_release *)&wcqe); | ||
8789 | break; | ||
8790 | case CQE_CODE_XRI_ABORTED: | ||
8791 | /* Process the WQ XRI abort event */ | ||
8792 | workposted = lpfc_sli4_sp_handle_abort_xri_wcqe(phba, cq, | ||
8793 | (struct sli4_wcqe_xri_aborted *)&wcqe); | ||
8794 | break; | ||
8795 | case CQE_CODE_RECEIVE: | ||
8796 | /* Process the RQ event */ | ||
8797 | workposted = lpfc_sli4_sp_handle_rcqe(phba, | ||
8798 | (struct lpfc_rcqe *)&wcqe); | ||
8799 | break; | ||
8800 | default: | ||
8801 | lpfc_printf_log(phba, KERN_ERR, LOG_SLI, | ||
8802 | "0388 Not a valid WCQE code: x%x\n", | ||
8803 | bf_get(lpfc_wcqe_c_code, &wcqe)); | ||
8804 | break; | ||
8805 | } | ||
8806 | return workposted; | ||
8807 | } | ||
8808 | |||
8809 | /** | ||
8801 | * lpfc_sli4_sp_handle_eqe - Process a slow-path event queue entry | 8810 | * lpfc_sli4_sp_handle_eqe - Process a slow-path event queue entry |
8802 | * @phba: Pointer to HBA context object. | 8811 | * @phba: Pointer to HBA context object. |
8803 | * @eqe: Pointer to fast-path event queue entry. | 8812 | * @eqe: Pointer to fast-path event queue entry. |
@@ -8858,14 +8867,7 @@ lpfc_sli4_sp_handle_eqe(struct lpfc_hba *phba, struct lpfc_eqe *eqe) | |||
8858 | break; | 8867 | break; |
8859 | case LPFC_WCQ: | 8868 | case LPFC_WCQ: |
8860 | while ((cqe = lpfc_sli4_cq_get(cq))) { | 8869 | while ((cqe = lpfc_sli4_cq_get(cq))) { |
8861 | workposted |= lpfc_sli4_sp_handle_wcqe(phba, cq, cqe); | 8870 | workposted |= lpfc_sli4_sp_handle_cqe(phba, cq, cqe); |
8862 | if (!(++ecount % LPFC_GET_QE_REL_INT)) | ||
8863 | lpfc_sli4_cq_release(cq, LPFC_QUEUE_NOARM); | ||
8864 | } | ||
8865 | break; | ||
8866 | case LPFC_RCQ: | ||
8867 | while ((cqe = lpfc_sli4_cq_get(cq))) { | ||
8868 | workposted |= lpfc_sli4_sp_handle_rcqe(phba, cqe); | ||
8869 | if (!(++ecount % LPFC_GET_QE_REL_INT)) | 8871 | if (!(++ecount % LPFC_GET_QE_REL_INT)) |
8870 | lpfc_sli4_cq_release(cq, LPFC_QUEUE_NOARM); | 8872 | lpfc_sli4_cq_release(cq, LPFC_QUEUE_NOARM); |
8871 | } | 8873 | } |
@@ -10823,6 +10825,7 @@ lpfc_fc_frame_add(struct lpfc_vport *vport, struct hbq_dmabuf *dmabuf) | |||
10823 | struct hbq_dmabuf *seq_dmabuf = NULL; | 10825 | struct hbq_dmabuf *seq_dmabuf = NULL; |
10824 | struct hbq_dmabuf *temp_dmabuf = NULL; | 10826 | struct hbq_dmabuf *temp_dmabuf = NULL; |
10825 | 10827 | ||
10828 | INIT_LIST_HEAD(&dmabuf->dbuf.list); | ||
10826 | new_hdr = (struct fc_frame_header *)dmabuf->hbuf.virt; | 10829 | new_hdr = (struct fc_frame_header *)dmabuf->hbuf.virt; |
10827 | /* Use the hdr_buf to find the sequence that this frame belongs to */ | 10830 | /* Use the hdr_buf to find the sequence that this frame belongs to */ |
10828 | list_for_each_entry(h_buf, &vport->rcv_buffer_list, list) { | 10831 | list_for_each_entry(h_buf, &vport->rcv_buffer_list, list) { |
@@ -10845,7 +10848,9 @@ lpfc_fc_frame_add(struct lpfc_vport *vport, struct hbq_dmabuf *dmabuf) | |||
10845 | } | 10848 | } |
10846 | temp_hdr = seq_dmabuf->hbuf.virt; | 10849 | temp_hdr = seq_dmabuf->hbuf.virt; |
10847 | if (new_hdr->fh_seq_cnt < temp_hdr->fh_seq_cnt) { | 10850 | if (new_hdr->fh_seq_cnt < temp_hdr->fh_seq_cnt) { |
10848 | list_add(&seq_dmabuf->dbuf.list, &dmabuf->dbuf.list); | 10851 | list_del_init(&seq_dmabuf->hbuf.list); |
10852 | list_add_tail(&dmabuf->hbuf.list, &vport->rcv_buffer_list); | ||
10853 | list_add_tail(&dmabuf->dbuf.list, &seq_dmabuf->dbuf.list); | ||
10849 | return dmabuf; | 10854 | return dmabuf; |
10850 | } | 10855 | } |
10851 | /* find the correct place in the sequence to insert this frame */ | 10856 | /* find the correct place in the sequence to insert this frame */ |
@@ -10957,7 +10962,8 @@ lpfc_prep_seq(struct lpfc_vport *vport, struct hbq_dmabuf *seq_dmabuf) | |||
10957 | LPFC_DATA_BUF_SIZE; | 10962 | LPFC_DATA_BUF_SIZE; |
10958 | first_iocbq->iocb.un.rcvels.remoteID = sid; | 10963 | first_iocbq->iocb.un.rcvels.remoteID = sid; |
10959 | first_iocbq->iocb.unsli3.rcvsli3.acc_len += | 10964 | first_iocbq->iocb.unsli3.rcvsli3.acc_len += |
10960 | bf_get(lpfc_rcqe_length, &seq_dmabuf->rcqe); | 10965 | bf_get(lpfc_rcqe_length, |
10966 | &seq_dmabuf->cq_event.cqe.rcqe_cmpl); | ||
10961 | } | 10967 | } |
10962 | iocbq = first_iocbq; | 10968 | iocbq = first_iocbq; |
10963 | /* | 10969 | /* |
@@ -10975,7 +10981,8 @@ lpfc_prep_seq(struct lpfc_vport *vport, struct hbq_dmabuf *seq_dmabuf) | |||
10975 | iocbq->iocb.unsli3.rcvsli3.bde2.tus.f.bdeSize = | 10981 | iocbq->iocb.unsli3.rcvsli3.bde2.tus.f.bdeSize = |
10976 | LPFC_DATA_BUF_SIZE; | 10982 | LPFC_DATA_BUF_SIZE; |
10977 | first_iocbq->iocb.unsli3.rcvsli3.acc_len += | 10983 | first_iocbq->iocb.unsli3.rcvsli3.acc_len += |
10978 | bf_get(lpfc_rcqe_length, &seq_dmabuf->rcqe); | 10984 | bf_get(lpfc_rcqe_length, |
10985 | &seq_dmabuf->cq_event.cqe.rcqe_cmpl); | ||
10979 | } else { | 10986 | } else { |
10980 | iocbq = lpfc_sli_get_iocbq(vport->phba); | 10987 | iocbq = lpfc_sli_get_iocbq(vport->phba); |
10981 | if (!iocbq) { | 10988 | if (!iocbq) { |
@@ -10994,7 +11001,8 @@ lpfc_prep_seq(struct lpfc_vport *vport, struct hbq_dmabuf *seq_dmabuf) | |||
10994 | iocbq->iocb.un.cont64[0].tus.f.bdeSize = | 11001 | iocbq->iocb.un.cont64[0].tus.f.bdeSize = |
10995 | LPFC_DATA_BUF_SIZE; | 11002 | LPFC_DATA_BUF_SIZE; |
10996 | first_iocbq->iocb.unsli3.rcvsli3.acc_len += | 11003 | first_iocbq->iocb.unsli3.rcvsli3.acc_len += |
10997 | bf_get(lpfc_rcqe_length, &seq_dmabuf->rcqe); | 11004 | bf_get(lpfc_rcqe_length, |
11005 | &seq_dmabuf->cq_event.cqe.rcqe_cmpl); | ||
10998 | iocbq->iocb.un.rcvels.remoteID = sid; | 11006 | iocbq->iocb.un.rcvels.remoteID = sid; |
10999 | list_add_tail(&iocbq->list, &first_iocbq->list); | 11007 | list_add_tail(&iocbq->list, &first_iocbq->list); |
11000 | } | 11008 | } |
@@ -11014,11 +11022,11 @@ lpfc_prep_seq(struct lpfc_vport *vport, struct hbq_dmabuf *seq_dmabuf) | |||
11014 | * Worker thread calls lpfc_sli4_handle_received_buffer, which will call the | 11022 | * Worker thread calls lpfc_sli4_handle_received_buffer, which will call the |
11015 | * appropriate receive function when the final frame in a sequence is received. | 11023 | * appropriate receive function when the final frame in a sequence is received. |
11016 | **/ | 11024 | **/ |
11017 | int | 11025 | void |
11018 | lpfc_sli4_handle_received_buffer(struct lpfc_hba *phba) | 11026 | lpfc_sli4_handle_received_buffer(struct lpfc_hba *phba, |
11027 | struct hbq_dmabuf *dmabuf) | ||
11019 | { | 11028 | { |
11020 | LIST_HEAD(cmplq); | 11029 | struct hbq_dmabuf *seq_dmabuf; |
11021 | struct hbq_dmabuf *dmabuf, *seq_dmabuf; | ||
11022 | struct fc_frame_header *fc_hdr; | 11030 | struct fc_frame_header *fc_hdr; |
11023 | struct lpfc_vport *vport; | 11031 | struct lpfc_vport *vport; |
11024 | uint32_t fcfi; | 11032 | uint32_t fcfi; |
@@ -11027,54 +11035,50 @@ lpfc_sli4_handle_received_buffer(struct lpfc_hba *phba) | |||
11027 | /* Clear hba flag and get all received buffers into the cmplq */ | 11035 | /* Clear hba flag and get all received buffers into the cmplq */ |
11028 | spin_lock_irq(&phba->hbalock); | 11036 | spin_lock_irq(&phba->hbalock); |
11029 | phba->hba_flag &= ~HBA_RECEIVE_BUFFER; | 11037 | phba->hba_flag &= ~HBA_RECEIVE_BUFFER; |
11030 | list_splice_init(&phba->rb_pend_list, &cmplq); | ||
11031 | spin_unlock_irq(&phba->hbalock); | 11038 | spin_unlock_irq(&phba->hbalock); |
11032 | 11039 | ||
11033 | /* Process each received buffer */ | 11040 | /* Process each received buffer */ |
11034 | while ((dmabuf = lpfc_sli_hbqbuf_get(&cmplq)) != NULL) { | 11041 | fc_hdr = (struct fc_frame_header *)dmabuf->hbuf.virt; |
11035 | fc_hdr = (struct fc_frame_header *)dmabuf->hbuf.virt; | 11042 | /* check to see if this a valid type of frame */ |
11036 | /* check to see if this a valid type of frame */ | 11043 | if (lpfc_fc_frame_check(phba, fc_hdr)) { |
11037 | if (lpfc_fc_frame_check(phba, fc_hdr)) { | 11044 | lpfc_in_buf_free(phba, &dmabuf->dbuf); |
11038 | lpfc_in_buf_free(phba, &dmabuf->dbuf); | 11045 | return; |
11039 | continue; | 11046 | } |
11040 | } | 11047 | fcfi = bf_get(lpfc_rcqe_fcf_id, &dmabuf->cq_event.cqe.rcqe_cmpl); |
11041 | fcfi = bf_get(lpfc_rcqe_fcf_id, &dmabuf->rcqe); | 11048 | vport = lpfc_fc_frame_to_vport(phba, fc_hdr, fcfi); |
11042 | vport = lpfc_fc_frame_to_vport(phba, fc_hdr, fcfi); | 11049 | if (!vport) { |
11043 | if (!vport) { | 11050 | /* throw out the frame */ |
11044 | /* throw out the frame */ | 11051 | lpfc_in_buf_free(phba, &dmabuf->dbuf); |
11045 | lpfc_in_buf_free(phba, &dmabuf->dbuf); | 11052 | return; |
11046 | continue; | 11053 | } |
11047 | } | 11054 | /* Link this frame */ |
11048 | /* Link this frame */ | 11055 | seq_dmabuf = lpfc_fc_frame_add(vport, dmabuf); |
11049 | seq_dmabuf = lpfc_fc_frame_add(vport, dmabuf); | 11056 | if (!seq_dmabuf) { |
11050 | if (!seq_dmabuf) { | 11057 | /* unable to add frame to vport - throw it out */ |
11051 | /* unable to add frame to vport - throw it out */ | 11058 | lpfc_in_buf_free(phba, &dmabuf->dbuf); |
11052 | lpfc_in_buf_free(phba, &dmabuf->dbuf); | 11059 | return; |
11053 | continue; | 11060 | } |
11054 | } | 11061 | /* If not last frame in sequence continue processing frames. */ |
11055 | /* If not last frame in sequence continue processing frames. */ | 11062 | if (!lpfc_seq_complete(seq_dmabuf)) { |
11056 | if (!lpfc_seq_complete(seq_dmabuf)) { | 11063 | /* |
11057 | /* | 11064 | * When saving off frames post a new one and mark this |
11058 | * When saving off frames post a new one and mark this | 11065 | * frame to be freed when it is finished. |
11059 | * frame to be freed when it is finished. | 11066 | **/ |
11060 | **/ | 11067 | lpfc_sli_hbqbuf_fill_hbqs(phba, LPFC_ELS_HBQ, 1); |
11061 | lpfc_sli_hbqbuf_fill_hbqs(phba, LPFC_ELS_HBQ, 1); | 11068 | dmabuf->tag = -1; |
11062 | dmabuf->tag = -1; | 11069 | return; |
11063 | continue; | 11070 | } |
11064 | } | 11071 | fc_hdr = (struct fc_frame_header *)seq_dmabuf->hbuf.virt; |
11065 | fc_hdr = (struct fc_frame_header *)seq_dmabuf->hbuf.virt; | 11072 | iocbq = lpfc_prep_seq(vport, seq_dmabuf); |
11066 | iocbq = lpfc_prep_seq(vport, seq_dmabuf); | 11073 | if (!lpfc_complete_unsol_iocb(phba, |
11067 | if (!lpfc_complete_unsol_iocb(phba, | 11074 | &phba->sli.ring[LPFC_ELS_RING], |
11068 | &phba->sli.ring[LPFC_ELS_RING], | 11075 | iocbq, fc_hdr->fh_r_ctl, |
11069 | iocbq, fc_hdr->fh_r_ctl, | 11076 | fc_hdr->fh_type)) |
11070 | fc_hdr->fh_type)) | 11077 | lpfc_printf_log(phba, KERN_WARNING, LOG_SLI, |
11071 | lpfc_printf_log(phba, KERN_WARNING, LOG_SLI, | 11078 | "2540 Ring %d handler: unexpected Rctl " |
11072 | "2540 Ring %d handler: unexpected Rctl " | 11079 | "x%x Type x%x received\n", |
11073 | "x%x Type x%x received\n", | 11080 | LPFC_ELS_RING, |
11074 | LPFC_ELS_RING, | 11081 | fc_hdr->fh_r_ctl, fc_hdr->fh_type); |
11075 | fc_hdr->fh_r_ctl, fc_hdr->fh_type); | ||
11076 | }; | ||
11077 | return 0; | ||
11078 | } | 11082 | } |
11079 | 11083 | ||
11080 | /** | 11084 | /** |
@@ -11542,7 +11546,8 @@ lpfc_sli4_read_fcf_record(struct lpfc_hba *phba, uint16_t fcf_index) | |||
11542 | lpfc_printf_log(phba, KERN_ERR, LOG_INIT, | 11546 | lpfc_printf_log(phba, KERN_ERR, LOG_INIT, |
11543 | "2000 Failed to allocate mbox for " | 11547 | "2000 Failed to allocate mbox for " |
11544 | "READ_FCF cmd\n"); | 11548 | "READ_FCF cmd\n"); |
11545 | return -ENOMEM; | 11549 | error = -ENOMEM; |
11550 | goto fail_fcfscan; | ||
11546 | } | 11551 | } |
11547 | 11552 | ||
11548 | req_len = sizeof(struct fcf_record) + | 11553 | req_len = sizeof(struct fcf_record) + |
@@ -11558,8 +11563,8 @@ lpfc_sli4_read_fcf_record(struct lpfc_hba *phba, uint16_t fcf_index) | |||
11558 | "0291 Allocated DMA memory size (x%x) is " | 11563 | "0291 Allocated DMA memory size (x%x) is " |
11559 | "less than the requested DMA memory " | 11564 | "less than the requested DMA memory " |
11560 | "size (x%x)\n", alloc_len, req_len); | 11565 | "size (x%x)\n", alloc_len, req_len); |
11561 | lpfc_sli4_mbox_cmd_free(phba, mboxq); | 11566 | error = -ENOMEM; |
11562 | return -ENOMEM; | 11567 | goto fail_fcfscan; |
11563 | } | 11568 | } |
11564 | 11569 | ||
11565 | /* Get the first SGE entry from the non-embedded DMA memory. This | 11570 | /* Get the first SGE entry from the non-embedded DMA memory. This |
@@ -11571,8 +11576,8 @@ lpfc_sli4_read_fcf_record(struct lpfc_hba *phba, uint16_t fcf_index) | |||
11571 | lpfc_printf_log(phba, KERN_ERR, LOG_MBOX, | 11576 | lpfc_printf_log(phba, KERN_ERR, LOG_MBOX, |
11572 | "2527 Failed to get the non-embedded SGE " | 11577 | "2527 Failed to get the non-embedded SGE " |
11573 | "virtual address\n"); | 11578 | "virtual address\n"); |
11574 | lpfc_sli4_mbox_cmd_free(phba, mboxq); | 11579 | error = -ENOMEM; |
11575 | return -ENOMEM; | 11580 | goto fail_fcfscan; |
11576 | } | 11581 | } |
11577 | virt_addr = mboxq->sge_array->addr[0]; | 11582 | virt_addr = mboxq->sge_array->addr[0]; |
11578 | read_fcf = (struct lpfc_mbx_read_fcf_tbl *)virt_addr; | 11583 | read_fcf = (struct lpfc_mbx_read_fcf_tbl *)virt_addr; |
@@ -11586,7 +11591,6 @@ lpfc_sli4_read_fcf_record(struct lpfc_hba *phba, uint16_t fcf_index) | |||
11586 | mboxq->mbox_cmpl = lpfc_mbx_cmpl_read_fcf_record; | 11591 | mboxq->mbox_cmpl = lpfc_mbx_cmpl_read_fcf_record; |
11587 | rc = lpfc_sli_issue_mbox(phba, mboxq, MBX_NOWAIT); | 11592 | rc = lpfc_sli_issue_mbox(phba, mboxq, MBX_NOWAIT); |
11588 | if (rc == MBX_NOT_FINISHED) { | 11593 | if (rc == MBX_NOT_FINISHED) { |
11589 | lpfc_sli4_mbox_cmd_free(phba, mboxq); | ||
11590 | error = -EIO; | 11594 | error = -EIO; |
11591 | } else { | 11595 | } else { |
11592 | spin_lock_irq(&phba->hbalock); | 11596 | spin_lock_irq(&phba->hbalock); |
@@ -11594,6 +11598,15 @@ lpfc_sli4_read_fcf_record(struct lpfc_hba *phba, uint16_t fcf_index) | |||
11594 | spin_unlock_irq(&phba->hbalock); | 11598 | spin_unlock_irq(&phba->hbalock); |
11595 | error = 0; | 11599 | error = 0; |
11596 | } | 11600 | } |
11601 | fail_fcfscan: | ||
11602 | if (error) { | ||
11603 | if (mboxq) | ||
11604 | lpfc_sli4_mbox_cmd_free(phba, mboxq); | ||
11605 | /* FCF scan failed, clear FCF_DISC_INPROGRESS flag */ | ||
11606 | spin_lock_irq(&phba->hbalock); | ||
11607 | phba->hba_flag &= ~FCF_DISC_INPROGRESS; | ||
11608 | spin_unlock_irq(&phba->hbalock); | ||
11609 | } | ||
11597 | return error; | 11610 | return error; |
11598 | } | 11611 | } |
11599 | 11612 | ||
diff --git a/drivers/scsi/lpfc/lpfc_sli.h b/drivers/scsi/lpfc/lpfc_sli.h index 3c53316cf6d0..ad8094966ff3 100644 --- a/drivers/scsi/lpfc/lpfc_sli.h +++ b/drivers/scsi/lpfc/lpfc_sli.h | |||
@@ -29,14 +29,17 @@ typedef enum _lpfc_ctx_cmd { | |||
29 | LPFC_CTX_HOST | 29 | LPFC_CTX_HOST |
30 | } lpfc_ctx_cmd; | 30 | } lpfc_ctx_cmd; |
31 | 31 | ||
32 | /* This structure is used to carry the needed response IOCB states */ | 32 | struct lpfc_cq_event { |
33 | struct lpfc_sli4_rspiocb_info { | 33 | struct list_head list; |
34 | uint8_t hw_status; | 34 | union { |
35 | uint8_t bfield; | 35 | struct lpfc_mcqe mcqe_cmpl; |
36 | #define LPFC_XB 0x1 | 36 | struct lpfc_acqe_link acqe_link; |
37 | #define LPFC_PV 0x2 | 37 | struct lpfc_acqe_fcoe acqe_fcoe; |
38 | uint8_t priority; | 38 | struct lpfc_acqe_dcbx acqe_dcbx; |
39 | uint8_t reserved; | 39 | struct lpfc_rcqe rcqe_cmpl; |
40 | struct sli4_wcqe_xri_aborted wcqe_axri; | ||
41 | struct lpfc_wcqe_complete wcqe_cmpl; | ||
42 | } cqe; | ||
40 | }; | 43 | }; |
41 | 44 | ||
42 | /* This structure is used to handle IOCB requests / responses */ | 45 | /* This structure is used to handle IOCB requests / responses */ |
@@ -76,7 +79,7 @@ struct lpfc_iocbq { | |||
76 | struct lpfc_iocbq *); | 79 | struct lpfc_iocbq *); |
77 | void (*iocb_cmpl) (struct lpfc_hba *, struct lpfc_iocbq *, | 80 | void (*iocb_cmpl) (struct lpfc_hba *, struct lpfc_iocbq *, |
78 | struct lpfc_iocbq *); | 81 | struct lpfc_iocbq *); |
79 | struct lpfc_sli4_rspiocb_info sli4_info; | 82 | struct lpfc_cq_event cq_event; |
80 | }; | 83 | }; |
81 | 84 | ||
82 | #define SLI_IOCB_RET_IOCB 1 /* Return IOCB if cmd ring full */ | 85 | #define SLI_IOCB_RET_IOCB 1 /* Return IOCB if cmd ring full */ |
diff --git a/drivers/scsi/lpfc/lpfc_sli4.h b/drivers/scsi/lpfc/lpfc_sli4.h index b5f4ba1a5c27..97da7589e038 100644 --- a/drivers/scsi/lpfc/lpfc_sli4.h +++ b/drivers/scsi/lpfc/lpfc_sli4.h | |||
@@ -110,18 +110,6 @@ struct lpfc_queue { | |||
110 | union sli4_qe qe[1]; /* array to index entries (must be last) */ | 110 | union sli4_qe qe[1]; /* array to index entries (must be last) */ |
111 | }; | 111 | }; |
112 | 112 | ||
113 | struct lpfc_cq_event { | ||
114 | struct list_head list; | ||
115 | union { | ||
116 | struct lpfc_mcqe mcqe_cmpl; | ||
117 | struct lpfc_acqe_link acqe_link; | ||
118 | struct lpfc_acqe_fcoe acqe_fcoe; | ||
119 | struct lpfc_acqe_dcbx acqe_dcbx; | ||
120 | struct lpfc_rcqe rcqe_cmpl; | ||
121 | struct sli4_wcqe_xri_aborted wcqe_axri; | ||
122 | } cqe; | ||
123 | }; | ||
124 | |||
125 | struct lpfc_sli4_link { | 113 | struct lpfc_sli4_link { |
126 | uint8_t speed; | 114 | uint8_t speed; |
127 | uint8_t duplex; | 115 | uint8_t duplex; |
@@ -325,7 +313,6 @@ struct lpfc_sli4_hba { | |||
325 | struct lpfc_queue **fcp_cq;/* Fast-path FCP compl queue */ | 313 | struct lpfc_queue **fcp_cq;/* Fast-path FCP compl queue */ |
326 | struct lpfc_queue *mbx_cq; /* Slow-path mailbox complete queue */ | 314 | struct lpfc_queue *mbx_cq; /* Slow-path mailbox complete queue */ |
327 | struct lpfc_queue *els_cq; /* Slow-path ELS response complete queue */ | 315 | struct lpfc_queue *els_cq; /* Slow-path ELS response complete queue */ |
328 | struct lpfc_queue *rxq_cq; /* Slow-path unsolicited complete queue */ | ||
329 | 316 | ||
330 | /* Setup information for various queue parameters */ | 317 | /* Setup information for various queue parameters */ |
331 | int eq_esize; | 318 | int eq_esize; |