git: c30251c44fb3 - main - net/daq: improve netmap support
Brad Davis
brd at FreeBSD.org
Thu Jun 10 22:22:26 UTC 2021
The branch main has been updated by brd (doc, src committer):
URL: https://cgit.FreeBSD.org/ports/commit/?id=c30251c44fb345a54b56ebdd9c482071306e5eff
commit c30251c44fb345a54b56ebdd9c482071306e5eff
Author: Bill Meeks <bmeeks8 at bellsouth.net>
AuthorDate: 2019-05-29 21:04:41 +0000
Commit: Brad Davis <brd at FreeBSD.org>
CommitDate: 2021-06-10 21:41:44 +0000
net/daq: improve netmap support
Cherry-pick c0c4b55 from pfsense/FreeBSD-ports:
Add host stack OS endpoint support for netmap module.
Cherry-pick 52b6dea from pfsense/FreeBSD-ports:
Support host stack interfaces with netmap and NETMAP_API versions 13 and 14.
Cherry-pick 6914d3a from pfsense/FreeBSD-ports:
Backport NS_MOREFRAG bug fix from netmap upstream demo app code.
Approved by: zi (maintainer)
Sponsored by: Rubicon Communications, LLC ("Netgate")
---
net/daq/Makefile | 2 +-
net/daq/files/patch-daq22-netmap.diff | 505 ++++++++++++++++++++++++++++++++++
2 files changed, 506 insertions(+), 1 deletion(-)
diff --git a/net/daq/Makefile b/net/daq/Makefile
index 28d8429e0e7d..81fe1f408578 100644
--- a/net/daq/Makefile
+++ b/net/daq/Makefile
@@ -2,7 +2,7 @@
PORTNAME= daq
PORTVERSION= 2.2.2
-PORTREVISION= 0
+PORTREVISION= 3
CATEGORIES= net
MASTER_SITES= https://snort.org/downloads/snortplus/ \
ZI
diff --git a/net/daq/files/patch-daq22-netmap.diff b/net/daq/files/patch-daq22-netmap.diff
new file mode 100644
index 000000000000..36e0e2c59f9a
--- /dev/null
+++ b/net/daq/files/patch-daq22-netmap.diff
@@ -0,0 +1,505 @@
+diff -ruN ./daq-2.2.2.orig/os-daq-modules/daq_netmap.c ./daq-2.2.2/os-daq-modules/daq_netmap.c
+--- ./daq-2.2.2.orig/os-daq-modules/daq_netmap.c 2017-02-08 17:04:18.000000000 -0500
++++ ./os-daq-modules/daq_netmap.c 2020-09-15 09:35:07.000000000 -0400
+@@ -1,6 +1,7 @@
+ /*
+-** Copyright (C) 2014 Cisco and/or its affiliates. All rights reserved.
++** Copyright (C) 2021 Cisco and/or its affiliates. All rights reserved.
+ ** Author: Michael R. Altizer <mialtize at cisco.com>
++** Author: Bill Meeks <billmeeks8 at gmail.com>
+ **
+ ** This program is free software; you can redistribute it and/or modify
+ ** it under the terms of the GNU General Public License Version 2 as
+@@ -37,6 +38,9 @@
+ #include <sfbpf_dlt.h>
+
+ #include <net/netmap.h>
++#if NETMAP_API < 14
++#define NETMAP_WITH_LIBS
++#endif
+ #include <net/netmap_user.h>
+
+ #define DAQ_NETMAP_VERSION 3
+@@ -44,12 +48,6 @@
+ /* Hi! I'm completely arbitrary! */
+ #define NETMAP_MAX_INTERFACES 32
+
+-/* FreeBSD 10.0 uses an old version of netmap, so work around it accordingly. */
+-#if NETMAP_API < 10
+-#define nm_ring_next(r, i) NETMAP_RING_NEXT(r, i)
+-#define nm_ring_empty(r) ((r)->avail == 0)
+-#endif
+-
+ typedef struct _netmap_instance
+ {
+ struct _netmap_instance *next;
+@@ -59,6 +57,10 @@
+ #define NMINST_TX_BLOCKED 0x2
+ uint32_t flags;
+ int index;
++ int done_mmap;
++ unsigned long long fwd_tx_blocked;
++ /* Netmap port name with suffix */
++ char nm_port_name[IFNAMSIZ + 1];
+ struct netmap_if *nifp;
+ /* TX ring info */
+ uint16_t first_tx_ring;
+@@ -71,8 +73,15 @@
+ /* MMAP'd memory */
+ void *mem;
+ uint32_t memsize;
+- struct nmreq req;
+- unsigned long long fwd_tx_blocked;
++ /* Use netmap API version-specific structures */
++#if NETMAP_API >= 14
++ /* Newer netmap Header and Request structures */
++ struct nmreq_header hdr;
++ struct nmreq_register reg;
++#else
++ /* Netmap descriptor structure */
++ struct nm_desc *ndesc;
++#endif
+ } NetmapInstance;
+
+ typedef struct _netmap_context
+@@ -109,6 +118,7 @@
+ {
+ if (instance)
+ {
++#if NETMAP_API >= 14
+ /* Unmap the packet memory region. If we had a peer, notify them that
+ the shared mapping has been freed and that we no longer exist. */
+ if (instance->mem)
+@@ -124,6 +134,24 @@
+ instance->peer->peer = NULL;
+ if (instance->fd != -1)
+ close(instance->fd);
++#else
++ /* Check for an active peer. */
++ if (instance->peer)
++ {
++ /* Close our peer's netmap connection first. */
++ if (instance->peer->ndesc != NULL)
++ {
++ nm_close(instance->peer->ndesc);
++ instance->peer->ndesc = NULL;
++ instance->peer->peer = NULL;
++ instance->peer->done_mmap = 0;
++ }
++ }
++
++ /* Now close our netmap connection if open. */
++ if (instance->ndesc != NULL)
++ nm_close(instance->ndesc);
++#endif
+ free(instance);
+ }
+ }
+@@ -142,7 +170,7 @@
+ if (nmc->debug)
+ {
+ printf("Netmap instance %s (%d) blocked %llu times on TX while forwarding.\n",
+- instance->req.nr_name, instance->index, instance->fwd_tx_blocked);
++ instance->nm_port_name, instance->index, instance->fwd_tx_blocked);
+ }
+ destroy_instance(instance);
+ }
+@@ -157,14 +185,18 @@
+ static NetmapInstance *create_instance(const char *device, NetmapInstance *parent, char *errbuf, size_t errlen)
+ {
+ NetmapInstance *instance;
+- struct nmreq *req;
++#if NETMAP_API >= 14
++ char *suffix = NULL;
++ char intf[IFNAMSIZ + 1];
++ uint32_t nr_mode = NR_REG_ALL_NIC;
++#endif
+ static int index = 0;
+
+ instance = calloc(1, sizeof(NetmapInstance));
+ if (!instance)
+ {
+ snprintf(errbuf, errlen, "%s: Could not allocate a new instance structure.", __func__);
+- goto err;
++ return NULL;
+ }
+
+ /* Initialize the instance, including an arbitrary and unique device index. */
+@@ -172,29 +204,45 @@
+ instance->index = index;
+ index++;
+
++ /* Initialize the netmap port name. */
++ strncpy(instance->nm_port_name, device, sizeof(instance->nm_port_name));
++
++#if NETMAP_API >= 14
+ /* Open /dev/netmap for communications to the driver. */
+ instance->fd = open("/dev/netmap", O_RDWR);
+ if (instance->fd < 0)
+ {
+ snprintf(errbuf, errlen, "%s: Could not open /dev/netmap: %s (%d)",
+ __func__, strerror(errno), errno);
+- goto err;
++ destroy_instance(instance);
++ return NULL;
+ }
+
+- /* Initialize the netmap request object. */
+- req = &instance->req;
+- strncpy(req->nr_name, device, sizeof(req->nr_name));
+- req->nr_version = NETMAP_API;
+- req->nr_ringid = 0;
+-#if NETMAP_API >= 11
+- req->nr_flags = NR_REG_ALL_NIC;
++ /* Check for suffix to see if this is a Host Stack interface spec. */
++ suffix = strchr(device, '^');
++ if (suffix != NULL)
++ {
++ /* Omit the trailing '^' suffix from the interface name we send netmap. */
++ snprintf(intf, strcspn(device, "^") + 1, "%s", device);
++ nr_mode = NR_REG_SW;
++ }
++ else
++ {
++ strncpy(intf, device, sizeof(intf));
++ nr_mode = NR_REG_ALL_NIC;
++ }
++
++ /* Initialize the netmap header object for the instance. */
++ instance->hdr.nr_version = NETMAP_API;
++ instance->hdr.nr_reqtype = NETMAP_REQ_REGISTER;
++ strncpy(instance->hdr.nr_name, intf, sizeof(instance->hdr.nr_name));
++ instance->hdr.nr_body = (uintptr_t)&instance->reg;
++
++ /* Initialize the netmap register object for the instance. */
++ instance->reg.nr_mode = nr_mode;
+ #endif
+
+ return instance;
+-
+-err:
+- destroy_instance(instance);
+- return NULL;
+ }
+
+ static int create_bridge(Netmap_Context_t *nmc, const char *device_name1, const char *device_name2)
+@@ -204,9 +252,9 @@
+ peer1 = peer2 = NULL;
+ for (instance = nmc->instances; instance; instance = instance->next)
+ {
+- if (!strcmp(instance->req.nr_name, device_name1))
++ if (!strcmp(instance->nm_port_name, device_name1))
+ peer1 = instance;
+- else if (!strcmp(instance->req.nr_name, device_name2))
++ else if (!strcmp(instance->nm_port_name, device_name2))
+ peer2 = instance;
+ }
+
+@@ -221,10 +269,13 @@
+
+ static int start_instance(Netmap_Context_t *nmc, NetmapInstance *instance)
+ {
+- if (ioctl(instance->fd, NIOCREGIF, &instance->req))
++#if NETMAP_API >= 14
++ u_int i, num_tx, num_rx;
++
++ if (ioctl(instance->fd, NIOCCTRL, &instance->hdr))
+ {
+ DPE(nmc->errbuf, "%s: Netmap registration for %s failed: %s (%d)",
+- __func__, instance->req.nr_name, strerror(errno), errno);
++ __func__, instance->nm_port_name, strerror(errno), errno);
+ return DAQ_ERROR;
+ }
+
+@@ -236,32 +287,96 @@
+ }
+ else
+ {
+- instance->memsize = instance->req.nr_memsize;
++ instance->memsize = instance->reg.nr_memsize;
+ instance->mem = mmap(0, instance->memsize, PROT_WRITE | PROT_READ, MAP_SHARED, instance->fd, 0);
+ if (instance->mem == MAP_FAILED)
+ {
+ DPE(nmc->errbuf, "%s: Could not MMAP the buffer memory region for %s: %s (%d)",
+- __func__, instance->req.nr_name, strerror(errno), errno);
++ __func__, instance->nm_port_name, strerror(errno), errno);
+ return DAQ_ERROR;
+ }
+ }
+
+- instance->nifp = NETMAP_IF(instance->mem, instance->req.nr_offset);
++ /* Calculate and set first and last TX and RX rings for this instance. */
++ instance->nifp = NETMAP_IF(instance->mem, instance->reg.nr_offset);
++
++ num_tx = instance->nifp->ni_tx_rings + instance->nifp->ni_host_tx_rings;
++ for (i = 0; i < num_tx && !instance->nifp->ring_ofs[i]; i++)
++ ;
++ instance->first_tx_ring = i;
++ instance->cur_tx_ring = i;
+
+- instance->first_tx_ring = 0;
+- instance->first_rx_ring = 0;
+- instance->last_tx_ring = instance->req.nr_tx_rings - 1;
+- instance->last_rx_ring = instance->req.nr_rx_rings - 1;
++ for ( ; i < num_tx && instance->nifp->ring_ofs[i]; i++)
++ ;
++ instance->last_tx_ring = i - 1;
+
++ num_rx = instance->nifp->ni_rx_rings + instance->nifp->ni_host_rx_rings;
++ for (i = 0; i < num_rx && !instance->nifp->ring_ofs[i + num_tx]; i++)
++ ;
++ instance->first_rx_ring = i;
++ instance->cur_rx_ring = i;
++
++ for ( ; i < num_rx && instance->nifp->ring_ofs[i + num_tx]; i++)
++ ;
++ instance->last_rx_ring = i - 1;
++
++#else
++ char nm_port[IFNAMSIZ + 9];
++
++ /* Create the proper netmap port name spec. */
++ snprintf(nm_port, sizeof(nm_port), "netmap:%s", instance->nm_port_name);
++
++ /* Only mmap the packet memory region for the first interface in a pair. */
++ if (instance->peer && instance->peer->done_mmap)
++ {
++ if ((instance->ndesc = nm_open(nm_port, NULL, NM_OPEN_NO_MMAP, instance->peer->ndesc)) == NULL)
++ {
++ DPE(nmc->errbuf, "%s: Netmap registration for port %s failed: %s (%d)",
++ __func__, instance->nm_port_name, strerror(errno), errno);
++ return DAQ_ERROR;
++ }
++ }
++ else
++ {
++ if ((instance->ndesc = nm_open(nm_port, NULL, 0, NULL)) == NULL)
++ {
++ DPE(nmc->errbuf, "%s: Netmap registration for port %s failed: %s (%d)",
++ __func__, instance->nm_port_name, strerror(errno), errno);
++ return DAQ_ERROR;
++ }
++ instance->done_mmap = instance->ndesc->done_mmap;
++ }
++
++ /* Copy important netmap parameters to our local instance variables. */
++ instance->fd = instance->ndesc->fd;
++ instance->nifp = instance->ndesc->nifp;
++ instance->mem = instance->ndesc->mem;
++ instance->memsize = instance->ndesc->memsize;
++ instance->first_tx_ring = instance->ndesc->first_tx_ring;
++ instance->last_tx_ring = instance->ndesc->last_tx_ring;
++ instance->cur_tx_ring = instance->ndesc->cur_tx_ring;
++ instance->first_rx_ring = instance->ndesc->first_rx_ring;
++ instance->last_rx_ring = instance->ndesc->last_rx_ring;
++ instance->cur_rx_ring = instance->ndesc->cur_rx_ring;
++#endif
++
+ if (nmc->debug)
+ {
+ struct netmap_ring *ring;
+ int i;
+
+- printf("[%s]\n", instance->req.nr_name);
+- printf(" nr_tx_slots: %u\n", instance->req.nr_tx_slots);
+- printf(" nr_rx_slots: %u\n", instance->req.nr_rx_slots);
+- printf(" nr_tx_rings: %hu\n", instance->req.nr_tx_rings);
++ printf("[%s]\n", instance->nm_port_name);
++#if NETMAP_API >= 14
++ printf(" nr_tx_slots: %u\n", instance->reg.nr_tx_slots);
++ printf(" nr_rx_slots: %u\n", instance->reg.nr_rx_slots);
++ printf(" nr_tx_rings: %hu\n", instance->reg.nr_tx_rings);
++ printf(" nr_host_tx_rings: %hu\n", instance->reg.nr_host_tx_rings);
++#else
++ printf(" nr_tx_slots: %u\n", instance->ndesc->req.nr_tx_slots);
++ printf(" nr_rx_slots: %u\n", instance->ndesc->req.nr_rx_slots);
++ printf(" nr_tx_rings: %hu\n", instance->ndesc->req.nr_tx_rings);
++#endif
++
+ for (i = instance->first_tx_ring; i <= instance->last_tx_ring; i++)
+ {
+ ring = NETMAP_TXRING(instance->nifp, i);
+@@ -271,7 +386,13 @@
+ printf(" nr_buf_size = %u\n", ring->nr_buf_size);
+ printf(" flags = 0x%x\n", ring->flags);
+ }
+- printf(" nr_rx_rings: %hu\n", instance->req.nr_rx_rings);
++
++#if NETMAP_API >= 14
++ printf(" nr_rx_rings: %hu\n", instance->reg.nr_rx_rings);
++ printf(" nr_host_rx_rings: %hu\n", instance->reg.nr_host_rx_rings);
++#else
++ printf(" nr_rx_rings: %hu\n", instance->ndesc->req.nr_rx_rings);
++#endif
+ for (i = instance->first_rx_ring; i <= instance->last_rx_ring; i++)
+ {
+ ring = NETMAP_RXRING(instance->nifp, i);
+@@ -293,11 +414,17 @@
+ Netmap_Context_t *nmc;
+ NetmapInstance *instance;
+ DAQ_Dict *entry;
+- char intf[IFNAMSIZ];
++ char intf[IFNAMSIZ + 1];
+ uint32_t num_intfs = 0;
+ size_t len;
+ char *name1, *name2, *dev;
+ int rval = DAQ_ERROR;
++#if NETMAP_API >= 14
++ int fd;
++ NetmapInstance *host_peer;
++ struct nmreq_header hdr;
++ struct nmreq_port_info_get nm_info;
++#endif
+
+ nmc = calloc(1, sizeof(Netmap_Context_t));
+ if (!nmc)
+@@ -343,6 +470,7 @@
+ __func__, NETMAP_MAX_INTERFACES);
+ goto err;
+ }
++
+ snprintf(intf, len + 1, "%s", dev);
+ instance = create_instance(intf, nmc->instances, errbuf, errlen);
+ if (!instance)
+@@ -355,8 +483,8 @@
+ {
+ if (num_intfs == 2)
+ {
+- name1 = nmc->instances->next->req.nr_name;
+- name2 = nmc->instances->req.nr_name;
++ name1 = nmc->instances->next->nm_port_name;
++ name2 = nmc->instances->nm_port_name;
+
+ if (create_bridge(nmc, name1, name2) != DAQ_SUCCESS)
+ {
+@@ -364,6 +492,51 @@
+ __func__, name1, name2);
+ goto err;
+ }
++
++#if NETMAP_API >= 14
++ /* Prepare netmap header object to query interface for number of HW rings. */
++ memset(&hdr, 0, sizeof(hdr));
++ memset(&nm_info, 0, sizeof(nm_info));
++ host_peer = NULL;
++ hdr.nr_version = NETMAP_API;
++ hdr.nr_reqtype = NETMAP_REQ_PORT_INFO_GET;
++ hdr.nr_body = (uintptr_t)&nm_info;
++
++ /* Find the NIC side of the bridge pair and query it for number of HW rings.
++ Then set the requested number of Host Rings on the SW Host Stack peer
++ (if present) to match the available HW rings. */
++ if (nmc->instances->reg.nr_mode == NR_REG_ALL_NIC && nmc->instances->next->reg.nr_mode == NR_REG_SW)
++ {
++ fd = nmc->instances->fd;
++ strncpy(hdr.nr_name, nmc->instances->hdr.nr_name, sizeof(hdr.nr_name));
++ host_peer = nmc->instances->next;
++ }
++
++ if (nmc->instances->next->reg.nr_mode == NR_REG_ALL_NIC && nmc->instances->reg.nr_mode == NR_REG_SW)
++ {
++ fd = nmc->instances->next->fd;
++ strncpy(hdr.nr_name, nmc->instances->next->hdr.nr_name, sizeof(hdr.nr_name));
++ host_peer = nmc->instances;
++ }
++
++ /* Did we find a HW NIC and SW Host Stack peer pair?
++ If not, skip requesting the SW Host Stack rings. */
++ if(host_peer)
++ {
++ /* Query the HW NIC netmap device for the number of TX and RX rings. */
++ if (ioctl(fd, NIOCCTRL, &hdr))
++ {
++ DPE(nmc->errbuf, "%s: Netmap NETMAP_REQ_PORT_INFO_GET for %s failed: %s (%d). Will run with default single pair of Host Rings.",
++ __func__, hdr.nr_name, strerror(errno), errno);
++ }
++ else
++ {
++ /* Set the Host Stack peer to request a matching number of Host Tx,Rx rings. */
++ host_peer->reg.nr_host_tx_rings = nm_info.nr_tx_rings;
++ host_peer->reg.nr_host_rx_rings = nm_info.nr_rx_rings;
++ }
++ }
++#endif
+ num_intfs = 0;
+ }
+ else if (num_intfs > 2)
+@@ -569,6 +742,7 @@
+ nminst_inc_tx_ring(peer);
+ if (nm_ring_empty(tx_ring))
+ continue;
++
+ /* Swap the RX buffer we want to forward with the next
+ unused buffer in the peer's TX ring. */
+ tx_cur = tx_ring->cur;
+@@ -577,16 +751,16 @@
+ tx_slot->len = len;
+ tx_slot->buf_idx = rx_slot->buf_idx;
+ rx_slot->buf_idx = tx_buf_idx;
++
+ /* Report the buffer change. */
+ tx_slot->flags |= NS_BUF_CHANGED;
+ rx_slot->flags |= NS_BUF_CHANGED;
+
++ /* copy the NS_MOREFRAG */
++ tx_slot->flags = (tx_slot->flags & ~NS_MOREFRAG) | (rx_slot->flags & NS_MOREFRAG);
++
+ tx_ring->cur = nm_ring_next(tx_ring, tx_cur);
+-#if NETMAP_API >= 10
+ tx_ring->head = tx_ring->cur;
+-#else
+- tx_ring->avail--;
+-#endif
+ sent = 1;
+ } while (peer->cur_tx_ring != start_tx_ring && !sent);
+
+@@ -604,11 +778,7 @@
+ }
+
+ rx_ring->cur = nm_ring_next(rx_ring, rx_cur);
+-#if NETMAP_API >= 10
+ rx_ring->head = rx_ring->cur;
+-#else
+- rx_ring->avail--;
+-#endif
+
+ /* Increment the current RX ring pointer on successfully completed processing. */
+ nminst_inc_rx_ring(instance);
+@@ -644,7 +814,7 @@
+ /* If the poll times out, return control to the caller. */
+ if (ret == 0)
+ break;
+- /* If some number of of sockets have events returned, check them all for badness. */
++ /* If some number of sockets have events returned, check them all for badness. */
+ if (ret > 0)
+ {
+ for (i = 0; i < nmc->intf_count; i++)
+@@ -715,18 +885,14 @@
+ memcpy(NETMAP_BUF(tx_ring, tx_buf_idx), packet_data, len);
+
+ tx_ring->cur = nm_ring_next(tx_ring, tx_cur);
+-#if NETMAP_API >= 10
+ tx_ring->head = tx_ring->cur;
+-#else
+- tx_ring->avail--;
+-#endif
+ nmc->stats.packets_injected++;
+
+ return DAQ_SUCCESS;
+ } while (instance->cur_tx_ring != start_tx_ring);
+
+ /* If we got here, it means we couldn't find an available TX slot, so tell the user to try again. */
+- DPE(nmc->errbuf, "%s: Could not find an available TX slot. Try again.", __func__);
++ DPE(nmc->errbuf, "%s: Could not find an available TX slot for interface %s. Try again.", __func__, instance->nm_port_name);
+ return DAQ_ERROR_AGAIN;
+ }
+
+@@ -826,7 +992,7 @@
+
+ for (instance = nmc->instances; instance; instance = instance->next)
+ {
+- if (!strcmp(device, instance->req.nr_name))
++ if (!strcmp(device, instance->nm_port_name))
+ return instance->index;
+ }
+
More information about the dev-commits-ports-all
mailing list