2846 lines
80 KiB
C
2846 lines
80 KiB
C
/*! \file ngknet_main.c
|
|
*
|
|
* NGKNET module entry.
|
|
*
|
|
*/
|
|
/*
|
|
* $Copyright: Copyright 2018-2022 Broadcom. All rights reserved.
|
|
* The term 'Broadcom' refers to Broadcom Inc. and/or its subsidiaries.
|
|
*
|
|
* 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 published by the Free Software Foundation.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* A copy of the GNU General Public License version 2 (GPLv2) can
|
|
* be found in the LICENSES folder.$
|
|
*/
|
|
|
|
/*
|
|
* This module implements a Linux network driver for Broadcom
|
|
* XGS switch devices. The driver simultaneously serves a
|
|
* number of virtual Linux network devices.
|
|
*
|
|
* Packets received from the switch device are sent to a virtual
|
|
* Linux network device based on a set of packet filters.
|
|
*
|
|
* Packets from the virtual Linux network devices are multiplexed
|
|
* with fifo mode if only one Tx queue enabled.
|
|
*
|
|
* A command-based IOCTL interface is used for managing the devices,
|
|
* packet filters and virtual Linux network interfaces.
|
|
*
|
|
* A virtual network interface can be configured to work in RCPU
|
|
* mode, which means that packets from the switch device will
|
|
* be encapsulated with a RCPU header and a block of meta data
|
|
* that basically contains the core DCB information. Likewise,
|
|
* packets received from the Linux network stack are assumed to
|
|
* be RCPU encapsulated when going out on an interface in RCPU
|
|
* mode. If a virtual network interface does not work in RCPU
|
|
* mode and transmits to this interface will unmodified go to
|
|
* specified physical switch port, DCB information should be
|
|
* provided when the interface is created.
|
|
*
|
|
* The module implements basic Rx DMA rate control. The rate is
|
|
* specified in packets per second, and different Rx DMA channels
|
|
* can be configured to use different maximum packet rates.
|
|
* The packet rate can be configure as a module parameter, and
|
|
* it can also be changed dynamically through the proc file
|
|
* system (syntax is described in function header comment).
|
|
*
|
|
* For a list of supported module parameters, please see below.
|
|
*/
|
|
|
|
#include <linux/kconfig.h>
|
|
#include <linux/kernel.h>
|
|
#include <linux/types.h>
|
|
#include <linux/init.h>
|
|
#include <linux/string.h>
|
|
#include <linux/errno.h>
|
|
#include <linux/unistd.h>
|
|
#include <asm/io.h>
|
|
#include <linux/uaccess.h>
|
|
#include <linux/module.h>
|
|
#include <linux/netdevice.h>
|
|
#include <linux/etherdevice.h>
|
|
#include <linux/skbuff.h>
|
|
#include <linux/if.h>
|
|
#include <linux/if_vlan.h>
|
|
#include <linux/net_tstamp.h>
|
|
#include <linux/mm.h>
|
|
#include <linux/dma-mapping.h>
|
|
#include <linux/vmalloc.h>
|
|
#include <linux/pagemap.h>
|
|
#include <linux/slab.h>
|
|
#include <linux/interrupt.h>
|
|
#include <linux/spinlock.h>
|
|
#include <linux/delay.h>
|
|
#include <linux/bitops.h>
|
|
#include <linux/time.h>
|
|
#include <linux/random.h>
|
|
|
|
#include <lkm/ngbde_kapi.h>
|
|
#include <lkm/ngknet_dev.h>
|
|
#include <lkm/ngknet_ioctl.h>
|
|
#include <bcmcnet/bcmcnet_core.h>
|
|
#include "ngknet_main.h"
|
|
#include "ngknet_extra.h"
|
|
#include "ngknet_procfs.h"
|
|
#include "ngknet_callback.h"
|
|
#include "ngknet_ptp.h"
|
|
|
|
/*! \cond */
|
|
MODULE_AUTHOR("Broadcom Corporation");
|
|
MODULE_DESCRIPTION("Network Device Driver Module");
|
|
MODULE_LICENSE("GPL");
|
|
/*! \endcond */
|
|
|
|
/*! \cond */
|
|
static int debug = 0;
|
|
MODULE_PARAM(debug, int, 0);
|
|
MODULE_PARM_DESC(debug,
|
|
"Debug level (default 0)");
|
|
/*! \endcond */
|
|
|
|
/*! \cond */
|
|
static char *base_dev_name = "bcm";
|
|
MODULE_PARAM(base_dev_name, charp, 0);
|
|
MODULE_PARM_DESC(base_dev_name,
|
|
"Base device name (default bcm0, bcm1, etc.)");
|
|
/*! \endcond */
|
|
|
|
/*! \cond */
|
|
static char *mac_addr = NULL;
|
|
MODULE_PARAM(mac_addr, charp, 0);
|
|
MODULE_PARM_DESC(mac_addr,
|
|
"Ethernet MAC address (default 02:10:18:xx:xx:xx)");
|
|
/*! \endcond */
|
|
|
|
/*! \cond */
|
|
static int default_mtu = 1500;
|
|
MODULE_PARAM(default_mtu, int, 0);
|
|
MODULE_PARM_DESC(default_mtu,
|
|
"Default MTU for NGKNET network interfaces (default 1500)");
|
|
/*! \endcond */
|
|
|
|
/*! \cond */
|
|
static int rx_buffer_size = RX_BUF_SIZE_DFLT;
|
|
MODULE_PARAM(rx_buffer_size, int, 0);
|
|
MODULE_PARM_DESC(rx_buffer_size,
|
|
"Default size of RX packet buffers (default 9216)");
|
|
/*! \endcond */
|
|
|
|
/*! \cond */
|
|
static int rx_rate_limit = -1;
|
|
MODULE_PARAM(rx_rate_limit, int, 0);
|
|
MODULE_PARM_DESC(rx_rate_limit,
|
|
"Rx rate limit (pps, default -1 no limit)");
|
|
/*! \endcond */
|
|
|
|
/*! \cond */
|
|
static int tx_polling = 0;
|
|
MODULE_PARAM(tx_polling, int, 0);
|
|
MODULE_PARM_DESC(tx_polling,
|
|
"Tx polling mode (default 0 in interrupt mode)");
|
|
/*! \endcond */
|
|
|
|
/*! \cond */
|
|
static int rx_batching = 0;
|
|
MODULE_PARAM(rx_batching, int, 0);
|
|
MODULE_PARM_DESC(rx_batching,
|
|
"Rx batching mode (default 0 in single fill mode)");
|
|
/*! \endcond */
|
|
|
|
/*! \cond */
|
|
static int page_buffer_mode = -1;
|
|
MODULE_PARAM(page_buffer_mode, int, 0);
|
|
MODULE_PARM_DESC(rx_batching,
|
|
"Page buffer mode (default -1 do not override, 0 forced disable, 1 forced enable)");
|
|
/*! \endcond */
|
|
|
|
typedef int (*drv_ops_attach)(struct pdma_dev *dev);
|
|
|
|
struct bcmcnet_drv_ops {
|
|
const char *drv_desc;
|
|
drv_ops_attach drv_attach;
|
|
drv_ops_attach drv_detach;
|
|
};
|
|
|
|
#define BCMDRD_DEVLIST_ENTRY(_nm,_vn,_dv,_rv,_md,_pi,_bd,_bc,_fn,_cn,_pf,_pd,_r0,_r1) \
|
|
static struct bcmcnet_drv_ops _bd##_cnet_drv_ops = { \
|
|
#_bd, \
|
|
_bd##_cnet_pdma_attach, \
|
|
_bd##_cnet_pdma_detach, \
|
|
};
|
|
#include <bcmdrd/bcmdrd_devlist.h>
|
|
|
|
#define BCMDRD_DEVLIST_ENTRY(_nm,_vn,_dv,_rv,_md,_pi,_bd,_bc,_fn,_cn,_pf,_pd,_r0,_r1) \
|
|
&_bd##_cnet_drv_ops,
|
|
static struct bcmcnet_drv_ops *drv_ops[] = {
|
|
NULL,
|
|
#include <bcmdrd/bcmdrd_devlist.h>
|
|
NULL
|
|
};
|
|
static int drv_num = sizeof(drv_ops) / sizeof(drv_ops[0]);
|
|
|
|
struct ngknet_dev ngknet_devices[NUM_PDMA_DEV_MAX];
|
|
|
|
/* Default random MAC address has Broadcom OUI with local admin bit set */
|
|
static uint8_t ngknet_dev_mac[6] = {0x02, 0x10, 0x18, 0x00, 0x00, 0x00};
|
|
|
|
/* Interrupt handles */
|
|
struct ngknet_intr_handle {
|
|
struct napi_struct napi;
|
|
struct intr_handle *hdl;
|
|
int napi_resched;
|
|
int napi_pending;
|
|
};
|
|
|
|
static struct ngknet_intr_handle priv_hdl[NUM_PDMA_DEV_MAX][NUM_Q_MAX];
|
|
|
|
/*!
|
|
* Dump packet content for debug
|
|
*/
|
|
static void
|
|
ngknet_pkt_dump(uint8_t *data, int len)
|
|
{
|
|
char str[128];
|
|
int i;
|
|
|
|
len = len > 256 ? 256 : len;
|
|
|
|
for (i = 0; i < len; i++) {
|
|
if ((i & 0x1f) == 0) {
|
|
sprintf(str, "%04x: ", i);
|
|
}
|
|
sprintf(&str[strlen(str)], "%02x", data[i]);
|
|
if ((i & 0x1f) == 0x1f) {
|
|
sprintf(&str[strlen(str)], "\n");
|
|
printk(str);
|
|
continue;
|
|
}
|
|
if ((i & 0x3) == 0x3) {
|
|
sprintf(&str[strlen(str)], " ");
|
|
}
|
|
}
|
|
if ((i & 0x1f) != 0) {
|
|
sprintf(&str[strlen(str)], "\n");
|
|
printk(str);
|
|
}
|
|
printk("\n");
|
|
}
|
|
|
|
/*!
|
|
* Rx packets rate test for debug
|
|
*/
|
|
static void
|
|
ngknet_pkt_stats(struct pdma_dev *pdev, int dir)
|
|
{
|
|
s64 ts0[2], ts1[2];
|
|
static uint32_t pkts[2] = {0}, prts[2] = {0};
|
|
static uint64_t intrs = 0;
|
|
uint32_t iv_time;
|
|
uint32_t pps;
|
|
uint32_t boudary;
|
|
|
|
if (rx_rate_limit == -1 || rx_rate_limit >= 100000) {
|
|
/* Dump every 100K packets */
|
|
boudary = 100000;
|
|
} else if (rx_rate_limit >= 10000) {
|
|
/* Dump every 10K packets */
|
|
boudary = 10000;
|
|
} else {
|
|
/* Dump every 1K packets */
|
|
boudary = 1000;
|
|
}
|
|
|
|
if (pkts[dir] == 0) {
|
|
ts0[dir] = kal_time_usecs();
|
|
intrs = pdev->stats.intrs;
|
|
}
|
|
if (++pkts[dir] >= boudary) {
|
|
ts1[dir] = kal_time_usecs();
|
|
iv_time = ts1[dir] - ts0[dir];
|
|
pps = boudary * 1000 / (iv_time / 1000);
|
|
prts[dir]++;
|
|
/* pdev->stats.intrs is reset and re-count from 0. */
|
|
if (intrs > pdev->stats.intrs) {
|
|
intrs = 0;
|
|
}
|
|
if (pps <= boudary || prts[dir] * boudary >= pps) {
|
|
printk(KERN_CRIT "%s - limit: %d pps, %dK pkts time: %d usec, "
|
|
"rate: %d pps, intrs: %llu\n",
|
|
dir == PDMA_Q_RX ? "Rx" : "Tx",
|
|
dir == PDMA_Q_RX ? rx_rate_limit : -1, (boudary / 1000),
|
|
iv_time, pps, pdev->stats.intrs - intrs);
|
|
prts[dir] = 0;
|
|
}
|
|
pkts[dir] = 0;
|
|
}
|
|
}
|
|
|
|
/*!
|
|
* Read 32-bit register callback
|
|
*/
|
|
static int
|
|
ngknet_dev_read32(struct pdma_dev *dev, uint32_t addr, uint32_t *data)
|
|
{
|
|
*data = ngbde_kapi_pio_read32(dev->unit, addr);
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*!
|
|
* Write 32-bit register callback
|
|
*/
|
|
static int
|
|
ngknet_dev_write32(struct pdma_dev *dev, uint32_t addr, uint32_t data)
|
|
{
|
|
ngbde_kapi_pio_write32(dev->unit, addr, data);
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*!
|
|
* Set Rx HW timestamping.
|
|
*/
|
|
static int
|
|
ngknet_ptp_rx_hwts_set(struct net_device *ndev, struct sk_buff *skb)
|
|
{
|
|
struct skb_shared_hwtstamps *shhwtstamps = skb_hwtstamps(skb);
|
|
uint64_t ts = 0;
|
|
int rv;
|
|
|
|
rv = ngknet_ptp_rx_hwts_get(ndev, skb, &ts);
|
|
if (SHR_FAILURE(rv) || !ts) {
|
|
return SHR_E_FAIL;
|
|
}
|
|
|
|
memset(shhwtstamps, 0, sizeof(*shhwtstamps));
|
|
shhwtstamps->hwtstamp = ns_to_ktime(ts);
|
|
|
|
return SHR_E_NONE;
|
|
}
|
|
|
|
/*!
|
|
* \brief Process Rx packet.
|
|
*
|
|
* Add RCPU encapsulation or strip matadata if needed
|
|
*
|
|
* \param [in] ndev Network device structure point.
|
|
* \param [in] oskb Rx packet SKB.
|
|
*
|
|
* \retval SHR_E_NONE No errors.
|
|
* \retval SHR_E_XXXX Operation failed.
|
|
*/
|
|
static int
|
|
ngknet_rx_frame_process(struct net_device *ndev, struct sk_buff **oskb)
|
|
{
|
|
struct ngknet_private *priv = netdev_priv(ndev);
|
|
struct ngknet_dev *dev = priv->bkn_dev;
|
|
struct sk_buff *skb = *oskb;
|
|
struct ngknet_rcpu_hdr *rch = (struct ngknet_rcpu_hdr *)skb->data;
|
|
struct pkt_hdr *pkh = (struct pkt_hdr *)skb->data;
|
|
uint8_t meta_len = pkh->meta_len;
|
|
|
|
/* Do Rx timestamping */
|
|
if (priv->hwts_rx_filter) {
|
|
ngknet_ptp_rx_hwts_set(ndev, skb);
|
|
}
|
|
|
|
/* Remove FCS from packet length */
|
|
skb_trim(skb, skb->len - ETH_FCS_LEN);
|
|
pkh->data_len -= ETH_FCS_LEN;
|
|
|
|
if (priv->netif.flags & NGKNET_NETIF_F_RCPU_ENCAP) {
|
|
/* Set up RCPU header */
|
|
memcpy(skb->data, skb->data + PKT_HDR_SIZE + meta_len, 2 * ETH_ALEN);
|
|
if (*(uint32_t *)&dev->rcpu_ctrl.dst_mac[0] != 0 ||
|
|
*(uint16_t *)&dev->rcpu_ctrl.dst_mac[4] != 0) {
|
|
memcpy(rch->dst_mac, dev->rcpu_ctrl.dst_mac, ETH_ALEN);
|
|
}
|
|
if (*(uint32_t *)&dev->rcpu_ctrl.src_mac[0] != 0 ||
|
|
*(uint16_t *)&dev->rcpu_ctrl.src_mac[4] != 0) {
|
|
memcpy(rch->src_mac, dev->rcpu_ctrl.src_mac, ETH_ALEN);
|
|
}
|
|
rch->vlan_tpid = htons(dev->rcpu_ctrl.vlan_tpid);
|
|
rch->vlan_tci = htons(dev->rcpu_ctrl.vlan_tci);
|
|
rch->eth_type = htons(dev->rcpu_ctrl.eth_type);
|
|
rch->pkt_sig = htons(dev->rcpu_ctrl.pkt_sig);
|
|
rch->op_code = RCPU_OPCODE_RX;
|
|
rch->flags = RCPU_FLAG_MODHDR;
|
|
rch->trans_id = htons(dev->rcpu_ctrl.trans_id);
|
|
rch->data_len = htons(pkh->data_len);
|
|
} else {
|
|
/* Remove packet header and meta data */
|
|
skb_pull(skb, PKT_HDR_SIZE + meta_len);
|
|
}
|
|
|
|
/* Check to ensure ngknet_callback_desc struct fits in sk_buff->cb */
|
|
BUILD_BUG_ON(sizeof(struct ngknet_callback_desc) > sizeof(skb->cb));
|
|
|
|
/* Optional callback handle */
|
|
if (dev->cbc->rx_cb) {
|
|
struct ngknet_callback_desc *cbd = NGKNET_SKB_CB(skb);
|
|
cbd->dinfo = &dev->dev_info;
|
|
cbd->netif = &priv->netif;
|
|
if (priv->netif.flags & NGKNET_NETIF_F_RCPU_ENCAP) {
|
|
cbd->pmd = skb->data + PKT_HDR_SIZE;
|
|
cbd->pkt_len = ntohs(rch->data_len);
|
|
} else {
|
|
cbd->pmd = skb->data - meta_len;
|
|
cbd->pkt_len = pkh->data_len;
|
|
}
|
|
cbd->pmd_len = meta_len;
|
|
skb = dev->cbc->rx_cb(ndev, skb);
|
|
if (!skb) {
|
|
*oskb = NULL;
|
|
return SHR_E_UNAVAIL;
|
|
}
|
|
if (priv->netif.flags & NGKNET_NETIF_F_RCPU_ENCAP) {
|
|
rch = (struct ngknet_rcpu_hdr *)skb->data;
|
|
rch->data_len = htons(skb->len - PKT_HDR_SIZE - meta_len);
|
|
}
|
|
}
|
|
|
|
/* Update SKB pointer */
|
|
*oskb = skb;
|
|
|
|
return SHR_E_NONE;
|
|
}
|
|
|
|
/*!
|
|
* \brief Network interface Rx function.
|
|
*
|
|
* After processing the packet, send it up to network stack.
|
|
*
|
|
* \param [in] ndev Network device structure point.
|
|
* \param [in] skb Rx packet SKB.
|
|
*
|
|
* \retval SHR_E_NONE No errors.
|
|
* \retval SHR_E_XXXX Operation failed.
|
|
*/
|
|
static int
|
|
ngknet_netif_recv(struct net_device *ndev, struct sk_buff *skb)
|
|
{
|
|
struct ngknet_private *priv = netdev_priv(ndev);
|
|
struct ngknet_dev *dev = priv->bkn_dev;
|
|
struct pdma_dev *pdev = &dev->pdma_dev;
|
|
struct pkt_hdr *pkh = (struct pkt_hdr *)skb->data;
|
|
struct napi_struct *napi = NULL;
|
|
uint16_t proto;
|
|
int chan_id, gi, qi, skb_len;
|
|
int rv;
|
|
|
|
/* Handle one incoming packet */
|
|
rv = ngknet_rx_frame_process(ndev, &skb);
|
|
if (SHR_FAILURE(rv)) {
|
|
if (!skb) {
|
|
return SHR_E_NONE;
|
|
}
|
|
}
|
|
|
|
DBG_VERB(("Rx packet sent up to ndev%d (%d bytes).\n",
|
|
priv->netif.id, skb->len));
|
|
if (debug & DBG_LVL_PDMP) {
|
|
ngknet_pkt_dump(skb->data, skb->len);
|
|
}
|
|
|
|
if (ndev->features & NETIF_F_RXCSUM) {
|
|
if ((pkh->attrs & (PDMA_RX_TU_CSUM | PDMA_RX_IP_CSUM)) ==
|
|
(PDMA_RX_TU_CSUM | PDMA_RX_IP_CSUM)) {
|
|
skb->ip_summed = CHECKSUM_UNNECESSARY;
|
|
} else {
|
|
skb_checksum_none_assert(skb);
|
|
}
|
|
}
|
|
|
|
proto = eth_type_trans(skb, ndev);
|
|
if (priv->netif.flags & NGKNET_NETIF_F_RCPU_ENCAP) {
|
|
skb->protocol = htons(dev->rcpu_ctrl.eth_type);
|
|
} else if (!(pkh->attrs & PDMA_RX_SET_PROTO) || !skb->protocol) {
|
|
skb->protocol = proto;
|
|
}
|
|
|
|
skb_record_rx_queue(skb, pkh->queue_id);
|
|
|
|
rv = bcmcnet_pdma_dev_queue_to_chan(pdev, pkh->queue_id, PDMA_Q_RX, &chan_id);
|
|
if (SHR_FAILURE(rv)) {
|
|
return rv;
|
|
}
|
|
|
|
gi = chan_id / pdev->grp_queues;
|
|
if (pdev->flags & PDMA_GROUP_INTR) {
|
|
napi = (struct napi_struct *)pdev->ctrl.grp[gi].intr_hdl[0].priv;
|
|
} else {
|
|
qi = pkh->queue_id;
|
|
napi = (struct napi_struct *)pdev->ctrl.grp[gi].intr_hdl[qi].priv;
|
|
}
|
|
|
|
/* FIXME: File CSP on KASAN warning on use-after-free in ngknet_netif_recv */
|
|
skb_len = skb->len;
|
|
napi_gro_receive(napi, skb);
|
|
/* Update accounting */
|
|
priv->stats.rx_packets++;
|
|
priv->stats.rx_bytes += skb_len;
|
|
|
|
|
|
/* Rate limit */
|
|
if (rx_rate_limit >= 0) {
|
|
if (!ngknet_rx_rate_limit_started()) {
|
|
ngknet_rx_rate_limit_start(dev);
|
|
}
|
|
ngknet_rx_rate_limit(dev, rx_rate_limit);
|
|
}
|
|
|
|
return SHR_E_NONE;
|
|
}
|
|
|
|
/*!
|
|
* \brief Driver Rx callback.
|
|
*
|
|
* After processing the packet, send it up to network stack.
|
|
*
|
|
* \param [in] pdev Packet DMA device structure point.
|
|
* \param [in] buf Raw Rx buffer.
|
|
*
|
|
* \retval SHR_E_NONE No errors.
|
|
* \retval SHR_E_XXXX Operation failed.
|
|
*/
|
|
static int
|
|
ngknet_frame_recv(struct pdma_dev *pdev, int queue, void *buf)
|
|
{
|
|
struct ngknet_dev *dev = (struct ngknet_dev *)pdev->priv;
|
|
struct sk_buff *skb = (struct sk_buff *)buf, *mskb = NULL;
|
|
struct net_device *ndev = NULL, *mndev = NULL;
|
|
struct ngknet_private *priv = NULL;
|
|
unsigned long flags;
|
|
int rv;
|
|
|
|
DBG_VERB(("Rx packet (%d bytes).\n", skb->len));
|
|
if (debug & DBG_LVL_PDMP) {
|
|
ngknet_pkt_dump(skb->data, skb->len);
|
|
}
|
|
|
|
DBG_NDEV(("Valid virtual network devices: %ld.\n", (long)dev->vdev[0]));
|
|
|
|
/* Go through the filters */
|
|
rv = ngknet_rx_pkt_filter(dev, skb, &ndev, &mndev, &mskb);
|
|
if (SHR_FAILURE(rv) || !ndev) {
|
|
return SHR_E_FAIL;
|
|
}
|
|
|
|
/* Populate header, checksum status, VLAN, and protocol */
|
|
priv = netdev_priv(ndev);
|
|
if (!netif_carrier_ok(ndev) ||
|
|
SHR_FAILURE(ngknet_netif_recv(ndev, skb))) {
|
|
priv->stats.rx_dropped++;
|
|
rv = SHR_E_UNAVAIL;
|
|
}
|
|
|
|
spin_lock_irqsave(&dev->lock, flags);
|
|
priv->users--;
|
|
if (!priv->users && priv->wait) {
|
|
wake_up(&dev->wq);
|
|
}
|
|
spin_unlock_irqrestore(&dev->lock, flags);
|
|
|
|
/* Handle mirrored packet */
|
|
if (mndev && mskb) {
|
|
priv = netdev_priv(mndev);
|
|
if (!netif_carrier_ok(mndev) ||
|
|
SHR_FAILURE(ngknet_netif_recv(mndev, mskb))) {
|
|
priv->stats.rx_dropped++;
|
|
dev_kfree_skb_any(mskb);
|
|
}
|
|
spin_lock_irqsave(&dev->lock, flags);
|
|
priv->users--;
|
|
if (!priv->users && priv->wait) {
|
|
wake_up(&dev->wq);
|
|
}
|
|
spin_unlock_irqrestore(&dev->lock, flags);
|
|
}
|
|
|
|
/* Measure speed */
|
|
if (debug & DBG_LVL_RATE) {
|
|
ngknet_pkt_stats(pdev, PDMA_Q_RX);
|
|
}
|
|
|
|
return rv;
|
|
}
|
|
|
|
/*!
|
|
* Set Tx HW timestamping.
|
|
*/
|
|
static int
|
|
ngknet_ptp_tx_hwts_set(struct net_device *ndev, struct sk_buff *skb)
|
|
{
|
|
struct skb_shared_hwtstamps shhwtstamps;
|
|
uint64_t ts = 0;
|
|
int rv;
|
|
|
|
rv = ngknet_ptp_tx_hwts_get(ndev, skb, &ts);
|
|
if (SHR_FAILURE(rv) || !ts) {
|
|
return SHR_E_FAIL;
|
|
}
|
|
|
|
memset(&shhwtstamps, 0, sizeof(shhwtstamps));
|
|
shhwtstamps.hwtstamp = ns_to_ktime(ts);
|
|
skb_tstamp_tx(skb, &shhwtstamps);
|
|
|
|
return SHR_E_NONE;
|
|
}
|
|
|
|
/*!
|
|
* PTP Tx worker.
|
|
*/
|
|
static void
|
|
ngknet_ptp_tx_work(struct work_struct *work)
|
|
{
|
|
struct ngknet_dev *dev = container_of(work, struct ngknet_dev, ptp_tx_work);
|
|
struct sk_buff *skb;
|
|
int rv;
|
|
|
|
while (skb_queue_len(&dev->ptp_tx_queue)) {
|
|
skb = skb_dequeue(&dev->ptp_tx_queue);
|
|
rv = ngknet_ptp_tx_hwts_set(dev->net_dev, skb);
|
|
if (SHR_FAILURE(rv)) {
|
|
printk("Timestamp value has not been set for current skb.\n");
|
|
}
|
|
dev_kfree_skb_any(skb);
|
|
}
|
|
}
|
|
|
|
/*!
|
|
* Config Tx metadata for HW timestamping.
|
|
*/
|
|
static int
|
|
ngknet_ptp_tx_config(struct net_device *ndev, struct sk_buff *skb)
|
|
{
|
|
struct ngknet_private *priv = netdev_priv(ndev);
|
|
struct ngknet_dev *dev = priv->bkn_dev;
|
|
int rv;
|
|
|
|
if (priv->netif.type == NGKNET_NETIF_T_PORT) {
|
|
rv = ngknet_ptp_tx_meta_set(ndev, skb);
|
|
if (SHR_FAILURE(rv)) {
|
|
return rv;
|
|
}
|
|
} else if (priv->hwts_tx_type != HWTSTAMP_TX_ONESTEP_SYNC) {
|
|
return SHR_E_UNAVAIL;
|
|
}
|
|
|
|
skb_shinfo(skb)->tx_flags |= SKBTX_IN_PROGRESS;
|
|
|
|
if (priv->hwts_tx_type == HWTSTAMP_TX_ONESTEP_SYNC) {
|
|
skb_queue_tail(&dev->ptp_tx_queue, skb_get(skb));
|
|
schedule_work(&dev->ptp_tx_work);
|
|
}
|
|
|
|
return SHR_E_NONE;
|
|
}
|
|
|
|
/*!
|
|
* \brief Process Tx packet.
|
|
*
|
|
* Strip RCPU encapsulation, setup CNET packet buffer, add vlan tag
|
|
* or pad the packet.
|
|
*
|
|
* \param [in] ndev Network device structure point.
|
|
* \param [in] oskb Tx packet SKB.
|
|
*
|
|
* \retval SHR_E_NONE No errors.
|
|
* \retval SHR_E_XXXX Operation failed.
|
|
*/
|
|
static int
|
|
ngknet_tx_frame_process(struct net_device *ndev, struct sk_buff **oskb)
|
|
{
|
|
struct ngknet_private *priv = netdev_priv(ndev);
|
|
struct ngknet_dev *dev = priv->bkn_dev;
|
|
struct sk_buff *skb = *oskb;
|
|
struct ngknet_rcpu_hdr *rch = (struct ngknet_rcpu_hdr *)skb->data;
|
|
struct pkt_hdr *pkh = (struct pkt_hdr *)skb->data;
|
|
struct sk_buff *nskb = NULL;
|
|
char *data = NULL;
|
|
uint32_t copy_len, meta_len, data_len, pkt_len, tag_len, pad_len;
|
|
uint16_t tpid;
|
|
|
|
/* Set up packet header */
|
|
if (priv->netif.flags & NGKNET_NETIF_F_RCPU_ENCAP) {
|
|
/* RCPU encapsulation packet */
|
|
data_len = pkh->attrs & PDMA_TX_HDR_COOKED ?
|
|
pkh->data_len - ETH_FCS_LEN : ntohs(rch->data_len);
|
|
pkt_len = PKT_HDR_SIZE + rch->meta_len + data_len;
|
|
if (skb->len != pkt_len || skb->len < (PKT_HDR_SIZE + ETH_HLEN)) {
|
|
DBG_WARN(("Tx drop: Invalid packet length\n"));
|
|
return SHR_E_PARAM;
|
|
}
|
|
if (dev->rcpu_ctrl.pkt_sig && dev->rcpu_ctrl.pkt_sig != ntohs(rch->pkt_sig)) {
|
|
DBG_WARN(("Tx drop: Invalid packet signature\n"));
|
|
return SHR_E_PARAM;
|
|
}
|
|
if (pkh->attrs & PDMA_TX_HDR_COOKED) {
|
|
/* Resumed packet */
|
|
return SHR_E_NONE;
|
|
}
|
|
pkh->data_len = data_len + ETH_FCS_LEN;
|
|
pkh->meta_len = rch->meta_len;
|
|
pkh->attrs = 0;
|
|
if (rch->flags & RCPU_FLAG_MODHDR) {
|
|
pkh->attrs |= PDMA_TX_HIGIG_PKT;
|
|
}
|
|
if (rch->flags & RCPU_FLAG_PAUSE) {
|
|
pkh->attrs |= PDMA_TX_PAUSE_PKT;
|
|
}
|
|
if (rch->flags & RCPU_FLAG_PURGE) {
|
|
pkh->attrs |= PDMA_TX_PURGE_PKT;
|
|
}
|
|
if (rch->flags & RCPU_FLAG_BIND_QUE) {
|
|
pkh->attrs |= PDMA_TX_BIND_QUE;
|
|
}
|
|
if (rch->flags & RCPU_FLAG_NO_PAD) {
|
|
pkh->attrs |= PDMA_TX_NO_PAD;
|
|
}
|
|
} else {
|
|
/* Non-RCPU encapsulation packet */
|
|
data_len = pkh->data_len - ETH_FCS_LEN;
|
|
pkt_len = PKT_HDR_SIZE + pkh->meta_len + data_len;
|
|
if (skb->len == pkt_len && pkh->attrs & PDMA_TX_HDR_COOKED &&
|
|
pkh->pkt_sig == dev->rcpu_ctrl.pkt_sig) {
|
|
/* Resumed packet */
|
|
return SHR_E_NONE;
|
|
}
|
|
meta_len = 0;
|
|
if (priv->netif.type == NGKNET_NETIF_T_PORT) {
|
|
meta_len = priv->netif.meta_len;
|
|
if (!meta_len) {
|
|
printk("Tx abort: no metadata\n");
|
|
return SHR_E_UNAVAIL;
|
|
}
|
|
}
|
|
if (skb_header_cloned(skb) ||
|
|
skb_headroom(skb) < (PKT_HDR_SIZE + meta_len + VLAN_HLEN) ||
|
|
skb_tailroom(skb) < ETH_FCS_LEN) {
|
|
nskb = skb_copy_expand(skb, PKT_HDR_SIZE + meta_len + VLAN_HLEN,
|
|
ETH_FCS_LEN, GFP_ATOMIC);
|
|
if (!nskb) {
|
|
return SHR_E_MEMORY;
|
|
}
|
|
skb_shinfo(nskb)->tx_flags = skb_shinfo(skb)->tx_flags;
|
|
skb = nskb;
|
|
}
|
|
skb_push(skb, PKT_HDR_SIZE + meta_len);
|
|
memset(skb->data, 0, PKT_HDR_SIZE + meta_len);
|
|
pkh = (struct pkt_hdr *)skb->data;
|
|
pkh->data_len = skb->len - PKT_HDR_SIZE - meta_len + ETH_FCS_LEN;
|
|
pkh->meta_len = meta_len;
|
|
pkh->attrs = 0;
|
|
if (priv->netif.type == NGKNET_NETIF_T_PORT) {
|
|
/* Send to physical port using netif metadata */
|
|
if (priv->netif.meta_off) {
|
|
memmove(skb->data + PKT_HDR_SIZE,
|
|
skb->data + PKT_HDR_SIZE + meta_len,
|
|
priv->netif.meta_off);
|
|
}
|
|
memcpy(skb->data + PKT_HDR_SIZE + priv->netif.meta_off,
|
|
priv->netif.meta_data, priv->netif.meta_len);
|
|
pkh->attrs |= PDMA_TX_HIGIG_PKT;
|
|
}
|
|
pkh->pkt_sig = dev->rcpu_ctrl.pkt_sig;
|
|
}
|
|
|
|
/* Packet header done here */
|
|
pkh->attrs |= PDMA_TX_HDR_COOKED;
|
|
|
|
data = skb->data + PKT_HDR_SIZE + pkh->meta_len;
|
|
tpid = data[12] << 8 | data[13];
|
|
tag_len = (tpid == ETH_P_8021Q || tpid == ETH_P_8021AD) ? VLAN_HLEN : 0;
|
|
|
|
/* Need to add VLAN tag if packet is untagged */
|
|
if (tag_len == 0 && (priv->netif.vlan & 0xfff) != 0 &&
|
|
(!(pkh->attrs & PDMA_TX_HIGIG_PKT) ||
|
|
priv->netif.flags & NGKNET_NETIF_F_ADD_TAG)) {
|
|
copy_len = PKT_HDR_SIZE + pkh->meta_len + 2 * ETH_ALEN;
|
|
if (skb_header_cloned(skb) || skb_headroom(skb) < VLAN_HLEN) {
|
|
nskb = skb_copy_expand(skb, VLAN_HLEN, 0, GFP_ATOMIC);
|
|
if (!nskb) {
|
|
return SHR_E_MEMORY;
|
|
}
|
|
skb_shinfo(nskb)->tx_flags = skb_shinfo(skb)->tx_flags;
|
|
skb = nskb;
|
|
}
|
|
skb_push(skb, VLAN_HLEN);
|
|
memmove(skb->data, skb->data + VLAN_HLEN, copy_len);
|
|
pkh = (struct pkt_hdr *)skb->data;
|
|
data = skb->data + PKT_HDR_SIZE + pkh->meta_len;
|
|
data[12] = 0x81;
|
|
data[13] = 0x00;
|
|
data[14] = priv->netif.vlan >> 8 & 0xf;
|
|
data[15] = priv->netif.vlan & 0xff;
|
|
pkh->data_len += VLAN_HLEN;
|
|
tag_len = VLAN_HLEN;
|
|
}
|
|
|
|
/* Optional callback handle */
|
|
if (dev->cbc->tx_cb) {
|
|
struct ngknet_callback_desc *cbd = NGKNET_SKB_CB(skb);
|
|
cbd->dinfo = &dev->dev_info;
|
|
cbd->netif = &priv->netif;
|
|
cbd->pmd = skb->data + PKT_HDR_SIZE;
|
|
cbd->pmd_len = pkh->meta_len;
|
|
cbd->pkt_len = skb->len - PKT_HDR_SIZE - pkh->meta_len;
|
|
skb = dev->cbc->tx_cb(skb);
|
|
if (!skb) {
|
|
if (!nskb) {
|
|
*oskb = NULL;
|
|
}
|
|
return SHR_E_UNAVAIL;
|
|
}
|
|
pkh = (struct pkt_hdr *)skb->data;
|
|
pkh->data_len = skb->len - PKT_HDR_SIZE - pkh->meta_len + ETH_FCS_LEN;
|
|
}
|
|
|
|
/* Pad packet if needed */
|
|
pad_len = ETH_ZLEN + ETH_FCS_LEN + tag_len;
|
|
if (pkh->data_len < pad_len && !(pkh->attrs & PDMA_TX_NO_PAD)) {
|
|
pkh->data_len = pad_len;
|
|
if (skb_padto(skb,
|
|
PKT_HDR_SIZE + pkh->meta_len + pkh->data_len - ETH_FCS_LEN)) {
|
|
if (!nskb) {
|
|
*oskb = NULL;
|
|
}
|
|
return SHR_E_MEMORY;
|
|
}
|
|
}
|
|
|
|
/* Update SKB pointer */
|
|
*oskb = skb;
|
|
|
|
return SHR_E_NONE;
|
|
}
|
|
|
|
/*!
|
|
* Network device detach callback
|
|
*/
|
|
static void
|
|
ngknet_ndev_detach(struct pdma_dev *pdev)
|
|
{
|
|
struct ngknet_dev *dev = (struct ngknet_dev *)pdev->priv;
|
|
int vdi;
|
|
|
|
netif_tx_lock(dev->net_dev);
|
|
netif_device_detach(dev->net_dev);
|
|
netif_tx_unlock(dev->net_dev);
|
|
|
|
for (vdi = 1; vdi <= NUM_VDEV_MAX; vdi++) {
|
|
if (!dev->vdev[vdi]) {
|
|
continue;
|
|
}
|
|
netif_tx_lock(dev->vdev[vdi]);
|
|
netif_device_detach(dev->vdev[vdi]);
|
|
netif_tx_unlock(dev->vdev[vdi]);
|
|
}
|
|
}
|
|
|
|
/*!
|
|
* Network device attach callback
|
|
*/
|
|
static void
|
|
ngknet_ndev_attach(struct pdma_dev *pdev)
|
|
{
|
|
struct ngknet_dev *dev = (struct ngknet_dev *)pdev->priv;
|
|
int vdi;
|
|
|
|
netif_tx_lock(dev->net_dev);
|
|
netif_device_attach(dev->net_dev);
|
|
netif_tx_unlock(dev->net_dev);
|
|
|
|
for (vdi = 1; vdi <= NUM_VDEV_MAX; vdi++) {
|
|
if (!dev->vdev[vdi]) {
|
|
continue;
|
|
}
|
|
netif_tx_lock(dev->vdev[vdi]);
|
|
netif_device_attach(dev->vdev[vdi]);
|
|
netif_tx_unlock(dev->vdev[vdi]);
|
|
}
|
|
}
|
|
|
|
/*!
|
|
* Suspend Tx queue callback
|
|
*/
|
|
static void
|
|
ngknet_tx_suspend(struct pdma_dev *pdev, int queue)
|
|
{
|
|
struct ngknet_dev *dev = (struct ngknet_dev *)pdev->priv;
|
|
unsigned long flags;
|
|
int vdi;
|
|
|
|
netif_stop_subqueue(dev->net_dev, queue);
|
|
|
|
spin_lock_irqsave(&dev->lock, flags);
|
|
for (vdi = 1; vdi <= NUM_VDEV_MAX; vdi++) {
|
|
if (!dev->vdev[vdi]) {
|
|
continue;
|
|
}
|
|
netif_stop_subqueue(dev->vdev[vdi], queue);
|
|
}
|
|
spin_unlock_irqrestore(&dev->lock, flags);
|
|
}
|
|
|
|
/*!
|
|
* Resume Tx queue callback
|
|
*/
|
|
static void
|
|
ngknet_tx_resume(struct pdma_dev *pdev, int queue)
|
|
{
|
|
struct ngknet_dev *dev = (struct ngknet_dev *)pdev->priv;
|
|
unsigned long flags;
|
|
int vdi;
|
|
|
|
if (__netif_subqueue_stopped(dev->net_dev, queue)) {
|
|
netif_wake_subqueue(dev->net_dev, queue);
|
|
}
|
|
|
|
spin_lock_irqsave(&dev->lock, flags);
|
|
for (vdi = 1; vdi <= NUM_VDEV_MAX; vdi++) {
|
|
if (!dev->vdev[vdi]) {
|
|
continue;
|
|
}
|
|
if (__netif_subqueue_stopped(dev->vdev[vdi], queue)) {
|
|
netif_wake_subqueue(dev->vdev[vdi], queue);
|
|
}
|
|
}
|
|
spin_unlock_irqrestore(&dev->lock, flags);
|
|
|
|
if (pdev->mode == DEV_MODE_HNET) {
|
|
atomic_set(&dev->hnet_active, 1);
|
|
wake_up_interruptible(&dev->hnet_wq);
|
|
}
|
|
}
|
|
|
|
/*!
|
|
* Enable interrupt callback
|
|
*/
|
|
static void
|
|
ngknet_intr_enable(struct pdma_dev *pdev, int cmc, int chan,
|
|
uint32_t reg, uint32_t val)
|
|
{
|
|
if (val) {
|
|
ngbde_kapi_iio_write32(pdev->unit, reg, val);
|
|
} else {
|
|
ngbde_kapi_intr_mask_write(pdev->unit, 0, reg, pdev->ctrl.grp[cmc].irq_mask);
|
|
}
|
|
}
|
|
|
|
/*!
|
|
* Disable interrupt callback
|
|
*/
|
|
static void
|
|
ngknet_intr_disable(struct pdma_dev *pdev, int cmc, int chan,
|
|
uint32_t reg, uint32_t val)
|
|
{
|
|
if (val) {
|
|
ngbde_kapi_iio_write32(pdev->unit, reg, val);
|
|
} else {
|
|
ngbde_kapi_intr_mask_write(pdev->unit, 0, reg, pdev->ctrl.grp[cmc].irq_mask);
|
|
}
|
|
}
|
|
|
|
/*!
|
|
* NAPI polling function
|
|
*/
|
|
static int
|
|
ngknet_poll(struct napi_struct *napi, int budget)
|
|
{
|
|
struct ngknet_intr_handle *kih = (struct ngknet_intr_handle *)napi;
|
|
struct intr_handle *hdl = kih->hdl;
|
|
struct pdma_dev *pdev = (struct pdma_dev *)hdl->dev;
|
|
struct ngknet_dev *dev = (struct ngknet_dev *)pdev->priv;
|
|
unsigned long flags;
|
|
int work_done;
|
|
|
|
DBG_NAPI(("Scheduled NAPI on queue %d.\n", hdl->queue));
|
|
|
|
kih->napi_resched = 0;
|
|
kih->napi_pending = 0;
|
|
|
|
if (pdev->flags & PDMA_GROUP_INTR) {
|
|
work_done = bcmcnet_group_poll(pdev, hdl->group, budget);
|
|
} else {
|
|
work_done = bcmcnet_queue_poll(pdev, hdl, budget);
|
|
}
|
|
|
|
if (work_done < budget) {
|
|
napi_complete(napi);
|
|
if (kih->napi_pending && napi_schedule_prep(napi)) {
|
|
__napi_schedule(napi);
|
|
return work_done;
|
|
}
|
|
spin_lock_irqsave(&dev->lock, flags);
|
|
if (!kih->napi_resched) {
|
|
if (pdev->flags & PDMA_GROUP_INTR) {
|
|
bcmcnet_group_intr_enable(pdev, hdl->group);
|
|
} else {
|
|
bcmcnet_queue_intr_enable(pdev, hdl);
|
|
}
|
|
}
|
|
spin_unlock_irqrestore(&dev->lock, flags);
|
|
}
|
|
|
|
return work_done;
|
|
}
|
|
|
|
/*!
|
|
* NGKNET ISR
|
|
*/
|
|
static int
|
|
ngknet_isr(void *isr_data)
|
|
{
|
|
struct ngknet_dev *dev = isr_data;
|
|
struct pdma_dev *pdev = &dev->pdma_dev;
|
|
struct intr_handle *hdl = NULL;
|
|
struct napi_struct *napi = NULL;
|
|
unsigned long flags;
|
|
int gi, qi;
|
|
int iv = 0;
|
|
|
|
for (gi = 0; gi < pdev->num_groups; gi++) {
|
|
if (!pdev->ctrl.grp[gi].attached) {
|
|
continue;
|
|
}
|
|
for (qi = 0; qi < pdev->grp_queues; qi++) {
|
|
hdl = &pdev->ctrl.grp[gi].intr_hdl[qi];
|
|
if (pdev->flags & PDMA_GROUP_INTR) {
|
|
if (!bcmcnet_group_intr_check(pdev, gi)) {
|
|
break;
|
|
}
|
|
} else {
|
|
if (!bcmcnet_queue_intr_check(pdev, hdl)) {
|
|
continue;
|
|
}
|
|
}
|
|
spin_lock_irqsave(&dev->lock, flags);
|
|
if (pdev->flags & PDMA_GROUP_INTR) {
|
|
bcmcnet_group_intr_disable(pdev, gi);
|
|
} else {
|
|
bcmcnet_queue_intr_disable(pdev, hdl);
|
|
}
|
|
spin_unlock_irqrestore(&dev->lock, flags);
|
|
napi = (struct napi_struct *)hdl->priv;
|
|
if (likely(napi_schedule_prep(napi))) {
|
|
__napi_schedule(napi);
|
|
}
|
|
iv++;
|
|
if (pdev->flags & PDMA_GROUP_INTR) {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (iv) {
|
|
DBG_IRQ(("Got interrupt on device %d.\n", dev->dev_info.dev_no));
|
|
pdev->stats.intrs++;
|
|
return IRQ_HANDLED;
|
|
} else {
|
|
return IRQ_NONE;
|
|
}
|
|
}
|
|
|
|
/*!
|
|
* Hypervisor network work handler
|
|
*/
|
|
static void
|
|
ngknet_dev_hnet_work(struct pdma_dev *pdev)
|
|
{
|
|
struct ngknet_dev *dev = (struct ngknet_dev *)pdev->priv;
|
|
struct intr_handle *hdl = NULL;
|
|
struct napi_struct *napi = NULL;
|
|
struct ngknet_intr_handle *kih = NULL;
|
|
unsigned long flags;
|
|
int gi, qi;
|
|
|
|
for (gi = 0; gi < pdev->num_groups; gi++) {
|
|
if (!pdev->ctrl.grp[gi].attached) {
|
|
continue;
|
|
}
|
|
for (qi = 0; qi < pdev->grp_queues; qi++) {
|
|
hdl = &pdev->ctrl.grp[gi].intr_hdl[qi];
|
|
napi = (struct napi_struct *)hdl->priv;
|
|
kih = (struct ngknet_intr_handle *)napi;
|
|
kih->napi_pending = 1;
|
|
if (napi_schedule_prep(napi)) {
|
|
spin_lock_irqsave(&dev->lock, flags);
|
|
kih->napi_resched = 1;
|
|
spin_unlock_irqrestore(&dev->lock, flags);
|
|
local_bh_disable();
|
|
__napi_schedule(napi);
|
|
local_bh_enable();
|
|
}
|
|
if (pdev->flags & PDMA_GROUP_INTR) {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/*!
|
|
* Hypervisor network wait handler
|
|
*/
|
|
static int
|
|
ngknet_dev_hnet_wait(struct pdma_dev *pdev)
|
|
{
|
|
struct ngknet_dev *dev = (struct ngknet_dev *)pdev->priv;
|
|
uint32_t bmp;
|
|
int budget, qi;
|
|
|
|
while (!kthread_should_stop()) {
|
|
wait_event_interruptible(dev->hnet_wq,
|
|
atomic_read(&dev->hnet_active) != 0);
|
|
if (!(dev->flags & NGKNET_DEV_ACTIVE)) {
|
|
schedule_timeout(HZ);
|
|
continue;
|
|
}
|
|
atomic_set(&dev->hnet_active, 0);
|
|
|
|
schedule_work(&dev->hnet_work);
|
|
|
|
do {
|
|
bmp = 0x0;
|
|
for (qi = 0; qi < pdev->ctrl.nb_txq; qi++) {
|
|
bmp |= 1 << qi;
|
|
budget = pdev->ctrl.budget;
|
|
while (budget--) {
|
|
if (SHR_FAILURE(pdev->pkt_xmit(pdev, qi, 0))) {
|
|
bmp &= ~(1 << qi);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
} while (bmp);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*!
|
|
* Hypervisor network wake handler
|
|
*/
|
|
static int
|
|
ngknet_dev_vnet_wake(struct pdma_dev *pdev)
|
|
{
|
|
struct ngknet_dev *dev = (struct ngknet_dev *)pdev->priv;
|
|
|
|
atomic_set(&dev->vnet_active, 1);
|
|
wake_up_interruptible(&dev->vnet_wq);
|
|
|
|
return SHR_E_NONE;
|
|
}
|
|
|
|
/*!
|
|
* Hypervisor network process
|
|
*/
|
|
static int
|
|
ngknet_dev_hnet_process(void *data)
|
|
{
|
|
return ngknet_dev_hnet_wait((struct pdma_dev *)data);
|
|
}
|
|
|
|
/*!
|
|
* Hypervisor network schedule
|
|
*/
|
|
static void
|
|
ngknet_dev_hnet_schedule(struct work_struct *work)
|
|
{
|
|
struct ngknet_dev *dev = container_of(work, struct ngknet_dev, hnet_work);
|
|
|
|
ngknet_dev_hnet_work(&dev->pdma_dev);
|
|
}
|
|
|
|
/*!
|
|
* Convert physical address to virtual address
|
|
*/
|
|
static void *
|
|
ngknet_sys_p2v(struct pdma_dev *pdev, uint64_t paddr)
|
|
{
|
|
return ngbde_kapi_dma_bus_to_virt(pdev->unit, (dma_addr_t)paddr);
|
|
}
|
|
|
|
/*!
|
|
* Convert virtual address to physical address
|
|
*/
|
|
static uint64_t
|
|
ngknet_sys_v2p(struct pdma_dev *pdev, void *vaddr)
|
|
{
|
|
return (uint64_t)ngbde_kapi_dma_virt_to_bus(pdev->unit, vaddr);
|
|
}
|
|
|
|
/*!
|
|
* Open network device
|
|
*/
|
|
static int
|
|
ngknet_enet_open(struct net_device *ndev)
|
|
{
|
|
struct ngknet_private *priv = netdev_priv(ndev);
|
|
struct ngknet_dev *dev = priv->bkn_dev;
|
|
struct pdma_dev *pdev = &dev->pdma_dev;
|
|
struct napi_struct *napi = NULL;
|
|
unsigned long bm_queue;
|
|
int gi, qi;
|
|
int rv;
|
|
|
|
if (!pdev->ctrl.bm_rxq || !pdev->ctrl.bm_txq) {
|
|
printk("Not config Rx or Tx queue yet!\n");
|
|
return -EPERM;
|
|
}
|
|
|
|
if (priv->netif.id <= 0) {
|
|
/* Register interrupt handler */
|
|
ngbde_kapi_intr_connect(dev->dev_info.dev_no, 0, ngknet_isr, dev);
|
|
|
|
/* Start PDMA device */
|
|
rv = bcmcnet_pdma_dev_start(pdev);
|
|
if (SHR_FAILURE(rv)) {
|
|
ngbde_kapi_intr_disconnect(dev->dev_info.dev_no, 0);
|
|
return -EPERM;
|
|
}
|
|
|
|
/* Start rate limit */
|
|
if (rx_rate_limit >= 0) {
|
|
ngknet_rx_rate_limit_start(dev);
|
|
}
|
|
|
|
/* Notify the stack of the actual queue counts. */
|
|
rv = netif_set_real_num_rx_queues(dev->net_dev, pdev->ctrl.nb_rxq);
|
|
if (rv < 0) {
|
|
ngbde_kapi_intr_disconnect(dev->dev_info.dev_no, 0);
|
|
return rv;
|
|
}
|
|
rv = netif_set_real_num_tx_queues(dev->net_dev, pdev->ctrl.nb_txq);
|
|
if (rv < 0) {
|
|
ngbde_kapi_intr_disconnect(dev->dev_info.dev_no, 0);
|
|
return rv;
|
|
}
|
|
|
|
for (gi = 0; gi < pdev->num_groups; gi++) {
|
|
if (!pdev->ctrl.grp[gi].attached) {
|
|
continue;
|
|
}
|
|
bm_queue = pdev->ctrl.grp[gi].bm_rxq | pdev->ctrl.grp[gi].bm_txq;
|
|
for (qi = 0; qi < pdev->grp_queues; qi++) {
|
|
napi = (struct napi_struct *)pdev->ctrl.grp[gi].intr_hdl[qi].priv;
|
|
if (pdev->flags & PDMA_GROUP_INTR) {
|
|
napi_enable(napi);
|
|
break;
|
|
}
|
|
if (1 << qi & bm_queue) {
|
|
napi_enable(napi);
|
|
}
|
|
}
|
|
}
|
|
} else {
|
|
/* Notify the stack of the actual queue counts. */
|
|
rv = netif_set_real_num_rx_queues(ndev, pdev->ctrl.nb_rxq);
|
|
if (rv < 0) {
|
|
return rv;
|
|
}
|
|
rv = netif_set_real_num_tx_queues(ndev, pdev->ctrl.nb_txq);
|
|
if (rv < 0) {
|
|
return rv;
|
|
}
|
|
}
|
|
|
|
/* Prevent tx timeout */
|
|
kal_netif_trans_update(ndev);
|
|
|
|
netif_tx_wake_all_queues(ndev);
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*!
|
|
* Stop network device
|
|
*/
|
|
static int
|
|
ngknet_enet_stop(struct net_device *ndev)
|
|
{
|
|
struct ngknet_private *priv = netdev_priv(ndev);
|
|
struct ngknet_dev *dev = priv->bkn_dev;
|
|
struct pdma_dev *pdev = &dev->pdma_dev;
|
|
struct napi_struct *napi = NULL;
|
|
unsigned long bm_queue;
|
|
int gi, qi;
|
|
|
|
netif_tx_stop_all_queues(ndev);
|
|
|
|
if (priv->netif.id <= 0) {
|
|
/* Stop rate limit */
|
|
if (rx_rate_limit >= 0) {
|
|
ngknet_rx_rate_limit_stop(dev);
|
|
}
|
|
|
|
for (gi = 0; gi < pdev->num_groups; gi++) {
|
|
if (!pdev->ctrl.grp[gi].attached) {
|
|
continue;
|
|
}
|
|
bm_queue = pdev->ctrl.grp[gi].bm_rxq | pdev->ctrl.grp[gi].bm_txq;
|
|
for (qi = 0; qi < pdev->grp_queues; qi++) {
|
|
napi = (struct napi_struct *)pdev->ctrl.grp[gi].intr_hdl[qi].priv;
|
|
if (pdev->flags & PDMA_GROUP_INTR) {
|
|
napi_disable(napi);
|
|
break;
|
|
}
|
|
if (1 << qi & bm_queue) {
|
|
napi_disable(napi);
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Stop PDMA device */
|
|
bcmcnet_pdma_dev_stop(pdev);
|
|
|
|
/* Unregister interrupt handler */
|
|
ngbde_kapi_intr_disconnect(dev->dev_info.dev_no, 0);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*!
|
|
* Start transmission
|
|
*/
|
|
static int
|
|
ngknet_start_xmit(struct sk_buff *skb, struct net_device *ndev)
|
|
{
|
|
struct ngknet_private *priv = netdev_priv(ndev);
|
|
struct ngknet_dev *dev = priv->bkn_dev;
|
|
struct pdma_dev *pdev = &dev->pdma_dev;
|
|
struct sk_buff *bskb = skb;
|
|
uint32_t len = skb->len;
|
|
int queue;
|
|
int rv;
|
|
|
|
DBG_VERB(("Tx packet from ndev%d (%d bytes).\n", priv->netif.id, skb->len));
|
|
if (debug & DBG_LVL_PDMP) {
|
|
ngknet_pkt_dump(skb->data, skb->len);
|
|
}
|
|
|
|
/* Do not transmit on base device */
|
|
if (priv->netif.id <= 0) {
|
|
priv->stats.tx_dropped++;
|
|
dev_kfree_skb_any(skb);
|
|
return NETDEV_TX_OK;
|
|
}
|
|
|
|
/* Measure speed */
|
|
if (debug & DBG_LVL_RATE) {
|
|
ngknet_pkt_stats(pdev, PDMA_Q_TX);
|
|
}
|
|
|
|
queue = skb->queue_mapping;
|
|
|
|
/* Handle one outgoing packet */
|
|
rv = ngknet_tx_frame_process(ndev, &skb);
|
|
if (SHR_FAILURE(rv)) {
|
|
priv->stats.tx_dropped++;
|
|
if (skb) {
|
|
dev_kfree_skb_any(skb);
|
|
}
|
|
return NETDEV_TX_OK;
|
|
}
|
|
|
|
/* Schedule Tx queue */
|
|
ngknet_tx_queue_schedule(dev, skb, &queue);
|
|
skb->queue_mapping = queue;
|
|
|
|
DBG_VERB(("Tx packet (%d bytes).\n", skb->len));
|
|
if (debug & DBG_LVL_PDMP) {
|
|
ngknet_pkt_dump(skb->data, skb->len);
|
|
}
|
|
|
|
/* Do Tx timestamping */
|
|
if (skb_shinfo(skb)->tx_flags & SKBTX_HW_TSTAMP) {
|
|
ngknet_ptp_tx_config(ndev, skb);
|
|
}
|
|
|
|
skb_tx_timestamp(skb);
|
|
|
|
rv = pdev->pkt_xmit(pdev, queue, skb);
|
|
|
|
if (rv == SHR_E_BUSY) {
|
|
DBG_WARN(("Tx suspend: DMA device is busy and temporarily "
|
|
"unavailable.\n"));
|
|
priv->stats.tx_fifo_errors++;
|
|
if (skb != bskb) {
|
|
dev_kfree_skb_any(skb);
|
|
}
|
|
return NETDEV_TX_BUSY;
|
|
} else if (rv != SHR_E_NONE) {
|
|
DBG_WARN(("Tx drop: DMA device not ready or not supported.\n"));
|
|
priv->stats.tx_dropped++;
|
|
if (skb != bskb) {
|
|
dev_kfree_skb_any(skb);
|
|
}
|
|
dev_kfree_skb_any(bskb);
|
|
return NETDEV_TX_OK;
|
|
} else {
|
|
if (skb != bskb) {
|
|
dev_kfree_skb_any(bskb);
|
|
}
|
|
}
|
|
|
|
/* Update accounting */
|
|
priv->stats.tx_packets++;
|
|
priv->stats.tx_bytes += len;
|
|
|
|
return NETDEV_TX_OK;
|
|
}
|
|
|
|
/*!
|
|
* Get network device stats
|
|
*/
|
|
static struct net_device_stats *
|
|
ngknet_get_stats(struct net_device *ndev)
|
|
{
|
|
struct ngknet_private *priv = netdev_priv(ndev);
|
|
|
|
return &priv->stats;
|
|
}
|
|
|
|
/*!
|
|
* Set network device MC list
|
|
*/
|
|
static void
|
|
ngknet_set_multicast_list(struct net_device *ndev)
|
|
{
|
|
return;
|
|
}
|
|
|
|
/*!
|
|
* Set network device MAC address
|
|
*/
|
|
static int
|
|
ngknet_set_mac_address(struct net_device *ndev, void *addr)
|
|
{
|
|
if (!is_valid_ether_addr(((struct sockaddr *)addr)->sa_data)) {
|
|
return -EINVAL;
|
|
}
|
|
|
|
netdev_info(ndev, "Setting new MAC address\n");
|
|
memcpy(ndev->dev_addr, ((struct sockaddr *)addr)->sa_data, ndev->addr_len);
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*!
|
|
* Change network device MTU
|
|
*/
|
|
static int
|
|
ngknet_change_mtu(struct net_device *ndev, int new_mtu)
|
|
{
|
|
int frame_size = new_mtu + ETH_HLEN + VLAN_HLEN + ETH_FCS_LEN;
|
|
|
|
if (frame_size < (ETH_ZLEN + ETH_FCS_LEN) || frame_size > rx_buffer_size) {
|
|
return -EINVAL;
|
|
}
|
|
|
|
netdev_info(ndev, "Changing MTU from %d to %d\n", ndev->mtu, new_mtu);
|
|
ndev->mtu = new_mtu;
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*!
|
|
* Do I/O control
|
|
*/
|
|
static int
|
|
ngknet_do_ioctl(struct net_device *ndev, struct ifreq *ifr, int cmd)
|
|
{
|
|
struct ngknet_private *priv = netdev_priv(ndev);
|
|
struct hwtstamp_config config;
|
|
int rv;
|
|
|
|
if (cmd == SIOCSHWTSTAMP) {
|
|
if (copy_from_user(&config, ifr->ifr_data, sizeof(config))) {
|
|
return -EFAULT;
|
|
}
|
|
|
|
if (priv->netif.type != NGKNET_NETIF_T_PORT) {
|
|
return -ENOSYS;
|
|
}
|
|
|
|
switch (config.tx_type) {
|
|
case HWTSTAMP_TX_OFF:
|
|
priv->hwts_tx_type = HWTSTAMP_TX_OFF;
|
|
rv = ngknet_ptp_tx_config_set(ndev, priv->hwts_tx_type);
|
|
if (SHR_FAILURE(rv)) {
|
|
return -ENOSYS;
|
|
}
|
|
break;
|
|
case HWTSTAMP_TX_ON:
|
|
priv->hwts_tx_type = HWTSTAMP_TX_ON;
|
|
rv = ngknet_ptp_tx_config_set(ndev, priv->hwts_tx_type);
|
|
if (SHR_FAILURE(rv)) {
|
|
return -ENOSYS;
|
|
}
|
|
break;
|
|
case HWTSTAMP_TX_ONESTEP_SYNC:
|
|
priv->hwts_tx_type = HWTSTAMP_TX_ONESTEP_SYNC;
|
|
rv = ngknet_ptp_tx_config_set(ndev, priv->hwts_tx_type);
|
|
if (SHR_FAILURE(rv)) {
|
|
return -ENOSYS;
|
|
}
|
|
break;
|
|
default:
|
|
return -ERANGE;
|
|
}
|
|
|
|
switch (config.rx_filter) {
|
|
case HWTSTAMP_FILTER_NONE:
|
|
rv = ngknet_ptp_rx_config_set(ndev, &config.rx_filter);
|
|
if (SHR_FAILURE(rv)) {
|
|
return -ENOSYS;
|
|
}
|
|
priv->hwts_rx_filter = HWTSTAMP_FILTER_NONE;
|
|
break;
|
|
default:
|
|
rv = ngknet_ptp_rx_config_set(ndev, &config.rx_filter);
|
|
if (SHR_FAILURE(rv)) {
|
|
return -ENOSYS;
|
|
}
|
|
priv->hwts_rx_filter = config.rx_filter;
|
|
break;
|
|
}
|
|
|
|
return copy_to_user(ifr->ifr_data, &config, sizeof(config)) ? -EFAULT : 0;
|
|
}
|
|
|
|
#if (LINUX_VERSION_CODE >= KERNEL_VERSION(3,14,0))
|
|
if (cmd == SIOCGHWTSTAMP) {
|
|
config.flags = 0;
|
|
config.tx_type = priv->hwts_tx_type;
|
|
config.rx_filter = priv->hwts_rx_filter;
|
|
|
|
return copy_to_user(ifr->ifr_data, &config, sizeof(config)) ? -EFAULT : 0;
|
|
}
|
|
#endif
|
|
|
|
return -EINVAL;
|
|
}
|
|
|
|
#ifdef CONFIG_NET_POLL_CONTROLLER
|
|
/*!
|
|
* Poll network device
|
|
*/
|
|
static void
|
|
ngknet_poll_controller(struct net_device *ndev)
|
|
{
|
|
struct ngknet_private *priv = netdev_priv(ndev);
|
|
|
|
disable_irq(ndev->irq);
|
|
ngknet_isr(priv->bkn_dev);
|
|
enable_irq(ndev->irq);
|
|
}
|
|
#endif
|
|
|
|
static const struct net_device_ops ngknet_netdev_ops = {
|
|
.ndo_open = ngknet_enet_open,
|
|
.ndo_stop = ngknet_enet_stop,
|
|
.ndo_start_xmit = ngknet_start_xmit,
|
|
.ndo_get_stats = ngknet_get_stats,
|
|
.ndo_validate_addr = eth_validate_addr,
|
|
.ndo_set_rx_mode = ngknet_set_multicast_list,
|
|
.ndo_set_mac_address = ngknet_set_mac_address,
|
|
.ndo_change_mtu = ngknet_change_mtu,
|
|
.ndo_set_features = NULL,
|
|
.ndo_do_ioctl = ngknet_do_ioctl,
|
|
.ndo_tx_timeout = NULL,
|
|
#ifdef CONFIG_NET_POLL_CONTROLLER
|
|
.ndo_poll_controller = ngknet_poll_controller,
|
|
#endif
|
|
};
|
|
|
|
static void
|
|
ngknet_get_drvinfo(struct net_device *ndev, struct ethtool_drvinfo *drvinfo)
|
|
{
|
|
strlcpy(drvinfo->driver, "linux_ngknet", sizeof(drvinfo->driver));
|
|
snprintf(drvinfo->version, sizeof(drvinfo->version), "%d", NGKNET_IOC_VERSION);
|
|
strlcpy(drvinfo->fw_version, "N/A", sizeof(drvinfo->fw_version));
|
|
strlcpy(drvinfo->bus_info, "N/A", sizeof(drvinfo->bus_info));
|
|
}
|
|
|
|
#if (LINUX_VERSION_CODE >= KERNEL_VERSION(3,5,0))
|
|
static int
|
|
ngknet_get_ts_info(struct net_device *ndev, struct ethtool_ts_info *info)
|
|
{
|
|
int rv;
|
|
|
|
info->so_timestamping = SOF_TIMESTAMPING_TX_HARDWARE |
|
|
SOF_TIMESTAMPING_TX_SOFTWARE |
|
|
SOF_TIMESTAMPING_RX_HARDWARE |
|
|
SOF_TIMESTAMPING_RX_SOFTWARE |
|
|
SOF_TIMESTAMPING_SOFTWARE |
|
|
SOF_TIMESTAMPING_RAW_HARDWARE;
|
|
info->tx_types = 1 << HWTSTAMP_TX_OFF | 1 << HWTSTAMP_TX_ON | 1 << HWTSTAMP_TX_ONESTEP_SYNC;
|
|
info->rx_filters = 1 << HWTSTAMP_FILTER_NONE | 1 << HWTSTAMP_FILTER_ALL;
|
|
rv = ngknet_ptp_phc_index_get(ndev, &info->phc_index);
|
|
if (SHR_FAILURE(rv)) {
|
|
info->phc_index = -1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
#endif
|
|
|
|
#if NGKNET_ETHTOOL_LINK_SETTINGS
|
|
static int
|
|
ngknet_get_link_ksettings(struct net_device *ndev,
|
|
struct ethtool_link_ksettings *cmd)
|
|
{
|
|
struct ngknet_private *priv = netdev_priv(ndev);
|
|
|
|
cmd->base.speed = priv->link_settings.speed;
|
|
cmd->base.duplex = priv->link_settings.duplex;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int
|
|
ngknet_set_link_ksettings(struct net_device *ndev,
|
|
const struct ethtool_link_ksettings *cmd)
|
|
{
|
|
struct ngknet_private *priv = netdev_priv(ndev);
|
|
|
|
priv->link_settings.speed = cmd->base.speed;
|
|
priv->link_settings.duplex = cmd->base.speed ? DUPLEX_FULL : 0;
|
|
|
|
return 0;
|
|
}
|
|
#endif
|
|
|
|
static const struct ethtool_ops ngknet_ethtool_ops = {
|
|
.get_drvinfo = ngknet_get_drvinfo,
|
|
#if (LINUX_VERSION_CODE >= KERNEL_VERSION(3,5,0))
|
|
.get_ts_info = ngknet_get_ts_info,
|
|
#endif
|
|
#if NGKNET_ETHTOOL_LINK_SETTINGS
|
|
.get_link_ksettings = ngknet_get_link_ksettings,
|
|
.set_link_ksettings = ngknet_set_link_ksettings,
|
|
#endif
|
|
};
|
|
|
|
/*!
|
|
* \brief Initialize network device.
|
|
*
|
|
* \param [in] name Network device name.
|
|
* \param [in] mac Network device MAC address.
|
|
* \param [out] nd New registered network device.
|
|
*
|
|
* \retval SHR_E_NONE No errors.
|
|
* \retval SHR_E_XXXX Operation failed.
|
|
*/
|
|
static int
|
|
ngknet_ndev_init(ngknet_netif_t *netif, struct net_device **nd)
|
|
{
|
|
struct net_device *ndev = NULL;
|
|
uint8_t *ma;
|
|
int rv;
|
|
|
|
if (!netif) {
|
|
DBG_WARN(("Network interface is NULL.\n"));
|
|
return SHR_E_PARAM;
|
|
}
|
|
if (!nd) {
|
|
DBG_WARN(("Network device is NULL.\n"));
|
|
return SHR_E_PARAM;
|
|
}
|
|
|
|
ndev = alloc_etherdev_mq(sizeof(struct ngknet_private), NUM_Q_MAX);
|
|
if (!ndev) {
|
|
DBG_WARN(("Error allocating network device.\n"));
|
|
return SHR_E_MEMORY;
|
|
}
|
|
if (!ndev->dev_addr) {
|
|
DBG_WARN(("ndev->dev_addr is NULL\n"));
|
|
free_netdev(ndev);
|
|
return SHR_E_INTERNAL;
|
|
}
|
|
|
|
/* Device information -- not available right now */
|
|
ndev->irq = 0;
|
|
ndev->base_addr = 0;
|
|
|
|
/* Fill in the dev structure */
|
|
ndev->watchdog_timeo = 5 * HZ;
|
|
|
|
/* Default MTU should not exceed MTU of switch front-panel ports */
|
|
ndev->mtu = netif->mtu;
|
|
if (!ndev->mtu) {
|
|
ndev->mtu = default_mtu ? default_mtu : rx_buffer_size;
|
|
}
|
|
|
|
/* MTU range: 32 - 9198 */
|
|
#if (LINUX_VERSION_CODE >= KERNEL_VERSION(4,10,0))
|
|
ndev->min_mtu = PKT_HDR_SIZE; /* Min 50-byte length of packet with RCPU-encap */
|
|
ndev->max_mtu = rx_buffer_size - (ETH_HLEN + ETH_FCS_LEN);
|
|
#endif
|
|
|
|
ndev->netdev_ops = &ngknet_netdev_ops;
|
|
ndev->ethtool_ops = &ngknet_ethtool_ops;
|
|
|
|
/* Network device name */
|
|
if (netif->name && *netif->name) {
|
|
strncpy(ndev->name, netif->name, IFNAMSIZ - 1);
|
|
}
|
|
|
|
/* Set the device MAC address */
|
|
ma = netif->macaddr;
|
|
if ((ma[0] | ma[1] | ma[2] | ma[3] | ma[4] | ma[5]) == 0) {
|
|
ngknet_dev_mac[5]++;
|
|
ma = ngknet_dev_mac;
|
|
}
|
|
memcpy(ndev->dev_addr, ma, ETH_ALEN);
|
|
|
|
/* Initialize the device features */
|
|
ndev->hw_features = NETIF_F_RXCSUM |
|
|
NETIF_F_HW_VLAN_CTAG_RX |
|
|
NETIF_F_HW_VLAN_CTAG_TX;
|
|
ndev->features = ndev->hw_features | NETIF_F_HIGHDMA;
|
|
|
|
/* Register the kernel network device */
|
|
rv = register_netdev(ndev);
|
|
if (rv < 0) {
|
|
DBG_WARN(("Error registering network device %s.\n", ndev->name));
|
|
free_netdev(ndev);
|
|
return SHR_E_FAIL;
|
|
}
|
|
|
|
*nd = ndev;
|
|
|
|
DBG_VERB(("Created network device %s.\n", ndev->name));
|
|
|
|
return SHR_E_NONE;
|
|
}
|
|
|
|
static int
|
|
ngknet_dev_remove(int dn);
|
|
|
|
static int
|
|
ngknet_bde_event_handler(int kdev, int event, void *data)
|
|
{
|
|
DBG_VERB(("%s: callback from BDE with kdev(%d) event(%d).\n",
|
|
__FUNCTION__, kdev, event));
|
|
|
|
if (event == NGBDE_EVENT_DEV_REMOVE) {
|
|
ngknet_dev_remove(kdev);
|
|
}
|
|
|
|
return SHR_E_NONE;
|
|
}
|
|
|
|
/*!
|
|
* \brief Initialize Packet DMA device.
|
|
*
|
|
* \param [in] dev NGKNET device structure point.
|
|
*
|
|
* \retval SHR_E_NONE No errors.
|
|
* \retval SHR_E_XXXX Operation failed.
|
|
*/
|
|
static int
|
|
ngknet_pdev_init(struct ngknet_dev *dev)
|
|
{
|
|
struct pdma_dev *pdev = &dev->pdma_dev;
|
|
int rv;
|
|
|
|
/* Initialize PDMA control structure */
|
|
pdev->unit = dev->dev_info.dev_no;
|
|
pdev->priv = dev;
|
|
pdev->ctrl.dev = pdev;
|
|
pdev->ctrl.hw_addr = dev->base_addr;
|
|
pdev->ctrl.rx_buf_size = rx_buffer_size;
|
|
|
|
/* Hook callbacks */
|
|
pdev->dev_read32 = ngknet_dev_read32;
|
|
pdev->dev_write32 = ngknet_dev_write32;
|
|
pdev->pkt_recv = ngknet_frame_recv;
|
|
pdev->ndev_detach = ngknet_ndev_detach;
|
|
pdev->ndev_attach = ngknet_ndev_attach;
|
|
pdev->tx_suspend = ngknet_tx_suspend;
|
|
pdev->tx_resume = ngknet_tx_resume;
|
|
pdev->intr_unmask = ngknet_intr_enable;
|
|
pdev->intr_mask = ngknet_intr_disable;
|
|
pdev->xnet_wait = ngknet_dev_hnet_wait;
|
|
pdev->xnet_wake = ngknet_dev_vnet_wake;
|
|
pdev->sys_p2v = ngknet_sys_p2v;
|
|
pdev->sys_v2p = ngknet_sys_v2p;
|
|
|
|
pdev->flags |= PDMA_GROUP_INTR;
|
|
if (tx_polling) {
|
|
pdev->flags |= PDMA_TX_POLLING;
|
|
}
|
|
if (rx_batching || pdev->mode == DEV_MODE_HNET) {
|
|
pdev->flags |= PDMA_RX_BATCHING;
|
|
}
|
|
|
|
/* Attach PDMA driver */
|
|
rv = drv_ops[pdev->dev_type]->drv_attach(pdev);
|
|
if (SHR_FAILURE(rv)) {
|
|
DBG_WARN(("Attach DMA driver failed.\n"));
|
|
return rv;
|
|
}
|
|
|
|
/* Initialize PDMA device */
|
|
rv = bcmcnet_pdma_dev_init(pdev);
|
|
if (SHR_FAILURE(rv)) {
|
|
DBG_WARN(("Init DMA device.failed.\n"));
|
|
return rv;
|
|
}
|
|
|
|
DBG_VERB(("Attached DMA device %s.\n", pdev->name));
|
|
|
|
return SHR_E_NONE;
|
|
}
|
|
|
|
/*!
|
|
* \brief Get device information from BDE.
|
|
*
|
|
* \param [in] dn Device number.
|
|
*
|
|
* \retval SHR_E_NONE No errors.
|
|
* \retval SHR_E_XXXX Operation failed.
|
|
*/
|
|
static int
|
|
ngknet_dev_info_get(int dn)
|
|
{
|
|
struct ngknet_dev *dev = &ngknet_devices[dn];
|
|
|
|
dev->base_addr = ngbde_kapi_pio_membase(dn);
|
|
dev->dev = ngbde_kapi_dma_dev_get(dn);
|
|
|
|
if (!dev->base_addr || !dev->dev) {
|
|
return SHR_E_ACCESS;
|
|
}
|
|
|
|
dev->dev_info.dev_no = dn;
|
|
strlcpy(dev->dev_info.type_str, drv_ops[dev->pdma_dev.dev_type]->drv_desc,
|
|
sizeof(dev->dev_info.type_str));
|
|
|
|
return SHR_E_NONE;
|
|
}
|
|
|
|
/*!
|
|
* \brief Probe device.
|
|
*
|
|
* Get the information from BDE, initialize Packet DMA device,
|
|
* initialize base network device and allocate other resources.
|
|
*
|
|
* \param [in] dn Device number.
|
|
*
|
|
* \retval SHR_E_NONE No errors.
|
|
* \retval SHR_E_XXXX Operation failed.
|
|
*/
|
|
static int
|
|
ngknet_dev_probe(int dn, ngknet_netif_t *netif)
|
|
{
|
|
struct ngknet_dev *dev = &ngknet_devices[dn];
|
|
struct pdma_dev *pdev = &dev->pdma_dev;
|
|
struct net_device *ndev = NULL;
|
|
struct ngknet_private *priv = NULL;
|
|
struct intr_handle *hdl = NULL;
|
|
struct cpumask mask;
|
|
int gi, qi;
|
|
int rv;
|
|
|
|
DBG_VERB(("%s: dev %d\n",__FUNCTION__, dn));
|
|
|
|
/* Get device information */
|
|
rv = ngknet_dev_info_get(dn);
|
|
if (SHR_FAILURE(rv)) {
|
|
return rv;
|
|
}
|
|
|
|
/* Initialize PDMA device */
|
|
rv = ngknet_pdev_init(dev);
|
|
if (SHR_FAILURE(rv)) {
|
|
return rv;
|
|
}
|
|
|
|
/* Get base network device name */
|
|
if (netif->name[0] == '\0') {
|
|
/* Reserve 6 vacancies for base&vitual device number, i.e. nameAB_XYZ */
|
|
if (strlen(base_dev_name) < IFNAMSIZ - 6) {
|
|
snprintf(netif->name, IFNAMSIZ, "%s%d", base_dev_name, dn);
|
|
} else {
|
|
DBG_WARN(("Too long network device name: %s.\n", base_dev_name));
|
|
return SHR_E_PARAM;
|
|
}
|
|
}
|
|
|
|
if (netif->chan >= NUM_Q_MAX) {
|
|
DBG_WARN(("Exceed max number of queues : %d.\n", netif->chan));
|
|
return SHR_E_PARAM;
|
|
}
|
|
|
|
rv = ngknet_ndev_init(netif, &ndev);
|
|
if (SHR_FAILURE(rv)) {
|
|
bcmcnet_pdma_dev_cleanup(pdev);
|
|
return rv;
|
|
}
|
|
dev->net_dev = ndev;
|
|
|
|
/* Initialize private information for base network device */
|
|
priv = netdev_priv(ndev);
|
|
priv->net_dev = ndev;
|
|
priv->bkn_dev = dev;
|
|
|
|
netif->id = 0;
|
|
memcpy(netif->macaddr, ndev->dev_addr, ETH_ALEN);
|
|
netif->mtu = ndev->mtu;
|
|
memcpy(netif->name, ndev->name, sizeof(netif->name) - 1);
|
|
memcpy(&priv->netif, netif, sizeof(priv->netif));
|
|
|
|
if (priv->netif.flags & NGKNET_NETIF_F_BIND_CHAN) {
|
|
dev->bdev[priv->netif.chan] = ndev;
|
|
}
|
|
|
|
/* Register for napi */
|
|
for (gi = 0; gi < pdev->num_groups; gi++) {
|
|
if (!pdev->ctrl.grp[gi].attached) {
|
|
continue;
|
|
}
|
|
for (qi = 0; qi < pdev->grp_queues; qi++) {
|
|
hdl = &pdev->ctrl.grp[gi].intr_hdl[qi];
|
|
priv_hdl[hdl->unit][hdl->chan].hdl = hdl;
|
|
hdl->priv = &priv_hdl[hdl->unit][hdl->chan];
|
|
netif_napi_add(ndev, (struct napi_struct *)hdl->priv,
|
|
ngknet_poll, pdev->ctrl.budget);
|
|
if (pdev->flags & PDMA_GROUP_INTR) {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Get callback control */
|
|
ngknet_callback_control_get(&dev->cbc);
|
|
|
|
INIT_LIST_HEAD(&dev->filt_list);
|
|
spin_lock_init(&dev->lock);
|
|
init_waitqueue_head(&dev->wq);
|
|
if (pdev->mode == DEV_MODE_HNET) {
|
|
init_waitqueue_head(&dev->vnet_wq);
|
|
atomic_set(&dev->vnet_active, 0);
|
|
init_waitqueue_head(&dev->hnet_wq);
|
|
atomic_set(&dev->hnet_active, 0);
|
|
dev->hnet_task = kthread_run(ngknet_dev_hnet_process, pdev, pdev->name);
|
|
if (IS_ERR(dev->hnet_task)) {
|
|
dev->hnet_task = NULL;
|
|
return SHR_E_INTERNAL;
|
|
}
|
|
cpumask_clear(&mask);
|
|
cpumask_set_cpu(num_online_cpus() / 2, &mask);
|
|
set_cpus_allowed_ptr(dev->hnet_task, &mask);
|
|
INIT_WORK(&dev->hnet_work, ngknet_dev_hnet_schedule);
|
|
}
|
|
|
|
skb_queue_head_init(&dev->ptp_tx_queue);
|
|
INIT_WORK(&dev->ptp_tx_work, ngknet_ptp_tx_work);
|
|
|
|
dev->flags |= NGKNET_DEV_ACTIVE;
|
|
|
|
DBG_NDEV(("Broadcom NGKNET Attached\n"));
|
|
DBG_NDEV(("MAC: %pM\n", ndev->dev_addr));
|
|
DBG_NDEV(("Running with NAPI enabled\n"));
|
|
|
|
/* Register handler for BDE events. */
|
|
ngbde_kapi_knet_connect(dn, ngknet_bde_event_handler, dev);
|
|
|
|
return SHR_E_NONE;
|
|
}
|
|
|
|
/*!
|
|
* \brief Remove device.
|
|
*
|
|
* Suspend device firstly, destroy all virtual network devices
|
|
* and filters, clean up Packet DMA device.
|
|
*
|
|
* \param [in] dn Device number.
|
|
*
|
|
* \retval SHR_E_NONE No errors.
|
|
* \retval SHR_E_XXXX Operation failed.
|
|
*/
|
|
static int
|
|
ngknet_dev_remove(int dn)
|
|
{
|
|
struct ngknet_dev *dev = &ngknet_devices[dn];
|
|
struct pdma_dev *pdev = &dev->pdma_dev;
|
|
struct net_device *ndev = NULL;
|
|
struct intr_handle *hdl = NULL;
|
|
int di, gi, qi;
|
|
int rv;
|
|
|
|
if (!(dev->flags & NGKNET_DEV_ACTIVE)) {
|
|
ngbde_kapi_knet_disconnect(dn);
|
|
return SHR_E_NONE;
|
|
}
|
|
|
|
DBG_VERB(("%s: dev %d\n",__FUNCTION__, dn));
|
|
|
|
dev->flags &= ~NGKNET_DEV_ACTIVE;
|
|
|
|
skb_queue_purge(&dev->ptp_tx_queue);
|
|
|
|
if (pdev->mode == DEV_MODE_HNET && dev->hnet_task) {
|
|
atomic_set(&dev->hnet_active, 1);
|
|
wake_up_interruptible(&dev->hnet_wq);
|
|
kthread_stop(dev->hnet_task);
|
|
dev->hnet_task = NULL;
|
|
}
|
|
|
|
/* Destroy all the filters */
|
|
ngknet_filter_destroy_all(dev);
|
|
|
|
/* Destroy all the virtual devices */
|
|
for (di = 1; di <= NUM_VDEV_MAX; di++) {
|
|
ndev = dev->vdev[di];
|
|
if (ndev) {
|
|
netif_carrier_off(ndev);
|
|
unregister_netdev(ndev);
|
|
free_netdev(ndev);
|
|
dev->vdev[di] = NULL;
|
|
}
|
|
}
|
|
dev->vdev[0] = NULL;
|
|
|
|
DBG_VERB(("Removing base network device %s.\n", dev->net_dev->name));
|
|
|
|
/* Destroy the base network device */
|
|
ndev = dev->net_dev;
|
|
unregister_netdev(ndev);
|
|
free_netdev(ndev);
|
|
|
|
for (qi = 0; qi < NUM_Q_MAX; qi++) {
|
|
dev->bdev[qi] = NULL;
|
|
}
|
|
|
|
for (gi = 0; gi < pdev->num_groups; gi++) {
|
|
if (!pdev->ctrl.grp[gi].attached) {
|
|
continue;
|
|
}
|
|
for (qi = 0; qi < pdev->grp_queues; qi++) {
|
|
hdl = &pdev->ctrl.grp[gi].intr_hdl[qi];
|
|
netif_napi_del((struct napi_struct *)hdl->priv);
|
|
priv_hdl[hdl->unit][hdl->chan].hdl = NULL;
|
|
if (pdev->flags & PDMA_GROUP_INTR) {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Clean up PDMA device */
|
|
bcmcnet_pdma_dev_cleanup(pdev);
|
|
|
|
/* Detach PDMA driver */
|
|
rv = drv_ops[pdev->dev_type]->drv_detach(pdev);
|
|
if (SHR_FAILURE(rv)) {
|
|
DBG_WARN(("Detach DMA driver failed.\n"));
|
|
}
|
|
ngbde_kapi_knet_disconnect(dn);
|
|
|
|
return rv;
|
|
}
|
|
|
|
/*!
|
|
* Network interface functions
|
|
*/
|
|
|
|
int
|
|
ngknet_netif_create(struct ngknet_dev *dev, ngknet_netif_t *netif)
|
|
{
|
|
struct net_device *ndev = NULL;
|
|
struct ngknet_private *priv = NULL;
|
|
unsigned long flags;
|
|
int num, id;
|
|
int rv;
|
|
|
|
switch (netif->type) {
|
|
case NGKNET_NETIF_T_VLAN:
|
|
case NGKNET_NETIF_T_PORT:
|
|
case NGKNET_NETIF_T_META:
|
|
break;
|
|
default:
|
|
return SHR_E_UNAVAIL;
|
|
}
|
|
|
|
/* Get vitual network device name */
|
|
if (netif->name[0] == '\0') {
|
|
/* Reserve 6 vacancies for base&vitual device number, i.e. nameAB_XYZ */
|
|
if (strlen(base_dev_name) < IFNAMSIZ - 6) {
|
|
snprintf(netif->name, IFNAMSIZ, "%s%d%s",
|
|
base_dev_name, dev->dev_info.dev_no, "_");
|
|
strncat(netif->name, "%d", 3);
|
|
} else {
|
|
DBG_WARN(("Too long network device name: %s.\n", base_dev_name));
|
|
return SHR_E_PARAM;
|
|
}
|
|
}
|
|
|
|
if (netif->chan >= NUM_Q_MAX) {
|
|
DBG_WARN(("Exceed max number of queues : %d.\n", netif->chan));
|
|
return SHR_E_PARAM;
|
|
}
|
|
|
|
rv = ngknet_ndev_init(netif, &ndev);
|
|
if (SHR_FAILURE(rv)) {
|
|
return rv;
|
|
}
|
|
|
|
spin_lock_irqsave(&dev->lock, flags);
|
|
|
|
num = (long)dev->vdev[0];
|
|
for (id = 1; id < num + 1; id++) {
|
|
if (!dev->vdev[id]) {
|
|
break;
|
|
}
|
|
}
|
|
if (id > NUM_VDEV_MAX) {
|
|
spin_unlock_irqrestore(&dev->lock, flags);
|
|
unregister_netdev(ndev);
|
|
free_netdev(ndev);
|
|
return SHR_E_RESOURCE;
|
|
}
|
|
|
|
dev->vdev[id] = ndev;
|
|
num += id == (num + 1) ? 1 : 0;
|
|
dev->vdev[0] = (struct net_device *)(long)num;
|
|
|
|
spin_unlock_irqrestore(&dev->lock, flags);
|
|
|
|
priv = netdev_priv(ndev);
|
|
priv->net_dev = ndev;
|
|
priv->bkn_dev = dev;
|
|
|
|
netif->id = id;
|
|
memcpy(netif->macaddr, ndev->dev_addr, ETH_ALEN);
|
|
netif->mtu = ndev->mtu;
|
|
memcpy(netif->name, ndev->name, sizeof(netif->name) - 1);
|
|
memcpy(&priv->netif, netif, sizeof(priv->netif));
|
|
|
|
if (priv->netif.flags & NGKNET_NETIF_F_BIND_CHAN) {
|
|
dev->bdev[priv->netif.chan] = ndev;
|
|
}
|
|
|
|
/* Optional netif create callback handle */
|
|
if (dev->cbc->netif_create_cb) {
|
|
if (dev->cbc->netif_create_cb(ndev)) {
|
|
DBG_WARN(("Network interface callback (create) failed for '%s'\n",
|
|
ndev->name));
|
|
}
|
|
}
|
|
|
|
DBG_VERB(("Created virtual network device %s (%d).\n",
|
|
ndev->name, priv->netif.id));
|
|
|
|
return SHR_E_NONE;
|
|
}
|
|
|
|
int
|
|
ngknet_netif_destroy(struct ngknet_dev *dev, int id)
|
|
{
|
|
struct net_device *ndev = NULL;
|
|
struct ngknet_private *priv = NULL;
|
|
unsigned long flags;
|
|
int num;
|
|
DECLARE_WAITQUEUE(wait, current);
|
|
|
|
if (id <= 0 || id > NUM_VDEV_MAX) {
|
|
return SHR_E_PARAM;
|
|
}
|
|
|
|
spin_lock_irqsave(&dev->lock, flags);
|
|
|
|
ndev = dev->vdev[id];
|
|
if (!ndev) {
|
|
spin_unlock_irqrestore(&dev->lock, flags);
|
|
return SHR_E_NOT_FOUND;
|
|
}
|
|
priv = netdev_priv(ndev);
|
|
|
|
add_wait_queue(&dev->wq, &wait);
|
|
|
|
while (priv->users) {
|
|
priv->wait = 1;
|
|
set_current_state(TASK_INTERRUPTIBLE);
|
|
spin_unlock_irqrestore(&dev->lock, flags);
|
|
schedule();
|
|
spin_lock_irqsave(&dev->lock, flags);
|
|
priv->wait = 0;
|
|
set_current_state(TASK_RUNNING);
|
|
}
|
|
|
|
if (priv->netif.flags & NGKNET_NETIF_F_BIND_CHAN) {
|
|
dev->bdev[priv->netif.chan] = NULL;
|
|
}
|
|
|
|
dev->vdev[id] = NULL;
|
|
num = (long)dev->vdev[0];
|
|
while (num-- == id--) {
|
|
if (dev->vdev[id]) {
|
|
dev->vdev[0] = (struct net_device *)(long)num;
|
|
break;
|
|
}
|
|
}
|
|
|
|
spin_unlock_irqrestore(&dev->lock, flags);
|
|
|
|
remove_wait_queue(&dev->wq, &wait);
|
|
|
|
/* Optional netif destroy callback handle */
|
|
if (dev->cbc->netif_destroy_cb) {
|
|
if (dev->cbc->netif_destroy_cb(ndev)) {
|
|
DBG_WARN(("Network interface callback (destroy) failed for '%s'\n",
|
|
ndev->name));
|
|
}
|
|
}
|
|
|
|
DBG_VERB(("Removing virtual network device %s (%d).\n",
|
|
ndev->name, priv->netif.id));
|
|
|
|
netif_carrier_off(ndev);
|
|
unregister_netdev(ndev);
|
|
free_netdev(ndev);
|
|
|
|
return SHR_E_NONE;
|
|
}
|
|
|
|
int
|
|
ngknet_netif_get(struct ngknet_dev *dev, int id, ngknet_netif_t *netif)
|
|
{
|
|
struct net_device *ndev = NULL;
|
|
struct ngknet_private *priv = NULL;
|
|
unsigned long flags;
|
|
int num;
|
|
|
|
if (id < 0 || id > NUM_VDEV_MAX) {
|
|
return SHR_E_PARAM;
|
|
}
|
|
|
|
spin_lock_irqsave(&dev->lock, flags);
|
|
|
|
ndev = id == 0 ? dev->net_dev : dev->vdev[id];
|
|
if (!ndev) {
|
|
spin_unlock_irqrestore(&dev->lock, flags);
|
|
return SHR_E_NOT_FOUND;
|
|
}
|
|
|
|
priv = netdev_priv(ndev);
|
|
memcpy(netif, &priv->netif, sizeof(*netif));
|
|
|
|
num = (long)dev->vdev[0];
|
|
for (id++; id < num + 1; id++) {
|
|
if (dev->vdev[id]) {
|
|
break;
|
|
}
|
|
}
|
|
netif->next = id == (num + 1) ? 0 : id;
|
|
|
|
spin_unlock_irqrestore(&dev->lock, flags);
|
|
|
|
DBG_VERB(("Got virtual network device %s (%d).\n",
|
|
ndev->name, priv->netif.id));
|
|
|
|
return SHR_E_NONE;
|
|
}
|
|
|
|
int
|
|
ngknet_netif_get_next(struct ngknet_dev *dev, ngknet_netif_t *netif)
|
|
{
|
|
return ngknet_netif_get(dev, netif->next, netif);
|
|
}
|
|
|
|
/*!
|
|
* System control interfaces
|
|
*/
|
|
|
|
int
|
|
ngknet_debug_level_get(void)
|
|
{
|
|
return debug;
|
|
}
|
|
|
|
void
|
|
ngknet_debug_level_set(int debug_level)
|
|
{
|
|
debug = debug_level;
|
|
}
|
|
|
|
int
|
|
ngknet_rx_rate_limit_get(void)
|
|
{
|
|
return rx_rate_limit;
|
|
}
|
|
|
|
void
|
|
ngknet_rx_rate_limit_set(int rate_limit)
|
|
{
|
|
rx_rate_limit = rate_limit;
|
|
}
|
|
|
|
int
|
|
ngknet_page_buffer_mode_get(void)
|
|
{
|
|
return page_buffer_mode;
|
|
}
|
|
|
|
/*!
|
|
* Generic module functions
|
|
*/
|
|
|
|
static int
|
|
ngknet_open(struct inode *inode, struct file *filp)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
static int
|
|
ngknet_release(struct inode *inode, struct file *filp)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
static long
|
|
ngknet_ioctl(struct file *file, unsigned int cmd, unsigned long arg)
|
|
{
|
|
struct ngknet_ioctl ioc;
|
|
struct ngknet_dev *dev = NULL;
|
|
struct net_device *ndev = NULL;
|
|
struct pdma_dev *pdev = NULL;
|
|
union {
|
|
ngknet_dev_cfg_t dev_cfg;
|
|
ngknet_chan_cfg_t chan_cfg;
|
|
ngknet_netif_t netif;
|
|
ngknet_filter_t filter;
|
|
} iod;
|
|
ngknet_dev_cfg_t *dev_cfg = &iod.dev_cfg;
|
|
ngknet_chan_cfg_t *chan_cfg = &iod.chan_cfg;
|
|
ngknet_netif_t *netif = &iod.netif;
|
|
ngknet_filter_t *filter = &iod.filter;
|
|
char *data = NULL;
|
|
int dt, gi, qi;
|
|
|
|
if (_IOC_TYPE(cmd) != NGKNET_IOC_MAGIC) {
|
|
DBG_WARN(("Unsupported command (cmd=%d)\n", cmd));
|
|
return -EINVAL;
|
|
}
|
|
|
|
if (copy_from_user(&ioc, (void *)arg, sizeof(ioc))) {
|
|
return -EFAULT;
|
|
}
|
|
|
|
ioc.rc = SHR_E_NONE;
|
|
|
|
dev = &ngknet_devices[ioc.unit];
|
|
pdev = &dev->pdma_dev;
|
|
|
|
if (cmd != NGKNET_VERSION_GET &&
|
|
cmd != NGKNET_RX_RATE_LIMIT &&
|
|
cmd != NGKNET_DEV_INIT &&
|
|
!(dev->flags & NGKNET_DEV_ACTIVE)) {
|
|
ioc.rc = SHR_E_UNAVAIL;
|
|
if (copy_to_user((void *)arg, &ioc, sizeof(ioc))) {
|
|
return -EFAULT;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
switch (cmd) {
|
|
case NGKNET_VERSION_GET:
|
|
DBG_CMD(("NGKNET_VERSION_GET\n"));
|
|
ioc.op.info.version = NGKNET_IOC_VERSION;
|
|
break;
|
|
case NGKNET_RX_RATE_LIMIT:
|
|
DBG_CMD(("NGKNET_RX_RATE_LIMIT\n"));
|
|
if (ioc.iarg[0]) {
|
|
ngknet_rx_rate_limit_set(ioc.iarg[1]);
|
|
} else {
|
|
ioc.iarg[1] = ngknet_rx_rate_limit_get();
|
|
}
|
|
break;
|
|
case NGKNET_DEV_INIT:
|
|
DBG_CMD(("NGKNET_DEV_INIT\n"));
|
|
if (dev->flags & NGKNET_DEV_ACTIVE) {
|
|
DBG_CMD(("NGKNET_DEV_INIT, retrieve device configurations.\n"));
|
|
strlcpy(dev_cfg->name, pdev->name, sizeof(dev_cfg->name));
|
|
dev_cfg->dev_id = pdev->dev_id;
|
|
dev_cfg->nb_grp = pdev->ctrl.nb_grp;
|
|
dev_cfg->bm_grp = pdev->ctrl.bm_grp;
|
|
ioc.rc = ngknet_netif_get(dev, 0, &dev_cfg->base_netif);
|
|
if (SHR_FAILURE((int)ioc.rc)) {
|
|
break;
|
|
}
|
|
if (kal_copy_to_user((void *)(unsigned long)ioc.op.data.buf, dev_cfg,
|
|
ioc.op.data.len, sizeof(*dev_cfg))) {
|
|
return -EFAULT;
|
|
}
|
|
break;
|
|
}
|
|
if (kal_copy_from_user(dev_cfg, (void *)(unsigned long)ioc.op.data.buf,
|
|
sizeof(*dev_cfg), ioc.op.data.len)) {
|
|
return -EFAULT;
|
|
}
|
|
if (!dev_cfg->name[0] || !dev_cfg->bm_grp ||
|
|
dev_cfg->bm_grp >= (1 << NUM_GRP_MAX)) {
|
|
DBG_WARN(("Invalid parameter: name=%s, bm_grp=0x%x\n",
|
|
dev_cfg->name, dev_cfg->bm_grp));
|
|
ioc.rc = SHR_E_PARAM;
|
|
break;
|
|
}
|
|
memset(pdev, 0, sizeof(*pdev));
|
|
strlcpy(pdev->name, dev_cfg->name, sizeof(pdev->name));
|
|
pdev->dev_id = dev_cfg->dev_id;
|
|
for (dt = 0; dt < drv_num; dt++) {
|
|
if (!drv_ops[dt]) {
|
|
continue;
|
|
}
|
|
if (!strcasecmp(dev_cfg->type_str, drv_ops[dt]->drv_desc)) {
|
|
pdev->dev_type = dt;
|
|
strlcpy(dev->dev_info.var_str, dev_cfg->var_str,
|
|
sizeof(dev->dev_info.var_str));
|
|
break;
|
|
}
|
|
}
|
|
if (pdev->dev_type <= NGKNET_DEV_T_NONE ||
|
|
pdev->dev_type >= NGKNET_DEV_T_COUNT) {
|
|
ioc.rc = SHR_E_PARAM;
|
|
break;
|
|
}
|
|
pdev->ctrl.bm_grp = dev_cfg->bm_grp;
|
|
for (gi = 0; gi < NUM_GRP_MAX; gi++) {
|
|
if (1 << gi & dev_cfg->bm_grp) {
|
|
pdev->ctrl.nb_grp++;
|
|
pdev->ctrl.grp[gi].attached = true;
|
|
pdev->num_groups = gi + 1;
|
|
}
|
|
}
|
|
pdev->rx_ph_size = dev_cfg->rx_ph_size;
|
|
pdev->tx_ph_size = dev_cfg->tx_ph_size;
|
|
pdev->mode = dev_cfg->mode;
|
|
if (pdev->mode != DEV_MODE_KNET && pdev->mode != DEV_MODE_HNET) {
|
|
pdev->mode = DEV_MODE_KNET;
|
|
}
|
|
ioc.rc = ngknet_dev_probe(ioc.unit, &dev_cfg->base_netif);
|
|
if (SHR_FAILURE((int)ioc.rc)) {
|
|
break;
|
|
}
|
|
if (dev->cbc->dev_init_cb) {
|
|
dev->cbc->dev_init_cb(&dev->dev_info);
|
|
}
|
|
|
|
if (kal_copy_to_user((void *)(unsigned long)ioc.op.data.buf, dev_cfg,
|
|
ioc.op.data.len, sizeof(*dev_cfg))) {
|
|
return -EFAULT;
|
|
}
|
|
break;
|
|
case NGKNET_DEV_DEINIT:
|
|
DBG_CMD(("NGKNET_DEV_DEINIT\n"));
|
|
if (dev->flags & NGKNET_DEV_ACTIVE) {
|
|
ioc.rc = ngknet_dev_remove(ioc.unit);
|
|
}
|
|
break;
|
|
case NGKNET_QUEUE_CONFIG:
|
|
DBG_CMD(("NGKNET_QUEUE_CONFIG\n"));
|
|
if (kal_copy_from_user(chan_cfg, (void *)(unsigned long)ioc.op.data.buf,
|
|
sizeof(*chan_cfg), ioc.op.data.len)) {
|
|
return -EFAULT;
|
|
}
|
|
gi = chan_cfg->chan / pdev->grp_queues;
|
|
if (!(1 << gi & pdev->ctrl.bm_grp)) {
|
|
DBG_WARN(("Invalid parameter: chan=%d (bm_grp=0x%x)\n",
|
|
chan_cfg->chan, pdev->ctrl.bm_grp));
|
|
ioc.rc = SHR_E_PARAM;
|
|
break;
|
|
}
|
|
if (chan_cfg->dir == PDMA_Q_RX) {
|
|
if (1 << chan_cfg->chan & pdev->ctrl.bm_txq) {
|
|
pdev->ctrl.bm_txq &= ~(1 << chan_cfg->chan);
|
|
pdev->ctrl.nb_txq--;
|
|
}
|
|
if (!(1 << chan_cfg->chan & pdev->ctrl.bm_rxq)) {
|
|
pdev->ctrl.bm_rxq |= 1 << chan_cfg->chan;
|
|
pdev->ctrl.nb_rxq++;
|
|
}
|
|
} else {
|
|
if (1 << chan_cfg->chan & pdev->ctrl.bm_rxq) {
|
|
pdev->ctrl.bm_rxq &= ~(1 << chan_cfg->chan);
|
|
pdev->ctrl.nb_rxq--;
|
|
}
|
|
if (!(1 << chan_cfg->chan & pdev->ctrl.bm_txq)) {
|
|
pdev->ctrl.bm_txq |= 1 << chan_cfg->chan;
|
|
pdev->ctrl.nb_txq++;
|
|
}
|
|
}
|
|
qi = chan_cfg->chan % pdev->grp_queues;
|
|
pdev->ctrl.grp[gi].nb_desc[qi] = chan_cfg->nb_desc;
|
|
pdev->ctrl.grp[gi].rx_size[qi] = chan_cfg->rx_buf_size;
|
|
pdev->ctrl.grp[gi].que_ctrl[qi] &= ~(PDMA_PKT_BYTE_SWAP |
|
|
PDMA_OTH_BYTE_SWAP |
|
|
PDMA_HDR_BYTE_SWAP);
|
|
if (chan_cfg->chan_ctrl & NGKNET_PKT_BYTE_SWAP) {
|
|
pdev->ctrl.grp[gi].que_ctrl[qi] |= PDMA_PKT_BYTE_SWAP;
|
|
}
|
|
if (chan_cfg->chan_ctrl & NGKNET_OTH_BYTE_SWAP) {
|
|
pdev->ctrl.grp[gi].que_ctrl[qi] |= PDMA_OTH_BYTE_SWAP;
|
|
}
|
|
if (chan_cfg->chan_ctrl & NGKNET_HDR_BYTE_SWAP) {
|
|
pdev->ctrl.grp[gi].que_ctrl[qi] |= PDMA_HDR_BYTE_SWAP;
|
|
}
|
|
break;
|
|
case NGKNET_QUEUE_QUERY:
|
|
DBG_CMD(("NGKNET_QUEUE_QUERY\n"));
|
|
if (kal_copy_from_user(chan_cfg, (void *)(unsigned long)ioc.op.data.buf,
|
|
sizeof(*chan_cfg), ioc.op.data.len)) {
|
|
return -EFAULT;
|
|
}
|
|
if (1 << chan_cfg->chan & pdev->ctrl.bm_rxq) {
|
|
chan_cfg->dir = PDMA_Q_RX;
|
|
} else if (1 << chan_cfg->chan & pdev->ctrl.bm_txq) {
|
|
chan_cfg->dir = PDMA_Q_TX;
|
|
} else {
|
|
ioc.rc = SHR_E_UNAVAIL;
|
|
break;
|
|
}
|
|
gi = chan_cfg->chan / pdev->grp_queues;
|
|
qi = chan_cfg->chan % pdev->grp_queues;
|
|
chan_cfg->nb_desc = pdev->ctrl.grp[gi].nb_desc[qi];
|
|
chan_cfg->chan_ctrl = pdev->ctrl.grp[gi].que_ctrl[qi];
|
|
if (chan_cfg->dir == PDMA_Q_RX) {
|
|
chan_cfg->rx_buf_size = pdev->ctrl.grp[gi].rx_size[qi];
|
|
} else {
|
|
chan_cfg->rx_buf_size = 0;
|
|
}
|
|
if (kal_copy_to_user((void *)(unsigned long)ioc.op.data.buf, chan_cfg,
|
|
ioc.op.data.len, sizeof(*chan_cfg))) {
|
|
return -EFAULT;
|
|
}
|
|
break;
|
|
case NGKNET_DEV_SUSPEND:
|
|
DBG_CMD(("NGKNET_DEV_SUSPEND\n"));
|
|
if (rx_rate_limit >= 0) {
|
|
ngknet_rx_rate_limit_stop(dev);
|
|
}
|
|
if (ioc.iarg[0]) {
|
|
/* Graceful suspend */
|
|
ioc.rc = bcmcnet_pdma_dev_suspend(pdev);
|
|
} else {
|
|
pdev->flags |= PDMA_ABORT;
|
|
ioc.rc = bcmcnet_pdma_dev_suspend(pdev);
|
|
}
|
|
break;
|
|
case NGKNET_DEV_RESUME:
|
|
DBG_CMD(("NGKNET_DEV_RESUME\n"));
|
|
ioc.rc = bcmcnet_pdma_dev_resume(pdev);
|
|
if (rx_rate_limit >= 0) {
|
|
ngknet_rx_rate_limit_start(dev);
|
|
}
|
|
break;
|
|
case NGKNET_DEV_VNET_WAIT:
|
|
DBG_CMD(("NGKNET_DEV_VNET_WAIT\n"));
|
|
if (pdev->mode != DEV_MODE_HNET) {
|
|
ioc.rc = SHR_E_UNAVAIL;
|
|
break;
|
|
}
|
|
wait_event_interruptible(dev->vnet_wq,
|
|
atomic_read(&dev->vnet_active) != 0);
|
|
atomic_set(&dev->vnet_active, 0);
|
|
break;
|
|
case NGKNET_DEV_HNET_WAKE:
|
|
DBG_CMD(("NGKNET_DEV_HNET_WAKE\n"));
|
|
if (pdev->mode != DEV_MODE_HNET) {
|
|
ioc.rc = SHR_E_UNAVAIL;
|
|
break;
|
|
}
|
|
atomic_set(&dev->hnet_active, 1);
|
|
wake_up_interruptible(&dev->hnet_wq);
|
|
break;
|
|
case NGKNET_DEV_VNET_DOCK:
|
|
DBG_CMD(("NGKNET_DEV_VNET_DOCK\n"));
|
|
if (pdev->mode != DEV_MODE_HNET) {
|
|
ioc.rc = SHR_E_UNAVAIL;
|
|
break;
|
|
}
|
|
if (kal_copy_from_user(&pdev->ctrl.vsync, (void *)(unsigned long)ioc.op.data.buf,
|
|
sizeof(pdev->ctrl.vsync), ioc.op.data.len)) {
|
|
return -EFAULT;
|
|
}
|
|
ioc.rc = bcmcnet_pdma_dev_dock(pdev);
|
|
break;
|
|
case NGKNET_DEV_VNET_UNDOCK:
|
|
DBG_CMD(("NGKNET_DEV_VNET_UNDOCK\n"));
|
|
if (pdev->mode != DEV_MODE_HNET) {
|
|
ioc.rc = SHR_E_UNAVAIL;
|
|
break;
|
|
}
|
|
ngknet_dev_vnet_wake(pdev);
|
|
ioc.rc = bcmcnet_pdma_dev_undock(pdev);
|
|
break;
|
|
case NGKNET_RCPU_CONFIG:
|
|
DBG_CMD(("NGKNET_RCPU_CONFIG\n"));
|
|
if (kal_copy_from_user(&dev->rcpu_ctrl, (void *)(unsigned long)ioc.op.data.buf,
|
|
sizeof(dev->rcpu_ctrl), ioc.op.data.len)) {
|
|
return -EFAULT;
|
|
}
|
|
break;
|
|
case NGKNET_RCPU_GET:
|
|
DBG_CMD(("NGKNET_RCPU_GET\n"));
|
|
if (kal_copy_to_user((void *)(unsigned long)ioc.op.data.buf, &dev->rcpu_ctrl,
|
|
ioc.op.data.len, sizeof(dev->rcpu_ctrl))) {
|
|
return -EFAULT;
|
|
}
|
|
break;
|
|
case NGKNET_INFO_GET:
|
|
DBG_CMD(("NGKNET_INFO_GET\n"));
|
|
bcmcnet_pdma_dev_info_get(pdev);
|
|
if (kal_copy_to_user((void *)(unsigned long)ioc.op.data.buf, &pdev->info,
|
|
ioc.op.data.len, sizeof(pdev->info))) {
|
|
return -EFAULT;
|
|
}
|
|
break;
|
|
case NGKNET_STATS_GET:
|
|
DBG_CMD(("NGKNET_STATS_GET\n"));
|
|
bcmcnet_pdma_dev_stats_get(pdev);
|
|
if (kal_copy_to_user((void *)(unsigned long)ioc.op.data.buf, &pdev->stats,
|
|
ioc.op.data.len, sizeof(pdev->stats))) {
|
|
return -EFAULT;
|
|
}
|
|
break;
|
|
case NGKNET_STATS_RESET:
|
|
DBG_CMD(("NGKNET_STATS_RESET\n"));
|
|
bcmcnet_pdma_dev_stats_reset(pdev);
|
|
break;
|
|
case NGKNET_NETIF_CREATE:
|
|
DBG_CMD(("NGKNET_NETIF_CREATE\n"));
|
|
if (kal_copy_from_user(netif, (void *)(unsigned long)ioc.op.data.buf,
|
|
sizeof(*netif), ioc.op.data.len)) {
|
|
return -EFAULT;
|
|
}
|
|
ioc.rc = ngknet_netif_create(dev, netif);
|
|
if (SHR_FAILURE((int)ioc.rc)) {
|
|
break;
|
|
}
|
|
if (kal_copy_to_user((void *)(unsigned long)ioc.op.data.buf, netif,
|
|
ioc.op.data.len, sizeof(*netif))) {
|
|
return -EFAULT;
|
|
}
|
|
break;
|
|
case NGKNET_NETIF_DESTROY:
|
|
DBG_CMD(("NGKNET_NETIF_DESTROY\n"));
|
|
ioc.rc = ngknet_netif_destroy(dev, ioc.iarg[0]);
|
|
break;
|
|
case NGKNET_NETIF_GET:
|
|
DBG_CMD(("NGKNET_NETIF_GET\n"));
|
|
ioc.rc = ngknet_netif_get(dev, ioc.iarg[0], netif);
|
|
if (SHR_FAILURE((int)ioc.rc)) {
|
|
break;
|
|
}
|
|
if (kal_copy_to_user((void *)(unsigned long)ioc.op.data.buf, netif,
|
|
ioc.op.data.len, sizeof(*netif))) {
|
|
return -EFAULT;
|
|
}
|
|
break;
|
|
case NGKNET_NETIF_NEXT:
|
|
DBG_CMD(("NGKNET_NETIF_NEXT\n"));
|
|
if (kal_copy_from_user(netif, (void *)(unsigned long)ioc.op.data.buf,
|
|
sizeof(*netif), ioc.op.data.len)) {
|
|
return -EFAULT;
|
|
}
|
|
ioc.rc = ngknet_netif_get_next(dev, netif);
|
|
if (SHR_FAILURE((int)ioc.rc)) {
|
|
break;
|
|
}
|
|
if (kal_copy_to_user((void *)(unsigned long)ioc.op.data.buf, netif,
|
|
ioc.op.data.len, sizeof(*netif))) {
|
|
return -EFAULT;
|
|
}
|
|
break;
|
|
case NGKNET_NETIF_LINK_SET:
|
|
DBG_CMD(("NGKNET_NETIF_LINK_SET\n"));
|
|
ioc.rc = ngknet_netif_get(dev, ioc.iarg[0], netif);
|
|
if (SHR_FAILURE((int)ioc.rc)) {
|
|
break;
|
|
}
|
|
ndev = dev->vdev[netif->id];
|
|
if (ioc.iarg[1]) {
|
|
if (!netif_carrier_ok(ndev)) {
|
|
netif_carrier_on(ndev);
|
|
netif_tx_wake_all_queues(ndev);
|
|
DBG_LINK(("%s: link up\n", netif->name));
|
|
}
|
|
} else {
|
|
if (netif_carrier_ok(ndev)) {
|
|
netif_carrier_off(ndev);
|
|
netif_tx_stop_all_queues(ndev);
|
|
DBG_LINK(("%s: link down\n", netif->name));
|
|
}
|
|
}
|
|
break;
|
|
case NGKNET_FILT_CREATE:
|
|
DBG_CMD(("NGKNET_FILT_CREATE\n"));
|
|
if (kal_copy_from_user(filter, (void *)(unsigned long)ioc.op.data.buf,
|
|
sizeof(*filter), ioc.op.data.len)) {
|
|
return -EFAULT;
|
|
}
|
|
ioc.rc = ngknet_filter_create(dev, filter);
|
|
if (SHR_FAILURE((int)ioc.rc)) {
|
|
break;
|
|
}
|
|
if (kal_copy_to_user((void *)(unsigned long)ioc.op.data.buf, filter,
|
|
ioc.op.data.len, sizeof(*filter))) {
|
|
return -EFAULT;
|
|
}
|
|
break;
|
|
case NGKNET_FILT_DESTROY:
|
|
DBG_CMD(("NGKNET_FILT_DESTROY\n"));
|
|
ioc.rc = ngknet_filter_destroy(dev, ioc.iarg[0]);
|
|
break;
|
|
case NGKNET_FILT_GET:
|
|
DBG_CMD(("NGKNET_FILT_GET\n"));
|
|
ioc.rc = ngknet_filter_get(dev, ioc.iarg[0], filter);
|
|
if (SHR_FAILURE((int)ioc.rc)) {
|
|
break;
|
|
}
|
|
if (kal_copy_to_user((void *)(unsigned long)ioc.op.data.buf, filter,
|
|
ioc.op.data.len, sizeof(*filter))) {
|
|
return -EFAULT;
|
|
}
|
|
break;
|
|
case NGKNET_FILT_NEXT:
|
|
DBG_CMD(("NGKNET_FILT_NEXT\n"));
|
|
if (kal_copy_from_user(filter, (void *)(unsigned long)ioc.op.data.buf,
|
|
sizeof(*filter), ioc.op.data.len)) {
|
|
return -EFAULT;
|
|
}
|
|
ioc.rc = ngknet_filter_get_next(dev, filter);
|
|
if (SHR_FAILURE((int)ioc.rc)) {
|
|
break;
|
|
}
|
|
if (kal_copy_to_user((void *)(unsigned long)ioc.op.data.buf, filter,
|
|
ioc.op.data.len, sizeof(*filter))) {
|
|
return -EFAULT;
|
|
}
|
|
break;
|
|
case NGKNET_PTP_DEV_CTRL:
|
|
DBG_CMD(("NGKNET_PTP_DEV_CTRL\n"));
|
|
if (ioc.op.data.len) {
|
|
data = kmalloc(ioc.op.data.len, GFP_ATOMIC);
|
|
if (data == NULL) {
|
|
printk("Fatal error: no memory for PTP device ioctl\n");
|
|
return -EFAULT;
|
|
}
|
|
if (copy_from_user(data, (void *)(unsigned long)ioc.op.data.buf,
|
|
ioc.op.data.len)) {
|
|
kfree(data);
|
|
return -EFAULT;
|
|
}
|
|
}
|
|
ioc.rc = ngknet_ptp_dev_ctrl(dev, ioc.iarg[0], data, ioc.op.data.len);
|
|
if (SHR_FAILURE((int)ioc.rc)) {
|
|
if (data) {
|
|
kfree(data);
|
|
}
|
|
break;
|
|
}
|
|
if (ioc.op.data.len) {
|
|
if (copy_to_user((void *)(unsigned long)ioc.op.data.buf, data,
|
|
ioc.op.data.len)) {
|
|
kfree(data);
|
|
return -EFAULT;
|
|
}
|
|
kfree(data);
|
|
}
|
|
break;
|
|
default:
|
|
ioc.rc = SHR_E_UNAVAIL;
|
|
printk("Invalid IOCTL");
|
|
break;
|
|
}
|
|
|
|
if (copy_to_user((void *)arg, &ioc, sizeof(ioc))) {
|
|
return -EFAULT;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int
|
|
ngknet_mmap(struct file *filp, struct vm_area_struct *vma)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
static struct file_operations ngknet_fops = {
|
|
.open = ngknet_open,
|
|
.release = ngknet_release,
|
|
.unlocked_ioctl = ngknet_ioctl,
|
|
.compat_ioctl = ngknet_ioctl,
|
|
.mmap = ngknet_mmap,
|
|
};
|
|
|
|
static int __init
|
|
ngknet_init_module(void)
|
|
{
|
|
int idx;
|
|
int rv;
|
|
|
|
rv = register_chrdev(NGKNET_MODULE_MAJOR, NGKNET_MODULE_NAME, &ngknet_fops);
|
|
if (rv < 0) {
|
|
printk(KERN_WARNING "%s: can't get major %d\n",
|
|
NGKNET_MODULE_NAME, NGKNET_MODULE_MAJOR);
|
|
return rv;
|
|
}
|
|
|
|
/* Randomize lower 3 bytes of the MAC address (TESTING ONLY) */
|
|
get_random_bytes(&ngknet_dev_mac[3], 3);
|
|
|
|
/* Check for user-supplied MAC address (recommended) */
|
|
if (mac_addr != NULL && strlen(mac_addr) == 17) {
|
|
for (idx = 0; idx < 6; idx++) {
|
|
ngknet_dev_mac[idx] = simple_strtoul(&mac_addr[idx * 3], NULL, 16);
|
|
}
|
|
/* Do not allow multicast address */
|
|
ngknet_dev_mac[0] &= ~0x01;
|
|
}
|
|
|
|
/* Initialize procfs */
|
|
ngknet_procfs_init();
|
|
|
|
/* Initialize Rx rate limit */
|
|
ngknet_rx_rate_limit_init(ngknet_devices);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void __exit
|
|
ngknet_exit_module(void)
|
|
{
|
|
int idx;
|
|
|
|
/* Cleanup Rx rate limit */
|
|
ngknet_rx_rate_limit_cleanup();
|
|
|
|
/* Cleanup procfs */
|
|
ngknet_procfs_cleanup();
|
|
|
|
/* Remove all the devices */
|
|
for (idx = 0; idx < NUM_PDMA_DEV_MAX; idx++) {
|
|
ngknet_dev_remove(idx);
|
|
}
|
|
|
|
unregister_chrdev(NGKNET_MODULE_MAJOR, NGKNET_MODULE_NAME);
|
|
}
|
|
|
|
module_init(ngknet_init_module);
|
|
module_exit(ngknet_exit_module);
|
|
|