Commit 3772a991 authored by James Smart's avatar James Smart Committed by James Bottomley

[SCSI] lpfc 8.3.2 : Reorganization for SLI4

Preps the organization of the driver so that the bottom half, which
interacts with the hardware, can share common code sequences for
attachment, detachment, initialization, teardown, etc with new hardware.

For very common code sections, which become specific to the interface
type, the driver uses an indirect function call. The function is set at
initialization. For less common sections, such as initialization, the
driver looks at the interface type and calls the routines relative to
the interface.
Signed-off-by: default avatarJames Smart <james.smart@emulex.com>
Signed-off-by: default avatarJames Bottomley <James.Bottomley@HansenPartnership.com>
parent a3666955
...@@ -23,6 +23,13 @@ ...@@ -23,6 +23,13 @@
struct lpfc_sli2_slim; struct lpfc_sli2_slim;
#define LPFC_PCI_DEV_LP 0x1
#define LPFC_PCI_DEV_OC 0x2
#define LPFC_SLI_REV2 2
#define LPFC_SLI_REV3 3
#define LPFC_SLI_REV4 4
#define LPFC_MAX_TARGET 4096 /* max number of targets supported */ #define LPFC_MAX_TARGET 4096 /* max number of targets supported */
#define LPFC_MAX_DISC_THREADS 64 /* max outstanding discovery els #define LPFC_MAX_DISC_THREADS 64 /* max outstanding discovery els
requests */ requests */
...@@ -264,8 +271,8 @@ enum hba_state { ...@@ -264,8 +271,8 @@ enum hba_state {
}; };
struct lpfc_vport { struct lpfc_vport {
struct list_head listentry;
struct lpfc_hba *phba; struct lpfc_hba *phba;
struct list_head listentry;
uint8_t port_type; uint8_t port_type;
#define LPFC_PHYSICAL_PORT 1 #define LPFC_PHYSICAL_PORT 1
#define LPFC_NPIV_PORT 2 #define LPFC_NPIV_PORT 2
...@@ -420,8 +427,66 @@ enum intr_type_t { ...@@ -420,8 +427,66 @@ enum intr_type_t {
}; };
struct lpfc_hba { struct lpfc_hba {
/* SCSI interface function jump table entries */
int (*lpfc_new_scsi_buf)
(struct lpfc_vport *, int);
struct lpfc_scsi_buf * (*lpfc_get_scsi_buf)
(struct lpfc_hba *);
int (*lpfc_scsi_prep_dma_buf)
(struct lpfc_hba *, struct lpfc_scsi_buf *);
void (*lpfc_scsi_unprep_dma_buf)
(struct lpfc_hba *, struct lpfc_scsi_buf *);
void (*lpfc_release_scsi_buf)
(struct lpfc_hba *, struct lpfc_scsi_buf *);
void (*lpfc_rampdown_queue_depth)
(struct lpfc_hba *);
void (*lpfc_scsi_prep_cmnd)
(struct lpfc_vport *, struct lpfc_scsi_buf *,
struct lpfc_nodelist *);
int (*lpfc_scsi_prep_task_mgmt_cmd)
(struct lpfc_vport *, struct lpfc_scsi_buf *,
unsigned int, uint8_t);
/* IOCB interface function jump table entries */
int (*__lpfc_sli_issue_iocb)
(struct lpfc_hba *, uint32_t,
struct lpfc_iocbq *, uint32_t);
void (*__lpfc_sli_release_iocbq)(struct lpfc_hba *,
struct lpfc_iocbq *);
int (*lpfc_hba_down_post)(struct lpfc_hba *phba);
IOCB_t * (*lpfc_get_iocb_from_iocbq)
(struct lpfc_iocbq *);
void (*lpfc_scsi_cmd_iocb_cmpl)
(struct lpfc_hba *, struct lpfc_iocbq *, struct lpfc_iocbq *);
/* MBOX interface function jump table entries */
int (*lpfc_sli_issue_mbox)
(struct lpfc_hba *, LPFC_MBOXQ_t *, uint32_t);
/* Slow-path IOCB process function jump table entries */
void (*lpfc_sli_handle_slow_ring_event)
(struct lpfc_hba *phba, struct lpfc_sli_ring *pring,
uint32_t mask);
/* INIT device interface function jump table entries */
int (*lpfc_sli_hbq_to_firmware)
(struct lpfc_hba *, uint32_t, struct hbq_dmabuf *);
int (*lpfc_sli_brdrestart)
(struct lpfc_hba *);
int (*lpfc_sli_brdready)
(struct lpfc_hba *, uint32_t);
void (*lpfc_handle_eratt)
(struct lpfc_hba *);
void (*lpfc_stop_port)
(struct lpfc_hba *);
/* SLI4 specific HBA data structure */
struct lpfc_sli4_hba sli4_hba;
struct lpfc_sli sli; struct lpfc_sli sli;
uint32_t sli_rev; /* SLI2 or SLI3 */ uint8_t pci_dev_grp; /* lpfc PCI dev group: 0x0, 0x1, 0x2,... */
uint32_t sli_rev; /* SLI2, SLI3, or SLI4 */
uint32_t sli3_options; /* Mask of enabled SLI3 options */ uint32_t sli3_options; /* Mask of enabled SLI3 options */
#define LPFC_SLI3_HBQ_ENABLED 0x01 #define LPFC_SLI3_HBQ_ENABLED 0x01
#define LPFC_SLI3_NPIV_ENABLED 0x02 #define LPFC_SLI3_NPIV_ENABLED 0x02
...@@ -526,11 +591,12 @@ struct lpfc_hba { ...@@ -526,11 +591,12 @@ struct lpfc_hba {
unsigned long data_flags; unsigned long data_flags;
uint32_t hbq_in_use; /* HBQs in use flag */ uint32_t hbq_in_use; /* HBQs in use flag */
struct list_head hbqbuf_in_list; /* in-fly hbq buffer list */ struct list_head rb_pend_list; /* Received buffers to be processed */
uint32_t hbq_count; /* Count of configured HBQs */ uint32_t hbq_count; /* Count of configured HBQs */
struct hbq_s hbqs[LPFC_MAX_HBQS]; /* local copy of hbq indicies */ struct hbq_s hbqs[LPFC_MAX_HBQS]; /* local copy of hbq indicies */
unsigned long pci_bar0_map; /* Physical address for PCI BAR0 */ unsigned long pci_bar0_map; /* Physical address for PCI BAR0 */
unsigned long pci_bar1_map; /* Physical address for PCI BAR1 */
unsigned long pci_bar2_map; /* Physical address for PCI BAR2 */ unsigned long pci_bar2_map; /* Physical address for PCI BAR2 */
void __iomem *slim_memmap_p; /* Kernel memory mapped address for void __iomem *slim_memmap_p; /* Kernel memory mapped address for
PCI BAR0 */ PCI BAR0 */
......
...@@ -267,8 +267,6 @@ lpfc_gen_req(struct lpfc_vport *vport, struct lpfc_dmabuf *bmp, ...@@ -267,8 +267,6 @@ lpfc_gen_req(struct lpfc_vport *vport, struct lpfc_dmabuf *bmp,
uint32_t tmo, uint8_t retry) uint32_t tmo, uint8_t retry)
{ {
struct lpfc_hba *phba = vport->phba; struct lpfc_hba *phba = vport->phba;
struct lpfc_sli *psli = &phba->sli;
struct lpfc_sli_ring *pring = &psli->ring[LPFC_ELS_RING];
IOCB_t *icmd; IOCB_t *icmd;
struct lpfc_iocbq *geniocb; struct lpfc_iocbq *geniocb;
int rc; int rc;
...@@ -331,7 +329,7 @@ lpfc_gen_req(struct lpfc_vport *vport, struct lpfc_dmabuf *bmp, ...@@ -331,7 +329,7 @@ lpfc_gen_req(struct lpfc_vport *vport, struct lpfc_dmabuf *bmp,
geniocb->drvrTimeout = icmd->ulpTimeout + LPFC_DRVR_TIMEOUT; geniocb->drvrTimeout = icmd->ulpTimeout + LPFC_DRVR_TIMEOUT;
geniocb->vport = vport; geniocb->vport = vport;
geniocb->retry = retry; geniocb->retry = retry;
rc = lpfc_sli_issue_iocb(phba, pring, geniocb, 0); rc = lpfc_sli_issue_iocb(phba, LPFC_ELS_RING, geniocb, 0);
if (rc == IOCB_ERROR) { if (rc == IOCB_ERROR) {
lpfc_sli_release_iocbq(phba, geniocb); lpfc_sli_release_iocbq(phba, geniocb);
......
...@@ -280,6 +280,8 @@ lpfc_debugfs_hbqinfo_data(struct lpfc_hba *phba, char *buf, int size) ...@@ -280,6 +280,8 @@ lpfc_debugfs_hbqinfo_data(struct lpfc_hba *phba, char *buf, int size)
struct lpfc_dmabuf *d_buf; struct lpfc_dmabuf *d_buf;
struct hbq_dmabuf *hbq_buf; struct hbq_dmabuf *hbq_buf;
if (phba->sli_rev != 3)
return 0;
cnt = LPFC_HBQINFO_SIZE; cnt = LPFC_HBQINFO_SIZE;
spin_lock_irq(&phba->hbalock); spin_lock_irq(&phba->hbalock);
...@@ -489,12 +491,15 @@ lpfc_debugfs_dumpHostSlim_data(struct lpfc_hba *phba, char *buf, int size) ...@@ -489,12 +491,15 @@ lpfc_debugfs_dumpHostSlim_data(struct lpfc_hba *phba, char *buf, int size)
pring->next_cmdidx, pring->local_getidx, pring->next_cmdidx, pring->local_getidx,
pring->flag, pgpp->rspPutInx, pring->numRiocb); pring->flag, pgpp->rspPutInx, pring->numRiocb);
} }
word0 = readl(phba->HAregaddr);
word1 = readl(phba->CAregaddr); if (phba->sli_rev <= LPFC_SLI_REV3) {
word2 = readl(phba->HSregaddr); word0 = readl(phba->HAregaddr);
word3 = readl(phba->HCregaddr); word1 = readl(phba->CAregaddr);
len += snprintf(buf+len, size-len, "HA:%08x CA:%08x HS:%08x HC:%08x\n", word2 = readl(phba->HSregaddr);
word0, word1, word2, word3); word3 = readl(phba->HCregaddr);
len += snprintf(buf+len, size-len, "HA:%08x CA:%08x HS:%08x "
"HC:%08x\n", word0, word1, word2, word3);
}
spin_unlock_irq(&phba->hbalock); spin_unlock_irq(&phba->hbalock);
return len; return len;
} }
......
This diff is collapsed.
...@@ -555,23 +555,24 @@ lpfc_work_done(struct lpfc_hba *phba) ...@@ -555,23 +555,24 @@ lpfc_work_done(struct lpfc_hba *phba)
/* /*
* Turn on Ring interrupts * Turn on Ring interrupts
*/ */
spin_lock_irq(&phba->hbalock); if (phba->sli_rev <= LPFC_SLI_REV3) {
control = readl(phba->HCregaddr); spin_lock_irq(&phba->hbalock);
if (!(control & (HC_R0INT_ENA << LPFC_ELS_RING))) { control = readl(phba->HCregaddr);
lpfc_debugfs_slow_ring_trc(phba, if (!(control & (HC_R0INT_ENA << LPFC_ELS_RING))) {
"WRK Enable ring: cntl:x%x hacopy:x%x", lpfc_debugfs_slow_ring_trc(phba,
control, ha_copy, 0); "WRK Enable ring: cntl:x%x hacopy:x%x",
control, ha_copy, 0);
control |= (HC_R0INT_ENA << LPFC_ELS_RING);
writel(control, phba->HCregaddr); control |= (HC_R0INT_ENA << LPFC_ELS_RING);
readl(phba->HCregaddr); /* flush */ writel(control, phba->HCregaddr);
} readl(phba->HCregaddr); /* flush */
else { } else {
lpfc_debugfs_slow_ring_trc(phba, lpfc_debugfs_slow_ring_trc(phba,
"WRK Ring ok: cntl:x%x hacopy:x%x", "WRK Ring ok: cntl:x%x hacopy:x%x",
control, ha_copy, 0); control, ha_copy, 0);
}
spin_unlock_irq(&phba->hbalock);
} }
spin_unlock_irq(&phba->hbalock);
} }
lpfc_work_list_done(phba); lpfc_work_list_done(phba);
} }
...@@ -689,7 +690,7 @@ lpfc_port_link_failure(struct lpfc_vport *vport) ...@@ -689,7 +690,7 @@ lpfc_port_link_failure(struct lpfc_vport *vport)
lpfc_can_disctmo(vport); lpfc_can_disctmo(vport);
} }
static void void
lpfc_linkdown_port(struct lpfc_vport *vport) lpfc_linkdown_port(struct lpfc_vport *vport)
{ {
struct Scsi_Host *shost = lpfc_shost_from_vport(vport); struct Scsi_Host *shost = lpfc_shost_from_vport(vport);
...@@ -1147,10 +1148,12 @@ lpfc_enable_la(struct lpfc_hba *phba) ...@@ -1147,10 +1148,12 @@ lpfc_enable_la(struct lpfc_hba *phba)
struct lpfc_sli *psli = &phba->sli; struct lpfc_sli *psli = &phba->sli;
spin_lock_irq(&phba->hbalock); spin_lock_irq(&phba->hbalock);
psli->sli_flag |= LPFC_PROCESS_LA; psli->sli_flag |= LPFC_PROCESS_LA;
control = readl(phba->HCregaddr); if (phba->sli_rev <= LPFC_SLI_REV3) {
control |= HC_LAINT_ENA; control = readl(phba->HCregaddr);
writel(control, phba->HCregaddr); control |= HC_LAINT_ENA;
readl(phba->HCregaddr); /* flush */ writel(control, phba->HCregaddr);
readl(phba->HCregaddr); /* flush */
}
spin_unlock_irq(&phba->hbalock); spin_unlock_irq(&phba->hbalock);
} }
...@@ -2919,11 +2922,13 @@ lpfc_disc_timeout_handler(struct lpfc_vport *vport) ...@@ -2919,11 +2922,13 @@ lpfc_disc_timeout_handler(struct lpfc_vport *vport)
* set port_state to PORT_READY if SLI2. * set port_state to PORT_READY if SLI2.
* cmpl_reg_vpi will set port_state to READY for SLI3. * cmpl_reg_vpi will set port_state to READY for SLI3.
*/ */
if (phba->sli3_options & LPFC_SLI3_NPIV_ENABLED) if (phba->sli_rev < LPFC_SLI_REV4) {
lpfc_issue_reg_vpi(phba, vport); if (phba->sli3_options & LPFC_SLI3_NPIV_ENABLED)
else { /* NPIV Not enabled */ lpfc_issue_reg_vpi(phba, vport);
lpfc_issue_clear_la(phba, vport); else { /* NPIV Not enabled */
vport->port_state = LPFC_VPORT_READY; lpfc_issue_clear_la(phba, vport);
vport->port_state = LPFC_VPORT_READY;
}
} }
/* Setup and issue mailbox INITIALIZE LINK command */ /* Setup and issue mailbox INITIALIZE LINK command */
...@@ -2959,11 +2964,13 @@ lpfc_disc_timeout_handler(struct lpfc_vport *vport) ...@@ -2959,11 +2964,13 @@ lpfc_disc_timeout_handler(struct lpfc_vport *vport)
* set port_state to PORT_READY if SLI2. * set port_state to PORT_READY if SLI2.
* cmpl_reg_vpi will set port_state to READY for SLI3. * cmpl_reg_vpi will set port_state to READY for SLI3.
*/ */
if (phba->sli3_options & LPFC_SLI3_NPIV_ENABLED) if (phba->sli_rev < LPFC_SLI_REV4) {
lpfc_issue_reg_vpi(phba, vport); if (phba->sli3_options & LPFC_SLI3_NPIV_ENABLED)
else { /* NPIV Not enabled */ lpfc_issue_reg_vpi(phba, vport);
lpfc_issue_clear_la(phba, vport); else { /* NPIV Not enabled */
vport->port_state = LPFC_VPORT_READY; lpfc_issue_clear_la(phba, vport);
vport->port_state = LPFC_VPORT_READY;
}
} }
break; break;
......
This diff is collapsed.
This diff is collapsed.
This diff is collapsed.
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