]> git.droids-corp.org - dpdk.git/commitdiff
net/nfp: support NFP3800 card
authorJin Liu <jin.liu@corigine.com>
Fri, 17 Jun 2022 09:34:37 +0000 (11:34 +0200)
committerFerruh Yigit <ferruh.yigit@xilinx.com>
Fri, 17 Jun 2022 13:32:41 +0000 (15:32 +0200)
Add support for a new type of NIC NFP3800 card, and update some
network card data acquisition interface functions.

Signed-off-by: Jin Liu <jin.liu@corigine.com>
Signed-off-by: Diana Wang <na.wang@corigine.com>
Signed-off-by: Peng Zhang <peng.zhang@corigine.com>
Signed-off-by: Chaoyong He <chaoyong.he@corigine.com>
Signed-off-by: Niklas Söderlund <niklas.soderlund@corigine.com>
doc/guides/nics/nfp.rst
doc/guides/rel_notes/release_22_07.rst
drivers/net/nfp/nfp_common.h
drivers/net/nfp/nfp_ethdev.c
drivers/net/nfp/nfp_ethdev_vf.c
drivers/net/nfp/nfpcore/nfp_cpp.h
drivers/net/nfp/nfpcore/nfp_nsp_eth.c

index 30cdc6920294785bc622c505d2722bcab91240be..dcefac3ef6e6d7533a73a4c12248336a49e0d401 100644 (file)
@@ -12,7 +12,8 @@ up to 400-Gb/s.
 
 This document explains how to use DPDK with the Netronome Poll Mode
 Driver (PMD) supporting Netronome's Network Flow Processor 6xxx
-(NFP-6xxx) and Netronome's Flow Processor 4xxx (NFP-4xxx).
+(NFP-6xxx), Netronome's Network Flow Processor 4xxx (NFP-4xxx) and
+Netronome's Network Flow Processor 38xx (NFP-38xx).
 
 NFP is a SRIOV capable device and the PMD supports the physical
 function (PF) and the virtual functions (VFs).
index 4d50aa2c028e884aabb621209a983bdf14baf62c..98db21c6e4935d707aba57cc311e35f19f43d778 100644 (file)
@@ -157,6 +157,10 @@ New Features
   * Added support for MTU on Windows.
   * Added matching and RSS on IPsec ESP.
 
+* **Updated Netronome nfp driver.**
+
+  * Added support for NFP3800 NIC.
+
 * **Updated VMware vmxnet3 networking driver.**
 
   * Added version 5 support.
index 8e1b4fb6a469eb7525998af243aaee729b36316a..19e96414e32811f5f06b6a20be788e82fff173fe 100644 (file)
 
 #define NFP_NET_PMD_VERSION "0.1"
 #define PCI_VENDOR_ID_NETRONOME         0x19ee
+#define PCI_DEVICE_ID_NFP3800_PF_NIC    0x3800
+#define PCI_DEVICE_ID_NFP3800_VF_NIC    0x3803
 #define PCI_DEVICE_ID_NFP4000_PF_NIC    0x4000
 #define PCI_DEVICE_ID_NFP6000_PF_NIC    0x6000
-#define PCI_DEVICE_ID_NFP6000_VF_NIC    0x6003
+#define PCI_DEVICE_ID_NFP6000_VF_NIC    0x6003  /* Include NFP4000VF */
 
 /* Forward declaration */
 struct nfp_net_adapter;
@@ -41,8 +43,16 @@ struct nfp_net_adapter;
 #define NFP_QCP_QUEUE_STS_HI                    0x000c
 #define NFP_QCP_QUEUE_STS_HI_WRITEPTR_mask    (0x3ffff)
 
+#define NFP_PCIE_QCP_NFP3800_OFFSET            0x400000
+#define NFP_PCIE_QCP_NFP6000_OFFSET            0x80000
+#define NFP_PCIE_QUEUE_NFP3800_MASK            0x1ff
+#define NFP_PCIE_QUEUE_NFP6000_MASK            0xff
+#define NFP_PCIE_QCP_PF_OFFSET                 0x0
+#define NFP_PCIE_QCP_VF_OFFSET                 0x0
+
 /* The offset of the queue controller queues in the PCIe Target */
-#define NFP_PCIE_QUEUE(_q) (0x80000 + (NFP_QCP_QUEUE_ADDR_SZ * ((_q) & 0xff)))
+#define NFP_PCIE_QUEUE(_offset, _q, _mask)    \
+               ((_offset) + (NFP_QCP_QUEUE_ADDR_SZ * ((_q) & (_mask))))
 
 /* Interrupt definitions */
 #define NFP_NET_IRQ_LSC_IDX             0
@@ -342,6 +352,26 @@ nfp_qcp_read(uint8_t *q, enum nfp_qcp_ptr ptr)
                return val & NFP_QCP_QUEUE_STS_HI_WRITEPTR_mask;
 }
 
