summaryrefslogtreecommitdiff
path: root/sys/dev/pci/if_mcx.c
diff options
context:
space:
mode:
Diffstat (limited to 'sys/dev/pci/if_mcx.c')
-rw-r--r--sys/dev/pci/if_mcx.c424
1 files changed, 254 insertions, 170 deletions
diff --git a/sys/dev/pci/if_mcx.c b/sys/dev/pci/if_mcx.c
index 4471acef409..69a5e6ace3c 100644
--- a/sys/dev/pci/if_mcx.c
+++ b/sys/dev/pci/if_mcx.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: if_mcx.c,v 1.53 2020/06/13 07:09:59 jmatthew Exp $ */
+/* $OpenBSD: if_mcx.c,v 1.54 2020/06/14 04:48:16 dlg Exp $ */
/*
* Copyright (c) 2017 David Gwynne <dlg@openbsd.org>
@@ -1919,6 +1919,41 @@ struct mcx_calibration {
#define MCX_CALIBRATE_FIRST 2
#define MCX_CALIBRATE_NORMAL 30
+struct mcx_rx {
+ struct mcx_softc *rx_softc;
+ struct ifiqueue *rx_ifiq;
+
+ int rx_tirn;
+ int rx_rqn;
+ struct mcx_dmamem rx_rq_mem;
+ struct mcx_slot *rx_slots;
+ uint32_t *rx_doorbell;
+
+ uint32_t rx_prod;
+ struct timeout rx_refill;
+ struct if_rxring rx_rxr;
+} __aligned(64);
+
+struct mcx_tx {
+ struct mcx_softc *tx_softc;
+ struct ifqueue *tx_ifq;
+
+ int tx_tisn;
+ int tx_sqn;
+ struct mcx_dmamem tx_sq_mem;
+ struct mcx_slot *tx_slots;
+ uint32_t *tx_doorbell;
+ int tx_bf_offset;
+
+ uint32_t tx_cons;
+ uint32_t tx_prod;
+} __aligned(64);
+
+struct mcx_queues {
+ struct mcx_rx q_rx;
+ struct mcx_tx q_tx;
+};
+
struct mcx_softc {
struct device sc_dev;
struct arpcom sc_ac;
@@ -1959,6 +1994,8 @@ struct mcx_softc {
int sc_hardmtu;
int sc_rxbufsz;
+ int sc_bf_size;
+
struct task sc_port_change;
int sc_flow_table_id;
@@ -1979,31 +2016,11 @@ struct mcx_softc {
unsigned int sc_calibration_gen;
struct timeout sc_calibrate;
+ struct mcx_queues sc_queues[1];
+ unsigned int sc_nqueues;
+
struct mcx_cq sc_cq[MCX_MAX_CQS];
int sc_num_cq;
-
- /* rx */
- int sc_tirn;
- int sc_rqn;
- struct mcx_dmamem sc_rq_mem;
- struct mcx_slot *sc_rx_slots;
- uint32_t *sc_rx_doorbell;
-
- uint32_t sc_rx_prod;
- struct timeout sc_rx_refill;
- struct if_rxring sc_rxr;
-
- /* tx */
- int sc_tisn;
- int sc_sqn;
- struct mcx_dmamem sc_sq_mem;
- struct mcx_slot *sc_tx_slots;
- uint32_t *sc_tx_doorbell;
- int sc_bf_size;
- int sc_bf_offset;
-
- uint32_t sc_tx_cons;
- uint32_t sc_tx_prod;
};
#define DEVNAME(_sc) ((_sc)->sc_dev.dv_xname)
@@ -2032,24 +2049,24 @@ static int mcx_query_special_contexts(struct mcx_softc *);
static int mcx_set_port_mtu(struct mcx_softc *, int);
static int mcx_create_cq(struct mcx_softc *, int);
static int mcx_destroy_cq(struct mcx_softc *, int);
-static int mcx_create_sq(struct mcx_softc *, int);
-static int mcx_destroy_sq(struct mcx_softc *);
-static int mcx_ready_sq(struct mcx_softc *);
-static int mcx_create_rq(struct mcx_softc *, int);
-static int mcx_destroy_rq(struct mcx_softc *);
-static int mcx_ready_rq(struct mcx_softc *);
-static int mcx_create_tir(struct mcx_softc *);
-static int mcx_destroy_tir(struct mcx_softc *);
-static int mcx_create_tis(struct mcx_softc *);
-static int mcx_destroy_tis(struct mcx_softc *);
+static int mcx_create_sq(struct mcx_softc *, struct mcx_tx *, int);
+static int mcx_destroy_sq(struct mcx_softc *, struct mcx_tx *);
+static int mcx_ready_sq(struct mcx_softc *, struct mcx_tx *);
+static int mcx_create_rq(struct mcx_softc *, struct mcx_rx *, int);
+static int mcx_destroy_rq(struct mcx_softc *, struct mcx_rx *);
+static int mcx_ready_rq(struct mcx_softc *, struct mcx_rx *);
+static int mcx_create_tir(struct mcx_softc *, struct mcx_rx *);
+static int mcx_destroy_tir(struct mcx_softc *, struct mcx_rx *);
+static int mcx_create_tis(struct mcx_softc *, struct mcx_tx *);
+static int mcx_destroy_tis(struct mcx_softc *, struct mcx_tx *);
static int mcx_create_flow_table(struct mcx_softc *, int);
static int mcx_set_flow_table_root(struct mcx_softc *);
static int mcx_destroy_flow_table(struct mcx_softc *);
static int mcx_create_flow_group(struct mcx_softc *, int, int,
int, int, struct mcx_flow_match *);
static int mcx_destroy_flow_group(struct mcx_softc *, int);
-static int mcx_set_flow_table_entry(struct mcx_softc *, int, int,
- uint8_t *);
+static int mcx_set_flow_table_entry(struct mcx_softc *,
+ struct mcx_rx *, int, int, uint8_t *);
static int mcx_delete_flow_table_entry(struct mcx_softc *, int, int);
#if 0
@@ -2070,7 +2087,8 @@ static int mcx_process_rx(struct mcx_softc *, struct mcx_cq_entry *,
struct mbuf_list *, const struct mcx_calibration *);
static void mcx_process_txeof(struct mcx_softc *, struct mcx_cq_entry *,
int *);
-static void mcx_process_cq(struct mcx_softc *, struct mcx_cq *);
+static void mcx_process_cq(struct mcx_softc *, struct mcx_queues *,
+ struct mcx_cq *);
static void mcx_arm_cq(struct mcx_softc *, struct mcx_cq *);
static void mcx_arm_eq(struct mcx_softc *);
@@ -2355,6 +2373,8 @@ mcx_attach(struct device *parent, struct device *self, void *aux)
printf(", %s, address %s\n", intrstr,
ether_sprintf(sc->sc_ac.ac_enaddr));
+ sc->sc_nqueues = nitems(sc->sc_queues);
+
strlcpy(ifp->if_xname, DEVNAME(sc), IFNAMSIZ);
ifp->if_softc = sc;
ifp->if_flags = IFF_BROADCAST | IFF_MULTICAST | IFF_SIMPLEX;
@@ -2380,7 +2400,24 @@ mcx_attach(struct device *parent, struct device *self, void *aux)
if_attach(ifp);
ether_ifattach(ifp);
- timeout_set(&sc->sc_rx_refill, mcx_refill, sc);
+ if_attach_iqueues(ifp, sc->sc_nqueues);
+ if_attach_queues(ifp, sc->sc_nqueues);
+ for (i = 0; i < sc->sc_nqueues; i++) {
+ struct ifiqueue *ifiq = ifp->if_iqs[i];
+ struct ifqueue *ifq = ifp->if_ifqs[i];
+ struct mcx_rx *rx = &sc->sc_queues[i].q_rx;
+ struct mcx_tx *tx = &sc->sc_queues[i].q_tx;
+
+ rx->rx_softc = sc;
+ rx->rx_ifiq = ifiq;
+ timeout_set(&rx->rx_refill, mcx_refill, rx);
+ ifiq->ifiq_softc = rx;
+
+ tx->tx_softc = sc;
+ tx->tx_ifq = ifq;
+ ifq->ifq_softc = tx;
+ }
+
timeout_set(&sc->sc_calibrate, mcx_calibrate, sc);
task_set(&sc->sc_port_change, mcx_port_change, sc);
@@ -3463,6 +3500,7 @@ free:
static int
mcx_iff(struct mcx_softc *sc)
{
+ struct mcx_rx *rx = &sc->sc_queues[0].q_rx;
struct ifnet *ifp = &sc->sc_ac.ac_if;
struct mcx_dmamem mxm;
struct mcx_cmdq_entry *cqe;
@@ -3476,7 +3514,7 @@ mcx_iff(struct mcx_softc *sc)
/* enable or disable the promisc flow */
if (ISSET(ifp->if_flags, IFF_PROMISC)) {
if (sc->sc_promisc_flow_enabled == 0) {
- mcx_set_flow_table_entry(sc, MCX_FLOW_GROUP_PROMISC,
+ mcx_set_flow_table_entry(sc, rx, MCX_FLOW_GROUP_PROMISC,
0, NULL);
sc->sc_promisc_flow_enabled = 1;
}
@@ -3492,8 +3530,8 @@ mcx_iff(struct mcx_softc *sc)
memset(mcast, 0, sizeof(mcast));
mcast[0] = 0x01;
- mcx_set_flow_table_entry(sc, MCX_FLOW_GROUP_ALLMULTI,
- 0, mcast);
+ mcx_set_flow_table_entry(sc, rx,
+ MCX_FLOW_GROUP_ALLMULTI, 0, mcast);
sc->sc_allmulti_flow_enabled = 1;
}
} else if (sc->sc_allmulti_flow_enabled != 0) {
@@ -4022,7 +4060,7 @@ mcx_destroy_cq(struct mcx_softc *sc, int index)
}
static int
-mcx_create_rq(struct mcx_softc *sc, int cqn)
+mcx_create_rq(struct mcx_softc *sc, struct mcx_rx *rx, int cqn)
{
struct mcx_cmdq_entry *cqe;
struct mcx_dmamem mxm;
@@ -4040,7 +4078,7 @@ mcx_create_rq(struct mcx_softc *sc, int cqn)
paslen = npages * sizeof(*pas);
insize = 0x10 + sizeof(struct mcx_rq_ctx) + paslen;
- if (mcx_dmamem_alloc(sc, &sc->sc_rq_mem, npages * MCX_PAGE_SIZE,
+ if (mcx_dmamem_alloc(sc, &rx->rx_rq_mem, npages * MCX_PAGE_SIZE,
MCX_PAGE_SIZE) != 0) {
printf("%s: unable to allocate receive queue memory\n",
DEVNAME(sc));
@@ -4079,7 +4117,7 @@ mcx_create_rq(struct mcx_softc *sc, int cqn)
mbin->rq_wq.wq_log_size = MCX_LOG_RQ_SIZE;
/* physical addresses follow the mailbox in data */
- mcx_cmdq_mboxes_pas(&mxm, sizeof(*mbin) + 0x10, npages, &sc->sc_rq_mem);
+ mcx_cmdq_mboxes_pas(&mxm, sizeof(*mbin) + 0x10, npages, &rx->rx_rq_mem);
mcx_cmdq_post(sc, cqe, 0);
error = mcx_cmdq_poll(sc, cqe, 1000);
@@ -4100,10 +4138,10 @@ mcx_create_rq(struct mcx_softc *sc, int cqn)
goto free;
}
- sc->sc_rqn = mcx_get_id(out->cmd_rqn);
+ rx->rx_rqn = mcx_get_id(out->cmd_rqn);
doorbell = MCX_DMA_KVA(&sc->sc_doorbell_mem);
- sc->sc_rx_doorbell = (uint32_t *)(doorbell + MCX_RQ_DOORBELL_OFFSET);
+ rx->rx_doorbell = (uint32_t *)(doorbell + MCX_RQ_DOORBELL_OFFSET);
free:
mcx_dmamem_free(sc, &mxm);
@@ -4111,7 +4149,7 @@ free:
}
static int
-mcx_ready_rq(struct mcx_softc *sc)
+mcx_ready_rq(struct mcx_softc *sc, struct mcx_rx *rx)
{
struct mcx_cmdq_entry *cqe;
struct mcx_dmamem mxm;
@@ -4129,7 +4167,7 @@ mcx_ready_rq(struct mcx_softc *sc)
in = mcx_cmdq_in(cqe);
in->cmd_opcode = htobe16(MCX_CMD_MODIFY_RQ);
in->cmd_op_mod = htobe16(0);
- in->cmd_rq_state = htobe32((MCX_QUEUE_STATE_RST << 28) | sc->sc_rqn);
+ in->cmd_rq_state = htobe32((MCX_QUEUE_STATE_RST << 28) | rx->rx_rqn);
if (mcx_cmdq_mboxes_alloc(sc, &mxm, 1,
&cqe->cq_input_ptr, token) != 0) {
@@ -4167,7 +4205,7 @@ free:
}
static int
-mcx_destroy_rq(struct mcx_softc *sc)
+mcx_destroy_rq(struct mcx_softc *sc, struct mcx_rx *rx)
{
struct mcx_cmdq_entry *cqe;
struct mcx_cmd_destroy_rq_in *in;
@@ -4182,7 +4220,7 @@ mcx_destroy_rq(struct mcx_softc *sc)
in = mcx_cmdq_in(cqe);
in->cmd_opcode = htobe16(MCX_CMD_DESTROY_RQ);
in->cmd_op_mod = htobe16(0);
- in->cmd_rqn = htobe32(sc->sc_rqn);
+ in->cmd_rqn = htobe32(rx->rx_rqn);
mcx_cmdq_post(sc, cqe, 0);
error = mcx_cmdq_poll(sc, cqe, 1000);
@@ -4202,12 +4240,12 @@ mcx_destroy_rq(struct mcx_softc *sc)
return -1;
}
- sc->sc_rqn = 0;
+ rx->rx_rqn = 0;
return 0;
}
static int
-mcx_create_tir(struct mcx_softc *sc)
+mcx_create_tir(struct mcx_softc *sc, struct mcx_rx *rx)
{
struct mcx_cmdq_entry *cqe;
struct mcx_dmamem mxm;
@@ -4234,7 +4272,7 @@ mcx_create_tir(struct mcx_softc *sc)
}
mbin = mcx_cq_mbox_data(mcx_cq_mbox(&mxm, 0));
/* leave disp_type = 0, so packets get sent to the inline rqn */
- mbin->cmd_inline_rqn = htobe32(sc->sc_rqn);
+ mbin->cmd_inline_rqn = htobe32(rx->rx_rqn);
mbin->cmd_tdomain = htobe32(sc->sc_tdomain);
mcx_cmdq_post(sc, cqe, 0);
@@ -4256,14 +4294,14 @@ mcx_create_tir(struct mcx_softc *sc)
goto free;
}
- sc->sc_tirn = mcx_get_id(out->cmd_tirn);
+ rx->rx_tirn = mcx_get_id(out->cmd_tirn);
free:
mcx_dmamem_free(sc, &mxm);
return (error);
}
static int
-mcx_destroy_tir(struct mcx_softc *sc)
+mcx_destroy_tir(struct mcx_softc *sc, struct mcx_rx *rx)
{
struct mcx_cmdq_entry *cqe;
struct mcx_cmd_destroy_tir_in *in;
@@ -4278,7 +4316,7 @@ mcx_destroy_tir(struct mcx_softc *sc)
in = mcx_cmdq_in(cqe);
in->cmd_opcode = htobe16(MCX_CMD_DESTROY_TIR);
in->cmd_op_mod = htobe16(0);
- in->cmd_tirn = htobe32(sc->sc_tirn);
+ in->cmd_tirn = htobe32(rx->rx_tirn);
mcx_cmdq_post(sc, cqe, 0);
error = mcx_cmdq_poll(sc, cqe, 1000);
@@ -4298,12 +4336,12 @@ mcx_destroy_tir(struct mcx_softc *sc)
return -1;
}
- sc->sc_tirn = 0;
- return 0;
+ rx->rx_tirn = 0;
+ return (0);
}
static int
-mcx_create_sq(struct mcx_softc *sc, int cqn)
+mcx_create_sq(struct mcx_softc *sc, struct mcx_tx *tx, int cqn)
{
struct mcx_cmdq_entry *cqe;
struct mcx_dmamem mxm;
@@ -4320,7 +4358,7 @@ mcx_create_sq(struct mcx_softc *sc, int cqn)
paslen = npages * sizeof(*pas);
insize = sizeof(struct mcx_sq_ctx) + paslen;
- if (mcx_dmamem_alloc(sc, &sc->sc_sq_mem, npages * MCX_PAGE_SIZE,
+ if (mcx_dmamem_alloc(sc, &tx->tx_sq_mem, npages * MCX_PAGE_SIZE,
MCX_PAGE_SIZE) != 0) {
printf("%s: unable to allocate send queue memory\n",
DEVNAME(sc));
@@ -4350,7 +4388,7 @@ mcx_create_sq(struct mcx_softc *sc, int cqn)
(1 << MCX_SQ_CTX_MIN_WQE_INLINE_SHIFT));
mbin->sq_cqn = htobe32(cqn);
mbin->sq_tis_lst_sz = htobe32(1 << MCX_SQ_CTX_TIS_LST_SZ_SHIFT);
- mbin->sq_tis_num = htobe32(sc->sc_tisn);
+ mbin->sq_tis_num = htobe32(tx->tx_tisn);
mbin->sq_wq.wq_type = MCX_WQ_CTX_TYPE_CYCLIC;
mbin->sq_wq.wq_pd = htobe32(sc->sc_pd);
mbin->sq_wq.wq_uar_page = htobe32(sc->sc_uar);
@@ -4361,7 +4399,7 @@ mcx_create_sq(struct mcx_softc *sc, int cqn)
/* physical addresses follow the mailbox in data */
mcx_cmdq_mboxes_pas(&mxm, sizeof(*mbin) + 0x10,
- npages, &sc->sc_sq_mem);
+ npages, &tx->tx_sq_mem);
mcx_cmdq_post(sc, cqe, 0);
error = mcx_cmdq_poll(sc, cqe, 1000);
@@ -4382,18 +4420,17 @@ mcx_create_sq(struct mcx_softc *sc, int cqn)
goto free;
}
- sc->sc_sqn = mcx_get_id(out->cmd_sqn);
+ tx->tx_sqn = mcx_get_id(out->cmd_sqn);
doorbell = MCX_DMA_KVA(&sc->sc_doorbell_mem);
- sc->sc_tx_doorbell =
- (uint32_t *)(doorbell + MCX_SQ_DOORBELL_OFFSET + 4);
+ tx->tx_doorbell = (uint32_t *)(doorbell + MCX_SQ_DOORBELL_OFFSET + 4);
free:
mcx_dmamem_free(sc, &mxm);
return (error);
}
static int
-mcx_destroy_sq(struct mcx_softc *sc)
+mcx_destroy_sq(struct mcx_softc *sc, struct mcx_tx *tx)
{
struct mcx_cmdq_entry *cqe;
struct mcx_cmd_destroy_sq_in *in;
@@ -4408,7 +4445,7 @@ mcx_destroy_sq(struct mcx_softc *sc)
in = mcx_cmdq_in(cqe);
in->cmd_opcode = htobe16(MCX_CMD_DESTROY_SQ);
in->cmd_op_mod = htobe16(0);
- in->cmd_sqn = htobe32(sc->sc_sqn);
+ in->cmd_sqn = htobe32(tx->tx_sqn);
mcx_cmdq_post(sc, cqe, 0);
error = mcx_cmdq_poll(sc, cqe, 1000);
@@ -4428,12 +4465,12 @@ mcx_destroy_sq(struct mcx_softc *sc)
return -1;
}
- sc->sc_sqn = 0;
+ tx->tx_sqn = 0;
return 0;
}
static int
-mcx_ready_sq(struct mcx_softc *sc)
+mcx_ready_sq(struct mcx_softc *sc, struct mcx_tx *tx)
{
struct mcx_cmdq_entry *cqe;
struct mcx_dmamem mxm;
@@ -4451,7 +4488,7 @@ mcx_ready_sq(struct mcx_softc *sc)
in = mcx_cmdq_in(cqe);
in->cmd_opcode = htobe16(MCX_CMD_MODIFY_SQ);
in->cmd_op_mod = htobe16(0);
- in->cmd_sq_state = htobe32((MCX_QUEUE_STATE_RST << 28) | sc->sc_sqn);
+ in->cmd_sq_state = htobe32((MCX_QUEUE_STATE_RST << 28) | tx->tx_sqn);
if (mcx_cmdq_mboxes_alloc(sc, &mxm, 1,
&cqe->cq_input_ptr, token) != 0) {
@@ -4489,7 +4526,7 @@ free:
}
static int
-mcx_create_tis(struct mcx_softc *sc)
+mcx_create_tis(struct mcx_softc *sc, struct mcx_tx *tx)
{
struct mcx_cmdq_entry *cqe;
struct mcx_dmamem mxm;
@@ -4537,14 +4574,14 @@ mcx_create_tis(struct mcx_softc *sc)
goto free;
}
- sc->sc_tisn = mcx_get_id(out->cmd_tisn);
+ tx->tx_tisn = mcx_get_id(out->cmd_tisn);
free:
mcx_dmamem_free(sc, &mxm);
return (error);
}
static int
-mcx_destroy_tis(struct mcx_softc *sc)
+mcx_destroy_tis(struct mcx_softc *sc, struct mcx_tx *tx)
{
struct mcx_cmdq_entry *cqe;
struct mcx_cmd_destroy_tis_in *in;
@@ -4559,7 +4596,7 @@ mcx_destroy_tis(struct mcx_softc *sc)
in = mcx_cmdq_in(cqe);
in->cmd_opcode = htobe16(MCX_CMD_DESTROY_TIS);
in->cmd_op_mod = htobe16(0);
- in->cmd_tisn = htobe32(sc->sc_tisn);
+ in->cmd_tisn = htobe32(tx->tx_tisn);
mcx_cmdq_post(sc, cqe, 0);
error = mcx_cmdq_poll(sc, cqe, 1000);
@@ -4579,7 +4616,7 @@ mcx_destroy_tis(struct mcx_softc *sc)
return -1;
}
- sc->sc_tirn = 0;
+ tx->tx_tisn = 0;
return 0;
}
@@ -4917,8 +4954,8 @@ free:
}
static int
-mcx_set_flow_table_entry(struct mcx_softc *sc, int group, int index,
- uint8_t *macaddr)
+mcx_set_flow_table_entry(struct mcx_softc *sc, struct mcx_rx *rx,
+ int group, int index, uint8_t *macaddr)
{
struct mcx_cmdq_entry *cqe;
struct mcx_dmamem mxm;
@@ -4955,7 +4992,7 @@ mcx_set_flow_table_entry(struct mcx_softc *sc, int group, int index,
(((char *)mcx_cq_mbox_data(mcx_cq_mbox(&mxm, 1))) + 0x130);
mbin->cmd_flow_ctx.fc_action = htobe32(MCX_FLOW_CONTEXT_ACTION_FORWARD);
mbin->cmd_flow_ctx.fc_dest_list_size = htobe32(1);
- *dest = htobe32(sc->sc_tirn | MCX_FLOW_CONTEXT_DEST_TYPE_TIR);
+ *dest = htobe32(rx->rx_tirn | MCX_FLOW_CONTEXT_DEST_TYPE_TIR);
/* the only thing we match on at the moment is the dest mac address */
if (macaddr != NULL) {
@@ -5368,7 +5405,7 @@ mcx_dump_sq(struct mcx_softc *sc)
in = mcx_cmdq_in(cqe);
in->cmd_opcode = htobe16(MCX_CMD_QUERY_SQ);
in->cmd_op_mod = htobe16(0);
- in->cmd_sqn = htobe32(sc->sc_sqn);
+ in->cmd_sqn = htobe32(tx->tx_sqn);
CTASSERT(sizeof(*mbout) <= MCX_CMDQ_MAILBOX_DATASIZE*2);
if (mcx_cmdq_mboxes_alloc(sc, &mxm, 2,
@@ -5565,8 +5602,8 @@ free:
#endif
int
-mcx_rx_fill_slots(struct mcx_softc *sc, void *ring, struct mcx_slot *slots,
- uint *prod, uint nslots)
+mcx_rx_fill_slots(struct mcx_softc *sc, struct mcx_rx *rx,
+ void *ring, struct mcx_slot *slots, uint *prod, uint nslots)
{
struct mcx_rq_entry *rqe;
struct mcx_slot *ms;
@@ -5604,7 +5641,7 @@ mcx_rx_fill_slots(struct mcx_softc *sc, void *ring, struct mcx_slot *slots,
}
if (fills != 0) {
- *sc->sc_rx_doorbell = htobe32(p & MCX_WQ_DOORBELL_MASK);
+ *rx->rx_doorbell = htobe32(p & MCX_WQ_DOORBELL_MASK);
/* barrier? */
}
@@ -5614,41 +5651,43 @@ mcx_rx_fill_slots(struct mcx_softc *sc, void *ring, struct mcx_slot *slots,
}
int
-mcx_rx_fill(struct mcx_softc *sc)
+mcx_rx_fill(struct mcx_softc *sc, struct mcx_rx *rx)
{
u_int slots;
- slots = if_rxr_get(&sc->sc_rxr, (1 << MCX_LOG_RQ_SIZE));
+ slots = if_rxr_get(&rx->rx_rxr, (1 << MCX_LOG_RQ_SIZE));
if (slots == 0)
return (1);
- slots = mcx_rx_fill_slots(sc, MCX_DMA_KVA(&sc->sc_rq_mem),
- sc->sc_rx_slots, &sc->sc_rx_prod, slots);
- if_rxr_put(&sc->sc_rxr, slots);
+ slots = mcx_rx_fill_slots(sc, rx, MCX_DMA_KVA(&rx->rx_rq_mem),
+ rx->rx_slots, &rx->rx_prod, slots);
+ if_rxr_put(&rx->rx_rxr, slots);
return (0);
}
void
-mcx_refill(void *xsc)
+mcx_refill(void *xrx)
{
- struct mcx_softc *sc = xsc;
+ struct mcx_rx *rx = xrx;
+ struct mcx_softc *sc = rx->rx_softc;
- mcx_rx_fill(sc);
+ mcx_rx_fill(sc, rx);
- if (if_rxr_inuse(&sc->sc_rxr) == 0)
- timeout_add(&sc->sc_rx_refill, 1);
+ if (if_rxr_inuse(&rx->rx_rxr) == 0)
+ timeout_add(&rx->rx_refill, 1);
}
void
mcx_process_txeof(struct mcx_softc *sc, struct mcx_cq_entry *cqe, int *txfree)
{
+ struct mcx_tx *tx = &sc->sc_queues[0].q_tx;
struct mcx_slot *ms;
bus_dmamap_t map;
int slot, slots;
slot = betoh16(cqe->cq_wqe_count) % (1 << MCX_LOG_SQ_SIZE);
- ms = &sc->sc_tx_slots[slot];
+ ms = &tx->tx_slots[slot];
map = ms->ms_map;
bus_dmamap_sync(sc->sc_dmat, map, 0, map->dm_mapsize,
BUS_DMASYNC_POSTWRITE);
@@ -5723,6 +5762,7 @@ static int
mcx_process_rx(struct mcx_softc *sc, struct mcx_cq_entry *cqe,
struct mbuf_list *ml, const struct mcx_calibration *c)
{
+ struct mcx_rx *rx = &sc->sc_queues[0].q_rx;
struct mcx_slot *ms;
struct mbuf *m;
uint32_t flags;
@@ -5730,7 +5770,7 @@ mcx_process_rx(struct mcx_softc *sc, struct mcx_cq_entry *cqe,
slot = betoh16(cqe->cq_wqe_count) % (1 << MCX_LOG_RQ_SIZE);
- ms = &sc->sc_rx_slots[slot];
+ ms = &rx->rx_slots[slot];
bus_dmamap_sync(sc->sc_dmat, ms->ms_map, 0, ms->ms_map->dm_mapsize,
BUS_DMASYNC_POSTREAD);
bus_dmamap_unload(sc->sc_dmat, ms->ms_map);
@@ -5815,9 +5855,10 @@ mcx_arm_cq(struct mcx_softc *sc, struct mcx_cq *cq)
}
void
-mcx_process_cq(struct mcx_softc *sc, struct mcx_cq *cq)
+mcx_process_cq(struct mcx_softc *sc, struct mcx_queues *q, struct mcx_cq *cq)
{
- struct ifnet *ifp = &sc->sc_ac.ac_if;
+ struct mcx_rx *rx = &q->q_rx;
+ struct mcx_tx *tx = &q->q_tx;
const struct mcx_calibration *c;
unsigned int gen;
struct mcx_cq_entry *cqe;
@@ -5861,18 +5902,18 @@ mcx_process_cq(struct mcx_softc *sc, struct mcx_cq *cq)
mcx_arm_cq(sc, cq);
if (rxfree > 0) {
- if_rxr_put(&sc->sc_rxr, rxfree);
- if (ifiq_input(&sc->sc_ac.ac_if.if_rcv, &ml))
- if_rxr_livelocked(&sc->sc_rxr);
+ if_rxr_put(&rx->rx_rxr, rxfree);
+ if (ifiq_input(rx->rx_ifiq, &ml))
+ if_rxr_livelocked(&rx->rx_rxr);
- mcx_rx_fill(sc);
- if (if_rxr_inuse(&sc->sc_rxr) == 0)
- timeout_add(&sc->sc_rx_refill, 1);
+ mcx_rx_fill(sc, rx);
+ if (if_rxr_inuse(&rx->rx_rxr) == 0)
+ timeout_add(&rx->rx_refill, 1);
}
if (txfree > 0) {
- sc->sc_tx_cons += txfree;
- if (ifq_is_oactive(&ifp->if_snd))
- ifq_restart(&ifp->if_snd);
+ tx->tx_cons += txfree;
+ if (ifq_is_oactive(tx->tx_ifq))
+ ifq_restart(tx->tx_ifq);
}
}
@@ -5918,7 +5959,8 @@ mcx_intr(void *xsc)
cq = betoh32(eqe->eq_event_data[6]);
for (i = 0; i < sc->sc_num_cq; i++) {
if (sc->sc_cq[i].cq_n == cq) {
- mcx_process_cq(sc, &sc->sc_cq[i]);
+ mcx_process_cq(sc, sc->sc_queues,
+ &sc->sc_cq[i]);
break;
}
}
@@ -5969,19 +6011,22 @@ static int
mcx_up(struct mcx_softc *sc)
{
struct ifnet *ifp = &sc->sc_ac.ac_if;
+ struct mcx_rx *rx;
+ struct mcx_tx *tx;
struct mcx_slot *ms;
int i, start;
struct mcx_flow_match match_crit;
- sc->sc_rx_slots = mallocarray(sizeof(*ms), (1 << MCX_LOG_RQ_SIZE),
+ rx = &sc->sc_queues[0].q_rx; /* XXX */
+ rx->rx_slots = mallocarray(sizeof(*ms), (1 << MCX_LOG_RQ_SIZE),
M_DEVBUF, M_WAITOK | M_ZERO);
- if (sc->sc_rx_slots == NULL) {
+ if (rx->rx_slots == NULL) {
printf("%s: failed to allocate rx slots\n", DEVNAME(sc));
return ENOMEM;
}
for (i = 0; i < (1 << MCX_LOG_RQ_SIZE); i++) {
- ms = &sc->sc_rx_slots[i];
+ ms = &rx->rx_slots[i];
if (bus_dmamap_create(sc->sc_dmat, sc->sc_hardmtu, 1,
sc->sc_hardmtu, 0,
BUS_DMA_WAITOK | BUS_DMA_ALLOCNOW | BUS_DMA_64BIT,
@@ -5992,15 +6037,16 @@ mcx_up(struct mcx_softc *sc)
}
}
- sc->sc_tx_slots = mallocarray(sizeof(*ms), (1 << MCX_LOG_SQ_SIZE),
+ tx = &sc->sc_queues[0].q_tx; /* XXX */
+ tx->tx_slots = mallocarray(sizeof(*ms), (1 << MCX_LOG_SQ_SIZE),
M_DEVBUF, M_WAITOK | M_ZERO);
- if (sc->sc_tx_slots == NULL) {
+ if (tx->tx_slots == NULL) {
printf("%s: failed to allocate tx slots\n", DEVNAME(sc));
goto destroy_rx_slots;
}
for (i = 0; i < (1 << MCX_LOG_SQ_SIZE); i++) {
- ms = &sc->sc_tx_slots[i];
+ ms = &tx->tx_slots[i];
if (bus_dmamap_create(sc->sc_dmat, sc->sc_hardmtu,
MCX_SQ_MAX_SEGMENTS, sc->sc_hardmtu, 0,
BUS_DMA_WAITOK | BUS_DMA_ALLOCNOW | BUS_DMA_64BIT,
@@ -6015,17 +6061,17 @@ mcx_up(struct mcx_softc *sc)
goto down;
/* send queue */
- if (mcx_create_tis(sc) != 0)
+ if (mcx_create_tis(sc, tx) != 0)
goto down;
- if (mcx_create_sq(sc, sc->sc_cq[0].cq_n) != 0)
+ if (mcx_create_sq(sc, tx, sc->sc_cq[0].cq_n) != 0)
goto down;
/* receive queue */
- if (mcx_create_rq(sc, sc->sc_cq[0].cq_n) != 0)
+ if (mcx_create_rq(sc, rx, sc->sc_cq[0].cq_n) != 0)
goto down;
- if (mcx_create_tir(sc) != 0)
+ if (mcx_create_tir(sc, rx) != 0)
goto down;
if (mcx_create_flow_table(sc, MCX_LOG_FLOW_TABLE_SIZE) != 0)
@@ -6057,12 +6103,12 @@ mcx_up(struct mcx_softc *sc)
/* flow table entries for unicast and broadcast */
start = 0;
- if (mcx_set_flow_table_entry(sc, MCX_FLOW_GROUP_MAC, start,
+ if (mcx_set_flow_table_entry(sc, rx, MCX_FLOW_GROUP_MAC, start,
sc->sc_ac.ac_enaddr) != 0)
goto down;
start++;
- if (mcx_set_flow_table_entry(sc, MCX_FLOW_GROUP_MAC, start,
+ if (mcx_set_flow_table_entry(sc, rx, MCX_FLOW_GROUP_MAC, start,
etherbroadcastaddr) != 0)
goto down;
start++;
@@ -6073,7 +6119,7 @@ mcx_up(struct mcx_softc *sc)
/* re-add any existing multicast flows */
for (i = 0; i < MCX_NUM_MCAST_FLOWS; i++) {
if (sc->sc_mcast_flows[i][0] != 0) {
- mcx_set_flow_table_entry(sc, MCX_FLOW_GROUP_MAC,
+ mcx_set_flow_table_entry(sc, rx, MCX_FLOW_GROUP_MAC,
sc->sc_mcast_flow_base + i,
sc->sc_mcast_flows[i]);
}
@@ -6082,35 +6128,40 @@ mcx_up(struct mcx_softc *sc)
if (mcx_set_flow_table_root(sc) != 0)
goto down;
- /* start the queues */
- if (mcx_ready_sq(sc) != 0)
- goto down;
+ for (i = 0; i < sc->sc_nqueues; i++) {
+ struct mcx_queues *q = &sc->sc_queues[i];
+ rx = &q->q_rx;
+ tx = &q->q_tx;
- if (mcx_ready_rq(sc) != 0)
- goto down;
+ /* start the queues */
+ if (mcx_ready_sq(sc, tx) != 0)
+ goto down;
+
+ if (mcx_ready_rq(sc, rx) != 0)
+ goto down;
- if_rxr_init(&sc->sc_rxr, 1, (1 << MCX_LOG_RQ_SIZE));
- sc->sc_rx_prod = 0;
- mcx_rx_fill(sc);
+ if_rxr_init(&rx->rx_rxr, 1, (1 << MCX_LOG_RQ_SIZE));
+ rx->rx_prod = 0;
+ mcx_rx_fill(sc, rx);
+
+ tx->tx_cons = 0;
+ tx->tx_prod = 0;
+ ifq_clr_oactive(tx->tx_ifq);
+ }
mcx_calibrate_first(sc);
SET(ifp->if_flags, IFF_RUNNING);
- sc->sc_tx_cons = 0;
- sc->sc_tx_prod = 0;
- ifq_clr_oactive(&ifp->if_snd);
- ifq_restart(&ifp->if_snd);
-
return ENETRESET;
destroy_tx_slots:
- mcx_free_slots(sc, sc->sc_tx_slots, i, (1 << MCX_LOG_SQ_SIZE));
- sc->sc_tx_slots = NULL;
+ mcx_free_slots(sc, tx->tx_slots, i, (1 << MCX_LOG_SQ_SIZE));
+ tx->tx_slots = NULL;
i = (1 << MCX_LOG_RQ_SIZE);
destroy_rx_slots:
- mcx_free_slots(sc, sc->sc_rx_slots, i, (1 << MCX_LOG_RQ_SIZE));
- sc->sc_rx_slots = NULL;
+ mcx_free_slots(sc, rx->rx_slots, i, (1 << MCX_LOG_RQ_SIZE));
+ rx->rx_slots = NULL;
down:
mcx_down(sc);
return ENOMEM;
@@ -6142,7 +6193,10 @@ mcx_down(struct mcx_softc *sc)
}
intr_barrier(sc->sc_ihc);
- ifq_barrier(&ifp->if_snd);
+ for (i = 0; i < sc->sc_nqueues; i++) {
+ struct ifqueue *ifq = sc->sc_queues[i].q_tx.tx_ifq;
+ ifq_barrier(ifq);
+ }
timeout_del_barrier(&sc->sc_calibrate);
@@ -6155,29 +6209,42 @@ mcx_down(struct mcx_softc *sc)
if (sc->sc_flow_table_id != -1)
mcx_destroy_flow_table(sc);
- if (sc->sc_tirn != 0)
- mcx_destroy_tir(sc);
- if (sc->sc_rqn != 0)
- mcx_destroy_rq(sc);
+ for (i = 0; i < sc->sc_nqueues; i++) {
+ struct mcx_queues *q = &sc->sc_queues[i];
+ struct mcx_rx *rx = &q->q_rx;
+ struct mcx_tx *tx = &q->q_tx;
+
+ if (rx->rx_tirn != 0)
+ mcx_destroy_tir(sc, rx);
+ if (rx->rx_rqn != 0)
+ mcx_destroy_rq(sc, rx);
- if (sc->sc_sqn != 0)
- mcx_destroy_sq(sc);
- if (sc->sc_tisn != 0)
- mcx_destroy_tis(sc);
+ if (tx->tx_sqn != 0)
+ mcx_destroy_sq(sc, tx);
+ if (tx->tx_tisn != 0)
+ mcx_destroy_tis(sc, tx);
+ }
+ /* XXX sc_queues */
for (i = 0; i < sc->sc_num_cq; i++)
mcx_destroy_cq(sc, i);
sc->sc_num_cq = 0;
- if (sc->sc_tx_slots != NULL) {
- mcx_free_slots(sc, sc->sc_tx_slots, (1 << MCX_LOG_SQ_SIZE),
- (1 << MCX_LOG_SQ_SIZE));
- sc->sc_tx_slots = NULL;
- }
- if (sc->sc_rx_slots != NULL) {
- mcx_free_slots(sc, sc->sc_rx_slots, (1 << MCX_LOG_RQ_SIZE),
- (1 << MCX_LOG_RQ_SIZE));
- sc->sc_rx_slots = NULL;
+ for (i = 0; i < sc->sc_nqueues; i++) {
+ struct mcx_queues *q = &sc->sc_queues[i];
+ struct mcx_rx *rx = &q->q_rx;
+ struct mcx_tx *tx = &q->q_tx;
+
+ if (tx->tx_slots != NULL) {
+ mcx_free_slots(sc, tx->tx_slots,
+ (1 << MCX_LOG_SQ_SIZE), (1 << MCX_LOG_SQ_SIZE));
+ tx->tx_slots = NULL;
+ }
+ if (rx->rx_slots != NULL) {
+ mcx_free_slots(sc, rx->rx_slots,
+ (1 << MCX_LOG_RQ_SIZE), (1 << MCX_LOG_RQ_SIZE));
+ rx->rx_slots = NULL;
+ }
}
}
@@ -6232,6 +6299,7 @@ mcx_ioctl(struct ifnet *ifp, u_long cmd, caddr_t data)
ETHER_ADDR_LEN);
if (ISSET(ifp->if_flags, IFF_RUNNING)) {
mcx_set_flow_table_entry(sc,
+ &sc->sc_queues[0].q_rx,
MCX_FLOW_GROUP_MAC,
sc->sc_mcast_flow_base + i,
sc->sc_mcast_flows[i]);
@@ -6348,13 +6416,28 @@ mcx_get_sffpage(struct ifnet *ifp, struct if_sffpage *sff)
static int
mcx_rxrinfo(struct mcx_softc *sc, struct if_rxrinfo *ifri)
{
- struct if_rxring_info ifr;
+ struct if_rxring_info *ifrs;
+ unsigned int i;
+ int error;
+
+ ifrs = mallocarray(sc->sc_nqueues, sizeof(*ifrs), M_TEMP,
+ M_WAITOK|M_ZERO|M_CANFAIL);
+ if (ifrs == NULL)
+ return (ENOMEM);
- memset(&ifr, 0, sizeof(ifr));
- ifr.ifr_size = sc->sc_hardmtu;
- ifr.ifr_info = sc->sc_rxr;
+ for (i = 0; i < sc->sc_nqueues; i++) {
+ struct mcx_rx *rx = &sc->sc_queues[i].q_rx;
+ struct if_rxring_info *ifr = &ifrs[i];
- return (if_rxr_info_ioctl(ifri, 1, &ifr));
+ snprintf(ifr->ifr_name, sizeof(ifr->ifr_name), "%u", i);
+ ifr->ifr_size = sc->sc_hardmtu;
+ ifr->ifr_info = rx->rx_rxr;
+ }
+
+ error = if_rxr_info_ioctl(ifri, i, ifrs);
+ free(ifrs, M_TEMP, i * sizeof(*ifrs));
+
+ return (error);
}
int
@@ -6382,6 +6465,7 @@ mcx_load_mbuf(struct mcx_softc *sc, struct mcx_slot *ms, struct mbuf *m)
static void
mcx_start(struct ifqueue *ifq)
{
+ struct mcx_tx *tx = ifq->ifq_softc;
struct ifnet *ifp = ifq->ifq_if;
struct mcx_softc *sc = ifp->if_softc;
struct mcx_sq_entry *sq, *sqe;
@@ -6397,12 +6481,12 @@ mcx_start(struct ifqueue *ifq)
bf_base = (sc->sc_uar * MCX_PAGE_SIZE) + MCX_UAR_BF;
- idx = sc->sc_tx_prod % (1 << MCX_LOG_SQ_SIZE);
- free = (sc->sc_tx_cons + (1 << MCX_LOG_SQ_SIZE)) - sc->sc_tx_prod;
+ idx = tx->tx_prod % (1 << MCX_LOG_SQ_SIZE);
+ free = (tx->tx_cons + (1 << MCX_LOG_SQ_SIZE)) - tx->tx_prod;
used = 0;
bf = NULL;
- sq = (struct mcx_sq_entry *)MCX_DMA_KVA(&sc->sc_sq_mem);
+ sq = (struct mcx_sq_entry *)MCX_DMA_KVA(&tx->tx_sq_mem);
for (;;) {
if (used + MCX_SQ_ENTRY_MAX_SLOTS >= free) {
@@ -6416,12 +6500,12 @@ mcx_start(struct ifqueue *ifq)
}
sqe = sq + idx;
- ms = &sc->sc_tx_slots[idx];
+ ms = &tx->tx_slots[idx];
memset(sqe, 0, sizeof(*sqe));
/* ctrl segment */
sqe->sqe_opcode_index = htobe32(MCX_SQE_WQE_OPCODE_SEND |
- ((sc->sc_tx_prod & 0xffff) << MCX_SQE_WQE_INDEX_SHIFT));
+ ((tx->tx_prod & 0xffff) << MCX_SQE_WQE_INDEX_SHIFT));
/* always generate a completion event */
sqe->sqe_signature = htobe32(MCX_SQE_CE_CQE_ALWAYS);
@@ -6472,7 +6556,7 @@ mcx_start(struct ifqueue *ifq)
BUS_DMASYNC_PREWRITE);
sqe->sqe_ds_sq_num =
- htobe32((sc->sc_sqn << MCX_SQE_SQ_NUM_SHIFT) |
+ htobe32((tx->tx_sqn << MCX_SQE_SQ_NUM_SHIFT) |
(map->dm_nsegs + 3));
/* data segment - first wqe has one segment */
@@ -6485,7 +6569,7 @@ mcx_start(struct ifqueue *ifq)
idx++;
if (idx == (1 << MCX_LOG_SQ_SIZE))
idx = 0;
- sc->sc_tx_prod++;
+ tx->tx_prod++;
used++;
sqs = (struct mcx_sq_entry_seg *)(sq + idx);
@@ -6502,12 +6586,12 @@ mcx_start(struct ifqueue *ifq)
idx++;
if (idx == (1 << MCX_LOG_SQ_SIZE))
idx = 0;
- sc->sc_tx_prod++;
+ tx->tx_prod++;
used++;
}
if (used) {
- *sc->sc_tx_doorbell = htobe32(sc->sc_tx_prod & MCX_WQ_DOORBELL_MASK);
+ htobem32(tx->tx_doorbell, tx->tx_prod & MCX_WQ_DOORBELL_MASK);
membar_sync();
@@ -6516,9 +6600,9 @@ mcx_start(struct ifqueue *ifq)
* to the blue flame buffer
*/
bus_space_write_raw_8(sc->sc_memt, sc->sc_memh,
- bf_base + sc->sc_bf_offset, *bf);
+ bf_base + tx->tx_bf_offset, *bf);
/* next write goes to the other buffer */
- sc->sc_bf_offset ^= sc->sc_bf_size;
+ tx->tx_bf_offset ^= sc->sc_bf_size;
membar_sync();
}