1. Added support to offline a port if is error recovery on successful.

2. Sysctls to enable/disable driver_state_dump and error_recovery.
3. Sysctl to control the delay between hw/fw reinitialization and
   restarting the fastpath.
4. Stop periodic stats retrieval if interface has IFF_DRV_RUNNING flag off.
5. Print contents of PEG_HALT_STATUS1 and PEG_HALT_STATUS2 on heartbeat
   failure.
6. Speed up slowpath shutdown during error recovery.
7. link_state update using atomic_store.
8. Added timestamp information on driver state and minidump captures.
9. Added support for Slowpath event logging
10.Added additional failure injection types to simulate failures.
This commit is contained in:
David C Somayajulu 2018-02-23 03:36:24 +00:00
parent ee4e69f1b9
commit b65c0c07b2
13 changed files with 811 additions and 176 deletions

View File

@ -44,17 +44,21 @@ extern void ql_dump_buf16(qla_host_t *ha, const char *str, void *dbuf,
extern void ql_dump_buf32(qla_host_t *ha, const char *str, void *dbuf, extern void ql_dump_buf32(qla_host_t *ha, const char *str, void *dbuf,
uint32_t len32); uint32_t len32);
#define INJCT_RX_RXB_INVAL 0x00001 #define INJCT_RX_RXB_INVAL 0x00001
#define INJCT_RX_MP_NULL 0x00002 #define INJCT_RX_MP_NULL 0x00002
#define INJCT_LRO_RXB_INVAL 0x00003 #define INJCT_LRO_RXB_INVAL 0x00003
#define INJCT_LRO_MP_NULL 0x00004 #define INJCT_LRO_MP_NULL 0x00004
#define INJCT_NUM_HNDLE_INVALID 0x00005 #define INJCT_NUM_HNDLE_INVALID 0x00005
#define INJCT_RDWR_INDREG_FAILURE 0x00006 #define INJCT_RDWR_INDREG_FAILURE 0x00006
#define INJCT_RDWR_OFFCHIPMEM_FAILURE 0x00007 #define INJCT_RDWR_OFFCHIPMEM_FAILURE 0x00007
#define INJCT_MBX_CMD_FAILURE 0x00008 #define INJCT_MBX_CMD_FAILURE 0x00008
#define INJCT_HEARTBEAT_FAILURE 0x00009 #define INJCT_HEARTBEAT_FAILURE 0x00009
#define INJCT_TEMPERATURE_FAILURE 0x0000A #define INJCT_TEMPERATURE_FAILURE 0x0000A
#define INJCT_M_GETCL_M_GETJCL_FAILURE 0x0000B #define INJCT_M_GETCL_M_GETJCL_FAILURE 0x0000B
#define INJCT_INV_CONT_OPCODE 0x0000C
#define INJCT_SGL_RCV_INV_DESC_COUNT 0x0000D
#define INJCT_SGL_LRO_INV_DESC_COUNT 0x0000E
#define INJCT_PEER_PORT_FAILURE_ERR_RECOVERY 0x0000F
#ifdef QL_DBG #ifdef QL_DBG

View File

@ -146,12 +146,12 @@ struct qla_host {
volatile uint32_t qla_watchdog_paused; volatile uint32_t qla_watchdog_paused;
volatile uint32_t qla_initiate_recovery; volatile uint32_t qla_initiate_recovery;
volatile uint32_t qla_detach_active; volatile uint32_t qla_detach_active;
volatile uint32_t offline;
device_t pci_dev; device_t pci_dev;
uint16_t watchdog_ticks; volatile uint16_t watchdog_ticks;
uint8_t pci_func; uint8_t pci_func;
uint8_t resvd;
/* ioctl related */ /* ioctl related */
struct cdev *ioctl_dev; struct cdev *ioctl_dev;
@ -184,6 +184,7 @@ struct qla_host {
/* hardware access lock */ /* hardware access lock */
struct mtx sp_log_lock;
struct mtx hw_lock; struct mtx hw_lock;
volatile uint32_t hw_lock_held; volatile uint32_t hw_lock_held;
uint64_t hw_lock_failed; uint64_t hw_lock_failed;
@ -241,6 +242,9 @@ struct qla_host {
volatile const char *qla_unlock; volatile const char *qla_unlock;
uint32_t dbg_level; uint32_t dbg_level;
uint32_t enable_minidump; uint32_t enable_minidump;
uint32_t enable_driverstate_dump;
uint32_t enable_error_recovery;
uint32_t ms_delay_after_init;
uint8_t fw_ver_str[32]; uint8_t fw_ver_str[32];
@ -275,4 +279,6 @@ typedef struct qla_host qla_host_t;
((((*(uint32_t *) mac1) == (*(uint32_t *) mac2) && \ ((((*(uint32_t *) mac1) == (*(uint32_t *) mac2) && \
(*(uint16_t *)(mac1 + 4)) == (*(uint16_t *)(mac2 + 4)))) ? 0 : 1) (*(uint16_t *)(mac1 + 4)) == (*(uint16_t *)(mac2 + 4)))) ? 0 : 1)
#define QL_INITIATE_RECOVERY(ha) qla_set_error_recovery(ha)
#endif /* #ifndef _QL_DEF_H_ */ #endif /* #ifndef _QL_DEF_H_ */

View File

@ -49,6 +49,7 @@ extern uint32_t ql_rcv_isr(qla_host_t *ha, uint32_t sds_idx, uint32_t count);
extern int ql_alloc_dmabuf(qla_host_t *ha, qla_dma_t *dma_buf); extern int ql_alloc_dmabuf(qla_host_t *ha, qla_dma_t *dma_buf);
extern void ql_free_dmabuf(qla_host_t *ha, qla_dma_t *dma_buf); extern void ql_free_dmabuf(qla_host_t *ha, qla_dma_t *dma_buf);
extern int ql_get_mbuf(qla_host_t *ha, qla_rx_buf_t *rxb, struct mbuf *nmp); extern int ql_get_mbuf(qla_host_t *ha, qla_rx_buf_t *rxb, struct mbuf *nmp);
extern void qla_set_error_recovery(qla_host_t *ha);
/* /*
* from ql_hw.c * from ql_hw.c
@ -117,5 +118,11 @@ extern unsigned int ql83xx_minidump_len;
extern void ql_alloc_drvr_state_buffer(qla_host_t *ha); extern void ql_alloc_drvr_state_buffer(qla_host_t *ha);
extern void ql_free_drvr_state_buffer(qla_host_t *ha); extern void ql_free_drvr_state_buffer(qla_host_t *ha);
extern void ql_capture_drvr_state(qla_host_t *ha); extern void ql_capture_drvr_state(qla_host_t *ha);
extern void ql_sp_log(qla_host_t *ha, uint16_t fmtstr_idx, uint16_t num_params,
uint32_t param0, uint32_t param1, uint32_t param2,
uint32_t param3, uint32_t param4);
extern void ql_alloc_sp_log_buffer(qla_host_t *ha);
extern void ql_free_sp_log_buffer(qla_host_t *ha);
#endif /* #ifndef_QL_GLBL_H_ */ #endif /* #ifndef_QL_GLBL_H_ */

View File

@ -51,7 +51,7 @@ __FBSDID("$FreeBSD$");
static void qla_del_rcv_cntxt(qla_host_t *ha); static void qla_del_rcv_cntxt(qla_host_t *ha);
static int qla_init_rcv_cntxt(qla_host_t *ha); static int qla_init_rcv_cntxt(qla_host_t *ha);
static void qla_del_xmt_cntxt(qla_host_t *ha); static int qla_del_xmt_cntxt(qla_host_t *ha);
static int qla_init_xmt_cntxt(qla_host_t *ha); static int qla_init_xmt_cntxt(qla_host_t *ha);
static int qla_mbx_cmd(qla_host_t *ha, uint32_t *h_mbox, uint32_t n_hmbox, static int qla_mbx_cmd(qla_host_t *ha, uint32_t *h_mbox, uint32_t n_hmbox,
uint32_t *fw_mbox, uint32_t n_fwmbox, uint32_t no_pause); uint32_t *fw_mbox, uint32_t n_fwmbox, uint32_t no_pause);
@ -648,12 +648,119 @@ qlnx_add_hw_xmt_stats_sysctls(qla_host_t *ha)
return; return;
} }
static void
qlnx_add_hw_mbx_cmpl_stats_sysctls(qla_host_t *ha)
{
struct sysctl_ctx_list *ctx;
struct sysctl_oid_list *node_children;
ctx = device_get_sysctl_ctx(ha->pci_dev);
node_children = SYSCTL_CHILDREN(device_get_sysctl_tree(ha->pci_dev));
SYSCTL_ADD_QUAD(ctx, node_children,
OID_AUTO, "mbx_completion_time_lt_200ms",
CTLFLAG_RD, &ha->hw.mbx_comp_msecs[0],
"mbx_completion_time_lt_200ms");
SYSCTL_ADD_QUAD(ctx, node_children,
OID_AUTO, "mbx_completion_time_200ms_400ms",
CTLFLAG_RD, &ha->hw.mbx_comp_msecs[1],
"mbx_completion_time_200ms_400ms");
SYSCTL_ADD_QUAD(ctx, node_children,
OID_AUTO, "mbx_completion_time_400ms_600ms",
CTLFLAG_RD, &ha->hw.mbx_comp_msecs[2],
"mbx_completion_time_400ms_600ms");
SYSCTL_ADD_QUAD(ctx, node_children,
OID_AUTO, "mbx_completion_time_600ms_800ms",
CTLFLAG_RD, &ha->hw.mbx_comp_msecs[3],
"mbx_completion_time_600ms_800ms");
SYSCTL_ADD_QUAD(ctx, node_children,
OID_AUTO, "mbx_completion_time_800ms_1000ms",
CTLFLAG_RD, &ha->hw.mbx_comp_msecs[4],
"mbx_completion_time_800ms_1000ms");
SYSCTL_ADD_QUAD(ctx, node_children,
OID_AUTO, "mbx_completion_time_1000ms_1200ms",
CTLFLAG_RD, &ha->hw.mbx_comp_msecs[5],
"mbx_completion_time_1000ms_1200ms");
SYSCTL_ADD_QUAD(ctx, node_children,
OID_AUTO, "mbx_completion_time_1200ms_1400ms",
CTLFLAG_RD, &ha->hw.mbx_comp_msecs[6],
"mbx_completion_time_1200ms_1400ms");
SYSCTL_ADD_QUAD(ctx, node_children,
OID_AUTO, "mbx_completion_time_1400ms_1600ms",
CTLFLAG_RD, &ha->hw.mbx_comp_msecs[7],
"mbx_completion_time_1400ms_1600ms");
SYSCTL_ADD_QUAD(ctx, node_children,
OID_AUTO, "mbx_completion_time_1600ms_1800ms",
CTLFLAG_RD, &ha->hw.mbx_comp_msecs[8],
"mbx_completion_time_1600ms_1800ms");
SYSCTL_ADD_QUAD(ctx, node_children,
OID_AUTO, "mbx_completion_time_1800ms_2000ms",
CTLFLAG_RD, &ha->hw.mbx_comp_msecs[9],
"mbx_completion_time_1800ms_2000ms");
SYSCTL_ADD_QUAD(ctx, node_children,
OID_AUTO, "mbx_completion_time_2000ms_2200ms",
CTLFLAG_RD, &ha->hw.mbx_comp_msecs[10],
"mbx_completion_time_2000ms_2200ms");
SYSCTL_ADD_QUAD(ctx, node_children,
OID_AUTO, "mbx_completion_time_2200ms_2400ms",
CTLFLAG_RD, &ha->hw.mbx_comp_msecs[11],
"mbx_completion_time_2200ms_2400ms");
SYSCTL_ADD_QUAD(ctx, node_children,
OID_AUTO, "mbx_completion_time_2400ms_2600ms",
CTLFLAG_RD, &ha->hw.mbx_comp_msecs[12],
"mbx_completion_time_2400ms_2600ms");
SYSCTL_ADD_QUAD(ctx, node_children,
OID_AUTO, "mbx_completion_time_2600ms_2800ms",
CTLFLAG_RD, &ha->hw.mbx_comp_msecs[13],
"mbx_completion_time_2600ms_2800ms");
SYSCTL_ADD_QUAD(ctx, node_children,
OID_AUTO, "mbx_completion_time_2800ms_3000ms",
CTLFLAG_RD, &ha->hw.mbx_comp_msecs[14],
"mbx_completion_time_2800ms_3000ms");
SYSCTL_ADD_QUAD(ctx, node_children,
OID_AUTO, "mbx_completion_time_3000ms_4000ms",
CTLFLAG_RD, &ha->hw.mbx_comp_msecs[15],
"mbx_completion_time_3000ms_4000ms");
SYSCTL_ADD_QUAD(ctx, node_children,
OID_AUTO, "mbx_completion_time_4000ms_5000ms",
CTLFLAG_RD, &ha->hw.mbx_comp_msecs[16],
"mbx_completion_time_4000ms_5000ms");
SYSCTL_ADD_QUAD(ctx, node_children,
OID_AUTO, "mbx_completion_host_mbx_cntrl_timeout",
CTLFLAG_RD, &ha->hw.mbx_comp_msecs[17],
"mbx_completion_host_mbx_cntrl_timeout");
SYSCTL_ADD_QUAD(ctx, node_children,
OID_AUTO, "mbx_completion_fw_mbx_cntrl_timeout",
CTLFLAG_RD, &ha->hw.mbx_comp_msecs[18],
"mbx_completion_fw_mbx_cntrl_timeout");
return;
}
static void static void
qlnx_add_hw_stats_sysctls(qla_host_t *ha) qlnx_add_hw_stats_sysctls(qla_host_t *ha)
{ {
qlnx_add_hw_mac_stats_sysctls(ha); qlnx_add_hw_mac_stats_sysctls(ha);
qlnx_add_hw_rcv_stats_sysctls(ha); qlnx_add_hw_rcv_stats_sysctls(ha);
qlnx_add_hw_xmt_stats_sysctls(ha); qlnx_add_hw_xmt_stats_sysctls(ha);
qlnx_add_hw_mbx_cmpl_stats_sysctls(ha);
return; return;
} }
@ -920,6 +1027,30 @@ ql_hw_add_sysctls(qla_host_t *ha)
"\t Any change requires ifconfig down/up to take effect\n" "\t Any change requires ifconfig down/up to take effect\n"
"\t Note that LRO may be turned off/on via ifconfig\n"); "\t Note that LRO may be turned off/on via ifconfig\n");
SYSCTL_ADD_UINT(device_get_sysctl_ctx(dev),
SYSCTL_CHILDREN(device_get_sysctl_tree(dev)),
OID_AUTO, "sp_log_index", CTLFLAG_RW, &ha->hw.sp_log_index,
ha->hw.sp_log_index, "sp_log_index");
SYSCTL_ADD_UINT(device_get_sysctl_ctx(dev),
SYSCTL_CHILDREN(device_get_sysctl_tree(dev)),
OID_AUTO, "sp_log_stop", CTLFLAG_RW, &ha->hw.sp_log_stop,
ha->hw.sp_log_stop, "sp_log_stop");
ha->hw.sp_log_stop_events = 0;
SYSCTL_ADD_UINT(device_get_sysctl_ctx(dev),
SYSCTL_CHILDREN(device_get_sysctl_tree(dev)),
OID_AUTO, "sp_log_stop_events", CTLFLAG_RW,
&ha->hw.sp_log_stop_events,
ha->hw.sp_log_stop_events, "Slow path event log is stopped"
" when OR of the following events occur \n"
"\t 0x01 : Heart beat Failure\n"
"\t 0x02 : Temperature Failure\n"
"\t 0x04 : HW Initialization Failure\n"
"\t 0x08 : Interface Initialization Failure\n"
"\t 0x10 : Error Recovery Failure\n");
ha->hw.mdump_active = 0; ha->hw.mdump_active = 0;
SYSCTL_ADD_UINT(device_get_sysctl_ctx(dev), SYSCTL_ADD_UINT(device_get_sysctl_ctx(dev),
SYSCTL_CHILDREN(device_get_sysctl_tree(dev)), SYSCTL_CHILDREN(device_get_sysctl_tree(dev)),
@ -1000,13 +1131,13 @@ ql_hw_link_status(qla_host_t *ha)
device_printf(ha->pci_dev, "link Down\n"); device_printf(ha->pci_dev, "link Down\n");
} }
if (ha->hw.flags.fduplex) { if (ha->hw.fduplex) {
device_printf(ha->pci_dev, "Full Duplex\n"); device_printf(ha->pci_dev, "Full Duplex\n");
} else { } else {
device_printf(ha->pci_dev, "Half Duplex\n"); device_printf(ha->pci_dev, "Half Duplex\n");
} }
if (ha->hw.flags.autoneg) { if (ha->hw.autoneg) {
device_printf(ha->pci_dev, "Auto Negotiation Enabled\n"); device_printf(ha->pci_dev, "Auto Negotiation Enabled\n");
} else { } else {
device_printf(ha->pci_dev, "Auto Negotiation Disabled\n"); device_printf(ha->pci_dev, "Auto Negotiation Disabled\n");
@ -1257,19 +1388,39 @@ qla_mbx_cmd(qla_host_t *ha, uint32_t *h_mbox, uint32_t n_hmbox,
uint32_t i; uint32_t i;
uint32_t data; uint32_t data;
int ret = 0; int ret = 0;
uint64_t start_usecs;
uint64_t end_usecs;
uint64_t msecs_200;
if (QL_ERR_INJECT(ha, INJCT_MBX_CMD_FAILURE)) { ql_sp_log(ha, 0, 5, no_pause, h_mbox[0], h_mbox[1], h_mbox[2], h_mbox[3]);
ret = -3;
ha->qla_initiate_recovery = 1; if (ha->offline || ha->qla_initiate_recovery) {
ql_sp_log(ha, 1, 2, ha->offline, ha->qla_initiate_recovery, 0, 0, 0);
goto exit_qla_mbx_cmd; goto exit_qla_mbx_cmd;
} }
if (((ha->err_inject & 0xFFFF) == INJCT_MBX_CMD_FAILURE) &&
(((ha->err_inject & ~0xFFFF) == ((h_mbox[0] & 0xFFFF) << 16))||
!(ha->err_inject & ~0xFFFF))) {
ret = -3;
QL_INITIATE_RECOVERY(ha);
goto exit_qla_mbx_cmd;
}
start_usecs = qla_get_usec_timestamp();
if (no_pause) if (no_pause)
i = 1000; i = 1000;
else else
i = Q8_MBX_MSEC_DELAY; i = Q8_MBX_MSEC_DELAY;
while (i) { while (i) {
if (ha->qla_initiate_recovery) {
ql_sp_log(ha, 2, 1, ha->qla_initiate_recovery, 0, 0, 0, 0);
return (-1);
}
data = READ_REG32(ha, Q8_HOST_MBOX_CNTRL); data = READ_REG32(ha, Q8_HOST_MBOX_CNTRL);
if (data == 0) if (data == 0)
break; break;
@ -1284,8 +1435,10 @@ qla_mbx_cmd(qla_host_t *ha, uint32_t *h_mbox, uint32_t n_hmbox,
if (i == 0) { if (i == 0) {
device_printf(ha->pci_dev, "%s: host_mbx_cntrl 0x%08x\n", device_printf(ha->pci_dev, "%s: host_mbx_cntrl 0x%08x\n",
__func__, data); __func__, data);
ql_sp_log(ha, 3, 1, data, 0, 0, 0, 0);
ret = -1; ret = -1;
ha->qla_initiate_recovery = 1; ha->hw.mbx_comp_msecs[(Q8_MBX_COMP_MSECS - 2)]++;
QL_INITIATE_RECOVERY(ha);
goto exit_qla_mbx_cmd; goto exit_qla_mbx_cmd;
} }
@ -1299,6 +1452,12 @@ qla_mbx_cmd(qla_host_t *ha, uint32_t *h_mbox, uint32_t n_hmbox,
i = Q8_MBX_MSEC_DELAY; i = Q8_MBX_MSEC_DELAY;
while (i) { while (i) {
if (ha->qla_initiate_recovery) {
ql_sp_log(ha, 4, 1, ha->qla_initiate_recovery, 0, 0, 0, 0);
return (-1);
}
data = READ_REG32(ha, Q8_FW_MBOX_CNTRL); data = READ_REG32(ha, Q8_FW_MBOX_CNTRL);
if ((data & 0x3) == 1) { if ((data & 0x3) == 1) {
@ -1316,18 +1475,44 @@ qla_mbx_cmd(qla_host_t *ha, uint32_t *h_mbox, uint32_t n_hmbox,
if (i == 0) { if (i == 0) {
device_printf(ha->pci_dev, "%s: fw_mbx_cntrl 0x%08x\n", device_printf(ha->pci_dev, "%s: fw_mbx_cntrl 0x%08x\n",
__func__, data); __func__, data);
ql_sp_log(ha, 5, 1, data, 0, 0, 0, 0);
ret = -2; ret = -2;
ha->qla_initiate_recovery = 1; ha->hw.mbx_comp_msecs[(Q8_MBX_COMP_MSECS - 1)]++;
QL_INITIATE_RECOVERY(ha);
goto exit_qla_mbx_cmd; goto exit_qla_mbx_cmd;
} }
for (i = 0; i < n_fwmbox; i++) { for (i = 0; i < n_fwmbox; i++) {
if (ha->qla_initiate_recovery) {
ql_sp_log(ha, 6, 1, ha->qla_initiate_recovery, 0, 0, 0, 0);
return (-1);
}
*fw_mbox++ = READ_REG32(ha, (Q8_FW_MBOX0 + (i << 2))); *fw_mbox++ = READ_REG32(ha, (Q8_FW_MBOX0 + (i << 2)));
} }
WRITE_REG32(ha, Q8_FW_MBOX_CNTRL, 0x0); WRITE_REG32(ha, Q8_FW_MBOX_CNTRL, 0x0);
WRITE_REG32(ha, ha->hw.mbx_intr_mask_offset, 0x0); WRITE_REG32(ha, ha->hw.mbx_intr_mask_offset, 0x0);
end_usecs = qla_get_usec_timestamp();
if (end_usecs > start_usecs) {
msecs_200 = (end_usecs - start_usecs)/(1000 * 200);
if (msecs_200 < 15)
ha->hw.mbx_comp_msecs[msecs_200]++;
else if (msecs_200 < 20)
ha->hw.mbx_comp_msecs[15]++;
else {
device_printf(ha->pci_dev, "%s: [%ld, %ld] %ld\n", __func__,
start_usecs, end_usecs, msecs_200);
ha->hw.mbx_comp_msecs[16]++;
}
}
ql_sp_log(ha, 7, 5, fw_mbox[0], fw_mbox[1], fw_mbox[2], fw_mbox[3], fw_mbox[4]);
exit_qla_mbx_cmd: exit_qla_mbx_cmd:
return (ret); return (ret);
} }
@ -1403,7 +1588,8 @@ qla_config_intr_cntxt(qla_host_t *ha, uint32_t start_idx, uint32_t num_intrs,
if (qla_mbx_cmd(ha, (uint32_t *)c_intr, if (qla_mbx_cmd(ha, (uint32_t *)c_intr,
(sizeof (q80_config_intr_t) >> 2), (sizeof (q80_config_intr_t) >> 2),
ha->hw.mbox, (sizeof (q80_config_intr_rsp_t) >> 2), 0)) { ha->hw.mbox, (sizeof (q80_config_intr_rsp_t) >> 2), 0)) {
device_printf(dev, "%s: failed0\n", __func__); device_printf(dev, "%s: %s failed0\n", __func__,
(create ? "create" : "delete"));
return (-1); return (-1);
} }
@ -1412,8 +1598,8 @@ qla_config_intr_cntxt(qla_host_t *ha, uint32_t start_idx, uint32_t num_intrs,
err = Q8_MBX_RSP_STATUS(c_intr_rsp->regcnt_status); err = Q8_MBX_RSP_STATUS(c_intr_rsp->regcnt_status);
if (err) { if (err) {
device_printf(dev, "%s: failed1 [0x%08x, %d]\n", __func__, err, device_printf(dev, "%s: %s failed1 [0x%08x, %d]\n", __func__,
c_intr_rsp->nentries); (create ? "create" : "delete"), err, c_intr_rsp->nentries);
for (i = 0; i < c_intr_rsp->nentries; i++) { for (i = 0; i < c_intr_rsp->nentries; i++) {
device_printf(dev, "%s: [%d]:[0x%x 0x%x 0x%x]\n", device_printf(dev, "%s: [%d]:[0x%x 0x%x 0x%x]\n",
@ -2017,7 +2203,8 @@ ql_get_stats(qla_host_t *ha)
cmd |= ((ha->pci_func & 0x1) << 16); cmd |= ((ha->pci_func & 0x1) << 16);
if (ha->qla_watchdog_pause) if (ha->qla_watchdog_pause || (!(ifp->if_drv_flags & IFF_DRV_RUNNING)) ||
ha->offline)
goto ql_get_stats_exit; goto ql_get_stats_exit;
if (qla_get_hw_stats(ha, cmd, sizeof (q80_get_stats_rsp_t)) == 0) { if (qla_get_hw_stats(ha, cmd, sizeof (q80_get_stats_rsp_t)) == 0) {
@ -2034,7 +2221,8 @@ ql_get_stats(qla_host_t *ha)
// cmd |= Q8_GET_STATS_CMD_CLEAR; // cmd |= Q8_GET_STATS_CMD_CLEAR;
cmd |= (ha->hw.rcv_cntxt_id << 16); cmd |= (ha->hw.rcv_cntxt_id << 16);
if (ha->qla_watchdog_pause) if (ha->qla_watchdog_pause || (!(ifp->if_drv_flags & IFF_DRV_RUNNING)) ||
ha->offline)
goto ql_get_stats_exit; goto ql_get_stats_exit;
if (qla_get_hw_stats(ha, cmd, sizeof (q80_get_stats_rsp_t)) == 0) { if (qla_get_hw_stats(ha, cmd, sizeof (q80_get_stats_rsp_t)) == 0) {
@ -2045,13 +2233,18 @@ ql_get_stats(qla_host_t *ha)
__func__, ha->hw.mbox[0]); __func__, ha->hw.mbox[0]);
} }
if (ha->qla_watchdog_pause) if (ha->qla_watchdog_pause || (!(ifp->if_drv_flags & IFF_DRV_RUNNING)) ||
ha->offline)
goto ql_get_stats_exit; goto ql_get_stats_exit;
/* /*
* Get XMT Statistics * Get XMT Statistics
*/ */
for (i = 0 ; ((i < ha->hw.num_tx_rings) && (!ha->qla_watchdog_pause)); for (i = 0 ; (i < ha->hw.num_tx_rings); i++) {
i++) { if (ha->qla_watchdog_pause ||
(!(ifp->if_drv_flags & IFF_DRV_RUNNING)) ||
ha->offline)
goto ql_get_stats_exit;
cmd = Q8_GET_STATS_CMD_XMT | Q8_GET_STATS_CMD_TYPE_CNTXT; cmd = Q8_GET_STATS_CMD_XMT | Q8_GET_STATS_CMD_TYPE_CNTXT;
// cmd |= Q8_GET_STATS_CMD_CLEAR; // cmd |= Q8_GET_STATS_CMD_CLEAR;
cmd |= (ha->hw.tx_cntxt[i].tx_cntxt_id << 16); cmd |= (ha->hw.tx_cntxt[i].tx_cntxt_id << 16);
@ -2681,7 +2874,8 @@ ql_del_hw_if(qla_host_t *ha)
qla_del_rcv_cntxt(ha); qla_del_rcv_cntxt(ha);
qla_del_xmt_cntxt(ha); if(qla_del_xmt_cntxt(ha))
goto ql_del_hw_if_exit;
if (ha->hw.flags.init_intr_cnxt) { if (ha->hw.flags.init_intr_cnxt) {
for (i = 0; i < ha->hw.num_sds_rings; ) { for (i = 0; i < ha->hw.num_sds_rings; ) {
@ -2690,7 +2884,9 @@ ql_del_hw_if(qla_host_t *ha)
num_msix = Q8_MAX_INTR_VECTORS; num_msix = Q8_MAX_INTR_VECTORS;
else else
num_msix = ha->hw.num_sds_rings - i; num_msix = ha->hw.num_sds_rings - i;
qla_config_intr_cntxt(ha, i, num_msix, 0);
if (qla_config_intr_cntxt(ha, i, num_msix, 0))
break;
i += num_msix; i += num_msix;
} }
@ -2698,6 +2894,7 @@ ql_del_hw_if(qla_host_t *ha)
ha->hw.flags.init_intr_cnxt = 0; ha->hw.flags.init_intr_cnxt = 0;
} }
ql_del_hw_if_exit:
if (ha->hw.enable_soft_lro) { if (ha->hw.enable_soft_lro) {
qla_drain_soft_lro(ha); qla_drain_soft_lro(ha);
qla_free_soft_lro(ha); qla_free_soft_lro(ha);
@ -3330,19 +3527,22 @@ qla_del_xmt_cntxt_i(qla_host_t *ha, uint32_t txr_idx)
return (0); return (0);
} }
static void static int
qla_del_xmt_cntxt(qla_host_t *ha) qla_del_xmt_cntxt(qla_host_t *ha)
{ {
uint32_t i; uint32_t i;
int ret = 0;
if (!ha->hw.flags.init_tx_cnxt) if (!ha->hw.flags.init_tx_cnxt)
return; return (ret);
for (i = 0; i < ha->hw.num_tx_rings; i++) { for (i = 0; i < ha->hw.num_tx_rings; i++) {
if (qla_del_xmt_cntxt_i(ha, i)) if ((ret = qla_del_xmt_cntxt_i(ha, i)) != 0)
break; break;
} }
ha->hw.flags.init_tx_cnxt = 0; ha->hw.flags.init_tx_cnxt = 0;
return (ret);
} }
static int static int
@ -3352,8 +3552,10 @@ qla_init_xmt_cntxt(qla_host_t *ha)
for (i = 0; i < ha->hw.num_tx_rings; i++) { for (i = 0; i < ha->hw.num_tx_rings; i++) {
if (qla_init_xmt_cntxt_i(ha, i) != 0) { if (qla_init_xmt_cntxt_i(ha, i) != 0) {
for (j = 0; j < i; j++) for (j = 0; j < i; j++) {
qla_del_xmt_cntxt_i(ha, j); if (qla_del_xmt_cntxt_i(ha, j))
break;
}
return (-1); return (-1);
} }
} }
@ -3629,21 +3831,22 @@ ql_hw_tx_done_locked(qla_host_t *ha, uint32_t txr_idx)
void void
ql_update_link_state(qla_host_t *ha) ql_update_link_state(qla_host_t *ha)
{ {
uint32_t link_state; uint32_t link_state = 0;
uint32_t prev_link_state; uint32_t prev_link_state;
if (!(ha->ifp->if_drv_flags & IFF_DRV_RUNNING)) {
ha->hw.link_up = 0;
return;
}
link_state = READ_REG32(ha, Q8_LINK_STATE);
prev_link_state = ha->hw.link_up; prev_link_state = ha->hw.link_up;
if (ha->pci_func == 0) if (ha->ifp->if_drv_flags & IFF_DRV_RUNNING) {
ha->hw.link_up = (((link_state & 0xF) == 1)? 1 : 0); link_state = READ_REG32(ha, Q8_LINK_STATE);
else
ha->hw.link_up = ((((link_state >> 4)& 0xF) == 1)? 1 : 0); if (ha->pci_func == 0) {
link_state = (((link_state & 0xF) == 1)? 1 : 0);
} else {
link_state = ((((link_state >> 4)& 0xF) == 1)? 1 : 0);
}
}
atomic_store_rel_8(&ha->hw.link_up, (uint8_t)link_state);
if (prev_link_state != ha->hw.link_up) { if (prev_link_state != ha->hw.link_up) {
if (ha->hw.link_up) { if (ha->hw.link_up) {
@ -3671,8 +3874,14 @@ ql_hw_check_health(qla_host_t *ha)
if (((val & 0xFFFF) == 2) || ((val & 0xFFFF) == 3) || if (((val & 0xFFFF) == 2) || ((val & 0xFFFF) == 3) ||
(QL_ERR_INJECT(ha, INJCT_TEMPERATURE_FAILURE))) { (QL_ERR_INJECT(ha, INJCT_TEMPERATURE_FAILURE))) {
device_printf(ha->pci_dev, "%s: Temperature Alert [0x%08x]\n", device_printf(ha->pci_dev, "%s: Temperature Alert"
__func__, val); " at ts_usecs %ld ts_reg = 0x%08x\n",
__func__, qla_get_usec_timestamp(), val);
if (ha->hw.sp_log_stop_events & Q8_SP_LOG_STOP_TEMP_FAILURE)
ha->hw.sp_log_stop = -1;
QL_INITIATE_RECOVERY(ha);
return -1; return -1;
} }
@ -3693,9 +3902,25 @@ ql_hw_check_health(qla_host_t *ha)
__func__, val); __func__, val);
if (ha->hw.hbeat_failure < 2) /* we ignore the first failure */ if (ha->hw.hbeat_failure < 2) /* we ignore the first failure */
return 0; return 0;
else else {
device_printf(ha->pci_dev, "%s: Heartbeat Failue [0x%08x]\n", uint32_t peg_halt_status1;
__func__, val); uint32_t peg_halt_status2;
peg_halt_status1 = READ_REG32(ha, Q8_PEG_HALT_STATUS1);
peg_halt_status2 = READ_REG32(ha, Q8_PEG_HALT_STATUS2);
device_printf(ha->pci_dev,
"%s: Heartbeat Failue at ts_usecs = %ld "
"fw_heart_beat = 0x%08x "
"peg_halt_status1 = 0x%08x "
"peg_halt_status2 = 0x%08x\n",
__func__, qla_get_usec_timestamp(), val,
peg_halt_status1, peg_halt_status2);
if (ha->hw.sp_log_stop_events & Q8_SP_LOG_STOP_HBEAT_FAILURE)
ha->hw.sp_log_stop = -1;
}
QL_INITIATE_RECOVERY(ha);
return -1; return -1;
} }
@ -4431,8 +4656,8 @@ ql_minidump(qla_host_t *ha)
if (ha->hw.mdump_done) if (ha->hw.mdump_done)
return; return;
ha->hw.mdump_usec_ts = qla_get_usec_timestamp();
ha->hw.mdump_start_seq_index = ql_stop_sequence(ha); ha->hw.mdump_start_seq_index = ql_stop_sequence(ha);
bzero(ha->hw.mdump_buffer, ha->hw.mdump_buffer_size); bzero(ha->hw.mdump_buffer, ha->hw.mdump_buffer_size);
bzero(ha->hw.mdump_template, ha->hw.mdump_template_size); bzero(ha->hw.mdump_template, ha->hw.mdump_template_size);

View File

@ -1602,26 +1602,26 @@ typedef struct _qla_hw {
uint32_t uint32_t
unicast_mac :1, unicast_mac :1,
bcast_mac :1, bcast_mac :1,
loopback_mode :2,
init_tx_cnxt :1, init_tx_cnxt :1,
init_rx_cnxt :1, init_rx_cnxt :1,
init_intr_cnxt :1, init_intr_cnxt :1,
fduplex :1,
autoneg :1,
fdt_valid :1; fdt_valid :1;
} flags; } flags;
uint16_t link_speed; volatile uint16_t link_speed;
uint16_t cable_length; volatile uint16_t cable_length;
uint32_t cable_oui; volatile uint32_t cable_oui;
uint8_t link_up; volatile uint8_t link_up;
uint8_t module_type; volatile uint8_t module_type;
uint8_t link_faults; volatile uint8_t link_faults;
volatile uint8_t loopback_mode;
volatile uint8_t fduplex;
volatile uint8_t autoneg;
uint8_t mac_rcv_mode; volatile uint8_t mac_rcv_mode;
uint32_t max_mtu; volatile uint32_t max_mtu;
uint8_t mac_addr[ETHER_ADDR_LEN]; uint8_t mac_addr[ETHER_ADDR_LEN];
@ -1705,9 +1705,25 @@ typedef struct _qla_hw {
uint32_t mdump_buffer_size; uint32_t mdump_buffer_size;
void *mdump_template; void *mdump_template;
uint32_t mdump_template_size; uint32_t mdump_template_size;
uint64_t mdump_usec_ts;
#define Q8_MBX_COMP_MSECS (19)
uint64_t mbx_comp_msecs[Q8_MBX_COMP_MSECS];
/* driver state related */ /* driver state related */
void *drvr_state; void *drvr_state;
/* slow path trace */
uint32_t sp_log_stop_events;
#define Q8_SP_LOG_STOP_HBEAT_FAILURE 0x001
#define Q8_SP_LOG_STOP_TEMP_FAILURE 0x002
#define Q8_SP_LOG_STOP_HW_INIT_FAILURE 0x004
#define Q8_SP_LOG_STOP_IF_START_FAILURE 0x008
#define Q8_SP_LOG_STOP_ERR_RECOVERY_FAILURE 0x010
uint32_t sp_log_stop;
uint32_t sp_log_index;
uint32_t sp_log_num_entries;
void *sp_log;
} qla_hw_t; } qla_hw_t;
#define QL_UPDATE_RDS_PRODUCER_INDEX(ha, prod_reg, val) \ #define QL_UPDATE_RDS_PRODUCER_INDEX(ha, prod_reg, val) \

View File

@ -168,7 +168,7 @@ qla_lock(qla_host_t *ha, const char *str, uint32_t timeout_ms,
while (1) { while (1) {
mtx_lock(&ha->hw_lock); mtx_lock(&ha->hw_lock);
if (ha->qla_detach_active) { if (ha->qla_detach_active || ha->offline) {
mtx_unlock(&ha->hw_lock); mtx_unlock(&ha->hw_lock);
break; break;
} }
@ -193,7 +193,10 @@ qla_lock(qla_host_t *ha, const char *str, uint32_t timeout_ms,
} }
} }
//device_printf(ha->pci_dev, "%s: %s ret = %d\n", __func__, str,ret); // if (!ha->enable_error_recovery)
// device_printf(ha->pci_dev, "%s: %s ret = %d\n", __func__,
// str,ret);
return (ret); return (ret);
} }
@ -204,7 +207,9 @@ qla_unlock(qla_host_t *ha, const char *str)
ha->hw_lock_held = 0; ha->hw_lock_held = 0;
ha->qla_unlock = str; ha->qla_unlock = str;
mtx_unlock(&ha->hw_lock); mtx_unlock(&ha->hw_lock);
//device_printf(ha->pci_dev, "%s: %s\n", __func__, str);
// if (!ha->enable_error_recovery)
// device_printf(ha->pci_dev, "%s: %s\n", __func__, str);
return; return;
} }

View File

@ -44,6 +44,7 @@ __FBSDID("$FreeBSD$");
#include "ql_ver.h" #include "ql_ver.h"
#include "ql_dbg.h" #include "ql_dbg.h"
static int ql_slowpath_log(qla_host_t *ha, qla_sp_log_t *log);
static int ql_drvr_state(qla_host_t *ha, qla_driver_state_t *drvr_state); static int ql_drvr_state(qla_host_t *ha, qla_driver_state_t *drvr_state);
static uint32_t ql_drvr_state_size(qla_host_t *ha); static uint32_t ql_drvr_state_size(qla_host_t *ha);
static int ql_eioctl(struct cdev *dev, u_long cmd, caddr_t data, int fflag, static int ql_eioctl(struct cdev *dev, u_long cmd, caddr_t data, int fflag,
@ -226,6 +227,7 @@ ql_eioctl(struct cdev *dev, u_long cmd, caddr_t data, int fflag,
case QLA_RD_FW_DUMP: case QLA_RD_FW_DUMP:
if (ha->hw.mdump_init == 0) { if (ha->hw.mdump_init == 0) {
device_printf(pci_dev, "%s: minidump not initialized\n", __func__);
rval = EINVAL; rval = EINVAL;
break; break;
} }
@ -235,45 +237,85 @@ ql_eioctl(struct cdev *dev, u_long cmd, caddr_t data, int fflag,
if ((fw_dump->minidump == NULL) || if ((fw_dump->minidump == NULL) ||
(fw_dump->minidump_size != (ha->hw.mdump_buffer_size + (fw_dump->minidump_size != (ha->hw.mdump_buffer_size +
ha->hw.mdump_template_size))) { ha->hw.mdump_template_size))) {
device_printf(pci_dev,
"%s: minidump buffer [%p] size = [%d, %d] invalid\n", __func__,
fw_dump->minidump, fw_dump->minidump_size,
(ha->hw.mdump_buffer_size + ha->hw.mdump_template_size));
rval = EINVAL; rval = EINVAL;
break; break;
} }
if (QLA_LOCK(ha, __func__, QLA_LOCK_DEFAULT_MS_TIMEOUT, 0) == 0) { if ((ha->pci_func & 0x1)) {
if (!ha->hw.mdump_done) device_printf(pci_dev, "%s: mindump allowed only on Port0\n", __func__);
ha->qla_initiate_recovery = 1;
QLA_UNLOCK(ha, __func__);
} else {
rval = ENXIO; rval = ENXIO;
break; break;
} }
fw_dump->saved = 1;
if (ha->offline) {
if (ha->enable_minidump)
ql_minidump(ha);
fw_dump->saved = 0;
fw_dump->usec_ts = ha->hw.mdump_usec_ts;
if (!ha->hw.mdump_done) {
device_printf(pci_dev,
"%s: port offline minidump failed\n", __func__);
rval = ENXIO;
break;
}
} else {
if (QLA_LOCK(ha, __func__, QLA_LOCK_DEFAULT_MS_TIMEOUT, 0) == 0) {
if (!ha->hw.mdump_done) {
fw_dump->saved = 0;
QL_INITIATE_RECOVERY(ha);
device_printf(pci_dev, "%s: recovery initiated "
" to trigger minidump\n",
__func__);
}
QLA_UNLOCK(ha, __func__);
} else {
device_printf(pci_dev, "%s: QLA_LOCK() failed0\n", __func__);
rval = ENXIO;
break;
}
#define QLNX_DUMP_WAIT_SECS 30 #define QLNX_DUMP_WAIT_SECS 30
count = QLNX_DUMP_WAIT_SECS * 1000; count = QLNX_DUMP_WAIT_SECS * 1000;
while (count) { while (count) {
if (ha->hw.mdump_done) if (ha->hw.mdump_done)
break;
qla_mdelay(__func__, 100);
count -= 100;
}
if (!ha->hw.mdump_done) {
device_printf(pci_dev,
"%s: port not offline minidump failed\n", __func__);
rval = ENXIO;
break; break;
qla_mdelay(__func__, 100); }
count -= 100; fw_dump->usec_ts = ha->hw.mdump_usec_ts;
}
if (!ha->hw.mdump_done) {
rval = ENXIO;
break;
}
if (QLA_LOCK(ha, __func__, QLA_LOCK_DEFAULT_MS_TIMEOUT, 0) == 0) { if (QLA_LOCK(ha, __func__, QLA_LOCK_DEFAULT_MS_TIMEOUT, 0) == 0) {
ha->hw.mdump_done = 0; ha->hw.mdump_done = 0;
QLA_UNLOCK(ha, __func__); QLA_UNLOCK(ha, __func__);
} else { } else {
rval = ENXIO; device_printf(pci_dev, "%s: QLA_LOCK() failed1\n", __func__);
break; rval = ENXIO;
break;
}
} }
if ((rval = copyout(ha->hw.mdump_template, if ((rval = copyout(ha->hw.mdump_template,
fw_dump->minidump, ha->hw.mdump_template_size))) { fw_dump->minidump, ha->hw.mdump_template_size))) {
device_printf(pci_dev, "%s: template copyout failed\n", __func__);
rval = ENXIO; rval = ENXIO;
break; break;
} }
@ -281,14 +323,20 @@ ql_eioctl(struct cdev *dev, u_long cmd, caddr_t data, int fflag,
if ((rval = copyout(ha->hw.mdump_buffer, if ((rval = copyout(ha->hw.mdump_buffer,
((uint8_t *)fw_dump->minidump + ((uint8_t *)fw_dump->minidump +
ha->hw.mdump_template_size), ha->hw.mdump_template_size),
ha->hw.mdump_buffer_size))) ha->hw.mdump_buffer_size))) {
device_printf(pci_dev, "%s: minidump copyout failed\n", __func__);
rval = ENXIO; rval = ENXIO;
}
break; break;
case QLA_RD_DRVR_STATE: case QLA_RD_DRVR_STATE:
rval = ql_drvr_state(ha, (qla_driver_state_t *)data); rval = ql_drvr_state(ha, (qla_driver_state_t *)data);
break; break;
case QLA_RD_SLOWPATH_LOG:
rval = ql_slowpath_log(ha, (qla_sp_log_t *)data);
break;
case QLA_RD_PCI_IDS: case QLA_RD_PCI_IDS:
pci_ids = (qla_rd_pci_ids_t *)data; pci_ids = (qla_rd_pci_ids_t *)data;
pci_ids->ven_id = pci_get_vendor(pci_dev); pci_ids->ven_id = pci_get_vendor(pci_dev);
@ -306,12 +354,12 @@ ql_eioctl(struct cdev *dev, u_long cmd, caddr_t data, int fflag,
} }
static int static int
ql_drvr_state(qla_host_t *ha, qla_driver_state_t *state) ql_drvr_state(qla_host_t *ha, qla_driver_state_t *state)
{ {
int rval = 0; int rval = 0;
uint32_t drvr_state_size; uint32_t drvr_state_size;
qla_drvr_state_hdr_t *hdr;
drvr_state_size = ql_drvr_state_size(ha); drvr_state_size = ql_drvr_state_size(ha);
@ -326,10 +374,7 @@ ql_drvr_state(qla_host_t *ha, qla_driver_state_t *state)
if (ha->hw.drvr_state == NULL) if (ha->hw.drvr_state == NULL)
return (ENOMEM); return (ENOMEM);
hdr = ha->hw.drvr_state; ql_capture_drvr_state(ha);
if (!hdr->drvr_version_major)
ql_capture_drvr_state(ha);
rval = copyout(ha->hw.drvr_state, state->buffer, drvr_state_size); rval = copyout(ha->hw.drvr_state, state->buffer, drvr_state_size);
@ -418,21 +463,25 @@ ql_capture_drvr_state(qla_host_t *ha)
{ {
uint8_t *state_buffer; uint8_t *state_buffer;
uint8_t *ptr; uint8_t *ptr;
uint32_t drvr_state_size;
qla_drvr_state_hdr_t *hdr; qla_drvr_state_hdr_t *hdr;
uint32_t size; uint32_t size;
int i; int i;
drvr_state_size = ql_drvr_state_size(ha);
state_buffer = ha->hw.drvr_state; state_buffer = ha->hw.drvr_state;
if (state_buffer == NULL) if (state_buffer == NULL)
return; return;
bzero(state_buffer, drvr_state_size);
hdr = (qla_drvr_state_hdr_t *)state_buffer; hdr = (qla_drvr_state_hdr_t *)state_buffer;
hdr->saved = 0;
if (hdr->drvr_version_major) {
hdr->saved = 1;
return;
}
hdr->usec_ts = qla_get_usec_timestamp();
hdr->drvr_version_major = QLA_VERSION_MAJOR; hdr->drvr_version_major = QLA_VERSION_MAJOR;
hdr->drvr_version_minor = QLA_VERSION_MINOR; hdr->drvr_version_minor = QLA_VERSION_MINOR;
@ -514,6 +563,9 @@ ql_alloc_drvr_state_buffer(qla_host_t *ha)
ha->hw.drvr_state = malloc(drvr_state_size, M_QLA83XXBUF, M_NOWAIT); ha->hw.drvr_state = malloc(drvr_state_size, M_QLA83XXBUF, M_NOWAIT);
if (ha->hw.drvr_state != NULL)
bzero(ha->hw.drvr_state, drvr_state_size);
return; return;
} }
@ -525,3 +577,93 @@ ql_free_drvr_state_buffer(qla_host_t *ha)
return; return;
} }
void
ql_sp_log(qla_host_t *ha, uint16_t fmtstr_idx, uint16_t num_params,
uint32_t param0, uint32_t param1, uint32_t param2, uint32_t param3,
uint32_t param4)
{
qla_sp_log_entry_t *sp_e, *sp_log;
if (((sp_log = ha->hw.sp_log) == NULL) || ha->hw.sp_log_stop)
return;
mtx_lock(&ha->sp_log_lock);
sp_e = &sp_log[ha->hw.sp_log_index];
bzero(sp_e, sizeof (qla_sp_log_entry_t));
sp_e->fmtstr_idx = fmtstr_idx;
sp_e->num_params = num_params;
sp_e->usec_ts = qla_get_usec_timestamp();
sp_e->params[0] = param0;
sp_e->params[1] = param1;
sp_e->params[2] = param2;
sp_e->params[3] = param3;
sp_e->params[4] = param4;
ha->hw.sp_log_index = (ha->hw.sp_log_index + 1) & (NUM_LOG_ENTRIES - 1);
if (ha->hw.sp_log_num_entries < NUM_LOG_ENTRIES)
ha->hw.sp_log_num_entries++;
mtx_unlock(&ha->sp_log_lock);
return;
}
void
ql_alloc_sp_log_buffer(qla_host_t *ha)
{
uint32_t size;
size = (sizeof(qla_sp_log_entry_t)) * NUM_LOG_ENTRIES;
ha->hw.sp_log = malloc(size, M_QLA83XXBUF, M_NOWAIT);
if (ha->hw.sp_log != NULL)
bzero(ha->hw.sp_log, size);
ha->hw.sp_log_index = 0;
ha->hw.sp_log_num_entries = 0;
return;
}
void
ql_free_sp_log_buffer(qla_host_t *ha)
{
if (ha->hw.sp_log != NULL)
free(ha->hw.sp_log, M_QLA83XXBUF);
return;
}
static int
ql_slowpath_log(qla_host_t *ha, qla_sp_log_t *log)
{
int rval = 0;
uint32_t size;
if ((ha->hw.sp_log == NULL) || (log->buffer == NULL))
return (EINVAL);
size = (sizeof(qla_sp_log_entry_t) * NUM_LOG_ENTRIES);
mtx_lock(&ha->sp_log_lock);
rval = copyout(ha->hw.sp_log, log->buffer, size);
if (!rval) {
log->next_idx = ha->hw.sp_log_index;
log->num_entries = ha->hw.sp_log_num_entries;
}
device_printf(ha->pci_dev,
"%s: exit [rval = %d][%p, next_idx = %d, %d entries, %d bytes]\n",
__func__, rval, log->buffer, log->next_idx, log->num_entries, size);
mtx_unlock(&ha->sp_log_lock);
return (rval);
}

View File

@ -75,6 +75,17 @@ struct qla_rd_pci_ids {
}; };
typedef struct qla_rd_pci_ids qla_rd_pci_ids_t; typedef struct qla_rd_pci_ids qla_rd_pci_ids_t;
#define NUM_LOG_ENTRY_PARAMS 5
#define NUM_LOG_ENTRIES 512
struct qla_sp_log_entry {
uint32_t fmtstr_idx;
uint32_t num_params;
uint64_t usec_ts;
uint32_t params[NUM_LOG_ENTRY_PARAMS];
};
typedef struct qla_sp_log_entry qla_sp_log_entry_t;
/* /*
* structure encapsulating the value to read/write from/to offchip (MS) memory * structure encapsulating the value to read/write from/to offchip (MS) memory
*/ */
@ -90,6 +101,8 @@ typedef struct qla_offchip_mem_val qla_offchip_mem_val_t;
struct qla_rd_fw_dump { struct qla_rd_fw_dump {
uint16_t pci_func; uint16_t pci_func;
uint16_t saved;
uint64_t usec_ts;
uint32_t minidump_size; uint32_t minidump_size;
void *minidump; void *minidump;
}; };
@ -124,6 +137,8 @@ struct qla_drvr_state_hdr {
uint32_t drvr_version_build; uint32_t drvr_version_build;
uint8_t mac_addr[ETHER_ADDR_LEN]; uint8_t mac_addr[ETHER_ADDR_LEN];
uint16_t saved;
uint64_t usec_ts;
uint16_t link_speed; uint16_t link_speed;
uint16_t cable_length; uint16_t cable_length;
uint32_t cable_oui; uint32_t cable_oui;
@ -161,6 +176,13 @@ struct qla_driver_state {
}; };
typedef struct qla_driver_state qla_driver_state_t; typedef struct qla_driver_state qla_driver_state_t;
struct qla_sp_log {
uint32_t next_idx; /* index of next entry in slowpath trace log */
uint32_t num_entries; /* number of entries in slowpath trace log */
void *buffer;
};
typedef struct qla_sp_log qla_sp_log_t;
/* /*
* Read/Write Register * Read/Write Register
*/ */
@ -206,5 +228,58 @@ typedef struct qla_driver_state qla_driver_state_t;
*/ */
#define QLA_RD_DRVR_STATE _IOWR('q', 9, qla_driver_state_t) #define QLA_RD_DRVR_STATE _IOWR('q', 9, qla_driver_state_t)
/*
* Read Slowpath Log
*/
#define QLA_RD_SLOWPATH_LOG _IOWR('q', 10, qla_sp_log_t)
/*
* Format Strings For Slowpath Trace Logs
*/
#define SP_TLOG_FMT_STR_0 \
"qla_mbx_cmd [%ld]: enter no_pause = %d [0x%08x 0x%08x 0x%08x 0x%08x]\n"
#define SP_TLOG_FMT_STR_1 \
"qla_mbx_cmd [%ld]: offline = 0x%08x qla_initiate_recovery = 0x%08x exit1\n"
#define SP_TLOG_FMT_STR_2 \
"qla_mbx_cmd [%ld]: qla_initiate_recovery = 0x%08x exit2\n"
#define SP_TLOG_FMT_STR_3 \
"qla_mbx_cmd [%ld]: timeout exit3 [host_mbx_cntrl = 0x%08x]\n"
#define SP_TLOG_FMT_STR_4 \
"qla_mbx_cmd [%ld]: qla_initiate_recovery = 0x%08x exit4\n"
#define SP_TLOG_FMT_STR_5 \
"qla_mbx_cmd [%ld]: timeout exit5 [fw_mbx_cntrl = 0x%08x]\n"
#define SP_TLOG_FMT_STR_6 \
"qla_mbx_cmd [%ld]: qla_initiate_recovery = 0x%08x exit6\n"
#define SP_TLOG_FMT_STR_7 \
"qla_mbx_cmd [%ld]: exit [0x%08x 0x%08x 0x%08x 0x%08x 0x%08x]\n"
#define SP_TLOG_FMT_STR_8 \
"qla_ioctl [%ld]: SIOCSIFADDR if_drv_flags = 0x%08x [0x%08x] ipv4 = 0x%08x\n"
#define SP_TLOG_FMT_STR_9 \
"qla_ioctl [%ld]: SIOCSIFMTU if_drv_flags = 0x%08x [0x%08x] max_frame_size = 0x%08x if_mtu = 0x%08x\n"
#define SP_TLOG_FMT_STR_10 \
"qla_ioctl [%ld]: SIOCSIFFLAGS if_drv_flags = 0x%08x [0x%08x] ha->if_flags = 0x%08x ifp->if_flags = 0x%08x\n"
#define SP_TLOG_FMT_STR_11 \
"qla_ioctl [%ld]: SIOCSIFCAP if_drv_flags = 0x%08x [0x%08x] mask = 0x%08x ifp->if_capenable = 0x%08x\n"
#define SP_TLOG_FMT_STR_12 \
"qla_set_multi [%ld]: if_drv_flags = 0x%08x [0x%08x] add_multi = 0x%08x mcnt = 0x%08x\n"
#define SP_TLOG_FMT_STR_13 \
"qla_stop [%ld]: \n"
#define SP_TLOG_FMT_STR_14 \
"qla_init_locked [%ld]: \n"
#endif /* #ifndef _QL_IOCTL_H_ */ #endif /* #ifndef _QL_IOCTL_H_ */

View File

@ -51,7 +51,7 @@ static void
qla_rcv_error(qla_host_t *ha) qla_rcv_error(qla_host_t *ha)
{ {
ha->stop_rcv = 1; ha->stop_rcv = 1;
ha->qla_initiate_recovery = 1; QL_INITIATE_RECOVERY(ha);
} }
@ -389,7 +389,7 @@ qla_rcv_cont_sds(qla_host_t *ha, uint32_t sds_idx, uint32_t comp_idx,
opcode = Q8_STAT_DESC_OPCODE((sdesc->data[1])); opcode = Q8_STAT_DESC_OPCODE((sdesc->data[1]));
if (!opcode) { if (!opcode || QL_ERR_INJECT(ha, INJCT_INV_CONT_OPCODE)) {
device_printf(ha->pci_dev, "%s: opcode=0 %p %p\n", device_printf(ha->pci_dev, "%s: opcode=0 %p %p\n",
__func__, (void *)sdesc->data[0], __func__, (void *)sdesc->data[0],
(void *)sdesc->data[1]); (void *)sdesc->data[1]);
@ -558,8 +558,9 @@ ql_rcv_isr(qla_host_t *ha, uint32_t sds_idx, uint32_t count)
sdesc0 = (q80_stat_desc_t *) sdesc0 = (q80_stat_desc_t *)
&hw->sds[sds_idx].sds_ring_base[c_idx]; &hw->sds[sds_idx].sds_ring_base[c_idx];
if (Q8_STAT_DESC_OPCODE((sdesc0->data[1])) != if ((Q8_STAT_DESC_OPCODE((sdesc0->data[1])) !=
Q8_STAT_DESC_OPCODE_CONT) { Q8_STAT_DESC_OPCODE_CONT) ||
QL_ERR_INJECT(ha, INJCT_SGL_RCV_INV_DESC_COUNT)) {
desc_count = 0; desc_count = 0;
break; break;
} }
@ -620,8 +621,9 @@ ql_rcv_isr(qla_host_t *ha, uint32_t sds_idx, uint32_t count)
sdesc0 = (q80_stat_desc_t *) sdesc0 = (q80_stat_desc_t *)
&hw->sds[sds_idx].sds_ring_base[c_idx]; &hw->sds[sds_idx].sds_ring_base[c_idx];
if (Q8_STAT_DESC_OPCODE((sdesc0->data[1])) != if ((Q8_STAT_DESC_OPCODE((sdesc0->data[1])) !=
Q8_STAT_DESC_OPCODE_CONT) { Q8_STAT_DESC_OPCODE_CONT) ||
QL_ERR_INJECT(ha, INJCT_SGL_LRO_INV_DESC_COUNT)) {
desc_count = 0; desc_count = 0;
break; break;
} }
@ -822,7 +824,13 @@ ql_mbx_isr(void *arg)
data = READ_REG32(ha, (Q8_FW_MBOX0 + 12)); data = READ_REG32(ha, (Q8_FW_MBOX0 + 12));
prev_link_state = ha->hw.link_up; prev_link_state = ha->hw.link_up;
ha->hw.link_up = (((data & 0xFF) == 0) ? 0 : 1);
data = (((data & 0xFF) == 0) ? 0 : 1);
atomic_store_rel_8(&ha->hw.link_up, (uint8_t)data);
device_printf(ha->pci_dev,
"%s: AEN[0x8001] data = 0x%08x, prev_link_state = 0x%08x\n",
__func__, data, prev_link_state);
if (prev_link_state != ha->hw.link_up) { if (prev_link_state != ha->hw.link_up) {
if (ha->hw.link_up) if (ha->hw.link_up)
@ -833,17 +841,18 @@ ql_mbx_isr(void *arg)
ha->hw.module_type = ((data >> 8) & 0xFF); ha->hw.module_type = ((data >> 8) & 0xFF);
ha->hw.flags.fduplex = (((data & 0xFF0000) == 0) ? 0 : 1); ha->hw.fduplex = (((data & 0xFF0000) == 0) ? 0 : 1);
ha->hw.flags.autoneg = (((data & 0xFF000000) == 0) ? 0 : 1); ha->hw.autoneg = (((data & 0xFF000000) == 0) ? 0 : 1);
data = READ_REG32(ha, (Q8_FW_MBOX0 + 16)); data = READ_REG32(ha, (Q8_FW_MBOX0 + 16));
ha->hw.flags.loopback_mode = data & 0x03; ha->hw.loopback_mode = data & 0x03;
ha->hw.link_faults = (data >> 3) & 0xFF; ha->hw.link_faults = (data >> 3) & 0xFF;
break; break;
case 0x8100: case 0x8100:
device_printf(ha->pci_dev, "%s: AEN[0x%08x]\n", __func__, data);
ha->hw.imd_compl=1; ha->hw.imd_compl=1;
break; break;
@ -854,6 +863,9 @@ ql_mbx_isr(void *arg)
ha->hw.aen_mb2 = READ_REG32(ha, (Q8_FW_MBOX0 + 8)); ha->hw.aen_mb2 = READ_REG32(ha, (Q8_FW_MBOX0 + 8));
ha->hw.aen_mb3 = READ_REG32(ha, (Q8_FW_MBOX0 + 12)); ha->hw.aen_mb3 = READ_REG32(ha, (Q8_FW_MBOX0 + 12));
ha->hw.aen_mb4 = READ_REG32(ha, (Q8_FW_MBOX0 + 16)); ha->hw.aen_mb4 = READ_REG32(ha, (Q8_FW_MBOX0 + 16));
device_printf(ha->pci_dev, "%s: AEN[0x%08x 0x%08x 0x%08x 0%08x 0x%08x]\n",
__func__, data, ha->hw.aen_mb1, ha->hw.aen_mb2,
ha->hw.aen_mb3, ha->hw.aen_mb4);
break; break;
case 0x8110: case 0x8110:

View File

@ -79,7 +79,7 @@ ql_rdwr_indreg32(qla_host_t *ha, uint32_t addr, uint32_t *val, uint32_t rd)
if (!count || QL_ERR_INJECT(ha, INJCT_RDWR_INDREG_FAILURE)) { if (!count || QL_ERR_INJECT(ha, INJCT_RDWR_INDREG_FAILURE)) {
device_printf(ha->pci_dev, "%s: [0x%08x, 0x%08x, %d] failed\n", device_printf(ha->pci_dev, "%s: [0x%08x, 0x%08x, %d] failed\n",
__func__, addr, *val, rd); __func__, addr, *val, rd);
ha->qla_initiate_recovery = 1; QL_INITIATE_RECOVERY(ha);
return -1; return -1;
} }
@ -212,7 +212,7 @@ ql_rdwr_offchip_mem(qla_host_t *ha, uint64_t addr, q80_offchip_mem_val_t *val,
(uint32_t)(addr), val->data_lo, val->data_hi, val->data_ulo, (uint32_t)(addr), val->data_lo, val->data_hi, val->data_ulo,
val->data_uhi, rd, step); val->data_uhi, rd, step);
ha->qla_initiate_recovery = 1; QL_INITIATE_RECOVERY(ha);
return (-1); return (-1);
} }
@ -775,12 +775,13 @@ ql_init_hw(qla_host_t *ha)
} }
qla_mdelay(__func__, 100); qla_mdelay(__func__, 100);
} }
return (-1); ret = -1;
goto ql_init_hw_exit;
} }
val = READ_REG32(ha, Q8_CMDPEG_STATE); val = READ_REG32(ha, Q8_CMDPEG_STATE);
if (!cold || (val != 0xFF01)) { if (!cold || (val != 0xFF01) || ha->qla_initiate_recovery) {
ret = qla_init_from_flash(ha); ret = qla_init_from_flash(ha);
qla_mdelay(__func__, 100); qla_mdelay(__func__, 100);
} }
@ -796,6 +797,13 @@ ql_init_hw(qla_host_t *ha)
ha->hw.flags.fdt_valid = 1; ha->hw.flags.fdt_valid = 1;
} }
ql_init_hw_exit:
if (ret) {
if (ha->hw.sp_log_stop_events & Q8_SP_LOG_STOP_HW_INIT_FAILURE)
ha->hw.sp_log_stop = -1;
}
return (ret); return (ret);
} }
@ -1282,6 +1290,7 @@ qla_ld_fw_init(qla_host_t *ha)
hdr = (q8_tmplt_hdr_t *)ql83xx_resetseq; hdr = (q8_tmplt_hdr_t *)ql83xx_resetseq;
device_printf(ha->pci_dev, "%s: reset sequence\n", __func__);
if (qla_tmplt_16bit_checksum(ha, (uint16_t *)ql83xx_resetseq, if (qla_tmplt_16bit_checksum(ha, (uint16_t *)ql83xx_resetseq,
(uint32_t)hdr->size)) { (uint32_t)hdr->size)) {
device_printf(ha->pci_dev, "%s: reset seq checksum failed\n", device_printf(ha->pci_dev, "%s: reset seq checksum failed\n",
@ -1292,7 +1301,7 @@ qla_ld_fw_init(qla_host_t *ha)
buf = ql83xx_resetseq + hdr->stop_seq_off; buf = ql83xx_resetseq + hdr->stop_seq_off;
// device_printf(ha->pci_dev, "%s: stop sequence\n", __func__); device_printf(ha->pci_dev, "%s: stop sequence\n", __func__);
if (qla_tmplt_execute(ha, buf, index , &end_idx, hdr->nentries)) { if (qla_tmplt_execute(ha, buf, index , &end_idx, hdr->nentries)) {
device_printf(ha->pci_dev, "%s: stop seq failed\n", __func__); device_printf(ha->pci_dev, "%s: stop seq failed\n", __func__);
return -1; return -1;
@ -1302,7 +1311,7 @@ qla_ld_fw_init(qla_host_t *ha)
buf = ql83xx_resetseq + hdr->init_seq_off; buf = ql83xx_resetseq + hdr->init_seq_off;
// device_printf(ha->pci_dev, "%s: init sequence\n", __func__); device_printf(ha->pci_dev, "%s: init sequence\n", __func__);
if (qla_tmplt_execute(ha, buf, index , &end_idx, hdr->nentries)) { if (qla_tmplt_execute(ha, buf, index , &end_idx, hdr->nentries)) {
device_printf(ha->pci_dev, "%s: init seq failed\n", __func__); device_printf(ha->pci_dev, "%s: init seq failed\n", __func__);
return -1; return -1;
@ -1324,7 +1333,7 @@ qla_ld_fw_init(qla_host_t *ha)
index = end_idx; index = end_idx;
buf = ql83xx_resetseq + hdr->start_seq_off; buf = ql83xx_resetseq + hdr->start_seq_off;
// device_printf(ha->pci_dev, "%s: start sequence\n", __func__); device_printf(ha->pci_dev, "%s: start sequence\n", __func__);
if (qla_tmplt_execute(ha, buf, index , &end_idx, hdr->nentries)) { if (qla_tmplt_execute(ha, buf, index , &end_idx, hdr->nentries)) {
device_printf(ha->pci_dev, "%s: init seq failed\n", __func__); device_printf(ha->pci_dev, "%s: init seq failed\n", __func__);
return -1; return -1;

View File

@ -192,7 +192,31 @@ qla_add_sysctls(qla_host_t *ha)
SYSCTL_CHILDREN(device_get_sysctl_tree(dev)), SYSCTL_CHILDREN(device_get_sysctl_tree(dev)),
OID_AUTO, "enable_minidump", CTLFLAG_RW, OID_AUTO, "enable_minidump", CTLFLAG_RW,
&ha->enable_minidump, ha->enable_minidump, &ha->enable_minidump, ha->enable_minidump,
"Minidump retrival is enabled only when this is set"); "Minidump retrival prior to error recovery "
"is enabled only when this is set");
ha->enable_driverstate_dump = 1;
SYSCTL_ADD_UINT(device_get_sysctl_ctx(dev),
SYSCTL_CHILDREN(device_get_sysctl_tree(dev)),
OID_AUTO, "enable_driverstate_dump", CTLFLAG_RW,
&ha->enable_driverstate_dump, ha->enable_driverstate_dump,
"Driver State retrival prior to error recovery "
"is enabled only when this is set");
ha->enable_error_recovery = 1;
SYSCTL_ADD_UINT(device_get_sysctl_ctx(dev),
SYSCTL_CHILDREN(device_get_sysctl_tree(dev)),
OID_AUTO, "enable_error_recovery", CTLFLAG_RW,
&ha->enable_error_recovery, ha->enable_error_recovery,
"when set error recovery is enabled on fatal errors "
"otherwise the port is turned offline");
ha->ms_delay_after_init = 1000;
SYSCTL_ADD_UINT(device_get_sysctl_ctx(dev),
SYSCTL_CHILDREN(device_get_sysctl_tree(dev)),
OID_AUTO, "ms_delay_after_init", CTLFLAG_RW,
&ha->ms_delay_after_init, ha->ms_delay_after_init,
"millisecond delay after hw_init");
ha->std_replenish = QL_STD_REPLENISH_THRES; ha->std_replenish = QL_STD_REPLENISH_THRES;
SYSCTL_ADD_UINT(device_get_sysctl_ctx(dev), SYSCTL_ADD_UINT(device_get_sysctl_ctx(dev),
@ -251,49 +275,47 @@ qla_watchdog(void *arg)
ha->qla_watchdog_exited = 0; ha->qla_watchdog_exited = 0;
if (!ha->qla_watchdog_pause) { if (!ha->qla_watchdog_pause) {
if (ql_hw_check_health(ha) || ha->qla_initiate_recovery || if (!ha->offline &&
(ha->msg_from_peer == QL_PEER_MSG_RESET)) { (ql_hw_check_health(ha) || ha->qla_initiate_recovery ||
(ha->msg_from_peer == QL_PEER_MSG_RESET))) {
if (!(ha->dbg_level & 0x8000)) { ifp->if_drv_flags &= ~IFF_DRV_RUNNING;
ql_update_link_state(ha);
if (ha->enable_error_recovery) {
ha->qla_watchdog_paused = 1; ha->qla_watchdog_paused = 1;
ha->qla_watchdog_pause = 1; ha->qla_watchdog_pause = 1;
ha->qla_initiate_recovery = 0;
ha->err_inject = 0; ha->err_inject = 0;
device_printf(ha->pci_dev, device_printf(ha->pci_dev,
"%s: taskqueue_enqueue(err_task) \n", "%s: taskqueue_enqueue(err_task) \n",
__func__); __func__);
taskqueue_enqueue(ha->err_tq, &ha->err_task); taskqueue_enqueue(ha->err_tq, &ha->err_task);
return; } else {
if (ifp != NULL)
ifp->if_drv_flags &= ~IFF_DRV_RUNNING;
ha->offline = 1;
} }
return;
} else if (ha->qla_interface_up) {
ha->watchdog_ticks++;
if (ha->watchdog_ticks > 1000)
ha->watchdog_ticks = 0;
if (!ha->watchdog_ticks && QL_RUNNING(ifp)) {
taskqueue_enqueue(ha->stats_tq, &ha->stats_task);
}
if (ha->async_event) {
taskqueue_enqueue(ha->async_event_tq,
&ha->async_event_task);
}
#if 0
for (i = 0; ((i < ha->hw.num_sds_rings) &&
!ha->watchdog_ticks); i++) {
qla_tx_fp_t *fp = &ha->tx_fp[i];
if (fp->fp_taskqueue != NULL)
taskqueue_enqueue(fp->fp_taskqueue,
&fp->fp_task);
}
#endif
ha->qla_watchdog_paused = 0;
} else { } else {
if (ha->qla_interface_up) {
ha->watchdog_ticks++;
if (ha->watchdog_ticks > 1000)
ha->watchdog_ticks = 0;
if (!ha->watchdog_ticks && QL_RUNNING(ifp)) {
taskqueue_enqueue(ha->stats_tq,
&ha->stats_task);
}
if (ha->async_event) {
taskqueue_enqueue(ha->async_event_tq,
&ha->async_event_task);
}
}
ha->qla_watchdog_paused = 0; ha->qla_watchdog_paused = 0;
} }
} else { } else {
@ -347,6 +369,7 @@ qla_pci_attach(device_t dev)
ha->reg_rid); ha->reg_rid);
mtx_init(&ha->hw_lock, "qla83xx_hw_lock", MTX_NETWORK_LOCK, MTX_DEF); mtx_init(&ha->hw_lock, "qla83xx_hw_lock", MTX_NETWORK_LOCK, MTX_DEF);
mtx_init(&ha->sp_log_lock, "qla83xx_sp_log_lock", MTX_NETWORK_LOCK, MTX_DEF);
ha->flags.lock_init = 1; ha->flags.lock_init = 1;
qla_add_sysctls(ha); qla_add_sysctls(ha);
@ -495,6 +518,7 @@ qla_pci_attach(device_t dev)
goto qla_pci_attach_err; goto qla_pci_attach_err;
} }
ql_alloc_drvr_state_buffer(ha); ql_alloc_drvr_state_buffer(ha);
ql_alloc_sp_log_buffer(ha);
/* create the o.s ethernet interface */ /* create the o.s ethernet interface */
qla_init_ifnet(dev, ha); qla_init_ifnet(dev, ha);
@ -540,6 +564,7 @@ qla_pci_attach(device_t dev)
if (ha->flags.lock_init) { if (ha->flags.lock_init) {
mtx_destroy(&ha->hw_lock); mtx_destroy(&ha->hw_lock);
mtx_destroy(&ha->sp_log_lock);
} }
QL_DPRINT2(ha, (dev, "%s: exit ENXIO\n", __func__)); QL_DPRINT2(ha, (dev, "%s: exit ENXIO\n", __func__));
@ -578,6 +603,7 @@ qla_pci_detach(device_t dev)
if (ha->flags.lock_init) { if (ha->flags.lock_init) {
mtx_destroy(&ha->hw_lock); mtx_destroy(&ha->hw_lock);
mtx_destroy(&ha->sp_log_lock);
} }
QL_DPRINT2(ha, (dev, "%s: exit\n", __func__)); QL_DPRINT2(ha, (dev, "%s: exit\n", __func__));
@ -619,17 +645,17 @@ qla_release(qla_host_t *ha)
dev = ha->pci_dev; dev = ha->pci_dev;
if (ha->async_event_tq) { if (ha->async_event_tq) {
taskqueue_drain(ha->async_event_tq, &ha->async_event_task); taskqueue_drain_all(ha->async_event_tq);
taskqueue_free(ha->async_event_tq); taskqueue_free(ha->async_event_tq);
} }
if (ha->err_tq) { if (ha->err_tq) {
taskqueue_drain(ha->err_tq, &ha->err_task); taskqueue_drain_all(ha->err_tq);
taskqueue_free(ha->err_tq); taskqueue_free(ha->err_tq);
} }
if (ha->stats_tq) { if (ha->stats_tq) {
taskqueue_drain(ha->stats_tq, &ha->stats_task); taskqueue_drain_all(ha->stats_tq);
taskqueue_free(ha->stats_tq); taskqueue_free(ha->stats_tq);
} }
@ -649,6 +675,7 @@ qla_release(qla_host_t *ha)
ether_ifdetach(ha->ifp); ether_ifdetach(ha->ifp);
ql_free_drvr_state_buffer(ha); ql_free_drvr_state_buffer(ha);
ql_free_sp_log_buffer(ha);
ql_free_dma(ha); ql_free_dma(ha);
qla_free_parent_dma_tag(ha); qla_free_parent_dma_tag(ha);
@ -679,10 +706,6 @@ qla_release(qla_host_t *ha)
if (ha->msix_count) if (ha->msix_count)
pci_release_msi(dev); pci_release_msi(dev);
// if (ha->flags.lock_init) {
// mtx_destroy(&ha->hw_lock);
// }
if (ha->pci_reg) if (ha->pci_reg)
(void) bus_release_resource(dev, SYS_RES_MEMORY, ha->reg_rid, (void) bus_release_resource(dev, SYS_RES_MEMORY, ha->reg_rid,
ha->pci_reg); ha->pci_reg);
@ -871,6 +894,7 @@ qla_init_ifnet(device_t dev, qla_host_t *ha)
ifp->if_capabilities |= IFCAP_HWCSUM | ifp->if_capabilities |= IFCAP_HWCSUM |
IFCAP_TSO4 | IFCAP_TSO4 |
IFCAP_TSO6 |
IFCAP_JUMBO_MTU | IFCAP_JUMBO_MTU |
IFCAP_VLAN_HWTAGGING | IFCAP_VLAN_HWTAGGING |
IFCAP_VLAN_MTU | IFCAP_VLAN_MTU |
@ -899,6 +923,8 @@ qla_init_locked(qla_host_t *ha)
{ {
struct ifnet *ifp = ha->ifp; struct ifnet *ifp = ha->ifp;
ql_sp_log(ha, 14, 0, 0, 0, 0, 0, 0);
qla_stop(ha); qla_stop(ha);
if (qla_alloc_xmt_bufs(ha) != 0) if (qla_alloc_xmt_bufs(ha) != 0)
@ -918,13 +944,17 @@ qla_init_locked(qla_host_t *ha)
if (ql_init_hw_if(ha) == 0) { if (ql_init_hw_if(ha) == 0) {
ifp = ha->ifp; ifp = ha->ifp;
ifp->if_drv_flags |= IFF_DRV_RUNNING; ifp->if_drv_flags |= IFF_DRV_RUNNING;
ha->qla_watchdog_pause = 0;
ha->hw_vlan_tx_frames = 0; ha->hw_vlan_tx_frames = 0;
ha->tx_tso_frames = 0; ha->tx_tso_frames = 0;
ha->qla_interface_up = 1; ha->qla_interface_up = 1;
ql_update_link_state(ha); ql_update_link_state(ha);
} else {
if (ha->hw.sp_log_stop_events & Q8_SP_LOG_STOP_IF_START_FAILURE)
ha->hw.sp_log_stop = -1;
} }
ha->qla_watchdog_pause = 0;
return; return;
} }
@ -978,6 +1008,10 @@ qla_set_multi(qla_host_t *ha, uint32_t add_multi)
QLA_LOCK_NO_SLEEP) != 0) QLA_LOCK_NO_SLEEP) != 0)
return (-1); return (-1);
ql_sp_log(ha, 12, 4, ifp->if_drv_flags,
(ifp->if_drv_flags & IFF_DRV_RUNNING),
add_multi, (uint32_t)mcnt, 0);
if (ifp->if_drv_flags & IFF_DRV_RUNNING) { if (ifp->if_drv_flags & IFF_DRV_RUNNING) {
if (!add_multi) { if (!add_multi) {
@ -1008,6 +1042,8 @@ qla_ioctl(struct ifnet *ifp, u_long cmd, caddr_t data)
qla_host_t *ha; qla_host_t *ha;
ha = (qla_host_t *)ifp->if_softc; ha = (qla_host_t *)ifp->if_softc;
if (ha->offline || ha->qla_initiate_recovery)
return (ret);
switch (cmd) { switch (cmd) {
case SIOCSIFADDR: case SIOCSIFADDR:
@ -1024,6 +1060,10 @@ qla_ioctl(struct ifnet *ifp, u_long cmd, caddr_t data)
ifp->if_flags |= IFF_UP; ifp->if_flags |= IFF_UP;
ql_sp_log(ha, 8, 3, ifp->if_drv_flags,
(ifp->if_drv_flags & IFF_DRV_RUNNING),
ntohl(IA_SIN(ifa)->sin_addr.s_addr), 0, 0);
if (!(ifp->if_drv_flags & IFF_DRV_RUNNING)) { if (!(ifp->if_drv_flags & IFF_DRV_RUNNING)) {
qla_init_locked(ha); qla_init_locked(ha);
} }
@ -1057,6 +1097,10 @@ qla_ioctl(struct ifnet *ifp, u_long cmd, caddr_t data)
ha->max_frame_size = ha->max_frame_size =
ifp->if_mtu + ETHER_HDR_LEN + ETHER_CRC_LEN; ifp->if_mtu + ETHER_HDR_LEN + ETHER_CRC_LEN;
ql_sp_log(ha, 9, 4, ifp->if_drv_flags,
(ifp->if_drv_flags & IFF_DRV_RUNNING),
ha->max_frame_size, ifp->if_mtu, 0);
if (ifp->if_drv_flags & IFF_DRV_RUNNING) { if (ifp->if_drv_flags & IFF_DRV_RUNNING) {
qla_init_locked(ha); qla_init_locked(ha);
} }
@ -1082,6 +1126,10 @@ qla_ioctl(struct ifnet *ifp, u_long cmd, caddr_t data)
if (ret) if (ret)
break; break;
ql_sp_log(ha, 10, 4, ifp->if_drv_flags,
(ifp->if_drv_flags & IFF_DRV_RUNNING),
ha->if_flags, ifp->if_flags, 0);
if (ifp->if_flags & IFF_UP) { if (ifp->if_flags & IFF_UP) {
ha->max_frame_size = ifp->if_mtu + ha->max_frame_size = ifp->if_mtu +
@ -1157,6 +1205,10 @@ qla_ioctl(struct ifnet *ifp, u_long cmd, caddr_t data)
if (ret) if (ret)
break; break;
ql_sp_log(ha, 11, 4, ifp->if_drv_flags,
(ifp->if_drv_flags & IFF_DRV_RUNNING),
mask, ifp->if_capenable, 0);
qla_init_locked(ha); qla_init_locked(ha);
QLA_UNLOCK(ha, __func__); QLA_UNLOCK(ha, __func__);
@ -1512,7 +1564,7 @@ qla_destroy_fp_taskqueues(qla_host_t *ha)
qla_tx_fp_t *fp = &ha->tx_fp[i]; qla_tx_fp_t *fp = &ha->tx_fp[i];
if (fp->fp_taskqueue != NULL) { if (fp->fp_taskqueue != NULL) {
taskqueue_drain(fp->fp_taskqueue, &fp->fp_task); taskqueue_drain_all(fp->fp_taskqueue);
taskqueue_free(fp->fp_taskqueue); taskqueue_free(fp->fp_taskqueue);
fp->fp_taskqueue = NULL; fp->fp_taskqueue = NULL;
} }
@ -1529,7 +1581,7 @@ qla_drain_fp_taskqueues(qla_host_t *ha)
qla_tx_fp_t *fp = &ha->tx_fp[i]; qla_tx_fp_t *fp = &ha->tx_fp[i];
if (fp->fp_taskqueue != NULL) { if (fp->fp_taskqueue != NULL) {
taskqueue_drain(fp->fp_taskqueue, &fp->fp_task); taskqueue_drain_all(fp->fp_taskqueue);
} }
} }
return; return;
@ -1614,6 +1666,8 @@ qla_stop(qla_host_t *ha)
device_t dev; device_t dev;
int i = 0; int i = 0;
ql_sp_log(ha, 13, 0, 0, 0, 0, 0, 0);
dev = ha->pci_dev; dev = ha->pci_dev;
ifp->if_drv_flags &= ~IFF_DRV_RUNNING; ifp->if_drv_flags &= ~IFF_DRV_RUNNING;
@ -2022,27 +2076,45 @@ qla_send_msg_to_peer(qla_host_t *ha, uint32_t msg_to_peer)
} }
} }
void
qla_set_error_recovery(qla_host_t *ha)
{
struct ifnet *ifp = ha->ifp;
if (!cold && ha->enable_error_recovery) {
if (ifp)
ifp->if_drv_flags &= ~IFF_DRV_RUNNING;
ha->qla_initiate_recovery = 1;
} else
ha->offline = 1;
return;
}
static void static void
qla_error_recovery(void *context, int pending) qla_error_recovery(void *context, int pending)
{ {
qla_host_t *ha = context; qla_host_t *ha = context;
uint32_t msecs_100 = 100; uint32_t msecs_100 = 400;
struct ifnet *ifp = ha->ifp; struct ifnet *ifp = ha->ifp;
int i = 0; int i = 0;
device_printf(ha->pci_dev, "%s: \n", __func__); device_printf(ha->pci_dev, "%s: enter\n", __func__);
ha->hw.imd_compl = 1; ha->hw.imd_compl = 1;
taskqueue_drain_all(ha->stats_tq);
taskqueue_drain_all(ha->async_event_tq);
if (QLA_LOCK(ha, __func__, -1, 0) != 0) if (QLA_LOCK(ha, __func__, -1, 0) != 0)
return; return;
device_printf(ha->pci_dev, "%s: enter\n", __func__); device_printf(ha->pci_dev, "%s: ts_usecs = %ld start\n",
__func__, qla_get_usec_timestamp());
if (ha->qla_interface_up) { if (ha->qla_interface_up) {
qla_mdelay(__func__, 300); qla_mdelay(__func__, 300);
ifp->if_drv_flags &= ~IFF_DRV_RUNNING; //ifp->if_drv_flags &= ~IFF_DRV_RUNNING;
for (i = 0; i < ha->hw.num_sds_rings; i++) { for (i = 0; i < ha->hw.num_sds_rings; i++) {
qla_tx_fp_t *fp; qla_tx_fp_t *fp;
@ -2059,7 +2131,6 @@ device_printf(ha->pci_dev, "%s: enter\n", __func__);
} }
} }
qla_drain_fp_taskqueues(ha); qla_drain_fp_taskqueues(ha);
if ((ha->pci_func & 0x1) == 0) { if ((ha->pci_func & 0x1) == 0) {
@ -2077,21 +2148,32 @@ device_printf(ha->pci_dev, "%s: enter\n", __func__);
if (ha->enable_minidump) if (ha->enable_minidump)
ql_minidump(ha); ql_minidump(ha);
(void) ql_init_hw(ha); if (ha->enable_driverstate_dump)
ql_capture_drvr_state(ha);
if (ql_init_hw(ha)) {
device_printf(ha->pci_dev,
"%s: ts_usecs = %ld exit: ql_init_hw failed\n",
__func__, qla_get_usec_timestamp());
ha->offline = 1;
goto qla_error_recovery_exit;
}
if (ha->qla_interface_up) { if (ha->qla_interface_up) {
qla_free_xmt_bufs(ha); qla_free_xmt_bufs(ha);
qla_free_rcv_bufs(ha); qla_free_rcv_bufs(ha);
} }
qla_send_msg_to_peer(ha, QL_PEER_MSG_ACK); if (!QL_ERR_INJECT(ha, INJCT_PEER_PORT_FAILURE_ERR_RECOVERY))
qla_send_msg_to_peer(ha, QL_PEER_MSG_ACK);
} else { } else {
if (ha->msg_from_peer == QL_PEER_MSG_RESET) { if (ha->msg_from_peer == QL_PEER_MSG_RESET) {
ha->msg_from_peer = 0; ha->msg_from_peer = 0;
qla_send_msg_to_peer(ha, QL_PEER_MSG_ACK); if (!QL_ERR_INJECT(ha, INJCT_PEER_PORT_FAILURE_ERR_RECOVERY))
qla_send_msg_to_peer(ha, QL_PEER_MSG_ACK);
} else { } else {
qla_send_msg_to_peer(ha, QL_PEER_MSG_RESET); qla_send_msg_to_peer(ha, QL_PEER_MSG_RESET);
} }
@ -2100,9 +2182,24 @@ device_printf(ha->pci_dev, "%s: enter\n", __func__);
qla_mdelay(__func__, 100); qla_mdelay(__func__, 100);
ha->msg_from_peer = 0; ha->msg_from_peer = 0;
(void) ql_init_hw(ha); if (ha->enable_driverstate_dump)
ql_capture_drvr_state(ha);
qla_mdelay(__func__, 1000); if (msecs_100 == 0) {
device_printf(ha->pci_dev,
"%s: ts_usecs = %ld exit: QL_PEER_MSG_ACK not received\n",
__func__, qla_get_usec_timestamp());
ha->offline = 1;
goto qla_error_recovery_exit;
}
if (ql_init_hw(ha)) {
device_printf(ha->pci_dev,
"%s: ts_usecs = %ld exit: ql_init_hw failed\n",
__func__, qla_get_usec_timestamp());
ha->offline = 1;
goto qla_error_recovery_exit;
}
if (ha->qla_interface_up) { if (ha->qla_interface_up) {
qla_free_xmt_bufs(ha); qla_free_xmt_bufs(ha);
@ -2110,14 +2207,22 @@ device_printf(ha->pci_dev, "%s: enter\n", __func__);
} }
} }
qla_mdelay(__func__, ha->ms_delay_after_init);
*((uint32_t *)&ha->hw.flags) = 0;
ha->qla_initiate_recovery = 0;
if (ha->qla_interface_up) { if (ha->qla_interface_up) {
if (qla_alloc_xmt_bufs(ha) != 0) { if (qla_alloc_xmt_bufs(ha) != 0) {
ha->offline = 1;
goto qla_error_recovery_exit; goto qla_error_recovery_exit;
} }
qla_confirm_9kb_enable(ha); qla_confirm_9kb_enable(ha);
if (qla_alloc_rcv_bufs(ha) != 0) { if (qla_alloc_rcv_bufs(ha) != 0) {
ha->offline = 1;
goto qla_error_recovery_exit; goto qla_error_recovery_exit;
} }
@ -2127,18 +2232,38 @@ device_printf(ha->pci_dev, "%s: enter\n", __func__);
ifp = ha->ifp; ifp = ha->ifp;
ifp->if_drv_flags |= IFF_DRV_RUNNING; ifp->if_drv_flags |= IFF_DRV_RUNNING;
ha->qla_watchdog_pause = 0; ha->qla_watchdog_pause = 0;
ql_update_link_state(ha);
} else {
ha->offline = 1;
if (ha->hw.sp_log_stop_events &
Q8_SP_LOG_STOP_IF_START_FAILURE)
ha->hw.sp_log_stop = -1;
} }
} else } else {
ha->qla_watchdog_pause = 0; ha->qla_watchdog_pause = 0;
}
qla_error_recovery_exit: qla_error_recovery_exit:
device_printf(ha->pci_dev, "%s: exit\n", __func__); if (ha->offline ) {
device_printf(ha->pci_dev, "%s: ts_usecs = %ld port offline\n",
__func__, qla_get_usec_timestamp());
if (ha->hw.sp_log_stop_events &
Q8_SP_LOG_STOP_ERR_RECOVERY_FAILURE)
ha->hw.sp_log_stop = -1;
}
QLA_UNLOCK(ha, __func__); QLA_UNLOCK(ha, __func__);
callout_reset(&ha->tx_callout, QLA_WATCHDOG_CALLOUT_TICKS, if (!ha->offline)
qla_watchdog, ha); callout_reset(&ha->tx_callout, QLA_WATCHDOG_CALLOUT_TICKS,
qla_watchdog, ha);
device_printf(ha->pci_dev,
"%s: ts_usecs = %ld exit\n",
__func__, qla_get_usec_timestamp());
return; return;
} }
@ -2168,6 +2293,7 @@ qla_stats(void *context, int pending)
ha = context; ha = context;
ql_get_stats(ha); ql_get_stats(ha);
return; return;
} }

View File

@ -122,6 +122,14 @@ static __inline int qla_sec_to_hz(int sec)
return (tvtohz(&t)); return (tvtohz(&t));
} }
static __inline uint64_t qla_get_usec_timestamp(void)
{
struct timeval tv;
microuptime(&tv);
return ((uint64_t)(((uint64_t)tv.tv_sec) * 1000000 + tv.tv_usec));
}
#define qla_host_to_le16(x) htole16(x) #define qla_host_to_le16(x) htole16(x)
#define qla_host_to_le32(x) htole32(x) #define qla_host_to_le32(x) htole32(x)

View File

@ -38,6 +38,6 @@
#define QLA_VERSION_MAJOR 3 #define QLA_VERSION_MAJOR 3
#define QLA_VERSION_MINOR 10 #define QLA_VERSION_MINOR 10
#define QLA_VERSION_BUILD 35 #define QLA_VERSION_BUILD 36
#endif /* #ifndef _QL_VER_H_ */ #endif /* #ifndef _QL_VER_H_ */