+static inline uint32_t
+nfp_pci_queue(struct rte_pci_device *pdev, uint16_t queue)
+{
+       switch (pdev->id.device_id) {
+       case PCI_DEVICE_ID_NFP4000_PF_NIC:
+       case PCI_DEVICE_ID_NFP6000_PF_NIC:
+               return NFP_PCIE_QUEUE(NFP_PCIE_QCP_PF_OFFSET, queue,
+                               NFP_PCIE_QUEUE_NFP6000_MASK);
+       case PCI_DEVICE_ID_NFP3800_VF_NIC:
+               return NFP_PCIE_QUEUE(NFP_PCIE_QCP_VF_OFFSET, queue,
+                               NFP_PCIE_QUEUE_NFP3800_MASK);
+       case PCI_DEVICE_ID_NFP6000_VF_NIC:
+               return NFP_PCIE_QUEUE(NFP_PCIE_QCP_VF_OFFSET, queue,
+                               NFP_PCIE_QUEUE_NFP6000_MASK);
+       default:
+               return NFP_PCIE_QUEUE(NFP_PCIE_QCP_PF_OFFSET, queue,
+                               NFP_PCIE_QUEUE_NFP3800_MASK);
+       }
+}
+
 /* Prototypes for common NFP functions */
 int nfp_net_reconfig(struct nfp_net_hw *hw, uint32_t ctrl, uint32_t update);
 int nfp_net_configure(struct rte_eth_dev *dev);
