sipcap/netmap/LINUX/forcedeth_netmap.h
farrokhi ccdba9490b - add netmap-libpcap
- add netmap (FreeBSD header files need to be updated with this)
- move prototype perl scripts to prototype/ folder
- create basic structure for sipcap app (no code yet)
2014-07-10 17:04:13 +04:30

408 lines
11 KiB
C

/*
* Copyright (C) 2012-2014 Luigi Rizzo. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
* ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
* OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE.
*/
/*
* $Id: forcedeth_netmap.h 10670 2012-02-27 21:15:38Z luigi $
*
* netmap support for: forcedeth (nfe, linux)
* For details on netmap support see ixgbe_netmap.h
The driver supports ORIGinal and EXtended descriptors through unions.
We remove the .orig and .ex suffix for brevity.
Pointers in the ring (N slots) are
first_rx = 0, last_rx = N-1, get_rx = put_rx = 0 at init
Following init there is a call to nv_alloc_rx_optimized() which does
less_rx = get_rx - 1
for (put_rx = 0; put_rx != less_rx; put_rx++)
put_rx.flags = LEN | NV_RX2_AVAIL;
so it leaves one free slot and put_rx pointing at the end.
Basically, get_rx is where new packets arrive, put_rx is where
new buffers are added.
The rx_intr aka nv_rx_process_optimized() scans
while (get_rx != put_rx && !(get_rx.flags & NV_RX2_AVAIL)) {
...
get_rx++
}
followed by a nv_alloc_rx_optimized().
This makes sure that there is always a free slot.
*/
#include <bsd_glue.h>
#include <net/netmap.h>
#include <netmap/netmap_kern.h>
#define SOFTC_T fe_priv
/*
* Register/unregister. We are already under netmap lock.
* only called on the first register or the last unregister.
* The "forcedeth" driver is poorly written, the reinit routine
* is replicated multiple times and one way to achieve it is to
* nv_change_mtu twice above ETH_DATA_LEN.
*/
static int
forcedeth_netmap_reg(struct netmap_adapter *na, int onoff)
{
struct ifnet *ifp = na->ifp;
struct SOFTC_T *np = netdev_priv(ifp);
u8 __iomem *base = get_hwbase(ifp);
// first half of nv_change_mtu() - down
nv_disable_irq(ifp);
nv_napi_disable(ifp);
netif_tx_lock_bh(ifp);
netif_addr_lock(ifp);
spin_lock(&np->lock);
/* stop engines */
nv_stop_rxtx(ifp);
nv_txrx_reset(ifp);
/* drain rx queue */
nv_drain_rxtx(ifp);
if (onoff) {
nm_set_native_flags(na);
} else {
nm_clear_native_flags(na);
}
// second half of nv_change_mtu() -- up
if (nv_init_ring(ifp)) {
if (!np->in_shutdown)
mod_timer(&np->oom_kick, jiffies + OOM_REFILL);
}
/* reinit nic view of the rx queue */
writel(np->rx_buf_sz, base + NvRegOffloadConfig);
setup_hw_rings(ifp, NV_SETUP_RX_RING | NV_SETUP_TX_RING);
writel(((np->rx_ring_size-1) << NVREG_RINGSZ_RXSHIFT) + ((np->tx_ring_size-1) << NVREG_RINGSZ_TXSHIFT),
base + NvRegRingSizes);
pci_push(base);
writel(NVREG_TXRXCTL_KICK|np->txrxctl_bits, get_hwbase(ifp) + NvRegTxRxControl);
pci_push(base);
/* restart rx engine */
nv_start_rxtx(ifp);
spin_unlock(&np->lock);
netif_addr_unlock(ifp);
netif_tx_unlock_bh(ifp);
nv_napi_enable(ifp);
nv_enable_irq(ifp);
return (0);
}
/*
* Reconcile kernel and user view of the transmit ring.
*/
static int
forcedeth_netmap_txsync(struct netmap_kring *kring, int flags)
{
struct netmap_adapter *na = kring->na;
struct ifnet *ifp = na->ifp;
struct netmap_ring *ring = kring->ring;
u_int nm_i; /* index into the netmap ring */
u_int nic_i; /* index into the NIC ring */
u_int n;
u_int const lim = kring->nkr_num_slots - 1;
u_int const head = kring->rhead;
/* device-specific */
struct SOFTC_T *np = netdev_priv(ifp);
struct ring_desc_ex *txr = np->tx_ring.ex;
uint32_t lastpkt = (np->desc_ver == DESC_VER_1 ? NV_TX_LASTPACKET : NV_TX2_LASTPACKET);
u_int k;
/*
* First part: process new packets to send.
*/
if (!netif_carrier_ok(ifp)) {
goto out;
}
nm_i = kring->nr_hwcur;
if (nm_i != head) { /* we have new packets to send */
nic_i = np->put_tx.ex - txr; // NIC pointer
for (n = 0; nm_i != head; n++) {
struct netmap_slot *slot = &ring->slot[nm_i];
u_int len = slot->len;
uint64_t paddr;
void *addr = PNMB(slot, &paddr);
/* device-specific */
struct ring_desc_ex *put_tx = txr + nic_i;
// XXX check who needs lastpkt
int cmd = (len - 1) | NV_TX2_VALID | lastpkt;
NM_CHECK_ADDR_LEN(addr, len);
if (slot->flags & NS_BUF_CHANGED) {
/* buffer has changed, reload map */
// netmap_reload_map(pdev, DMA_TO_DEVICE, old_paddr, addr);
}
slot->flags &= ~(NS_REPORT | NS_BUF_CHANGED);
/* Fill the slot in the NIC ring. */
put_tx->bufhigh = htole32(dma_high(paddr));
put_tx->buflow = htole32(dma_low(paddr));
put_tx->flaglen = htole32(cmd);
put_tx->txvlan = 0;
nm_i = nm_next(nm_i, lim);
nic_i = nm_next(nic_i, lim);
}
np->put_tx.ex = txr + nic_i;
kring->nr_hwcur = head;
wmb(); /* synchronize writes to the NIC ring */
/* restart tx unit where is the new index ? */
writel(NVREG_TXRXCTL_KICK|np->txrxctl_bits,
get_hwbase(ifp) + NvRegTxRxControl);
}
/*
* Second part: reclaim buffers for completed transmissions
*/
/* Sync the TX descriptor list */
rmb();
nic_i = np->get_tx.ex - txr;
k = np->put_tx.ex - txr;
if (nic_i != k) {
for (n = 0; nic_i != k; n++) {
uint32_t cmdstat = le32toh(txr[nic_i].flaglen);
if (cmdstat & NV_TX2_VALID)
break;
if (++nic_i == np->tx_ring_size)
nic_i = 0;
}
if (n > 0) {
np->get_tx.ex = txr + nic_i;
kring->nr_hwtail = nm_prev(netmap_idx_n2k(kring, nic_i), lim);
}
}
out:
nm_txsync_finalize(kring);
return 0;
}
/*
* Reconcile kernel and user view of the receive ring.
*/
static int
forcedeth_netmap_rxsync(struct netmap_kring *kring, int flags)
{
struct netmap_adapter *na = kring->na;
struct ifnet *ifp = na->ifp;
struct netmap_ring *ring = kring->ring;
u_int nm_i; /* index into the netmap ring */
u_int nic_i; /* index into the NIC ring */
u_int n;
u_int const lim = kring->nkr_num_slots - 1;
u_int const head = nm_rxsync_prologue(kring);
int force_update = (flags & NAF_FORCE_READ) || kring->nr_kflags & NKR_PENDINTR;
/* device-specific */
struct SOFTC_T *np = netdev_priv(ifp);
struct ring_desc_ex *rxr = np->rx_ring.ex;
u_int refill; // refill position
if (head > lim)
return netmap_ring_reinit(kring);
/*
* First part: import newly received packets.
*/
rmb();
if (netmap_no_pendintr || force_update) {
uint16_t slot_flags = kring->nkr_slot_flags;
nic_i = np->get_rx.ex - rxr; /* next pkt to check */
/* put_rx is the refill position, one before nr_hwcur.
* This slot is not available
*/
refill = np->put_rx.ex - rxr; /* refill position */
nm_i = netmap_idx_n2k(kring, nic_i);
while (nic_i != refill) {
uint32_t statlen = le32toh(rxr[nic_i].flaglen);
if (statlen & NV_RX2_AVAIL) /* still owned by the NIC */
break;
ring->slot[nm_i].len = statlen & LEN_MASK_V2; // XXX crc?
ring->slot[nm_i].flags = slot_flags;
// ifp->stats.rx_packets++;
nm_i = nm_next(nm_i, lim);
nic_i = nm_next(nic_i, lim);
}
np->get_rx.ex = rxr + nic_i;
kring->nr_hwtail = nm_i;
}
/*
* Second part: skip past packets that userspace has released.
*/
nm_i = kring->nr_hwcur; // refill is one before nic_i
if (nm_i != head) {
nic_i = netmap_idx_k2n(kring, nm_i);
refill = np->put_rx.ex - rxr; /* refill position */
for (n = 0; nm_i != head; n++) {
struct netmap_slot *slot = &ring->slot[nm_i];
uint64_t paddr;
void *addr = PNMB(slot, &paddr);
struct ring_desc_ex *desc = rxr + nic_i;
if (addr == netmap_buffer_base) /* bad buf */
goto ring_reset;
if (slot->flags & NS_BUF_CHANGED) {
/* buffer has changed, reload map */
// netmap_reload_map(pdev, DMA_TO_DEVICE, old_paddr, addr);
slot->flags &= ~NS_BUF_CHANGED;
}
desc->flaglen = htole32(NETMAP_BUF_SIZE);
desc->bufhigh = htole32(dma_high(paddr));
desc->buflow = htole32(dma_low(paddr));
// enable the previous buffer
rxr[refill].flaglen |= htole32(NV_RX2_AVAIL);
refill = nm_next(refill, lim);
nm_i = nm_next(nm_i, lim);
nic_i = nm_next(nic_i, lim);
}
kring->nr_hwcur = head;
np->put_rx.ex = rxr + refill;
/* Flush the RX DMA ring */
wmb();
}
/* tell userspace that there are might be new packets */
nm_rxsync_finalize(kring);
return 0;
ring_reset:
return netmap_ring_reinit(kring);
}
/*
* Additional routines to init the tx and rx rings.
* In other drivers we do that inline in the main code.
*/
static int
forcedeth_netmap_tx_init(struct SOFTC_T *np)
{
struct ring_desc_ex *desc;
int i, n;
struct netmap_adapter *na = NA(np->dev);
struct netmap_slot *slot;
if (!na || !(na->na_flags & NAF_NATIVE_ON)) {
return 0;
}
slot = netmap_reset(na, NR_TX, 0, 0);
/* slot is NULL if we are not in netmap mode */
if (!slot)
return 0;
/* in netmap mode, overwrite addresses and maps */
//txd = np->rl_ldata.rl_tx_desc;
desc = np->tx_ring.ex;
n = np->tx_ring_size;
/* l points in the netmap ring, i points in the NIC ring */
for (i = 0; i < n; i++) {
int l = netmap_idx_n2k(&na->tx_rings[0], i);
uint64_t paddr;
PNMB(slot + l, &paddr);
desc[i].flaglen = 0;
desc[i].bufhigh = htole32(dma_high(paddr));
desc[i].buflow = htole32(dma_low(paddr));
}
return 1;
}
static int
forcedeth_netmap_rx_init(struct SOFTC_T *np)
{
struct netmap_adapter *na = NA(np->dev);
struct netmap_slot *slot = netmap_reset(na, NR_RX, 0, 0);
struct ring_desc_ex *desc = np->rx_ring.ex;
uint32_t cmdstat;
int i, lim;
if (!slot)
return 0;
/*
* Do not release the slots owned by userspace,
* and also keep one empty.
*/
lim = np->rx_ring_size - 1 - nm_kr_rxspace(&na->rx_rings[0]);
for (i = 0; i < np->rx_ring_size; i++) {
void *addr;
uint64_t paddr;
int l = netmap_idx_n2k(&na->rx_rings[0], i);
addr = PNMB(slot + l, &paddr);
netmap_reload_map(np->rl_ldata.rl_rx_mtag,
np->rl_ldata.rl_rx_desc[i].rx_dmamap, addr);
desc[i].bufhigh = htole32(dma_high(paddr));
desc[i].buflow = htole32(dma_low(paddr));
cmdstat = NETMAP_BUF_SIZE;
if (i < lim)
cmdstat |= NV_RX2_AVAIL;
desc[i].flaglen = htole32(cmdstat);
}
// XXX ring end anywhere ?
np->get_rx.ex = desc;
np->put_rx.ex = desc + lim;
return 1;
}
static void
forcedeth_netmap_attach(struct SOFTC_T *np)
{
struct netmap_adapter na;
bzero(&na, sizeof(na));
na.ifp = np->dev;
na.num_tx_desc = np->tx_ring_size;
na.num_rx_desc = np->tx_ring_size;
na.nm_txsync = forcedeth_netmap_txsync;
na.nm_rxsync = forcedeth_netmap_rxsync;
na.nm_register = forcedeth_netmap_reg;
na.num_tx_rings = na.num_rx_rings = 1;
netmap_attach(&na);
}
/* end of file */