Source-Changes-HG archive

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]

[src/trunk]: src/sys/arch/xen/xen add support for scatter-gather also for fro...



details:   https://anonhg.NetBSD.org/src/rev/cc2b094e93b3
branches:  trunk
changeset: 1009803:cc2b094e93b3
user:      jdolecek <jdolecek%NetBSD.org@localhost>
date:      Sun May 03 17:56:19 2020 +0000

description:
add support for scatter-gather also for frontend Rx path (backend -> frontend)

enable ETHERCAP_JUMBO_MTU and feature-sg

diffstat:

 sys/arch/xen/xen/xennetback_xenbus.c |  324 +++++++++++++++++++++-------------
 1 files changed, 200 insertions(+), 124 deletions(-)

diffs (truncated from 452 to 300 lines):

diff -r b0aeb0767551 -r cc2b094e93b3 sys/arch/xen/xen/xennetback_xenbus.c
--- a/sys/arch/xen/xen/xennetback_xenbus.c      Sun May 03 17:54:28 2020 +0000
+++ b/sys/arch/xen/xen/xennetback_xenbus.c      Sun May 03 17:56:19 2020 +0000
@@ -1,4 +1,4 @@
-/*      $NetBSD: xennetback_xenbus.c,v 1.102 2020/05/02 11:28:02 jdolecek Exp $      */
+/*      $NetBSD: xennetback_xenbus.c,v 1.103 2020/05/03 17:56:19 jdolecek Exp $      */
 
 /*
  * Copyright (c) 2006 Manuel Bouyer.
@@ -25,7 +25,7 @@
  */
 
 #include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: xennetback_xenbus.c,v 1.102 2020/05/02 11:28:02 jdolecek Exp $");
+__KERNEL_RCSID(0, "$NetBSD: xennetback_xenbus.c,v 1.103 2020/05/03 17:56:19 jdolecek Exp $");
 
 #include "opt_xen.h"
 
@@ -60,8 +60,6 @@
 
 #include <uvm/uvm.h>
 
-/* #define notyet 1 */
-
 /*
  * Backend network device driver for Xen.
  */
@@ -263,11 +261,7 @@
        /* create pseudo-interface */
        aprint_verbose_ifnet(ifp, "Ethernet address %s\n",
            ether_sprintf(xneti->xni_enaddr));
-       xneti->xni_ec.ec_capabilities |= ETHERCAP_VLAN_MTU
-#ifdef notyet
-               | ETHERCAP_JUMBO_MTU
-#endif
-       ;
+       xneti->xni_ec.ec_capabilities |= ETHERCAP_VLAN_MTU | ETHERCAP_JUMBO_MTU;
        ifp->if_flags = IFF_BROADCAST | IFF_SIMPLEX | IFF_MULTICAST;
        ifp->if_snd.ifq_maxlen =
            uimax(ifqmaxlen, NET_TX_RING_SIZE * 2);
@@ -327,7 +321,6 @@
                            xbusd->xbusd_path, err);
                        goto abort_xbt;
                }
-#if notyet
                err = xenbus_printf(xbt, xbusd->xbusd_path,
                    "feature-sg", "%d", 1);
                if (err) {
@@ -336,7 +329,6 @@
                            xbusd->xbusd_path, err);
                        goto abort_xbt;
                }
-#endif
        } while ((err = xenbus_transaction_end(xbt, 0)) == EAGAIN);
        if (err) {
                aprint_error_ifnet(ifp,
@@ -678,21 +670,22 @@
 }
 
 static int
-xennetback_copy(struct ifnet *ifp, gnttab_copy_t *gop, int copycnt)
+xennetback_copy(struct ifnet *ifp, gnttab_copy_t *gop, int copycnt,
+    const char *dir)
 {
        /*
         * Copy the data and ack it. Delaying it until the mbuf is
         * freed will stall transmit.
         */
        if (HYPERVISOR_grant_table_op(GNTTABOP_copy, gop, copycnt) != 0) {
-               printf("%s: GNTTABOP_copy Tx failed", ifp->if_xname);
+               printf("%s: GNTTABOP_copy %s failed", ifp->if_xname, dir);
                return EINVAL;
        }
 
        for (int i = 0; i < copycnt; i++) {
                if (gop->status != GNTST_okay) {
-                       printf("%s GNTTABOP_copy[%d] Tx %d\n",
-                           ifp->if_xname, i, gop->status);
+                       printf("%s GNTTABOP_copy[%d] %s %d\n",
+                           ifp->if_xname, i, dir, gop->status);
                        return EINVAL;
                }
        }
@@ -758,7 +751,7 @@
                        KASSERT(copycnt <= NB_XMIT_PAGES_BATCH);
                        if (copycnt == NB_XMIT_PAGES_BATCH) {
                                if (xennetback_copy(ifp, xneti->xni_gop_copy,
-                                   copycnt) != 0)
+                                   copycnt, "Tx") != 0)
                                        goto abort;
                                copycnt = 0;
                        }
