diff options
author | davidch <davidch@FreeBSD.org> | 2011-04-21 23:06:00 +0000 |
---|---|---|
committer | davidch <davidch@FreeBSD.org> | 2011-04-21 23:06:00 +0000 |
commit | 0f13b0e3bfeafa2d83b5c05c2ff47acb45d02a8b (patch) | |
tree | 28f91e89f407b28ee63b9ec31535b0b4f05efa2d /sys/dev/bxe | |
parent | 72db64ceeaeef4fc6ff9287d3f3da51be1657bf0 (diff) | |
download | FreeBSD-src-0f13b0e3bfeafa2d83b5c05c2ff47acb45d02a8b.zip FreeBSD-src-0f13b0e3bfeafa2d83b5c05c2ff47acb45d02a8b.tar.gz |
- Centralize driver tunables initialization/validation.
- Centralize PCI resource allocation/release.
- Enable flowid (TSS) support.
- Added "per-fastpath" locks and watchdog timeouts.
- Fixed problem where the CQ producer index was advanced beyond
the size of the CQ ring during initialization.
- Replaced hard-coded debug levels in some debug print statements.
- More style(9) fixes.
MFC after: Two weeks
Diffstat (limited to 'sys/dev/bxe')
-rw-r--r-- | sys/dev/bxe/bxe_debug.h | 48 | ||||
-rw-r--r-- | sys/dev/bxe/bxe_include.h | 70 | ||||
-rw-r--r-- | sys/dev/bxe/if_bxe.c | 1137 | ||||
-rw-r--r-- | sys/dev/bxe/if_bxe.h | 56 |
4 files changed, 896 insertions, 415 deletions
diff --git a/sys/dev/bxe/bxe_debug.h b/sys/dev/bxe/bxe_debug.h index 47cabb5..a08641b 100644 --- a/sys/dev/bxe/bxe_debug.h +++ b/sys/dev/bxe/bxe_debug.h @@ -159,50 +159,50 @@ extern uint32_t bxe_debug; #ifdef BXE_DEBUG /* Print a message based on the logging level and code path. */ -#define DBPRINT(sc, level, format, args...) \ - do { \ - if (BXE_LOG_MSG(level)) { \ - device_printf(sc->bxe_dev, format, ## args); \ - } \ +#define DBPRINT(sc, level, format, args...) \ + do { \ + if (BXE_LOG_MSG(level)) { \ + device_printf(sc->dev, format, ## args); \ + } \ } while (0) /* Runs a particular command when debugging is enabled. */ -#define DBRUN(args...) \ - do { \ - args; \ +#define DBRUN(args...) \ + do { \ + args; \ } while (0) /* Runs a particular command based on the logging level. */ -#define DBRUNLV(level, args...) \ - if (BXE_MSG_LEVEL(level)) { \ - args; \ +#define DBRUNLV(level, args...) \ + if (BXE_MSG_LEVEL(level)) { \ + args; \ } /* Runs a particular command based on the code path. */ -#define DBRUNCP(cp, args...) \ - if (BXE_CODE_PATH(cp)) { \ - args; \ +#define DBRUNCP(cp, args...) \ + if (BXE_CODE_PATH(cp)) { \ + args; \ } /* Runs a particular command based on a condition. */ -#define DBRUNIF(cond, args...) \ - if (cond) { \ - args; \ +#define DBRUNIF(cond, args...) \ + if (cond) { \ + args; \ } /* Runs a particular command based on the logging level and code path. */ -#define DBRUNMSG(msg, args...) \ - if (BXE_LOG_MSG(msg)) { \ - args; \ +#define DBRUNMSG(msg, args...) \ + if (BXE_LOG_MSG(msg)) { \ + args; \ } /* Announces function entry. */ -#define DBENTER(cond) \ - DBPRINT(sc, (cond), "%s(enter:%d)\n", __FUNCTION__, curcpu) \ +#define DBENTER(cond) \ + DBPRINT(sc, (cond), "%s(enter:%d)\n", __FUNCTION__, curcpu) /* Announces function exit. */ -#define DBEXIT(cond) \ - DBPRINT(sc, (cond), "%s(exit:%d)\n", __FUNCTION__, curcpu) \ +#define DBEXIT(cond) \ + DBPRINT(sc, (cond), "%s(exit:%d)\n", __FUNCTION__, curcpu) /* Needed for random() function which is only used in debugging. */ #include <sys/random.h> diff --git a/sys/dev/bxe/bxe_include.h b/sys/dev/bxe/bxe_include.h index 79e632b..67673f5 100644 --- a/sys/dev/bxe/bxe_include.h +++ b/sys/dev/bxe/bxe_include.h @@ -61,21 +61,21 @@ /* * Convenience definitions used in multiple files. */ -#define BXE_PRINTF(fmt, args...) \ - do { \ - device_printf(sc->bxe_dev, fmt, ##args);\ +#define BXE_PRINTF(fmt, args...) \ + do { \ + device_printf(sc->dev, fmt, ##args); \ }while(0) #ifdef BXE_DEBUG -#define REG_WR(sc, offset, val) \ +#define REG_WR(sc, offset, val) \ bxe_reg_write32(sc, offset, val) -#define REG_WR8(sc, offset, val) \ +#define REG_WR8(sc, offset, val) \ bxe_reg_write8(sc, offset, val) -#define REG_WR16(sc, offset, val) \ +#define REG_WR16(sc, offset, val) \ bxe_reg_write16(sc, offset, val) -#define REG_WR32(sc, offset, val) \ +#define REG_WR32(sc, offset, val) \ bxe_reg_write32(sc, offset, val) #define REG_RD(sc, offset) \ @@ -87,77 +87,77 @@ #define REG_RD32(sc, offset) \ bxe_reg_read32(sc, offset) -#define REG_RD_IND(sc, offset) \ +#define REG_RD_IND(sc, offset) \ bxe_reg_rd_ind(sc, offset) -#define REG_WR_IND(sc, offset, val) \ +#define REG_WR_IND(sc, offset, val) \ bxe_reg_wr_ind(sc, offset, val) #else -#define REG_WR(sc, offset, val) \ +#define REG_WR(sc, offset, val) \ bus_space_write_4(sc->bxe_btag, sc->bxe_bhandle, offset, val) -#define REG_WR8(sc, offset, val) \ +#define REG_WR8(sc, offset, val) \ bus_space_write_1(sc->bxe_btag, sc->bxe_bhandle, offset, val) -#define REG_WR16(sc, offset, val) \ +#define REG_WR16(sc, offset, val) \ bus_space_write_2(sc->bxe_btag, sc->bxe_bhandle, offset, val) -#define REG_WR32(sc, offset, val) \ +#define REG_WR32(sc, offset, val) \ bus_space_write_4(sc->bxe_btag, sc->bxe_bhandle, offset, val) -#define REG_RD(sc, offset) \ +#define REG_RD(sc, offset) \ bus_space_read_4(sc->bxe_btag, sc->bxe_bhandle, offset) -#define REG_RD8(sc, offset) \ +#define REG_RD8(sc, offset) \ bus_space_read_1(sc->bxe_btag, sc->bxe_bhandle, offset) -#define REG_RD16(sc, offset) \ +#define REG_RD16(sc, offset) \ bus_space_read_2(sc->bxe_btag, sc->bxe_bhandle, offset) -#define REG_RD32(sc, offset) \ +#define REG_RD32(sc, offset) \ bus_space_read_4(sc->bxe_btag, sc->bxe_bhandle, offset) -#define REG_RD_IND(sc, offset) \ +#define REG_RD_IND(sc, offset) \ bxe_reg_rd_ind(sc, offset) -#define REG_WR_IND(sc, offset, val) \ +#define REG_WR_IND(sc, offset, val) \ bxe_reg_wr_ind(sc, offset, val) #endif /* BXE_DEBUG */ #define REG_RD_DMAE(sc, offset, val, len32) \ - do { \ - bxe_read_dmae(sc, offset, len32); \ - memcpy(val, BXE_SP(sc, wb_data[0]), len32 * 4); \ + do { \ + bxe_read_dmae(sc, offset, len32); \ + memcpy(val, BXE_SP(sc, wb_data[0]), len32 * 4); \ } while (0) #define REG_WR_DMAE(sc, offset, val, len32) \ - do { \ - memcpy(BXE_SP(sc, wb_data[0]), val, len32 * 4); \ - bxe_write_dmae(sc, BXE_SP_MAPPING(sc, wb_data), \ - offset, len32); \ + do { \ + memcpy(BXE_SP(sc, wb_data[0]), val, len32 * 4); \ + bxe_write_dmae(sc, BXE_SP_MAPPING(sc, wb_data), \ + offset, len32); \ } while (0) -#define SHMEM_ADDR(sc, field) (sc->common.shmem_base + \ +#define SHMEM_ADDR(sc, field) (sc->common.shmem_base + \ offsetof(struct shmem_region, field)) -#define SHMEM_RD(sc, field) \ +#define SHMEM_RD(sc, field) \ REG_RD(sc, SHMEM_ADDR(sc, field)) -#define SHMEM_RD16(sc, field) \ +#define SHMEM_RD16(sc, field) \ REG_RD16(sc, SHMEM_ADDR(sc, field)) -#define SHMEM_WR(sc, field, val) \ +#define SHMEM_WR(sc, field, val) \ REG_WR(sc, SHMEM_ADDR(sc, field), val) -#define SHMEM2_ADDR(sc, field) (sc->common.shmem2_base + \ - offsetof(struct shmem2_region, field)) +#define SHMEM2_ADDR(sc, field) (sc->common.shmem2_base + \ + offsetof(struct shmem2_region, field)) #define SHMEM2_RD(sc, field) REG_RD(sc, SHMEM2_ADDR(sc, field)) #define SHMEM2_WR(sc, field, val) REG_WR(sc, SHMEM2_ADDR(sc, field), val) -#define EMAC_RD(sc, reg) \ +#define EMAC_RD(sc, reg) \ REG_RD(sc, emac_base + (uint32_t) reg) -#define EMAC_WR(sc, reg, val) \ +#define EMAC_WR(sc, reg, val) \ REG_WR(sc, emac_base + (uint32_t) reg, val) -#define BMAC_WR(sc, reg, val) \ +#define BMAC_WR(sc, reg, val) \ REG_WR(sc, GRCBASE_NIG + bmac_addr + reg, val) #endif /* _BXE_INCLUDE_H */ diff --git a/sys/dev/bxe/if_bxe.c b/sys/dev/bxe/if_bxe.c index 5750cc0..a877ba0 100644 --- a/sys/dev/bxe/if_bxe.c +++ b/sys/dev/bxe/if_bxe.c @@ -68,7 +68,7 @@ __FBSDID("$FreeBSD$"); /* BXE Debug Options */ #ifdef BXE_DEBUG -uint32_t bxe_debug = BXE_INFO; +uint32_t bxe_debug = BXE_WARN; /* 0 = Never */ @@ -132,6 +132,7 @@ static int bxe_attach(device_t); static int bxe_detach(device_t); static int bxe_shutdown(device_t); +static void bxe_set_tunables(struct bxe_softc *); static void bxe_print_adapter_info(struct bxe_softc *); static void bxe_probe_pci_caps(struct bxe_softc *); static void bxe_link_settings_supported(struct bxe_softc *, uint32_t); @@ -145,6 +146,8 @@ static int bxe_stop_leading(struct bxe_softc *); static int bxe_setup_multi(struct bxe_softc *, int); static int bxe_stop_multi(struct bxe_softc *, int); static int bxe_stop_locked(struct bxe_softc *, int); +static int bxe_alloc_buf_rings(struct bxe_softc *); +static void bxe_free_buf_rings(struct bxe_softc *); static void bxe_init_locked(struct bxe_softc *, int); static int bxe_wait_ramrod(struct bxe_softc *, int, int, int *, int); static void bxe_init_str_wr(struct bxe_softc *, uint32_t, const uint32_t *, @@ -237,6 +240,10 @@ static void bxe_stats_handle(struct bxe_softc *, enum bxe_stats_event); static int bxe_tx_encap(struct bxe_fastpath *, struct mbuf **); static void bxe_tx_start(struct ifnet *); static void bxe_tx_start_locked(struct ifnet *, struct bxe_fastpath *); +static int bxe_tx_mq_start(struct ifnet *, struct mbuf *); +static int bxe_tx_mq_start_locked(struct ifnet *, struct bxe_fastpath *, + struct mbuf *); +static void bxe_mq_flush(struct ifnet *ifp); static int bxe_ioctl(struct ifnet *, u_long, caddr_t); static __inline int bxe_has_rx_work(struct bxe_fastpath *); static __inline int bxe_has_tx_work(struct bxe_fastpath *); @@ -266,6 +273,8 @@ static struct mbuf *bxe_alloc_mbuf(struct bxe_fastpath *, int); static int bxe_map_mbuf(struct bxe_fastpath *, struct mbuf *, bus_dma_tag_t, bus_dmamap_t, bus_dma_segment_t *); static struct mbuf *bxe_alloc_tpa_mbuf(struct bxe_fastpath *, int, int); +static void bxe_alloc_mutexes(struct bxe_softc *); +static void bxe_free_mutexes(struct bxe_softc *); static int bxe_alloc_rx_sge(struct bxe_softc *, struct bxe_fastpath *, uint16_t); static void bxe_init_rx_chains(struct bxe_softc *); @@ -322,7 +331,7 @@ static void bxe_tpa_stop(struct bxe_softc *, struct bxe_fastpath *, uint16_t, static void bxe_rxeof(struct bxe_fastpath *); static void bxe_txeof(struct bxe_fastpath *); static int bxe_get_buf(struct bxe_fastpath *, struct mbuf *, uint16_t); -static void bxe_watchdog(struct bxe_softc *); +static int bxe_watchdog(struct bxe_fastpath *fp); static int bxe_change_mtu(struct bxe_softc *, int); static void bxe_tick(void *); static void bxe_add_sysctls(struct bxe_softc *); @@ -481,8 +490,7 @@ SYSCTL_UINT(_hw_bxe, OID_AUTO, queue_count, CTLFLAG_RDTUN, &bxe_queue_count, * destination IP address and the source/destination TCP port). * */ -/* static int bxe_multi_mode = ETH_RSS_MODE_REGULAR; */ -static int bxe_multi_mode = ETH_RSS_MODE_DISABLED; +static int bxe_multi_mode = ETH_RSS_MODE_REGULAR; TUNABLE_INT("hw.bxe.multi_mode", &bxe_multi_mode); SYSCTL_UINT(_hw_bxe, OID_AUTO, multi_mode, CTLFLAG_RDTUN, &bxe_multi_mode, 0, "Multi-Queue Mode"); @@ -738,7 +746,7 @@ bxe_calc_vn_wsum(struct bxe_softc *sc) uint32_t vn_cfg, vn_min_rate; int all_zero, vn; - DBENTER(1); + DBENTER(BXE_VERBOSE_LOAD); all_zero = 1; sc->vn_wsum = 0; @@ -764,7 +772,7 @@ bxe_calc_vn_wsum(struct bxe_softc *sc) else sc->cmng.flags.cmng_enables |= CMNG_FLAGS_PER_PORT_FAIRNESS_VN; - DBEXIT(1); + DBEXIT(BXE_VERBOSE_LOAD); } /* @@ -784,7 +792,7 @@ bxe_init_vn_minmax(struct bxe_softc *sc, int vn) vn_cfg = sc->mf_config[vn]; func = 2 * vn + BP_PORT(sc); - DBENTER(1); + DBENTER(BXE_VERBOSE_LOAD); /* If function is hidden - set min and max to zeroes. */ if (vn_cfg & FUNC_MF_CFG_FUNC_HIDE) { @@ -807,7 +815,7 @@ bxe_init_vn_minmax(struct bxe_softc *sc, int vn) if (vn_max_rate == 0) return; } - DBPRINT(sc, 1, + DBPRINT(sc, BXE_INFO_LOAD, "%s(): func %d: vn_min_rate = %d, vn_max_rate = %d, wsum = %d.\n", __FUNCTION__, func, vn_min_rate, vn_max_rate, sc->vn_wsum); @@ -846,7 +854,8 @@ bxe_init_vn_minmax(struct bxe_softc *sc, int vn) REG_WR(sc, BAR_XSTORM_INTMEM + XSTORM_FAIRNESS_PER_VN_VARS_OFFSET(func) + (i * 4), ((uint32_t *)(&m_fair_vn))[i]); - DBEXIT(1); + + DBEXIT(BXE_VERBOSE_LOAD); } static void @@ -854,6 +863,8 @@ bxe_congestionmgmt(struct bxe_softc *sc, uint8_t readshm) { int vn; + DBENTER(BXE_VERBOSE_LOAD); + /* Read mf conf from shmem. */ if (readshm) bxe_read_mf_cfg(sc); @@ -871,11 +882,14 @@ bxe_congestionmgmt(struct bxe_softc *sc, uint8_t readshm) /* Always enable rate shaping and fairness. */ sc->cmng.flags.cmng_enables |= CMNG_FLAGS_PER_PORT_RATE_SHAPING_VN; - DBPRINT(sc, 1, "rate shaping set\n"); + DBPRINT(sc, BXE_VERBOSE_LOAD, + "%s(): Rate shaping set\n", __FUNCTION__); if (!sc->vn_wsum) - DBPRINT(sc, 1, - "All MIN values are zeroes fairness is disabled\n"); + DBPRINT(sc, BXE_INFO_LOAD, "%s(): All MIN values " + "are zeroes, fairness is disabled\n", __FUNCTION__); + + DBEXIT(BXE_VERBOSE_LOAD); } static void @@ -883,19 +897,18 @@ bxe_dcc_event(struct bxe_softc *sc, uint32_t dcc_event) { int i, port; + DBENTER(BXE_VERBOSE_LOAD); + if (dcc_event & DRV_STATUS_DCC_DISABLE_ENABLE_PF) { - /* - * This is the only place besides the function initialization - * where the sc->bxe_flags can change so it is done without any - * locks - */ if (sc->mf_config[BP_E1HVN(sc)] & FUNC_MF_CFG_FUNC_DISABLED) { - DBPRINT(sc, 1, "mf_cfg function disabled\n"); - sc->bxe_flags = BXE_STATE_DISABLED; + DBPRINT(sc, BXE_INFO_LOAD, "%s(): mf_cfg function " + "disabled\n", __FUNCTION__); + sc->state = BXE_STATE_DISABLED; bxe_e1h_disable(sc); } else { - DBPRINT(sc, 1, "mf_cfg function enabled\n"); - sc->bxe_flags = BXE_STATE_OPEN; + DBPRINT(sc, BXE_INFO_LOAD, "%s(): mf_cfg function " + "enabled\n", __FUNCTION__); + sc->state = BXE_STATE_OPEN; bxe_e1h_enable(sc); } dcc_event &= ~DRV_STATUS_DCC_DISABLE_ENABLE_PF; @@ -915,6 +928,8 @@ bxe_dcc_event(struct bxe_softc *sc, uint32_t dcc_event) bxe_fw_command(sc, DRV_MSG_CODE_DCC_FAILURE); else bxe_fw_command(sc, DRV_MSG_CODE_DCC_OK); + + DBEXIT(BXE_VERBOSE_LOAD); } /* @@ -935,7 +950,7 @@ bxe_probe(device_t dev) uint16_t did, sdid, svid, vid; sc = device_get_softc(dev); - sc->bxe_dev = dev; + sc->dev = dev; t = bxe_devs; /* Get the data for the device to be probed. */ @@ -1028,7 +1043,7 @@ bxe_print_adapter_info(struct bxe_softc *sc) printf("TPA"); i++; } - printf(") Queues ("); + printf("); Queues ("); switch (sc->multi_mode) { case ETH_RSS_MODE_DISABLED: printf("None"); @@ -1079,7 +1094,7 @@ bxe_interrupt_allocate(struct bxe_softc *sc) int msix_count, msix_required, msix_allocated; rc = 0; - dev = sc->bxe_dev; + dev = sc->dev; msi_count = 0; msi_required = 0; msi_allocated = 0; @@ -1096,70 +1111,6 @@ bxe_interrupt_allocate(struct bxe_softc *sc) for (i = 0; i < BXE_MAX_PRIORITY; i++) sc->pri_map[i] = 0; - /* - * Get our starting point for interrupt mode/number of queues. - * We will progressively step down from MSI-X to MSI to INTx - * and reduce the number of receive queues as necessary to - * match the system capabilities. - */ - sc->multi_mode = bxe_multi_mode; - sc->int_mode = bxe_int_mode; - - /* - * Verify the Priority -> Receive Queue mappings. - */ - if (sc->int_mode > 0) { - /* Multi-queue modes require MSI/MSI-X. */ - switch (sc->multi_mode) { - case ETH_RSS_MODE_DISABLED: - /* No multi-queue mode requested. */ - sc->num_queues = 1; - break; - case ETH_RSS_MODE_REGULAR: - if (sc->int_mode > 1) { - /* - * Assume we can use MSI-X - * (max of 16 receive queues). - */ - sc->num_queues = min((bxe_queue_count ? - bxe_queue_count : mp_ncpus), MAX_CONTEXT); - } else { - /* - * Assume we can use MSI - * (max of 7 receive queues). - */ - sc->num_queues = min((bxe_queue_count ? - bxe_queue_count : mp_ncpus), - BXE_MSI_VECTOR_COUNT - 1); - } - break; - default: - BXE_PRINTF( - "%s(%d): Unsupported multi_mode parameter (%d), " - "disabling multi-queue support!\n", __FILE__, - __LINE__, sc->multi_mode); - sc->multi_mode = ETH_RSS_MODE_DISABLED; - sc->num_queues = 1; - break; - } - } else { - /* User has forced INTx mode. */ - sc->multi_mode = ETH_RSS_MODE_DISABLED; - sc->num_queues = 1; - } - - DBPRINT(sc, (BXE_VERBOSE_LOAD | BXE_VERBOSE_INTR), - "%s(): Requested: int_mode = %d, multi_mode = %d num_queues = %d\n", - __FUNCTION__, sc->int_mode, sc->multi_mode, sc->num_queues); - -#ifdef BXE_DEBUG - for (i = 0; i < BXE_MAX_PRIORITY; i++) { - DBPRINT(sc, (BXE_VERBOSE_LOAD | BXE_VERBOSE_INTR), - "%s(): sc->pri_map[%d] = %d.\n", __FUNCTION__, i, - sc->pri_map[i]); - } -#endif - /* Get the number of available MSI/MSI-X interrupts from the OS. */ if (sc->int_mode > 0) { if (sc->bxe_cap_flags & BXE_MSIX_CAPABLE_FLAG) @@ -1331,7 +1282,7 @@ bxe_interrupt_detach(struct bxe_softc *sc) device_t dev; int i; - dev = sc->bxe_dev; + dev = sc->dev; DBENTER(BXE_VERBOSE_RESET | BXE_VERBOSE_UNLOAD); /* Release interrupt resources. */ if ((sc->bxe_flags & BXE_USING_MSIX_FLAG) && sc->msix_count) { @@ -1381,7 +1332,7 @@ bxe_interrupt_attach(struct bxe_softc *sc) sc->tq = taskqueue_create_fast("bxe_spq", M_NOWAIT, taskqueue_thread_enqueue, &sc->tq); taskqueue_start_threads(&sc->tq, 1, PI_NET, "%s spq", - device_get_nameunit(sc->bxe_dev)); + device_get_nameunit(sc->dev)); #endif /* Setup interrupt handlers. */ @@ -1393,7 +1344,7 @@ bxe_interrupt_attach(struct bxe_softc *sc) * driver instance to the interrupt handler for the * slowpath. */ - rc = bus_setup_intr(sc->bxe_dev, + rc = bus_setup_intr(sc->dev, sc->bxe_msix_res[0], INTR_TYPE_NET | INTR_MPSAFE, NULL, @@ -1420,7 +1371,7 @@ bxe_interrupt_attach(struct bxe_softc *sc) * fastpath context to the interrupt handler in this * case. Also the first msix_res was used by the sp. */ - rc = bus_setup_intr(sc->bxe_dev, + rc = bus_setup_intr(sc->dev, sc->bxe_msix_res[i + 1], INTR_TYPE_NET | INTR_MPSAFE, NULL, @@ -1440,7 +1391,7 @@ bxe_interrupt_attach(struct bxe_softc *sc) fp->tq = taskqueue_create_fast("bxe_fpq", M_NOWAIT, taskqueue_thread_enqueue, &fp->tq); taskqueue_start_threads(&fp->tq, 1, PI_NET, "%s fpq", - device_get_nameunit(sc->bxe_dev)); + device_get_nameunit(sc->dev)); #endif fp->state = BXE_FP_STATE_IRQ; } @@ -1452,7 +1403,7 @@ bxe_interrupt_attach(struct bxe_softc *sc) * Setup the interrupt handler. Note that we pass the driver * instance to the interrupt handler for the slowpath. */ - rc = bus_setup_intr(sc->bxe_dev,sc->bxe_msi_res[0], + rc = bus_setup_intr(sc->dev,sc->bxe_msi_res[0], INTR_TYPE_NET | INTR_MPSAFE, NULL, bxe_intr_sp, @@ -1479,7 +1430,7 @@ bxe_interrupt_attach(struct bxe_softc *sc) * fastpath context to the interrupt handler in this * case. */ - rc = bus_setup_intr(sc->bxe_dev, + rc = bus_setup_intr(sc->dev, sc->bxe_msi_res[i + 1], INTR_TYPE_NET | INTR_MPSAFE, NULL, @@ -1499,7 +1450,7 @@ bxe_interrupt_attach(struct bxe_softc *sc) fp->tq = taskqueue_create_fast("bxe_fpq", M_NOWAIT, taskqueue_thread_enqueue, &fp->tq); taskqueue_start_threads(&fp->tq, 1, PI_NET, "%s fpq", - device_get_nameunit(sc->bxe_dev)); + device_get_nameunit(sc->dev)); #endif } @@ -1515,7 +1466,7 @@ bxe_interrupt_attach(struct bxe_softc *sc) * driver instance to the interrupt handler which * will handle both the slowpath and fastpath. */ - rc = bus_setup_intr(sc->bxe_dev,sc->bxe_irq_res, + rc = bus_setup_intr(sc->dev,sc->bxe_irq_res, INTR_TYPE_NET | INTR_MPSAFE, NULL, bxe_intr_legacy, @@ -1529,13 +1480,10 @@ bxe_interrupt_attach(struct bxe_softc *sc) } #ifdef BXE_TASK TASK_INIT(&fp->task, 0, bxe_task_fp, fp); - fp->tq = taskqueue_create_fast("bxe_fpq", M_NOWAIT, - taskqueue_thread_enqueue, - &fp->tq - ); - taskqueue_start_threads(&fp->tq, 1, PI_NET, "%s fpq", - device_get_nameunit(sc->bxe_dev) - ); + fp->tq = taskqueue_create_fast("bxe_fpq", + M_NOWAIT, taskqueue_thread_enqueue, &fp->tq); + taskqueue_start_threads(&fp->tq, 1, + PI_NET, "%s fpq", device_get_nameunit(sc->dev)); #endif } @@ -1562,7 +1510,7 @@ bxe_probe_pci_caps(struct bxe_softc *sc) uint32_t reg; uint16_t link_status; - dev = sc->bxe_dev; + dev = sc->dev; DBENTER(BXE_EXTREME_LOAD); /* Check if PCI Power Management capability is enabled. */ @@ -1679,46 +1627,118 @@ bxe_init_firmware(struct bxe_softc *sc) return (0); } -/* - * Device attach function. - * - * Allocates device resources, performs secondary chip identification, - * resets and initializes the hardware, and initializes driver instance - * variables. - * - * Returns: - * 0 = Success, Positive value on failure. - */ -static int -bxe_attach(device_t dev) + +static void +bxe_set_tunables(struct bxe_softc *sc) { - struct bxe_softc *sc; - struct ifnet *ifp; - int rid, rc; + /* + * Get our starting point for interrupt mode/number of queues. + * We will progressively step down from MSI-X to MSI to INTx + * and reduce the number of receive queues as necessary to + * match the system capabilities. + */ + sc->multi_mode = bxe_multi_mode; + sc->int_mode = bxe_int_mode; + sc->tso_enable = bxe_tso_enable; - sc = device_get_softc(dev); - DBENTER(BXE_VERBOSE_LOAD | BXE_VERBOSE_RESET); + /* + * Verify the Priority -> Receive Queue mappings. + */ + if (sc->int_mode > 0) { + /* Multi-queue modes require MSI/MSI-X. */ + switch (sc->multi_mode) { + case ETH_RSS_MODE_DISABLED: + /* No multi-queue mode requested. */ + sc->num_queues = 1; + break; + case ETH_RSS_MODE_REGULAR: + if (sc->int_mode > 1) { + /* + * Assume we can use MSI-X + * (max of 16 receive queues). + */ + sc->num_queues = min((bxe_queue_count ? + bxe_queue_count : mp_ncpus), MAX_CONTEXT); + } else { + /* + * Assume we can use MSI + * (max of 7 receive queues). + */ + sc->num_queues = min((bxe_queue_count ? + bxe_queue_count : mp_ncpus), + BXE_MSI_VECTOR_COUNT - 1); + } + break; + default: + BXE_PRINTF( + "%s(%d): Unsupported multi_mode parameter (%d), " + "disabling multi-queue support!\n", __FILE__, + __LINE__, sc->multi_mode); + sc->multi_mode = ETH_RSS_MODE_DISABLED; + sc->num_queues = 1; + break; + } + } else { + /* User has forced INTx mode. */ + sc->multi_mode = ETH_RSS_MODE_DISABLED; + sc->num_queues = 1; + } - sc->bxe_dev = dev; - sc->bxe_unit = device_get_unit(dev); - sc->bxe_func = pci_get_function(dev); - sc->bxe_flags = 0; - sc->state = BXE_STATE_CLOSED; - rc = 0; + DBPRINT(sc, (BXE_VERBOSE_LOAD | BXE_VERBOSE_INTR), + "%s(): Requested: int_mode = %d, multi_mode = %d num_queues = %d\n", + __FUNCTION__, sc->int_mode, sc->multi_mode, sc->num_queues); - /* Initialize mutexes. */ - BXE_CORE_LOCK_INIT(sc, device_get_nameunit(dev)); - BXE_SP_LOCK_INIT(sc, "bxe_sp_lock"); - BXE_DMAE_LOCK_INIT(sc, "bxe_dmae_lock"); - BXE_PHY_LOCK_INIT(sc, "bxe_phy_lock"); - BXE_FWMB_LOCK_INIT(sc, "bxe_fwmb_lock"); - BXE_PRINT_LOCK_INIT(sc, "bxe_print_lock"); + /* Set transparent packet aggregation (TPA), aka LRO, flag. */ + if (bxe_tpa_enable!= FALSE) + sc->bxe_flags |= BXE_TPA_ENABLE_FLAG; - /* Prepare the tick routine. */ - callout_init(&sc->bxe_tick_callout, CALLOUT_MPSAFE); + /* Capture the stats enable/disable setting. */ + if (bxe_stats_enable == FALSE) + sc->stats_enable = FALSE; + else + sc->stats_enable = TRUE; + + /* Select the host coalescing tick count values (limit values). */ + if (bxe_tx_ticks > 100) { + BXE_PRINTF("%s(%d): bxe_tx_ticks too large " + "(%d), setting default value of 50.\n", + __FILE__, __LINE__, bxe_tx_ticks); + sc->tx_ticks = 50; + } else + sc->tx_ticks = bxe_tx_ticks; + + if (bxe_rx_ticks > 100) { + BXE_PRINTF("%s(%d): bxe_rx_ticks too large " + "(%d), setting default value of 25.\n", + __FILE__, __LINE__, bxe_rx_ticks); + sc->rx_ticks = 25; + } else + sc->rx_ticks = bxe_rx_ticks; + + /* Select the PCIe maximum read request size (MRRS). */ + if (bxe_mrrs > 3) + sc->mrrs = 3; + else + sc->mrrs = bxe_mrrs; + + /* Check for DCC support. */ + if (bxe_dcc_enable == FALSE) + sc->dcc_enable = FALSE; + else + sc->dcc_enable = TRUE; +} - /* Enable bus master capability */ - pci_enable_busmaster(dev); + +/* + * Returns: + * 0 = Success, !0 = Failure + */ +static int +bxe_alloc_pci_resources(struct bxe_softc *sc) +{ + int rid, rc = 0; + + DBENTER(BXE_VERBOSE_LOAD); /* * Allocate PCI memory resources for BAR0. @@ -1726,32 +1746,32 @@ bxe_attach(device_t dev) * processor memory. */ rid = PCIR_BAR(0); - sc->bxe_res = bus_alloc_resource_any(dev, - SYS_RES_MEMORY, &rid, RF_ACTIVE); + sc->bxe_res = bus_alloc_resource_any( + sc->dev, SYS_RES_MEMORY, &rid, RF_ACTIVE); if (sc->bxe_res == NULL) { BXE_PRINTF("%s(%d):PCI BAR0 memory allocation failed\n", __FILE__, __LINE__); rc = ENXIO; - goto bxe_attach_fail; + goto bxe_alloc_pci_resources_exit; } /* Get OS resource handles for BAR0 memory. */ - sc->bxe_btag = rman_get_bustag(sc->bxe_res); - sc->bxe_bhandle = rman_get_bushandle(sc->bxe_res); - sc->bxe_vhandle = (vm_offset_t) rman_get_virtual(sc->bxe_res); + sc->bxe_btag = rman_get_bustag(sc->bxe_res); + sc->bxe_bhandle = rman_get_bushandle(sc->bxe_res); + sc->bxe_vhandle = (vm_offset_t) rman_get_virtual(sc->bxe_res); /* * Allocate PCI memory resources for BAR2. * Doorbell (DB) memory. */ rid = PCIR_BAR(2); - sc->bxe_db_res = bus_alloc_resource_any(dev, - SYS_RES_MEMORY, &rid, RF_ACTIVE); + sc->bxe_db_res = bus_alloc_resource_any( + sc->dev, SYS_RES_MEMORY, &rid, RF_ACTIVE); if (sc->bxe_db_res == NULL) { BXE_PRINTF("%s(%d): PCI BAR2 memory allocation failed\n", __FILE__, __LINE__); rc = ENXIO; - goto bxe_attach_fail; + goto bxe_alloc_pci_resources_exit; } /* Get OS resource handles for BAR2 memory. */ @@ -1759,22 +1779,46 @@ bxe_attach(device_t dev) sc->bxe_db_bhandle = rman_get_bushandle(sc->bxe_db_res); sc->bxe_db_vhandle = (vm_offset_t) rman_get_virtual(sc->bxe_db_res); - /* Put indirect address registers into a sane state. */ - pci_write_config(sc->bxe_dev, PCICFG_GRC_ADDRESS, - PCICFG_VENDOR_ID_OFFSET, 4); - REG_WR(sc, PXP2_REG_PGL_ADDR_88_F0 + BP_PORT(sc) * 16, 0); - REG_WR(sc, PXP2_REG_PGL_ADDR_8C_F0 + BP_PORT(sc) * 16, 0); - REG_WR(sc, PXP2_REG_PGL_ADDR_90_F0 + BP_PORT(sc) * 16, 0); - REG_WR(sc, PXP2_REG_PGL_ADDR_94_F0 + BP_PORT(sc) * 16, 0); +bxe_alloc_pci_resources_exit: + DBEXIT(BXE_VERBOSE_LOAD); + return(rc); +} - /* Get hardware info from shared memory and validate data. */ - if (bxe_get_function_hwinfo(sc)) { - DBPRINT(sc, BXE_WARN, - "%s(): Failed to get hardware info!\n", __FUNCTION__); - rc = ENODEV; - goto bxe_attach_fail; + +/* + * Returns: + * None + */ +static void +bxe_release_pci_resources(struct bxe_softc *sc) +{ + /* Release the PCIe BAR0 mapped memory. */ + if (sc->bxe_res != NULL) { + DBPRINT(sc, (BXE_VERBOSE_LOAD | BXE_VERBOSE_RESET), + "%s(): Releasing PCI BAR0 memory.\n", __FUNCTION__); + bus_release_resource(sc->dev, + SYS_RES_MEMORY, PCIR_BAR(0), sc->bxe_res); } + /* Release the PCIe BAR2 (doorbell) mapped memory. */ + if (sc->bxe_db_res != NULL) { + DBPRINT(sc, (BXE_VERBOSE_LOAD | BXE_VERBOSE_RESET), + "%s(): Releasing PCI BAR2 memory.\n", __FUNCTION__); + bus_release_resource(sc->dev, + SYS_RES_MEMORY, PCIR_BAR(2), sc->bxe_db_res); + } +} + + +/* + * Returns: + * 0 = Success, !0 = Failure + */ +static int +bxe_media_detect(struct bxe_softc *sc) +{ + int rc = 0; + /* Identify supported media based on the PHY type. */ switch (XGXS_EXT_PHY_TYPE(sc->link_params.ext_phy_config)) { case PORT_HW_CFG_XGXS_EXT_PHY_TYPE_DIRECT: @@ -1782,21 +1826,21 @@ bxe_attach(device_t dev) "%s(): Found 10GBase-CX4 media.\n", __FUNCTION__); sc->media = IFM_10G_CX4; break; -#if 0 - /* ToDo: Configure correct media types for these PHYs. */ - case PORT_HW_CFG_XGXS_EXT_PHY_TYPE_BCM8071 - case PORT_HW_CFG_XGXS_EXT_PHY_TYPE_BCM8072 - case PORT_HW_CFG_XGXS_EXT_PHY_TYPE_BCM8073 - case PORT_HW_CFG_XGXS_EXT_PHY_TYPE_BCM8705 - case PORT_HW_CFG_XGXS_EXT_PHY_TYPE_BCM8706 - case PORT_HW_CFG_XGXS_EXT_PHY_TYPE_BCM8726 -#endif + case PORT_HW_CFG_XGXS_EXT_PHY_TYPE_BCM8073: + /* Technically 10GBase-KR but report as 10GBase-SR*/ + case PORT_HW_CFG_XGXS_EXT_PHY_TYPE_BCM8726: case PORT_HW_CFG_XGXS_EXT_PHY_TYPE_BCM8727: case PORT_HW_CFG_XGXS_EXT_PHY_TYPE_BCM8727_NOC: DBPRINT(sc, BXE_INFO_LOAD, "%s(): Found 10GBase-SR media.\n", __FUNCTION__); sc->media = IFM_10G_SR; break; + case PORT_HW_CFG_XGXS_EXT_PHY_TYPE_BCM8705: + case PORT_HW_CFG_XGXS_EXT_PHY_TYPE_BCM8706: + DBPRINT(sc, BXE_INFO_LOAD, + "%s(): Found 10Gb twinax media.\n", __FUNCTION__); + sc->media = IFM_10G_TWINAX; + break; case PORT_HW_CFG_XGXS_EXT_PHY_TYPE_BCM8481: case PORT_HW_CFG_XGXS_EXT_PHY_TYPE_SFX7101: case PORT_HW_CFG_XGXS_EXT_PHY_TYPE_BCM84823: @@ -1811,10 +1855,71 @@ bxe_attach(device_t dev) __FILE__, __LINE__); sc->media = 0; rc = ENODEV; + } + + return (rc); +} + + +/* + * Device attach function. + * + * Allocates device resources, performs secondary chip identification, + * resets and initializes the hardware, and initializes driver instance + * variables. + * + * Returns: + * 0 = Success, Positive value on failure. + */ +static int +bxe_attach(device_t dev) +{ + struct bxe_softc *sc; + struct ifnet *ifp; + int rc; + + sc = device_get_softc(dev); + DBENTER(BXE_VERBOSE_LOAD | BXE_VERBOSE_RESET); + + sc->dev = dev; + sc->bxe_unit = device_get_unit(dev); + sc->bxe_func = pci_get_function(dev); + sc->bxe_flags = 0; + sc->state = BXE_STATE_CLOSED; + rc = 0; + bxe_set_tunables(sc); + + bxe_alloc_mutexes(sc); + + /* Prepare the tick routine. */ + callout_init(&sc->bxe_tick_callout, CALLOUT_MPSAFE); + + /* Enable bus master capability */ + pci_enable_busmaster(dev); + + if ((rc = bxe_alloc_pci_resources(sc)) != 0) + goto bxe_attach_fail; + + /* Put indirect address registers into a sane state. */ + pci_write_config(sc->dev, PCICFG_GRC_ADDRESS, + PCICFG_VENDOR_ID_OFFSET, 4); + REG_WR(sc, PXP2_REG_PGL_ADDR_88_F0 + BP_PORT(sc) * 16, 0); + REG_WR(sc, PXP2_REG_PGL_ADDR_8C_F0 + BP_PORT(sc) * 16, 0); + REG_WR(sc, PXP2_REG_PGL_ADDR_90_F0 + BP_PORT(sc) * 16, 0); + REG_WR(sc, PXP2_REG_PGL_ADDR_94_F0 + BP_PORT(sc) * 16, 0); + + /* Get hardware info from shared memory and validate data. */ + if (bxe_get_function_hwinfo(sc)) { + DBPRINT(sc, BXE_WARN, + "%s(): Failed to get hardware info!\n", __FUNCTION__); + rc = ENODEV; goto bxe_attach_fail; } /* Setup supported media options. */ + if ((rc = bxe_media_detect(sc)) != 0) + goto bxe_attach_fail; + ifmedia_init(&sc->bxe_ifmedia, IFM_IMASK, bxe_ifmedia_upd, bxe_ifmedia_status); ifmedia_add(&sc->bxe_ifmedia, @@ -1823,7 +1928,8 @@ bxe_attach(device_t dev) IFM_ETHER | IFM_AUTO, 0, NULL); ifmedia_set(&sc->bxe_ifmedia, IFM_ETHER | IFM_AUTO); - sc->bxe_ifmedia.ifm_media = sc->bxe_ifmedia.ifm_cur->ifm_media; + sc->bxe_ifmedia.ifm_media = + sc->bxe_ifmedia.ifm_cur->ifm_media; /* Set init arrays */ rc = bxe_init_firmware(sc); @@ -1877,18 +1983,6 @@ bxe_attach(device_t dev) if (!BP_NOMCP(sc)) bxe_undi_unload(sc); - /* Set TPA flag. */ - if (bxe_tpa_enable){ - sc->bxe_flags |= BXE_TPA_ENABLE_FLAG; - }else - sc->bxe_flags &= ~BXE_TPA_ENABLE_FLAG; - - /* Select the PCIe maximum read request size. */ - if (bxe_mrrs > 3) - sc->mrrs = 3; - else - sc->mrrs = bxe_mrrs; - /* * Select the RX and TX ring sizes. The actual * ring size for TX is complicated by the fact @@ -1904,10 +1998,6 @@ bxe_attach(device_t dev) /* Assume receive IP/TCP/UDP checksum is enabled. */ sc->rx_csum = 1; - /* Select the host coalescing tick count values. */ - sc->tx_ticks = bxe_tx_ticks; - sc->rx_ticks = bxe_rx_ticks; - /* Disable WoL. */ sc->wol = 0; @@ -1915,7 +2005,7 @@ bxe_attach(device_t dev) sc->mbuf_alloc_size = MCLBYTES; /* Allocate DMA memory resources. */ - if (bxe_dma_alloc(sc->bxe_dev)) { + if (bxe_dma_alloc(sc->dev)) { BXE_PRINTF("%s(%d): DMA memory allocation failed!\n", __FILE__, __LINE__); rc = ENOMEM; @@ -1937,13 +2027,23 @@ bxe_attach(device_t dev) ifp->if_flags = IFF_BROADCAST | IFF_SIMPLEX | IFF_MULTICAST; ifp->if_ioctl = bxe_ioctl; ifp->if_start = bxe_tx_start; + +#if __FreeBSD_version >= 800000 + ifp->if_transmit = bxe_tx_mq_start; + ifp->if_qflush = bxe_mq_flush; +#endif + #ifdef FreeBSD8_0 ifp->if_timer = 0; #endif + ifp->if_init = bxe_init; ifp->if_mtu = ETHERMTU; ifp->if_hwassist = BXE_IF_HWASSIST; ifp->if_capabilities = BXE_IF_CAPABILITIES; + if (TPA_ENABLED(sc)) { + ifp->if_capabilities |= IFCAP_LRO; + } ifp->if_capenable = ifp->if_capabilities; ifp->if_baudrate = IF_Gbps(10UL); @@ -2892,6 +2992,7 @@ bxe_detach(device_t dev) if (ifp != NULL) ether_ifdetach(ifp); ifmedia_removeall(&sc->bxe_ifmedia); + /* Release all remaining resources. */ bxe_release_resources(sc); pci_disable_busmaster(dev); @@ -3425,7 +3526,7 @@ bxe__link_status_update(struct bxe_softc *sc) { DBENTER(BXE_VERBOSE_PHY); - if (bxe_stats_enable == FALSE || sc->state != BXE_STATE_OPEN) + if (sc->stats_enable == FALSE || sc->state != BXE_STATE_OPEN) return; bxe_link_status_update(&sc->link_params, &sc->link_vars); @@ -3515,9 +3616,8 @@ bxe_initial_phy_init(struct bxe_softc *sc) } } else { - DBPRINT(sc, 1, - "%s(): Bootcode is not running, not initializing link!\n", - __FUNCTION__); + DBPRINT(sc, BXE_FATAL, "%s(): Bootcode is not running, " + "not initializing link!\n", __FUNCTION__); rc = EINVAL; } @@ -3526,6 +3626,65 @@ bxe_initial_phy_init(struct bxe_softc *sc) } +#if __FreeBSD_version >= 800000 +/* + * Allocate buffer rings used for multiqueue. + * + * Returns: + * 0 = Success, !0 = Failure. + */ +static int +bxe_alloc_buf_rings(struct bxe_softc *sc) +{ + struct bxe_fastpath *fp; + int i, rc = 0; + + DBENTER(BXE_VERBOSE_LOAD); + + for (i = 0; i < sc->num_queues; i++) { + fp = &sc->fp[i]; + + if (fp != NULL) { + fp->br = buf_ring_alloc(BXE_BR_SIZE, + M_DEVBUF, M_WAITOK, &fp->mtx); + if (fp->br == NULL) { + rc = ENOMEM; + return(rc); + } + } else + BXE_PRINTF("%s(%d): Bug!\n", __FILE__, __LINE__); + } + + DBEXIT(BXE_VERBOSE_LOAD); + return(rc); +} + +/* + * Releases buffer rings used for multiqueue. + * + * Returns: + * None + */ +static void +bxe_free_buf_rings(struct bxe_softc *sc) +{ + struct bxe_fastpath *fp; + int i; + + DBENTER(BXE_VERBOSE_UNLOAD); + + for (i = 0; i < sc->num_queues; i++) { + fp = &sc->fp[i]; + if (fp != NULL) { + if (fp->br != NULL) + buf_ring_free(fp->br, M_DEVBUF); + } + } + + DBEXIT(BXE_VERBOSE_UNLOAD); +} +#endif + /* * Handles controller initialization. @@ -3558,7 +3717,7 @@ bxe_init_locked(struct bxe_softc *sc, int load_mode) /* Check if the driver is still running and bail out if it is. */ if (ifp->if_drv_flags & IFF_DRV_RUNNING) { - DBPRINT(sc, BXE_WARN, + DBPRINT(sc, BXE_INFO, "%s(): Init called while driver is running!\n", __FUNCTION__); goto bxe_init_locked_exit; @@ -3605,9 +3764,8 @@ bxe_init_locked(struct bxe_softc *sc, int load_mode) if ((load_code == 0) || (load_code == FW_MSG_CODE_DRV_LOAD_REFUSED)) { - DBPRINT(sc, (BXE_WARN_LOAD | BXE_WARN_RESET), - "%s(): Bootcode refused load request.!\n", - __FUNCTION__); + BXE_PRINTF("%s(%d): Bootcode refused load request.!\n", + __FILE__, __LINE__); goto bxe_init_locked_failed1; } } @@ -3632,6 +3790,8 @@ bxe_init_locked(struct bxe_softc *sc, int load_mode) /* Calculate and save the Ethernet MTU size. */ sc->port.ether_mtu = ifp->if_mtu + ETHER_HDR_LEN + (ETHER_VLAN_ENCAP_LEN * 2) + ETHER_CRC_LEN + 4; + DBPRINT(sc, BXE_INFO, "%s(): Setting MTU = %d\n", + __FUNCTION__, sc->port.ether_mtu); /* Setup the mbuf allocation size for RX frames. */ if (sc->port.ether_mtu <= MCLBYTES) @@ -3640,8 +3800,6 @@ bxe_init_locked(struct bxe_softc *sc, int load_mode) sc->mbuf_alloc_size = PAGE_SIZE; else sc->mbuf_alloc_size = MJUM9BYTES; - - DBPRINT(sc, BXE_INFO, "%s(): mbuf_alloc_size = %d, " "max_frame_size = %d\n", __FUNCTION__, sc->mbuf_alloc_size, sc->port.ether_mtu); @@ -3651,7 +3809,7 @@ bxe_init_locked(struct bxe_softc *sc, int load_mode) if ((load_code == FW_MSG_CODE_DRV_LOAD_COMMON) && (sc->common.shmem2_base)){ - if (bxe_dcc_enable) { + if (sc->dcc_enable == TRUE) { BXE_PRINTF("Enabing DCC support\n"); SHMEM2_WR(sc, dcc_support, (SHMEM_DCC_SUPPORT_DISABLE_ENABLE_PF_TLV | @@ -3659,6 +3817,15 @@ bxe_init_locked(struct bxe_softc *sc, int load_mode) } } +#if __FreeBSD_version >= 800000 + /* Allocate buffer rings for multiqueue operation. */ + if (bxe_alloc_buf_rings(sc)) { + BXE_PRINTF("%s(%d): Buffer ring initialization failed, " + "aborting!\n", __FILE__, __LINE__); + goto bxe_init_locked_failed1; + } +#endif + /* Tell MCP that driver load is done. */ if (!BP_NOMCP(sc)) { load_code = bxe_fw_command(sc, DRV_MSG_CODE_LOAD_DONE); @@ -3676,8 +3843,8 @@ bxe_init_locked(struct bxe_softc *sc, int load_mode) /* Setup the leading connection for the controller. */ if (bxe_setup_leading(sc)) - DBPRINT(sc, 1, "%s(): Initial PORT_SETUP ramrod failed. " - "State is not OPEN!\n", __FUNCTION__); + DBPRINT(sc, BXE_FATAL, "%s(): Initial PORT_SETUP ramrod " + "failed. State is not OPEN!\n", __FUNCTION__); if (CHIP_IS_E1H(sc)) { @@ -3778,6 +3945,10 @@ bxe_init_locked_failed1: } sc->port.pmf = 0; +#if __FreeBSD_version >= 800000 + bxe_free_buf_rings(sc); +#endif + DBPRINT(sc, BXE_INFO, "%s(): Initialization failed!\n", __FUNCTION__); bxe_init_locked_exit: @@ -3832,7 +4003,7 @@ bxe_wait_ramrod(struct bxe_softc *sc, int state, int idx, int *state_p, } /* We timed out polling for a completion. */ - DBPRINT(sc, 1, "%s(): Timeout %s for state 0x%08X on fp[%d]. " + DBPRINT(sc, BXE_FATAL, "%s(): Timeout %s for state 0x%08X on fp[%d]. " "Got 0x%x instead\n", __FUNCTION__, poll ? "polling" : "waiting", state, idx, *state_p); @@ -4186,6 +4357,7 @@ bxe_init(void *xsc) DBEXIT(BXE_VERBOSE_RESET | BXE_VERBOSE_UNLOAD); } + /* * Release all resources used by the driver. * @@ -4203,10 +4375,15 @@ bxe_release_resources(struct bxe_softc *sc) DBENTER(BXE_VERBOSE_RESET | BXE_VERBOSE_UNLOAD); - dev = sc->bxe_dev; + dev = sc->dev; + + /* Release the FreeBSD interface. */ + if (sc->bxe_ifp != NULL) + if_free(sc->bxe_ifp); /* Release interrupt resources. */ bxe_interrupt_detach(sc); + if ((sc->bxe_flags & BXE_USING_MSIX_FLAG) && sc->msix_count) { for (i = 0; i < sc->msix_count; i++) { @@ -4239,39 +4416,22 @@ bxe_release_resources(struct bxe_softc *sc) BXE_VERBOSE_INTR), "%s(): Releasing legacy interrupt.\n", __FUNCTION__); if (sc->bxe_irq_res != NULL) - bus_release_resource(dev, SYS_RES_IRQ, sc->bxe_irq_rid, - sc->bxe_irq_res); + bus_release_resource(dev, SYS_RES_IRQ, + sc->bxe_irq_rid, sc->bxe_irq_res); } - /* Free the DMA memory */ + /* Free the DMA resources. */ bxe_dma_free(sc); - /* Release the PCIe BAR0 mapped memory */ - if (sc->bxe_res != NULL) { - DBPRINT(sc, (BXE_VERBOSE_LOAD | BXE_VERBOSE_RESET), - "%s(): Releasing PCI BAR0 memory.\n", __FUNCTION__); - bus_release_resource(dev, SYS_RES_MEMORY, PCIR_BAR(0), - sc->bxe_res); - } - - /* Release the PCIe BAR2 mapped memory */ - if (sc->bxe_db_res != NULL) { - DBPRINT(sc, (BXE_VERBOSE_LOAD | BXE_VERBOSE_RESET), - "%s(): Releasing PCI BAR2 memory.\n", __FUNCTION__); - bus_release_resource(dev, SYS_RES_MEMORY, PCIR_BAR(2), - sc->bxe_db_res); - } + bxe_release_pci_resources(sc); - /* Release the FreeBSD interface. */ - if (sc->bxe_ifp != NULL) - if_free(sc->bxe_ifp); +#if __FreeBSD_version >= 800000 + /* Free multiqueue buffer rings. */ + bxe_free_buf_rings(sc); +#endif - BXE_CORE_LOCK_DESTROY(sc); - BXE_SP_LOCK_DESTROY(sc); - BXE_DMAE_LOCK_DESTROY(sc); - BXE_PHY_LOCK_DESTROY(sc); - BXE_FWMB_LOCK_DESTROY(sc); - BXE_PRINT_LOCK_DESTROY(sc); + /* Free remaining fastpath resources. */ + bxe_free_mutexes(sc); } @@ -4291,11 +4451,11 @@ bxe_reg_wr_ind(struct bxe_softc *sc, uint32_t offset, uint32_t val) DBPRINT(sc, BXE_INSANE, "%s(); offset = 0x%08X, val = 0x%08X\n", __FUNCTION__, offset, val); - pci_write_config(sc->bxe_dev, PCICFG_GRC_ADDRESS, offset, 4); - pci_write_config(sc->bxe_dev, PCICFG_GRC_DATA, val, 4); + pci_write_config(sc->dev, PCICFG_GRC_ADDRESS, offset, 4); + pci_write_config(sc->dev, PCICFG_GRC_DATA, val, 4); /* Return to a safe address. */ - pci_write_config(sc->bxe_dev, PCICFG_GRC_ADDRESS, + pci_write_config(sc->dev, PCICFG_GRC_ADDRESS, PCICFG_VENDOR_ID_OFFSET, 4); } @@ -4315,11 +4475,11 @@ bxe_reg_rd_ind(struct bxe_softc *sc, uint32_t offset) { uint32_t val; - pci_write_config(sc->bxe_dev, PCICFG_GRC_ADDRESS, offset, 4); - val = pci_read_config(sc->bxe_dev, PCICFG_GRC_DATA, 4); + pci_write_config(sc->dev, PCICFG_GRC_ADDRESS, offset, 4); + val = pci_read_config(sc->dev, PCICFG_GRC_DATA, 4); /* Return to a safe address. */ - pci_write_config(sc->bxe_dev, PCICFG_GRC_ADDRESS, + pci_write_config(sc->dev, PCICFG_GRC_ADDRESS, PCICFG_VENDOR_ID_OFFSET, 4); DBPRINT(sc, BXE_INSANE, "%s(); offset = 0x%08X, val = 0x%08X\n", @@ -4393,8 +4553,8 @@ bxe_write_dmae(struct bxe_softc *sc, bus_addr_t dma_addr, uint32_t dst_addr, if (!sc->dmae_ready) { data = BXE_SP(sc, wb_data[0]); - DBPRINT(sc, 1, "%s(): DMAE not ready, using indirect.\n", - __FUNCTION__); + DBPRINT(sc, BXE_WARN, "%s(): DMAE not ready, " + "using indirect.\n", __FUNCTION__); bxe_init_ind_wr(sc, dst_addr, data, len32); goto bxe_write_dmae_exit; @@ -4433,7 +4593,7 @@ bxe_write_dmae(struct bxe_softc *sc, bus_addr_t dma_addr, uint32_t dst_addr, timeout = 4000; while (*wb_comp != BXE_WB_COMP_VAL) { if (!timeout) { - DBPRINT(sc, 1, + DBPRINT(sc, BXE_FATAL, "%s(): DMAE timeout (dst_addr = 0x%08X, len = %d)!\n", __FUNCTION__, dst_addr, len32); break; @@ -4477,8 +4637,8 @@ bxe_read_dmae(struct bxe_softc *sc, uint32_t src_addr, if (!sc->dmae_ready) { data = BXE_SP(sc, wb_data[0]); - DBPRINT(sc, 1, "%s(): DMAE not ready, using indirect.\n", - __FUNCTION__); + DBPRINT(sc, BXE_WARN, "%s(): DMAE not ready, " + "using indirect.\n", __FUNCTION__); for (i = 0; i < len32; i++) data[i] = bxe_reg_rd_ind(sc, src_addr + i * 4); @@ -4520,7 +4680,7 @@ bxe_read_dmae(struct bxe_softc *sc, uint32_t src_addr, timeout = 4000; while (*wb_comp != BXE_WB_COMP_VAL) { if (!timeout) { - DBPRINT(sc, 1, + DBPRINT(sc, BXE_FATAL, "%s(): DMAE timeout (src_addr = 0x%08X, len = %d)!\n", __FUNCTION__, src_addr, len32); break; @@ -5578,7 +5738,7 @@ bxe_acquire_hw_lock(struct bxe_softc *sc, uint32_t resource) int cnt, rc; DBENTER(BXE_VERBOSE_MISC); - DBPRINT(sc, BXE_VERBOSE, "%s(): Locking resource 0x%08X\n", + DBPRINT(sc, BXE_VERBOSE_MISC, "%s(): Locking resource 0x%08X\n", __FUNCTION__, resource); func = BP_FUNC(sc); @@ -5642,7 +5802,7 @@ bxe_release_hw_lock(struct bxe_softc *sc, uint32_t resource) int rc; DBENTER(BXE_VERBOSE_MISC); - DBPRINT(sc, BXE_VERBOSE, "%s(): Unlocking resource 0x%08X\n", + DBPRINT(sc, BXE_VERBOSE_MISC, "%s(): Unlocking resource 0x%08X\n", __FUNCTION__, resource); resource_bit = 1 << resource; @@ -6013,7 +6173,7 @@ bxe_link_attn(struct bxe_softc *sc) if (IS_E1HMF(sc)) { port = BP_PORT(sc); if (sc->link_vars.link_up) { - if (bxe_dcc_enable) { + if (sc->dcc_enable == TRUE) { bxe_congestionmgmt(sc, TRUE); /* Store in internal memory. */ for (i = 0; i < @@ -6739,7 +6899,7 @@ bxe_attn_int_deasserted3(struct bxe_softc *sc, uint32_t attn) SHMEM_RD(sc, mf_cfg.func_mf_config[(sc->bxe_func & 1)].config); val = SHMEM_RD(sc, func_mb[func].drv_status); - if (bxe_dcc_enable) { + if (sc->dcc_enable == TRUE) { if (val & DRV_STATUS_DCC_EVENT_MASK) bxe_dcc_event(sc, val & DRV_STATUS_DCC_EVENT_MASK); @@ -6830,11 +6990,16 @@ bxe_attn_int_deasserted(struct bxe_softc *sc, uint32_t deasserted) port = BP_PORT(sc); /* Get the current attention signal bits. */ - attn.sig[0] = REG_RD(sc, MISC_REG_AEU_AFTER_INVERT_1_FUNC_0 + port * 4); - attn.sig[1] = REG_RD(sc, MISC_REG_AEU_AFTER_INVERT_2_FUNC_0 + port * 4); - attn.sig[2] = REG_RD(sc, MISC_REG_AEU_AFTER_INVERT_3_FUNC_0 + port * 4); - attn.sig[3] = REG_RD(sc, MISC_REG_AEU_AFTER_INVERT_4_FUNC_0 + port * 4); - DBPRINT(sc, BXE_VERBOSE, + attn.sig[0] = REG_RD(sc, + MISC_REG_AEU_AFTER_INVERT_1_FUNC_0 + port * 4); + attn.sig[1] = REG_RD(sc, + MISC_REG_AEU_AFTER_INVERT_2_FUNC_0 + port * 4); + attn.sig[2] = REG_RD(sc, + MISC_REG_AEU_AFTER_INVERT_3_FUNC_0 + port * 4); + attn.sig[3] = REG_RD(sc, + MISC_REG_AEU_AFTER_INVERT_4_FUNC_0 + port * 4); + + DBPRINT(sc, BXE_EXTREME_INTR, "%s(): attention = 0x%08X 0x%08X 0x%08X 0x%08X\n", __FUNCTION__, attn.sig[0], attn.sig[1], attn.sig[2], attn.sig[3]); @@ -6846,7 +7011,7 @@ bxe_attn_int_deasserted(struct bxe_softc *sc, uint32_t deasserted) if (deasserted & (1 << index)) { group_mask = sc->attn_group[index]; - DBPRINT(sc, 1, /*BXE_EXTREME_INTR,*/ + DBPRINT(sc, BXE_EXTREME_INTR, "%s(): group[%02d] = 0x%08X 0x%08X 0x%08x 0X%08x\n", __FUNCTION__, index, group_mask.sig[0], group_mask.sig[1], group_mask.sig[2], @@ -6875,10 +7040,11 @@ bxe_attn_int_deasserted(struct bxe_softc *sc, uint32_t deasserted) bxe_release_alr(sc); - reg_addr = (HC_REG_COMMAND_REG + port * 32 + COMMAND_REG_ATTN_BITS_CLR); + reg_addr = (HC_REG_COMMAND_REG + + port * 32 + COMMAND_REG_ATTN_BITS_CLR); val = ~deasserted; - DBPRINT(sc, BXE_VERBOSE_INTR, + DBPRINT(sc, BXE_EXTREME_INTR, "%s(): About to mask 0x%08X at HC addr 0x%08X\n", __FUNCTION__, deasserted, reg_addr); REG_WR(sc, reg_addr, val); @@ -6896,6 +7062,7 @@ bxe_attn_int_deasserted(struct bxe_softc *sc, uint32_t deasserted) "%s(): Current aeu_mask = 0x%08X, newly deasserted = 0x%08X\n", __FUNCTION__, aeu_mask, deasserted); aeu_mask |= (deasserted & 0xff); + DBPRINT(sc, BXE_EXTREME_INTR, "%s(): New aeu_mask = 0x%08X\n", __FUNCTION__, aeu_mask); @@ -6904,6 +7071,7 @@ bxe_attn_int_deasserted(struct bxe_softc *sc, uint32_t deasserted) DBPRINT(sc, BXE_EXTREME_INTR, "%s(): Current attn_state = 0x%08X\n", __FUNCTION__, sc->attn_state); + sc->attn_state &= ~deasserted; DBPRINT(sc, BXE_EXTREME_INTR, "%s(): New attn_state = 0x%08X\n", __FUNCTION__, sc->attn_state); @@ -7223,7 +7391,7 @@ bxe_stats_init(struct bxe_softc *sc) DBENTER(BXE_VERBOSE_STATS); - if (bxe_stats_enable == FALSE) + if (sc->stats_enable == FALSE) return; port = BP_PORT(sc); @@ -8940,10 +9108,10 @@ bxe_tx_encap(struct bxe_fastpath *fp, struct mbuf **m_head) * FW job easy... */ tx_start_bd->nbd++; - DBPRINT(sc, 1, /*BXE_EXTREME_SEND,*/ - "%s(): TSO split headr size is %d (%x:%x) nbds %d\n", - __FUNCTION__, tx_start_bd->nbytes, tx_start_bd->addr_hi, - tx_start_bd->addr_lo, nbds); + DBPRINT(sc, BXE_EXTREME_SEND, + "%s(): TSO split headr size is %d (%x:%x) nbds %d\n", + __FUNCTION__, tx_start_bd->nbytes, tx_start_bd->addr_hi, + tx_start_bd->addr_lo, nbds); bd_prod = TX_BD(NEXT_TX_BD(bd_prod)); @@ -8962,10 +9130,10 @@ bxe_tx_encap(struct bxe_fastpath *fp, struct mbuf **m_head) * not marked with the start flag. */ - DBPRINT(sc, BXE_EXTREME_SEND, "%s(): TSO split data " - "size is %d (%x:%x)\n", __FUNCTION__, - tx_data_bd->nbytes, tx_data_bd->addr_hi, - tx_data_bd->addr_lo); + DBPRINT(sc, BXE_EXTREME_SEND, + "%s(): TSO split data size is %d (%x:%x)\n", + __FUNCTION__, tx_data_bd->nbytes, + tx_data_bd->addr_hi, tx_data_bd->addr_lo); } /* @@ -9068,12 +9236,9 @@ bxe_tx_encap(struct bxe_fastpath *fp, struct mbuf **m_head) return(rc); } + /* - * Transmit dispatch routine. - * - * This routine acts as a dispatcher, deciding which TX chain will be used - * to send an outgoing frame and then adding the frame to that chain. It is - * called by the OS when it's ready to transmit a frame. + * Legacy (non-RSS) dispatch routine. * * Returns: * Nothing. @@ -9087,36 +9252,29 @@ bxe_tx_start(struct ifnet *ifp) sc = ifp->if_softc; DBENTER(BXE_EXTREME_SEND); - /* Exit if there's no link. */ - if (!sc->link_vars.link_up) { + /* Exit if the transmit queue is full or link down. */ + if (((ifp->if_drv_flags & (IFF_DRV_RUNNING | IFF_DRV_OACTIVE)) != + IFF_DRV_RUNNING) || !sc->link_vars.link_up) { DBPRINT(sc, BXE_VERBOSE_SEND, - "%s(): No link, ignoring transmit request.\n", - __FUNCTION__); - goto bxe_tx_start_exit; - } - - /* Exit if the transmit queue is full. */ - if ((ifp->if_drv_flags & (IFF_DRV_RUNNING | IFF_DRV_OACTIVE)) != - IFF_DRV_RUNNING) { - DBPRINT(sc, BXE_VERBOSE_SEND, "%s(): Driver running but " - "transmit queue full, ignoring transmit request.\n", - __FUNCTION__); + "%s(): No link or TX queue full, ignoring " + "transmit request.\n", __FUNCTION__); goto bxe_tx_start_exit; } /* Set the TX queue for the frame. */ fp = &sc->fp[0]; - BXE_CORE_LOCK(sc); + BXE_FP_LOCK(fp); bxe_tx_start_locked(ifp, fp); - BXE_CORE_UNLOCK(sc); + BXE_FP_UNLOCK(fp); bxe_tx_start_exit: DBEXIT(BXE_EXTREME_SEND); } + /* - * Main transmit routine. + * Legacy (non-RSS) transmit routine. * * Returns: * Nothing. @@ -9131,6 +9289,8 @@ bxe_tx_start_locked(struct ifnet *ifp, struct bxe_fastpath *fp) sc = fp->sc; DBENTER(BXE_EXTREME_SEND); + BXE_FP_LOCK_ASSERT(fp); + /* Keep adding entries while there are frames to send. */ while (!IFQ_DRV_IS_EMPTY(&ifp->if_snd)) { @@ -9178,13 +9338,221 @@ bxe_tx_start_locked(struct ifnet *ifp, struct bxe_fastpath *fp) /* No TX packets were dequeued. */ if (tx_count > 0) /* Reset the TX watchdog timeout timer. */ - sc->watchdog_timer = BXE_TX_TIMEOUT; + fp->watchdog_timer = BXE_TX_TIMEOUT; else fp->tx_start_called_on_empty_queue++; DBEXIT(BXE_EXTREME_SEND); } +#if __FreeBSD_version >= 800000 +/* + * Multiqueue (RSS) dispatch routine. + * + * Returns: + * 0 if transmit succeeds, !0 otherwise. + */ +static int +bxe_tx_mq_start(struct ifnet *ifp, struct mbuf *m) +{ + struct bxe_softc *sc; + struct bxe_fastpath *fp; + int fp_index, rc; + + sc = ifp->if_softc; + fp_index = 0; + + DBENTER(BXE_EXTREME_SEND); + + /* Map the flow ID to a queue number. */ + if ((m->m_flags & M_FLOWID) != 0) { + fp_index = m->m_pkthdr.flowid % sc->num_queues; + DBPRINT(sc, BXE_EXTREME_SEND, + "%s(): Found flowid %d\n", + __FUNCTION__, fp_index); + } else { + DBPRINT(sc, BXE_EXTREME_SEND, + "%s(): No flowid found, using %d\n", + __FUNCTION__, fp_index); + } + + + /* Select the fastpath TX queue for the frame. */ + fp = &sc->fp[fp_index]; + + /* Exit if the transmit queue is full or link down. */ + if (((ifp->if_drv_flags & (IFF_DRV_RUNNING | IFF_DRV_OACTIVE)) != + IFF_DRV_RUNNING) || !sc->link_vars.link_up) { + /* We're stuck with the mbuf. Stash it for now. */ + DBPRINT(sc, BXE_EXTREME_SEND, + "%s(): TX queue full/link down, " + "parking mbuf...\n", __FUNCTION__); + rc = drbr_enqueue(ifp, fp->br, m); + /* DRC - Setup a task to try again. */ + /* taskqueue_enqueue(tq, task); */ + goto bxe_tx_mq_start_exit; + } + + BXE_FP_LOCK(fp); + rc = bxe_tx_mq_start_locked(ifp, fp, m); + BXE_FP_UNLOCK(fp); + +bxe_tx_mq_start_exit: + DBEXIT(BXE_EXTREME_SEND); + return(rc); +} + + +/* + * Multiqueue (RSS) transmit routine. + * + * Returns: + * 0 if transmit succeeds, !0 otherwise. + */ +static int +bxe_tx_mq_start_locked(struct ifnet *ifp, + struct bxe_fastpath *fp, struct mbuf *m) +{ + struct bxe_softc *sc; + struct mbuf *next; + int rc = 0, tx_count = 0; + + sc = fp->sc; + + DBENTER(BXE_EXTREME_SEND); + DBPRINT(sc, BXE_EXTREME_SEND, + "%s(): fp[%02d], drbr queue depth=%d\n", + __FUNCTION__, fp->index, drbr_inuse(ifp, fp->br)); + + BXE_FP_LOCK_ASSERT(fp); + + if (m == NULL) { + /* Check for any other work. */ + DBPRINT(sc, BXE_EXTREME_SEND, + "%s(): No initial work, dequeue mbuf...\n", + __FUNCTION__); + next = drbr_dequeue(ifp, fp->br); + } else if (drbr_needs_enqueue(ifp, fp->br)) { + /* Work pending, queue mbuf to maintain packet order. */ + DBPRINT(sc, BXE_EXTREME_SEND, + "%s(): Found queued data pending...\n", + __FUNCTION__); + if ((rc = drbr_enqueue(ifp, fp->br, m)) != 0) { + DBPRINT(sc, BXE_EXTREME_SEND, + "%s(): Enqueue failed...\n", + __FUNCTION__); + goto bxe_tx_mq_start_locked_exit; + } + DBPRINT(sc, BXE_EXTREME_SEND, + "%s(): Dequeueing old mbuf...\n", + __FUNCTION__); + next = drbr_dequeue(ifp, fp->br); + } else { + /* Work with the mbuf we have. */ + DBPRINT(sc, BXE_EXTREME_SEND, + "%s(): Start with current mbuf...\n", + __FUNCTION__); + next = m; + } + + /* Keep adding entries while there are frames to send. */ + while (next != NULL) { + + /* The transmit mbuf now belongs to us, keep track of it. */ + DBRUN(fp->tx_mbuf_alloc++); + + /* + * Pack the data into the transmit ring. If we + * don't have room, place the mbuf back at the + * head of the TX queue, set the OACTIVE flag, + * and wait for the NIC to drain the chain. + */ + if (__predict_false(bxe_tx_encap(fp, &next))) { + DBPRINT(sc, BXE_WARN, "%s(): TX encap failure...\n", + __FUNCTION__); + fp->tx_encap_failures++; + /* Very Bad Frames(tm) may have been dropped. */ + if (next != NULL) { + /* + * Mark the TX queue as full and save + * the frame. + */ + ifp->if_drv_flags |= IFF_DRV_OACTIVE; + DBPRINT(sc, BXE_EXTREME_SEND, + "%s(): Save mbuf for another time...\n", + __FUNCTION__); + rc = drbr_enqueue(ifp, fp->br, next); + DBRUN(fp->tx_mbuf_alloc--); + sc->eth_stats.driver_xoff++; + } + + /* Stop looking for more work. */ + break; + } + + /* The transmit frame was enqueued successfully. */ + tx_count++; + + /* Send a copy of the frame to any BPF listeners. */ + BPF_MTAP(ifp, next); + + DBPRINT(sc, BXE_EXTREME_SEND, + "%s(): Check for queued mbufs...\n", + __FUNCTION__); + next = drbr_dequeue(ifp, fp->br); + } + + DBPRINT(sc, BXE_EXTREME_SEND, + "%s(): Enqueued %d mbufs...\n", + __FUNCTION__, tx_count); + + /* No TX packets were dequeued. */ + if (tx_count > 0) { + /* Reset the TX watchdog timeout timer. */ + fp->watchdog_timer = BXE_TX_TIMEOUT; + } else { + fp->tx_start_called_on_empty_queue++; + } + +bxe_tx_mq_start_locked_exit: + DBEXIT(BXE_EXTREME_SEND); + return(rc); +} + + +static void +bxe_mq_flush(struct ifnet *ifp) +{ + struct bxe_softc *sc; + struct bxe_fastpath *fp; + struct mbuf *m; + int i; + + sc = ifp->if_softc; + + DBENTER(BXE_VERBOSE_UNLOAD); + + for (i = 0; i < sc->num_queues; i++) { + fp = &sc->fp[i]; + + DBPRINT(sc, BXE_VERBOSE_UNLOAD, "%s(): Clearing fp[%02d]...\n", + __FUNCTION__, fp->index); + + if (fp->br != NULL) { + BXE_FP_LOCK(fp); + while ((m = buf_ring_dequeue_sc(fp->br)) != NULL) + m_freem(m); + BXE_FP_UNLOCK(fp); + } + } + + if_qflush(ifp); + + DBEXIT(BXE_VERBOSE_UNLOAD); +} +#endif /* FreeBSD_version >= 800000 */ + + /* * Handles any IOCTL calls from the operating system. * @@ -9289,11 +9657,13 @@ bxe_ioctl(struct ifnet *ifp, u_long command, caddr_t data) /* Toggle the LRO capabilites enable flag. */ if (mask & IFCAP_LRO) { - ifp->if_capenable ^= IFCAP_LRO; - sc->bxe_flags ^= BXE_TPA_ENABLE_FLAG; - DBPRINT(sc, BXE_INFO_MISC, - "%s(): Toggling LRO (bxe_flags = 0x%08X).\n", - __FUNCTION__, sc->bxe_flags); + if (TPA_ENABLED(sc)) { + ifp->if_capenable ^= IFCAP_LRO; + sc->bxe_flags ^= BXE_TPA_ENABLE_FLAG; + DBPRINT(sc, BXE_INFO_MISC, + "%s(): Toggling LRO (bxe_flags = " + "0x%08X).\n", __FUNCTION__, sc->bxe_flags); + } reinit = 1; } @@ -9633,8 +10003,9 @@ bxe_intr_fp (void *xfp) DBENTER(BXE_EXTREME_INTR); - DBPRINT(sc, BXE_EXTREME_INTR, - "%s(%d): MSI-X vector on fp[%d].sb_id = %d\n", __FUNCTION__, curcpu, fp->index, fp->sb_id); + DBPRINT(sc, BXE_VERBOSE_INTR, + "%s(%d): MSI-X vector on fp[%d].sb_id = %d\n", + __FUNCTION__, curcpu, fp->index, fp->sb_id); /* Don't handle any interrupts if we're not ready. */ if (__predict_false(sc->intr_sem != 0)) @@ -9674,17 +10045,17 @@ bxe_task_fp (void *xfp, int pending) DBPRINT(sc, BXE_EXTREME_INTR, "%s(): pending = %d.\n", __FUNCTION__, pending); - DBPRINT(sc, BXE_EXTREME_INTR, "%s(%d): FP task on fp[%d].sb_id = %d\n", - __FUNCTION__, curcpu, fp->index, fp->sb_id); + DBPRINT(sc, BXE_EXTREME_INTR, "%s(%d): Fastpath task on fp[%d]" + ".sb_id = %d\n", __FUNCTION__, curcpu, fp->index, fp->sb_id); /* Update the fast path indices */ bxe_update_fpsb_idx(fp); /* Service any completed TX frames. */ if (bxe_has_tx_work(fp)) { - BXE_CORE_LOCK(sc); + BXE_FP_LOCK(fp); bxe_txeof(fp); - BXE_CORE_UNLOCK(sc); + BXE_FP_UNLOCK(fp); } /* Service any completed RX frames. */ @@ -10292,11 +10663,77 @@ bxe_alloc_rx_sge_exit: return (rc); } + +/* + * Returns: + * None. + */ +static void +bxe_alloc_mutexes(struct bxe_softc *sc) +{ + struct bxe_fastpath *fp; + int i; + + DBENTER(BXE_VERBOSE_LOAD); + + BXE_CORE_LOCK_INIT(sc, device_get_nameunit(sc->dev)); + BXE_SP_LOCK_INIT(sc, "bxe_sp_lock"); + BXE_DMAE_LOCK_INIT(sc, "bxe_dmae_lock"); + BXE_PHY_LOCK_INIT(sc, "bxe_phy_lock"); + BXE_FWMB_LOCK_INIT(sc, "bxe_fwmb_lock"); + BXE_PRINT_LOCK_INIT(sc, "bxe_print_lock"); + + /* Allocate one mutex for each fastpath structure. */ + for (i=0; i < sc->num_queues; i++ ) { + fp = &sc->fp[i]; + + /* Allocate per fastpath mutexes. */ + snprintf(fp->mtx_name, sizeof(fp->mtx_name), "%s:fp[%02d]", + device_get_nameunit(sc->dev), fp->index); + mtx_init(&fp->mtx, fp->mtx_name, NULL, MTX_DEF); + } + + DBEXIT(BXE_VERBOSE_LOAD); +} + +/* + * Returns: + * None. + */ +static void +bxe_free_mutexes(struct bxe_softc *sc) +{ + struct bxe_fastpath *fp; + int i; + + DBENTER(BXE_VERBOSE_UNLOAD); + + for (i=0; i < sc->num_queues; i++ ) { + fp = &sc->fp[i]; + + /* Release per fastpath mutexes. */ + if (mtx_initialized(&(fp->mtx))) + mtx_destroy(&(fp->mtx)); + } + + BXE_PRINT_LOCK_DESTROY(sc); + BXE_FWMB_LOCK_DESTROY(sc); + BXE_PHY_LOCK_DESTROY(sc); + BXE_DMAE_LOCK_DESTROY(sc); + BXE_SP_LOCK_DESTROY(sc); + BXE_CORE_LOCK_DESTROY(sc); + + DBEXIT(BXE_VERBOSE_UNLOAD); + +} + + + /* * Initialize the receive rings. * * Returns: - * 0 = Success, !0 = Failure. + * None. */ static void bxe_init_rx_chains(struct bxe_softc *sc) @@ -10478,7 +10915,7 @@ bxe_init_rx_chains(struct bxe_softc *sc) /* Update the driver's copy of the producer indices. */ fp->rx_bd_prod = rx_bd_prod; - fp->rx_cq_prod = MAX_RCQ_ENTRIES; + fp->rx_cq_prod = TOTAL_RCQ_ENTRIES; fp->rx_pkts = fp->rx_calls = 0; DBPRINT(sc, (BXE_VERBOSE_LOAD | BXE_VERBOSE_RESET), @@ -10544,6 +10981,7 @@ bxe_init_tx_chains(struct bxe_softc *sc) for (i = 0; i < sc->num_queues; i++) { fp = &sc->fp[i]; + DBPRINT(sc, (BXE_INSANE_LOAD | BXE_INSANE_RESET), "%s(): Linking fp[%d] TX chain pages.\n", __FUNCTION__, i); @@ -10779,7 +11217,7 @@ bxe_init_context(struct bxe_softc *sc) /* Enable packet alignment/pad and statistics. */ context->ustorm_st_context.common.flags = USTORM_ETH_ST_CONTEXT_CONFIG_ENABLE_MC_ALIGNMENT; - if (bxe_stats_enable) + if (sc->stats_enable == TRUE) context->ustorm_st_context.common.flags |= USTORM_ETH_ST_CONTEXT_CONFIG_ENABLE_STATISTICS; context->ustorm_st_context.common.statistics_counter_id=cl_id; @@ -11323,7 +11761,7 @@ bxe_init_internal(struct bxe_softc *sc, uint32_t load_code) * Perform driver instance specific initialization. * * Returns: - * 0 = Success, !0 = Failure + * None */ static void bxe_init_nic(struct bxe_softc *sc, uint32_t load_code) @@ -11978,7 +12416,7 @@ bxe_init_pxp(struct bxe_softc *sc) uint16_t devctl; int r_order, w_order; - devctl = pci_read_config(sc->bxe_dev, + devctl = pci_read_config(sc->dev, sc->pcie_cap + PCI_EXP_DEVCTL, 2); DBPRINT(sc, (BXE_VERBOSE_LOAD | BXE_VERBOSE_RESET), "%s(): Read 0x%x from devctl\n", __FUNCTION__, devctl); @@ -13254,7 +13692,7 @@ bxe_dma_alloc(device_t dev) /* * Check required size before mapping to conserve resources. */ - if (bxe_tso_enable) { + if (sc->tso_enable == TRUE) { max_size = BXE_TSO_MAX_SIZE; max_segments = BXE_TSO_MAX_SEGMENTS; max_seg_size = BXE_TSO_MAX_SEG_SIZE; @@ -13988,21 +14426,21 @@ bxe_set_rx_mode(struct bxe_softc *sc) * multicast address filtering. */ if (ifp->if_flags & IFF_PROMISC) { - DBPRINT(sc, BXE_INFO, "%s(): Enabling promiscuous mode.\n", - __FUNCTION__); + DBPRINT(sc, BXE_VERBOSE_MISC, + "%s(): Enabling promiscuous mode.\n", __FUNCTION__); /* Enable promiscuous mode. */ rx_mode = BXE_RX_MODE_PROMISC; } else if ((ifp->if_flags & IFF_ALLMULTI) || (ifp->if_amcount > BXE_MAX_MULTICAST)) { - DBPRINT(sc, BXE_INFO, "%s(): Enabling all multicast mode.\n", - __FUNCTION__); + DBPRINT(sc, BXE_VERBOSE_MISC, + "%s(): Enabling all multicast mode.\n", __FUNCTION__); /* Enable all multicast addresses. */ rx_mode = BXE_RX_MODE_ALLMULTI; } else { /* Enable selective multicast mode. */ - DBPRINT(sc, BXE_INFO, + DBPRINT(sc, BXE_VERBOSE_MISC, "%s(): Enabling selective multicast mode.\n", __FUNCTION__); @@ -15016,6 +15454,12 @@ bxe_rxeof(struct bxe_fastpath *fp) m->m_flags |= M_VLANTAG; } +#if __FreeBSD_version >= 800000 + /* Tell OS what RSS queue was used for this flow. */ + m->m_pkthdr.flowid = fp->index; + m->m_flags |= M_FLOWID; +#endif + /* Last chance to check for problems. */ DBRUN(bxe_validate_rx_packet(fp, rx_cq_cons, cqe, m)); @@ -15087,6 +15531,8 @@ bxe_txeof(struct bxe_fastpath *fp) ifp = sc->bxe_ifp; DBENTER(BXE_EXTREME_SEND); + DBPRINT(sc, BXE_EXTREME_SEND, "%s(): Servicing fp[%d]\n", + __FUNCTION__, fp->index); /* Get the hardware's view of the TX packet consumer index. */ hw_pkt_cons = le16toh(*fp->tx_cons_sb); @@ -15167,13 +15613,13 @@ bxe_txeof(struct bxe_fastpath *fp) * Clear the watchdog timer if we've emptied * the TX chain. */ - sc->watchdog_timer = 0; + fp->watchdog_timer = 0; } else { /* * Reset the watchdog timer if we still have * transmits pending. */ - sc->watchdog_timer = BXE_TX_TIMEOUT; + fp->watchdog_timer = BXE_TX_TIMEOUT; } } @@ -15259,23 +15705,31 @@ bxe_get_buf_exit: * Transmit timeout handler. * * Returns: - * None. + * 0 = No timeout, !0 = timeout occurred. */ -static void -bxe_watchdog(struct bxe_softc *sc) +static int +bxe_watchdog(struct bxe_fastpath *fp) { + struct bxe_softc *sc = fp->sc; + int rc = 0; DBENTER(BXE_INSANE_SEND); - BXE_CORE_LOCK_ASSERT(sc); - - if (sc->watchdog_timer == 0 || --sc->watchdog_timer) + BXE_FP_LOCK(fp); + if (fp->watchdog_timer == 0 || --fp->watchdog_timer) { + rc = EINVAL; + BXE_FP_UNLOCK(fp); goto bxe_watchdog_exit; + } + BXE_FP_UNLOCK(fp); - BXE_PRINTF("TX watchdog timeout occurred, resetting!\n"); + BXE_PRINTF("TX watchdog timeout occurred on fp[%02d], " + "resetting!\n", fp->index); /* DBRUNLV(BXE_FATAL, bxe_breakpoint(sc)); */ + BXE_CORE_LOCK(sc); + /* Mark the interface as down. */ sc->bxe_ifp->if_drv_flags &= ~IFF_DRV_RUNNING; @@ -15283,10 +15737,14 @@ bxe_watchdog(struct bxe_softc *sc) DELAY(10000); bxe_init_locked(sc, LOAD_OPEN); + BXE_CORE_UNLOCK(sc); + bxe_watchdog_exit: DBEXIT(BXE_INSANE_SEND); + return(rc); } + /* * Change the MTU size for the port. The MTU should be validated before * calling this routine. @@ -15328,21 +15786,28 @@ static void bxe_tick(void *xsc) { struct bxe_softc *sc; -//zz uint32_t drv_pulse, mcp_pulse; - int func; + struct bxe_fastpath *fp; +#if 0 + /* Re-enable at a later time. */ + uint32_t drv_pulse, mcp_pulse; +#endif + int i, func; sc = xsc; DBENTER(BXE_INSANE_MISC); + + /* Check for TX timeouts on any fastpath. */ + for (i = 0; i < sc->num_queues; i++) { + fp = &sc->fp[i]; + if (bxe_watchdog(fp) != 0) + break; + } + BXE_CORE_LOCK(sc); -func = BP_FUNC(sc); + func = BP_FUNC(sc); + /* Schedule the next tick. */ callout_reset(&sc->bxe_tick_callout, hz, bxe_tick, sc); - /* - * Check if a transmit timeout has - * occurred and reset the device if - * necessary to get back in service. - */ - bxe_watchdog(sc); #if 0 if (!BP_NOMCP(sc)) { @@ -15633,9 +16098,9 @@ static void bxe_add_sysctls(struct bxe_softc *sc) { struct sysctl_ctx_list *ctx = - device_get_sysctl_ctx(sc->bxe_dev); + device_get_sysctl_ctx(sc->dev); struct sysctl_oid_list *children = - SYSCTL_CHILDREN(device_get_sysctl_tree(sc->bxe_dev)); + SYSCTL_CHILDREN(device_get_sysctl_tree(sc->dev)); struct bxe_eth_stats *estats = &sc->eth_stats; SYSCTL_ADD_UINT(ctx, children, OID_AUTO, @@ -15669,14 +16134,14 @@ bxe_add_sysctls(struct bxe_softc *sc) 0, "Total unicast packets received (lo)"); SYSCTL_ADD_UINT(ctx, children, OID_AUTO, - "estats_total_bytes_transmitted_hi", - CTLFLAG_RD, &estats->total_bytes_transmitted_hi, - 0, "Total bytes transmitted (hi)"); + "estats_total_bytes_transmitted_hi", + CTLFLAG_RD, &estats->total_bytes_transmitted_hi, + 0, "Total bytes transmitted (hi)"); SYSCTL_ADD_UINT(ctx, children, OID_AUTO, - "estats_total_bytes_transmitted_lo", - CTLFLAG_RD, &estats->total_bytes_transmitted_lo, - 0, "Total bytes transmitted (lo)"); + "estats_total_bytes_transmitted_lo", + CTLFLAG_RD, &estats->total_bytes_transmitted_lo, + 0, "Total bytes transmitted (lo)"); SYSCTL_ADD_UINT(ctx, children, OID_AUTO, "estats_total_unicast_packets_transmitted_hi", @@ -16494,8 +16959,8 @@ void bxe_dump_tx_chain(struct bxe_fastpath * fp, int tx_bd_prod, int count) val_hi = U64_HI(fp->tx_bd_chain_paddr); val_lo = U64_LO(fp->tx_bd_chain_paddr); BXE_PRINTF( - "0x%08X:%08X - (fp->tx_bd_chain_paddr) TX Chain physical address\n", - val_hi, val_lo); + "0x%08X:%08X - (fp[%02d]->tx_bd_chain_paddr) TX Chain physical address\n", + val_hi, val_lo, fp->index); BXE_PRINTF( "page size = 0x%08X, tx chain pages = 0x%08X\n", (uint32_t)BCM_PAGE_SIZE, (uint32_t)NUM_TX_PAGES); @@ -16532,8 +16997,6 @@ void bxe_dump_tx_chain(struct bxe_fastpath * fp, int tx_bd_prod, int count) } /* Don't skip next page pointers. */ tx_bd_prod = ((tx_bd_prod + 1) & MAX_TX_BD); - - /* tx_bd_prod = TX_BD(NEXT_TX_BD(tx_bd_prod)); */ } BXE_PRINTF( diff --git a/sys/dev/bxe/if_bxe.h b/sys/dev/bxe/if_bxe.h index e1d6e46..2ef9c66 100644 --- a/sys/dev/bxe/if_bxe.h +++ b/sys/dev/bxe/if_bxe.h @@ -76,12 +76,12 @@ /* * Device identification definitions. */ -#define BRCM_VENDORID 0x14E4 +#define BRCM_VENDORID 0x14E4 #define BRCM_DEVICEID_BCM57710 0x164E #define BRCM_DEVICEID_BCM57711 0x164F #define BRCM_DEVICEID_BCM57711E 0x1650 -#define PCI_ANY_ID (u_int16_t) (~0U) +#define PCI_ANY_ID (u_int16_t) (~0U) struct bxe_type { @@ -137,6 +137,8 @@ struct bxe_type { mtx_lock(&(sc->bxe_core_mtx)) #define BXE_SP_LOCK(sc) \ mtx_lock(&(sc->bxe_sp_mtx)) +#define BXE_FP_LOCK(fp) \ + mtx_lock(&(fp->mtx)) #define BXE_DMAE_LOCK(sc) \ mtx_lock(&(sc->bxe_dmae_mtx)) #define BXE_PHY_LOCK(sc) \ @@ -151,6 +153,8 @@ struct bxe_type { mtx_assert(&(sc->bxe_core_mtx), MA_OWNED) #define BXE_SP_LOCK_ASSERT(sc) \ mtx_assert(&(sc->bxe_sp_mtx), MA_OWNED) +#define BXE_FP_LOCK_ASSERT(fp) \ + mtx_assert(&(fp->mtx), MA_OWNED) #define BXE_DMAE_LOCK_ASSERT(sc) \ mtx_assert(&(sc->bxe_dmae_mtx), MA_OWNED) #define BXE_PHY_LOCK_ASSERT(sc) \ @@ -160,6 +164,8 @@ struct bxe_type { mtx_unlock(&(sc->bxe_core_mtx)) #define BXE_SP_UNLOCK(sc) \ mtx_unlock(&(sc->bxe_sp_mtx)) +#define BXE_FP_UNLOCK(fp) \ + mtx_unlock(&(fp->mtx)) #define BXE_DMAE_UNLOCK(sc) \ mtx_unlock(&(sc->bxe_dmae_mtx)) #define BXE_PHY_UNLOCK(sc) \ @@ -414,12 +420,12 @@ struct bxe_type { #if __FreeBSD_version < 700000 #define BXE_IF_CAPABILITIES \ (IFCAP_VLAN_MTU | IFCAP_VLAN_HWTAGGING | IFCAP_HWCSUM | \ - IFCAP_JUMBO_MTU | IFCAP_LRO) + IFCAP_JUMBO_MTU) #else /* TSO was introduced in FreeBSD 7 */ #define BXE_IF_CAPABILITIES \ (IFCAP_VLAN_MTU | IFCAP_VLAN_HWTAGGING | IFCAP_HWCSUM | \ - IFCAP_JUMBO_MTU | IFCAP_LRO | IFCAP_TSO4 | IFCAP_VLAN_HWCSUM) + IFCAP_JUMBO_MTU | IFCAP_TSO4 | IFCAP_VLAN_HWCSUM) #endif /* Some typical Ethernet frame sizes */ @@ -458,6 +464,7 @@ struct bxe_type { #define MIN_BXE_BC_VER 0x00040200 +#define BXE_BR_SIZE 4096 #define BXE_NO_RX_FLAGS \ (TSTORM_ETH_DROP_FLAGS_DROP_ALL_PACKETS) @@ -923,6 +930,9 @@ struct bxe_fastpath { /* Pointer back to parent structure. */ struct bxe_softc *sc; + struct mtx mtx; + char mtx_name[16]; + /* Hardware maintained status block. */ bus_dma_tag_t status_block_tag; bus_dmamap_t status_block_map; @@ -968,6 +978,9 @@ struct bxe_fastpath { union eth_rx_cqe *rx_cq_chain[NUM_RCQ_PAGES]; bus_addr_t rx_cq_chain_paddr[NUM_RCQ_PAGES]; + /* Ticks until chip reset. */ + int watchdog_timer; + /* Taskqueue reqources. */ struct task task; struct taskqueue *tq; @@ -1077,6 +1090,10 @@ struct bxe_fastpath { uint16_t free_rx_bd; +#if __FreeBSD_version >= 800000 + struct buf_ring *br; +#endif + /* Recieve/transmit packet counters. */ unsigned long rx_pkts; unsigned long tx_pkts; @@ -1133,11 +1150,14 @@ struct bxe_fastpath { /* ToDo: Audit this structure for unused varaibles. */ struct bxe_softc { + /* + * MUST start with ifnet pointer (see definition of miibus_statchg()). + */ struct ifnet *bxe_ifp; int media; /* Parent device handle. */ - device_t bxe_dev; + device_t dev; /* Driver instance number. */ u_int8_t bxe_unit; @@ -1309,9 +1329,6 @@ struct bxe_softc { uint16_t pcie_cap; uint16_t pm_cap; - /* PCIe maximum read request size. */ - int mrrs; - /* ToDo: Is this really needed? */ uint16_t sp_running; @@ -1362,8 +1379,20 @@ struct bxe_softc { #define BXE_STATE_DIAG 0xE000 #define BXE_STATE_ERROR 0xF000 +/* Driver tunable options. */ int int_mode; int multi_mode; + int tso_enable; + int num_queues; + int stats_enable; + int mrrs; + int dcc_enable; + +#define BXE_NUM_QUEUES(cos) \ + ((bxe_qs_per_cos & (0xff << (cos * 8))) >> (cos * 8)) +#define BXE_MAX_QUEUES(sc) \ + (IS_E1HMF(sc) ? (MAX_CONTEXT / E1HVN_MAX) : MAX_CONTEXT) + #define BXE_MAX_COS 3 #define BXE_MAX_PRIORITY 8 @@ -1381,14 +1410,6 @@ struct bxe_softc { /* Class of service to queue mapping. */ uint8_t cos_map[BXE_MAX_COS]; - /* The number of fastpath queues (for RSS/multi-queue). */ - int num_queues; - -#define BXE_NUM_QUEUES(cos) \ - ((bxe_qs_per_cos & (0xff << (cos * 8))) >> (cos * 8)) -#define BXE_MAX_QUEUES(sc) \ - (IS_E1HMF(sc) ? (MAX_CONTEXT / E1HVN_MAX) : MAX_CONTEXT) - /* Used for multiple function devices. */ uint32_t mf_config[E1HVN_MAX]; @@ -1477,9 +1498,6 @@ struct bxe_softc { uint16_t tx_driver; - /* Ticks until chip reset. */ - int watchdog_timer; - /* Verify bxe_function_init is run before handling interrupts. */ uint8_t intr_sem; |