index ae6cb5943f3d3a0ced57cecf4be0540d2442d28b..cb84dc31889e42a336186136ba72688e0f07edf6 100644 (file)
@@ -446,12 +446,13 @@ nfp_net_init(struct rte_eth_dev *eth_dev)
 
        /* Work out where in the BAR the queues start. */
        switch (pci_dev->id.device_id) {
+       case PCI_DEVICE_ID_NFP3800_PF_NIC:
        case PCI_DEVICE_ID_NFP4000_PF_NIC:
        case PCI_DEVICE_ID_NFP6000_PF_NIC:
                start_q = nn_cfg_readl(hw, NFP_NET_CFG_START_TXQ);
-               tx_bar_off = (uint64_t)start_q * NFP_QCP_QUEUE_ADDR_SZ;
+               tx_bar_off = nfp_pci_queue(pci_dev, start_q);
                start_q = nn_cfg_readl(hw, NFP_NET_CFG_START_RXQ);
-               rx_bar_off = (uint64_t)start_q * NFP_QCP_QUEUE_ADDR_SZ;
+               rx_bar_off = nfp_pci_queue(pci_dev, start_q);
                break;
        default:
                PMD_DRV_LOG(ERR, "nfp_net: no device ID matching");
@@ -764,6 +765,7 @@ nfp_pf_init(struct rte_pci_device *pci_dev)
 {
        int err;
        int ret = 0;
+       uint64_t addr;
        int total_ports;
        struct nfp_cpp *cpp;
        struct nfp_pf_dev *pf_dev;
@@ -867,8 +869,24 @@ nfp_pf_init(struct rte_pci_device *pci_dev)
        PMD_INIT_LOG(DEBUG, "ctrl bar: %p", pf_dev->ctrl_bar);
 
        /* configure access to tx/rx vNIC BARs */
+       switch (pci_dev->id.device_id) {
+       case PCI_DEVICE_ID_NFP3800_PF_NIC:
+               addr = NFP_PCIE_QUEUE(NFP_PCIE_QCP_NFP3800_OFFSET,
+                                       0, NFP_PCIE_QUEUE_NFP3800_MASK);
+               break;
+       case PCI_DEVICE_ID_NFP4000_PF_NIC:
+       case PCI_DEVICE_ID_NFP6000_PF_NIC:
+               addr = NFP_PCIE_QUEUE(NFP_PCIE_QCP_NFP6000_OFFSET,
+                                       0, NFP_PCIE_QUEUE_NFP6000_MASK);
+               break;
+       default:
+               PMD_INIT_LOG(ERR, "nfp_net: no device ID matching");
+               err = -ENODEV;
+               goto ctrl_area_cleanup;
+       }
+
        pf_dev->hw_queues = nfp_cpp_map_area(pf_dev->cpp, 0, 0,
-                       NFP_PCIE_QUEUE(0), NFP_QCP_QUEUE_AREA_SZ,
+                       addr, NFP_QCP_QUEUE_AREA_SZ,
                        &pf_dev->hwqueues_area);
        if (pf_dev->hw_queues == NULL) {
                PMD_INIT_LOG(ERR, "nfp_rtsym_map fails for net.qc");
@@ -995,6 +1013,10 @@ nfp_pf_pci_probe(struct rte_pci_driver *pci_drv __rte_unused,
 }
 
 static const struct rte_pci_id pci_id_nfp_pf_net_map[] = {
+       {
+               RTE_PCI_DEVICE(PCI_VENDOR_ID_NETRONOME,
+                              PCI_DEVICE_ID_NFP3800_PF_NIC)
+       },
        {
                RTE_PCI_DEVICE(PCI_VENDOR_ID_NETRONOME,
                               PCI_DEVICE_ID_NFP4000_PF_NIC)
index d0fa1df24d9bb79b49e9e745795d5223e413a37a..c46ee0f91342fbd3f4e17b71aef563d70ead7090 100644 (file)
@@ -327,11 +327,12 @@ nfp_netvf_init(struct rte_eth_dev *eth_dev)
 
        /* Work out where in the BAR the queues start. */
        switch (pci_dev->id.device_id) {
+       case PCI_DEVICE_ID_NFP3800_VF_NIC:
        case PCI_DEVICE_ID_NFP6000_VF_NIC:
                start_q = nn_cfg_readl(hw, NFP_NET_CFG_START_TXQ);
-               tx_bar_off = (uint64_t)start_q * NFP_QCP_QUEUE_ADDR_SZ;
+               tx_bar_off = nfp_pci_queue(pci_dev, start_q);
                start_q = nn_cfg_readl(hw, NFP_NET_CFG_START_RXQ);
-               rx_bar_off = (uint64_t)start_q * NFP_QCP_QUEUE_ADDR_SZ;
+               rx_bar_off = nfp_pci_queue(pci_dev, start_q);
                break;
        default:
                PMD_DRV_LOG(ERR, "nfp_net: no device ID matching");
@@ -456,6 +457,10 @@ dev_err_ctrl_map:
 }
 
 static const struct rte_pci_id pci_id_nfp_vf_net_map[] = {
+       {
+               RTE_PCI_DEVICE(PCI_VENDOR_ID_NETRONOME,
+                              PCI_DEVICE_ID_NFP3800_VF_NIC)
+       },
        {
                RTE_PCI_DEVICE(PCI_VENDOR_ID_NETRONOME,
                               PCI_DEVICE_ID_NFP6000_VF_NIC)
index 720d3989e601c551c86366f3baac78c5a8b38e6d..a04a68f546b82b311a31b23d81fc571486fa0578 100644 (file)
@@ -214,7 +214,7 @@ void nfp_cpp_free(struct nfp_cpp *cpp);
  * @return             true if model is in the NFP6000 family, false otherwise.
  */
 #define NFP_CPP_MODEL_IS_6000(model)                \
-               ((NFP_CPP_MODEL_CHIP_of(model) >= 0x4000) && \
+               ((NFP_CPP_MODEL_CHIP_of(model) >= 0x3800) && \
                (NFP_CPP_MODEL_CHIP_of(model) < 0x7000))
 
 /*
index 67946891ab0584a48c2eb79eb4eb934d7b6fdfda..f8f3c372ac160a7b2d6cd1fc04de12efe078e890 100644 (file)
@@ -266,6 +266,7 @@ __nfp_eth_read_ports(struct nfp_nsp *nsp)
        struct nfp_eth_table *table;
        uint32_t table_sz;
        int i, j, ret, cnt = 0;
+       const struct rte_ether_addr *mac;
 
        entries = malloc(NSP_ETH_TABLE_SIZE);
        if (!entries)
@@ -278,9 +279,15 @@ __nfp_eth_read_ports(struct nfp_nsp *nsp)
                goto err;
        }
 
-       for (i = 0; i < NSP_ETH_MAX_COUNT; i++)
-               if (entries[i].port & NSP_ETH_PORT_LANES_MASK)
+       /* The NFP3800 NIC support 8 ports, but only 2 ports are valid,
+        * the rest 6 ports mac are all 0, ensure we don't use these port
+        */
+       for (i = 0; i < NSP_ETH_MAX_COUNT; i++) {
+               mac = (const struct rte_ether_addr *)entries[i].mac_addr;
+               if ((entries[i].port & NSP_ETH_PORT_LANES_MASK) &&
+                               (!rte_is_zero_ether_addr(mac)))
                        cnt++;
+       }
 
        /* Some versions of flash will give us 0 instead of port count. For
         * those that give a port count, verify it against the value calculated
@@ -299,10 +306,13 @@ __nfp_eth_read_ports(struct nfp_nsp *nsp)
 
        memset(table, 0, table_sz);
        table->count = cnt;
-       for (i = 0, j = 0; i < NSP_ETH_MAX_COUNT; i++)
-               if (entries[i].port & NSP_ETH_PORT_LANES_MASK)
+       for (i = 0, j = 0; i < NSP_ETH_MAX_COUNT; i++) {
+               mac = (const struct rte_ether_addr *)entries[i].mac_addr;
+               if ((entries[i].port & NSP_ETH_PORT_LANES_MASK) &&
+                               (!rte_is_zero_ether_addr(mac)))
                        nfp_eth_port_translate(nsp, &entries[i], i,
-                                              &table->ports[j++]);
+                                       &table->ports[j++]);
+       }
 
        nfp_eth_calc_port_geometry(table);
        for (i = 0; i < (int)table->count; i++)