@@ -791,7 +784,7 @@
                KASSERT(goff == xst->xs_tx_size);
        }
        if (copycnt > 0) {
-               if (xennetback_copy(ifp, xneti->xni_gop_copy, copycnt) != 0)
+               if (xennetback_copy(ifp, xneti->xni_gop_copy, copycnt, "Tx"))
                        goto abort;
                copycnt = 0;
        }
@@ -1058,19 +1051,151 @@
 }
 
 static void
+xennetback_rx_copy_process(struct ifnet *ifp, struct xnetback_instance *xneti,
+       int queued, int copycnt)
+{
+       int notify;
+       struct xnetback_xstate *xst;
+
+       if (xennetback_copy(ifp, xneti->xni_gop_copy, copycnt, "Rx") != 0) {
+               /* message already displayed */
+               goto free_mbufs;
+       }
+
+       /* update pointer */
+       xen_rmb();
+       xneti->xni_rxring.req_cons += queued;
+       xneti->xni_rxring.rsp_prod_pvt += queued;
+       RING_PUSH_RESPONSES_AND_CHECK_NOTIFY(&xneti->xni_rxring, notify);
+
+       /* send event */
+       if (notify) {
+               xen_rmb();
+               XENPRINTF(("%s receive event\n",
+                   xneti->xni_if.if_xname));
+               hypervisor_notify_via_evtchn(xneti->xni_evtchn);
+       }
+
+free_mbufs:
+       /* now that data was copied we can free the mbufs */
+       for (int j = 0; j < queued; j++) {
+               xst = &xneti->xni_xstate[j];
+               if (xst->xs_loaded) {
+                       bus_dmamap_unload(xneti->xni_xbusd->xbusd_dmat,
+                           xst->xs_dmamap);
+                       xst->xs_loaded = false;
+               }
+               if (xst->xs_m != NULL) {
+                       m_freem(xst->xs_m);
+                       xst->xs_m = NULL;
+               }
+       }
+}
+
+static void
+xennetback_rx_copy_queue(struct xnetback_instance *xneti,
+    struct xnetback_xstate *xst0, int rsp_prod_pvt, int *queued, int *copycntp)
+{
+       struct xnetback_xstate *xst = xst0;
+       gnttab_copy_t *gop;
+       struct netif_rx_request rxreq;
+       netif_rx_response_t *rxresp;
+       paddr_t ma;
+       size_t goff, segoff, segsize, take, totsize;
+       int copycnt = *copycntp, reqcnt = *queued;
+       const bus_dmamap_t dm = xst0->xs_dmamap;
+       const bool multiseg = (dm->dm_nsegs > 1);
+
+       KASSERT(xst0 == &xneti->xni_xstate[reqcnt]);
+
+       RING_COPY_REQUEST(&xneti->xni_rxring,
+           xneti->xni_rxring.req_cons + reqcnt, &rxreq);
+       goff = 0;
+       rxresp = RING_GET_RESPONSE(&xneti->xni_rxring, rsp_prod_pvt + reqcnt);
+       reqcnt++;
+
+       rxresp->id = rxreq.id;
+       rxresp->offset = 0;
+       if ((xst0->xs_m->m_pkthdr.csum_flags & XN_M_CSUM_SUPPORTED) != 0) {
+               rxresp->flags = NETRXF_csum_blank;
+       } else {
+               rxresp->flags = NETRXF_data_validated;
+       }
+       if (multiseg)
+               rxresp->flags |= NETRXF_more_data;
+
+       totsize = xst0->xs_m->m_pkthdr.len;
+
+       /*
+        * Arrange for the mbuf contents to be copied into one or more
+        * provided memory pages.
+        */
+       for (int seg = 0; seg < dm->dm_nsegs; seg++) {
+               ma = dm->dm_segs[seg].ds_addr;
+               segsize = dm->dm_segs[seg].ds_len;
+               segoff = 0;
+
+               while (segoff < segsize) {
+                       take = uimin(PAGE_SIZE - goff, segsize - segoff);
+                       KASSERT(take <= totsize);
+
+                       /* add copy request */
+                       gop = &xneti->xni_gop_copy[copycnt++];
+                       gop->flags = GNTCOPY_dest_gref;
+                       gop->source.offset = (ma & PAGE_MASK) + segoff;
+                       gop->source.domid = DOMID_SELF;
+                       gop->source.u.gmfn = ma >> PAGE_SHIFT;
+
+                       gop->dest.u.ref = rxreq.gref;
+                       gop->dest.offset = goff;
+                       gop->dest.domid = xneti->xni_domid;
+
+                       gop->len = take;
+
+                       segoff += take;
+                       goff += take;
+                       totsize -= take;
+
+                       if (goff == PAGE_SIZE && totsize > 0) {
+                               rxresp->status = goff;
+
+                               /* Take next grant */
+                               RING_COPY_REQUEST(&xneti->xni_rxring,
+                                   xneti->xni_rxring.req_cons + reqcnt,
+                                   &rxreq);
+                               goff = 0;
+                               rxresp = RING_GET_RESPONSE(&xneti->xni_rxring,
+                                   rsp_prod_pvt + reqcnt);
+                               reqcnt++;
+
+                               rxresp->id = rxreq.id;
+                               rxresp->offset = 0;
+                               rxresp->flags = NETRXF_more_data;
+
+                               xst++;
+                               xst->xs_m = NULL;
+                       }
+               }
+       }
+       rxresp->flags &= ~NETRXF_more_data;
+       rxresp->status = goff;
+       KASSERT(totsize == 0);
+
+       KASSERT(copycnt > *copycntp);
+       KASSERT(reqcnt > *queued);
+       *copycntp = copycnt;
+       *queued = reqcnt;
+}
+
+static void
 xennetback_ifsoftstart_copy(struct xnetback_instance *xneti)
 {
        struct ifnet *ifp = &xneti->xni_if;
        struct mbuf *m;
-       paddr_t xmit_ma;
-       int i, j;
-       netif_rx_response_t *rxresp;
-       netif_rx_request_t rxreq;
-       RING_IDX req_prod, resp_prod;
-       int do_event = 0;
-       gnttab_copy_t *gop;
+       int queued = 0;
+       RING_IDX req_prod, rsp_prod_pvt;
        struct xnetback_xstate *xst;
-       int id;
+       int copycnt = 0;
        bool abort;
 
        XENPRINTF(("xennetback_ifsoftstart_copy "));
@@ -1083,22 +1208,26 @@
        while (!IFQ_IS_EMPTY(&ifp->if_snd)) {
                XENPRINTF(("pkt\n"));
                req_prod = xneti->xni_rxring.sring->req_prod;
-               resp_prod = xneti->xni_rxring.rsp_prod_pvt;
+               rsp_prod_pvt = xneti->xni_rxring.rsp_prod_pvt;
                xen_rmb();
 
-               gop = xneti->xni_gop_copy;
                abort = false;
-               for (i = 0; i < NB_XMIT_PAGES_BATCH; i++) {
-                       if (__predict_false(
-                           req_prod == xneti->xni_rxring.req_cons ||
-                           xneti->xni_rxring.req_cons - resp_prod ==
-                           NET_RX_RING_SIZE)) {
+               KASSERT(queued == 0);
+               KASSERT(copycnt == 0);
+               while (copycnt < NB_XMIT_PAGES_BATCH) {
+#define XN_RING_FULL(cnt)      \
+                       req_prod == xneti->xni_rxring.req_cons + (cnt) ||  \
+                       xneti->xni_rxring.req_cons - (rsp_prod_pvt + cnt) ==  \
+                       NET_RX_RING_SIZE
+
+                       if (__predict_false(XN_RING_FULL(1))) {
                                /* out of ring space */
                                XENPRINTF(("xennetback_ifstart: ring full "
-                                   "req_prod 0x%x req_cons 0x%x resp_prod "
+                                   "req_prod 0x%x req_cons 0x%x rsp_prod_pvt "
                                    "0x%x\n",
-                                   req_prod, xneti->xni_rxring.req_cons,
-                                   resp_prod));
+                                   req_prod,
+                                   xneti->xni_rxring.req_cons + queued,
+                                   rsp_prod_pvt + queued));
                                abort = true;
                                break;
                        }
@@ -1107,7 +1236,8 @@
                        if (m == NULL)



Home | Main Index | Thread Index | Old Index