2017-04-21 10:43:26 +00:00
|
|
|
/*
|
2021-09-18 08:05:45 +00:00
|
|
|
* Copyright (C) 2017-2021 THL A29 Limited, a Tencent company.
|
2017-04-21 10:43:26 +00:00
|
|
|
* 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 COPYRIGHT HOLDERS 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 COPYRIGHT OWNER 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.
|
|
|
|
*
|
|
|
|
*/
|
2017-08-28 08:51:06 +00:00
|
|
|
#include <assert.h>
|
2018-08-17 17:45:11 +00:00
|
|
|
#include <unistd.h>
|
2019-03-29 02:42:02 +00:00
|
|
|
#include <sys/mman.h>
|
|
|
|
#include <errno.h>
|
2017-04-21 10:43:26 +00:00
|
|
|
|
|
|
|
#include <rte_common.h>
|
|
|
|
#include <rte_byteorder.h>
|
|
|
|
#include <rte_log.h>
|
|
|
|
#include <rte_memory.h>
|
|
|
|
#include <rte_memcpy.h>
|
|
|
|
#include <rte_memzone.h>
|
|
|
|
#include <rte_config.h>
|
|
|
|
#include <rte_eal.h>
|
|
|
|
#include <rte_pci.h>
|
|
|
|
#include <rte_mbuf.h>
|
|
|
|
#include <rte_memory.h>
|
|
|
|
#include <rte_lcore.h>
|
|
|
|
#include <rte_launch.h>
|
|
|
|
#include <rte_ethdev.h>
|
|
|
|
#include <rte_debug.h>
|
|
|
|
#include <rte_common.h>
|
|
|
|
#include <rte_ether.h>
|
|
|
|
#include <rte_malloc.h>
|
|
|
|
#include <rte_cycles.h>
|
|
|
|
#include <rte_timer.h>
|
|
|
|
#include <rte_thash.h>
|
2017-05-06 13:52:25 +00:00
|
|
|
#include <rte_ip.h>
|
|
|
|
#include <rte_tcp.h>
|
|
|
|
#include <rte_udp.h>
|
2019-09-04 13:17:10 +00:00
|
|
|
#include <rte_eth_bond.h>
|
2022-07-05 06:50:22 +00:00
|
|
|
#include <rte_eth_bond_8023ad.h>
|
2017-04-21 10:43:26 +00:00
|
|
|
|
|
|
|
#include "ff_dpdk_if.h"
|
|
|
|
#include "ff_dpdk_pcap.h"
|
|
|
|
#include "ff_dpdk_kni.h"
|
|
|
|
#include "ff_config.h"
|
|
|
|
#include "ff_veth.h"
|
|
|
|
#include "ff_host_interface.h"
|
2017-05-23 15:13:49 +00:00
|
|
|
#include "ff_msg.h"
|
|
|
|
#include "ff_api.h"
|
2019-04-01 06:54:36 +00:00
|
|
|
#include "ff_memory.h"
|
2017-04-21 10:43:26 +00:00
|
|
|
|
2018-06-21 09:51:57 +00:00
|
|
|
#ifdef FF_KNI
|
2017-06-12 08:12:22 +00:00
|
|
|
#define KNI_MBUF_MAX 2048
|
|
|
|
#define KNI_QUEUE_SIZE 2048
|
|
|
|
|
2019-07-17 09:31:47 +00:00
|
|
|
int enable_kni;
|
2017-05-02 10:05:26 +00:00
|
|
|
static int kni_accept;
|
2020-05-14 10:30:34 +00:00
|
|
|
static int knictl_action = FF_KNICTL_ACTION_DEFAULT;
|
2018-06-21 09:39:17 +00:00
|
|
|
#endif
|
2017-04-21 10:43:26 +00:00
|
|
|
|
2017-09-04 03:42:18 +00:00
|
|
|
static int numa_on;
|
|
|
|
|
2018-08-17 17:45:11 +00:00
|
|
|
static unsigned idle_sleep;
|
2019-06-27 09:44:19 +00:00
|
|
|
static unsigned pkt_tx_delay;
|
2021-09-02 04:04:32 +00:00
|
|
|
static uint64_t usr_cb_tsc;
|
2018-08-17 17:45:11 +00:00
|
|
|
|
2017-04-21 10:43:26 +00:00
|
|
|
static struct rte_timer freebsd_clock;
|
|
|
|
|
|
|
|
// Mellanox Linux's driver key
|
|
|
|
static uint8_t default_rsskey_40bytes[40] = {
|
|
|
|
0xd1, 0x81, 0xc6, 0x2c, 0xf7, 0xf4, 0xdb, 0x5b,
|
|
|
|
0x19, 0x83, 0xa2, 0xfc, 0x94, 0x3e, 0x1a, 0xdb,
|
|
|
|
0xd9, 0x38, 0x9e, 0x6b, 0xd1, 0x03, 0x9c, 0x2c,
|
|
|
|
0xa7, 0x44, 0x99, 0xad, 0x59, 0x3d, 0x56, 0xd9,
|
|
|
|
0xf3, 0x25, 0x3c, 0x06, 0x2a, 0xdc, 0x1f, 0xfc
|
|
|
|
};
|
|
|
|
|
2019-07-25 03:27:30 +00:00
|
|
|
static uint8_t default_rsskey_52bytes[52] = {
|
|
|
|
0x44, 0x39, 0x79, 0x6b, 0xb5, 0x4c, 0x50, 0x23,
|
|
|
|
0xb6, 0x75, 0xea, 0x5b, 0x12, 0x4f, 0x9f, 0x30,
|
|
|
|
0xb8, 0xa2, 0xc0, 0x3d, 0xdf, 0xdc, 0x4d, 0x02,
|
|
|
|
0xa0, 0x8c, 0x9b, 0x33, 0x4a, 0xf6, 0x4a, 0x4c,
|
|
|
|
0x05, 0xc6, 0xfa, 0x34, 0x39, 0x58, 0xd8, 0x55,
|
|
|
|
0x7d, 0x99, 0x58, 0x3a, 0xe1, 0x38, 0xc9, 0x2e,
|
|
|
|
0x81, 0x15, 0x03, 0x66
|
|
|
|
};
|
|
|
|
|
2020-08-29 16:55:08 +00:00
|
|
|
static uint8_t symmetric_rsskey[52] = {
|
|
|
|
0x6d, 0x5a, 0x6d, 0x5a, 0x6d, 0x5a, 0x6d, 0x5a,
|
|
|
|
0x6d, 0x5a, 0x6d, 0x5a, 0x6d, 0x5a, 0x6d, 0x5a,
|
|
|
|
0x6d, 0x5a, 0x6d, 0x5a, 0x6d, 0x5a, 0x6d, 0x5a,
|
|
|
|
0x6d, 0x5a, 0x6d, 0x5a, 0x6d, 0x5a, 0x6d, 0x5a,
|
|
|
|
0x6d, 0x5a, 0x6d, 0x5a, 0x6d, 0x5a, 0x6d, 0x5a,
|
|
|
|
0x6d, 0x5a, 0x6d, 0x5a, 0x6d, 0x5a, 0x6d, 0x5a,
|
|
|
|
0x6d, 0x5a, 0x6d, 0x5a
|
|
|
|
};
|
|
|
|
|
|
|
|
static int rsskey_len = sizeof(default_rsskey_40bytes);
|
|
|
|
static uint8_t *rsskey = default_rsskey_40bytes;
|
|
|
|
|
2019-04-01 06:54:36 +00:00
|
|
|
struct lcore_conf lcore_conf;
|
2017-04-21 10:43:26 +00:00
|
|
|
|
2019-04-01 06:54:36 +00:00
|
|
|
struct rte_mempool *pktmbuf_pool[NB_SOCKETS];
|
2017-04-21 10:43:26 +00:00
|
|
|
|
2021-01-22 09:12:05 +00:00
|
|
|
static pcblddr_func_t pcblddr_fun;
|
|
|
|
|
2017-11-09 13:22:13 +00:00
|
|
|
static struct rte_ring **dispatch_ring[RTE_MAX_ETHPORTS];
|
|
|
|
static dispatch_func_t packet_dispatcher;
|
2017-04-21 10:43:26 +00:00
|
|
|
|
2017-08-28 08:51:06 +00:00
|
|
|
static uint16_t rss_reta_size[RTE_MAX_ETHPORTS];
|
|
|
|
|
2019-09-04 13:17:10 +00:00
|
|
|
#define BOND_DRIVER_NAME "net_bonding"
|
|
|
|
|
2019-03-08 07:12:57 +00:00
|
|
|
static inline int send_single_packet(struct rte_mbuf *m, uint8_t port);
|
|
|
|
|
2017-05-23 15:13:49 +00:00
|
|
|
struct ff_msg_ring {
|
2019-07-25 14:33:42 +00:00
|
|
|
char ring_name[FF_MSG_NUM][RTE_RING_NAMESIZE];
|
2017-05-23 15:13:49 +00:00
|
|
|
/* ring[0] for lcore recv msg, other send */
|
|
|
|
/* ring[1] for lcore send msg, other read */
|
2019-07-25 14:33:42 +00:00
|
|
|
struct rte_ring *ring[FF_MSG_NUM];
|
2017-05-23 15:13:49 +00:00
|
|
|
} __rte_cache_aligned;
|
|
|
|
|
|
|
|
static struct ff_msg_ring msg_ring[RTE_MAX_LCORE];
|
|
|
|
static struct rte_mempool *message_pool;
|
2017-04-21 10:43:26 +00:00
|
|
|
static struct ff_dpdk_if_context *veth_ctx[RTE_MAX_ETHPORTS];
|
|
|
|
|
2018-08-16 12:29:12 +00:00
|
|
|
static struct ff_top_args ff_top_status;
|
|
|
|
static struct ff_traffic_args ff_traffic;
|
2017-04-21 10:43:26 +00:00
|
|
|
extern void ff_hardclock(void);
|
|
|
|
|
|
|
|
static void
|
2017-08-04 09:40:50 +00:00
|
|
|
ff_hardclock_job(__rte_unused struct rte_timer *timer,
|
2017-04-21 10:43:26 +00:00
|
|
|
__rte_unused void *arg) {
|
|
|
|
ff_hardclock();
|
2017-08-04 09:40:50 +00:00
|
|
|
ff_update_current_ts();
|
2017-04-21 10:43:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
struct ff_dpdk_if_context *
|
|
|
|
ff_dpdk_register_if(void *sc, void *ifp, struct ff_port_cfg *cfg)
|
|
|
|
{
|
|
|
|
struct ff_dpdk_if_context *ctx;
|
|
|
|
|
|
|
|
ctx = calloc(1, sizeof(struct ff_dpdk_if_context));
|
|
|
|
if (ctx == NULL)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
ctx->sc = sc;
|
|
|
|
ctx->ifp = ifp;
|
|
|
|
ctx->port_id = cfg->port_id;
|
2017-05-06 13:52:25 +00:00
|
|
|
ctx->hw_features = cfg->hw_features;
|
2017-04-21 10:43:26 +00:00
|
|
|
|
|
|
|
return ctx;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
ff_dpdk_deregister_if(struct ff_dpdk_if_context *ctx)
|
|
|
|
{
|
|
|
|
free(ctx);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
check_all_ports_link_status(void)
|
|
|
|
{
|
|
|
|
#define CHECK_INTERVAL 100 /* 100ms */
|
|
|
|
#define MAX_CHECK_TIME 90 /* 9s (90 * 100ms) in total */
|
|
|
|
|
2018-05-15 09:49:22 +00:00
|
|
|
uint16_t portid;
|
|
|
|
uint8_t count, all_ports_up, print_flag = 0;
|
2017-04-21 10:43:26 +00:00
|
|
|
struct rte_eth_link link;
|
|
|
|
|
|
|
|
printf("\nChecking link status");
|
|
|
|
fflush(stdout);
|
|
|
|
|
|
|
|
int i, nb_ports;
|
|
|
|
nb_ports = ff_global_cfg.dpdk.nb_ports;
|
|
|
|
for (count = 0; count <= MAX_CHECK_TIME; count++) {
|
|
|
|
all_ports_up = 1;
|
|
|
|
for (i = 0; i < nb_ports; i++) {
|
2018-05-15 09:49:22 +00:00
|
|
|
uint16_t portid = ff_global_cfg.dpdk.portid_list[i];
|
2017-04-21 10:43:26 +00:00
|
|
|
memset(&link, 0, sizeof(link));
|
|
|
|
rte_eth_link_get_nowait(portid, &link);
|
2017-09-14 11:51:07 +00:00
|
|
|
|
2017-04-21 10:43:26 +00:00
|
|
|
/* print link status if flag set */
|
|
|
|
if (print_flag == 1) {
|
|
|
|
if (link.link_status) {
|
|
|
|
printf("Port %d Link Up - speed %u "
|
|
|
|
"Mbps - %s\n", (int)portid,
|
|
|
|
(unsigned)link.link_speed,
|
|
|
|
(link.link_duplex == ETH_LINK_FULL_DUPLEX) ?
|
|
|
|
("full-duplex") : ("half-duplex\n"));
|
|
|
|
} else {
|
|
|
|
printf("Port %d Link Down\n", (int)portid);
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
/* clear all_ports_up flag if any link down */
|
|
|
|
if (link.link_status == 0) {
|
|
|
|
all_ports_up = 0;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* after finally printing all link status, get out */
|
|
|
|
if (print_flag == 1)
|
|
|
|
break;
|
|
|
|
|
|
|
|
if (all_ports_up == 0) {
|
|
|
|
printf(".");
|
|
|
|
fflush(stdout);
|
|
|
|
rte_delay_ms(CHECK_INTERVAL);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* set the print_flag if all ports up or timeout */
|
|
|
|
if (all_ports_up == 1 || count == (MAX_CHECK_TIME - 1)) {
|
|
|
|
print_flag = 1;
|
|
|
|
printf("done\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
init_lcore_conf(void)
|
|
|
|
{
|
2019-06-25 11:12:58 +00:00
|
|
|
uint8_t nb_dev_ports = rte_eth_dev_count_avail();
|
2017-09-14 11:51:07 +00:00
|
|
|
if (nb_dev_ports == 0) {
|
2017-04-21 10:43:26 +00:00
|
|
|
rte_exit(EXIT_FAILURE, "No probed ethernet devices\n");
|
|
|
|
}
|
2017-09-22 09:35:50 +00:00
|
|
|
|
2017-09-14 11:51:07 +00:00
|
|
|
if (ff_global_cfg.dpdk.max_portid >= nb_dev_ports) {
|
|
|
|
rte_exit(EXIT_FAILURE, "this machine doesn't have port %d.\n",
|
|
|
|
ff_global_cfg.dpdk.max_portid);
|
|
|
|
}
|
2017-04-21 10:43:26 +00:00
|
|
|
|
2017-09-22 09:35:50 +00:00
|
|
|
lcore_conf.port_cfgs = ff_global_cfg.dpdk.port_cfgs;
|
2017-04-21 10:43:26 +00:00
|
|
|
lcore_conf.proc_id = ff_global_cfg.dpdk.proc_id;
|
|
|
|
|
|
|
|
uint16_t socket_id = 0;
|
2017-09-04 03:42:18 +00:00
|
|
|
if (numa_on) {
|
2017-04-21 10:43:26 +00:00
|
|
|
socket_id = rte_lcore_to_socket_id(rte_lcore_id());
|
|
|
|
}
|
|
|
|
|
|
|
|
lcore_conf.socket_id = socket_id;
|
|
|
|
|
2017-09-14 11:51:07 +00:00
|
|
|
uint16_t lcore_id = ff_global_cfg.dpdk.proc_lcore[lcore_conf.proc_id];
|
2020-06-18 16:55:50 +00:00
|
|
|
if (!rte_lcore_is_enabled(lcore_id)) {
|
|
|
|
rte_exit(EXIT_FAILURE, "lcore %u unavailable\n", lcore_id);
|
|
|
|
}
|
|
|
|
|
2017-09-14 11:51:07 +00:00
|
|
|
int j;
|
|
|
|
for (j = 0; j < ff_global_cfg.dpdk.nb_ports; ++j) {
|
|
|
|
uint16_t port_id = ff_global_cfg.dpdk.portid_list[j];
|
|
|
|
struct ff_port_cfg *pconf = &ff_global_cfg.dpdk.port_cfgs[port_id];
|
|
|
|
|
|
|
|
int queueid = -1;
|
|
|
|
int i;
|
|
|
|
for (i = 0; i < pconf->nb_lcores; i++) {
|
|
|
|
if (pconf->lcore_list[i] == lcore_id) {
|
|
|
|
queueid = i;
|
|
|
|
}
|
2017-04-21 10:43:26 +00:00
|
|
|
}
|
2017-09-14 11:51:07 +00:00
|
|
|
if (queueid < 0) {
|
|
|
|
continue;
|
2017-04-21 10:43:26 +00:00
|
|
|
}
|
2017-09-14 11:51:07 +00:00
|
|
|
printf("lcore: %u, port: %u, queue: %u\n", lcore_id, port_id, queueid);
|
2017-04-21 10:43:26 +00:00
|
|
|
uint16_t nb_rx_queue = lcore_conf.nb_rx_queue;
|
|
|
|
lcore_conf.rx_queue_list[nb_rx_queue].port_id = port_id;
|
2017-09-14 11:51:07 +00:00
|
|
|
lcore_conf.rx_queue_list[nb_rx_queue].queue_id = queueid;
|
2017-04-21 10:43:26 +00:00
|
|
|
lcore_conf.nb_rx_queue++;
|
|
|
|
|
2017-09-14 11:51:07 +00:00
|
|
|
lcore_conf.tx_queue_id[port_id] = queueid;
|
2017-09-22 09:35:50 +00:00
|
|
|
lcore_conf.tx_port_id[lcore_conf.nb_tx_port] = port_id;
|
|
|
|
lcore_conf.nb_tx_port++;
|
2017-04-21 10:43:26 +00:00
|
|
|
|
2020-01-08 04:09:40 +00:00
|
|
|
/* Enable pcap dump */
|
|
|
|
if (ff_global_cfg.pcap.enable) {
|
2020-01-08 13:13:12 +00:00
|
|
|
ff_enable_pcap(ff_global_cfg.pcap.save_path, ff_global_cfg.pcap.snap_len);
|
2020-01-08 04:09:40 +00:00
|
|
|
}
|
|
|
|
|
2017-09-14 11:51:07 +00:00
|
|
|
lcore_conf.nb_queue_list[port_id] = pconf->nb_lcores;
|
2017-04-21 10:43:26 +00:00
|
|
|
}
|
|
|
|
|
2017-11-09 13:22:13 +00:00
|
|
|
if (lcore_conf.nb_rx_queue == 0) {
|
|
|
|
rte_exit(EXIT_FAILURE, "lcore %u has nothing to do\n", lcore_id);
|
|
|
|
}
|
|
|
|
|
2017-04-21 10:43:26 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
init_mem_pool(void)
|
|
|
|
{
|
|
|
|
uint8_t nb_ports = ff_global_cfg.dpdk.nb_ports;
|
|
|
|
uint32_t nb_lcores = ff_global_cfg.dpdk.nb_procs;
|
|
|
|
uint32_t nb_tx_queue = nb_lcores;
|
|
|
|
uint32_t nb_rx_queue = lcore_conf.nb_rx_queue * nb_lcores;
|
2019-09-04 13:17:10 +00:00
|
|
|
uint16_t max_portid = ff_global_cfg.dpdk.max_portid;
|
2017-04-21 10:43:26 +00:00
|
|
|
|
2019-06-27 04:12:59 +00:00
|
|
|
unsigned nb_mbuf = RTE_ALIGN_CEIL (
|
2019-09-04 13:17:10 +00:00
|
|
|
(nb_rx_queue * (max_portid + 1) * 2 * RX_QUEUE_SIZE +
|
|
|
|
nb_ports * (max_portid + 1) * 2 * nb_lcores * MAX_PKT_BURST +
|
|
|
|
nb_ports * (max_portid + 1) * 2 * nb_tx_queue * TX_QUEUE_SIZE +
|
|
|
|
nb_lcores * MEMPOOL_CACHE_SIZE +
|
2018-06-21 09:51:57 +00:00
|
|
|
#ifdef FF_KNI
|
2019-09-04 13:17:10 +00:00
|
|
|
nb_ports * KNI_MBUF_MAX +
|
|
|
|
nb_ports * KNI_QUEUE_SIZE +
|
2018-06-21 09:51:57 +00:00
|
|
|
#endif
|
2019-09-04 13:17:10 +00:00
|
|
|
nb_lcores * nb_ports * DISPATCH_RING_SIZE),
|
2017-04-21 10:43:26 +00:00
|
|
|
(unsigned)8192);
|
|
|
|
|
|
|
|
unsigned socketid = 0;
|
|
|
|
uint16_t i, lcore_id;
|
|
|
|
char s[64];
|
|
|
|
|
2017-09-14 11:51:07 +00:00
|
|
|
for (i = 0; i < ff_global_cfg.dpdk.nb_procs; i++) {
|
|
|
|
lcore_id = ff_global_cfg.dpdk.proc_lcore[i];
|
2017-04-21 10:43:26 +00:00
|
|
|
if (numa_on) {
|
|
|
|
socketid = rte_lcore_to_socket_id(lcore_id);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (socketid >= NB_SOCKETS) {
|
|
|
|
rte_exit(EXIT_FAILURE, "Socket %d of lcore %u is out of range %d\n",
|
|
|
|
socketid, i, NB_SOCKETS);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (pktmbuf_pool[socketid] != NULL) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (rte_eal_process_type() == RTE_PROC_PRIMARY) {
|
|
|
|
snprintf(s, sizeof(s), "mbuf_pool_%d", socketid);
|
|
|
|
pktmbuf_pool[socketid] =
|
|
|
|
rte_pktmbuf_pool_create(s, nb_mbuf,
|
|
|
|
MEMPOOL_CACHE_SIZE, 0,
|
|
|
|
RTE_MBUF_DEFAULT_BUF_SIZE, socketid);
|
|
|
|
} else {
|
|
|
|
snprintf(s, sizeof(s), "mbuf_pool_%d", socketid);
|
|
|
|
pktmbuf_pool[socketid] = rte_mempool_lookup(s);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (pktmbuf_pool[socketid] == NULL) {
|
|
|
|
rte_exit(EXIT_FAILURE, "Cannot create mbuf pool on socket %d\n", socketid);
|
|
|
|
} else {
|
|
|
|
printf("create mbuf pool on socket %d\n", socketid);
|
|
|
|
}
|
2022-09-01 11:18:12 +00:00
|
|
|
|
2019-04-01 06:54:36 +00:00
|
|
|
#ifdef FF_USE_PAGE_ARRAY
|
2019-06-27 04:12:59 +00:00
|
|
|
nb_mbuf = RTE_ALIGN_CEIL (
|
2019-04-01 07:42:01 +00:00
|
|
|
nb_ports*nb_lcores*MAX_PKT_BURST +
|
|
|
|
nb_ports*nb_tx_queue*TX_QUEUE_SIZE +
|
|
|
|
nb_lcores*MEMPOOL_CACHE_SIZE,
|
|
|
|
(unsigned)4096);
|
|
|
|
ff_init_ref_pool(nb_mbuf, socketid);
|
2019-03-29 02:42:02 +00:00
|
|
|
#endif
|
2017-04-21 10:43:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-05-23 15:13:49 +00:00
|
|
|
static struct rte_ring *
|
|
|
|
create_ring(const char *name, unsigned count, int socket_id, unsigned flags)
|
|
|
|
{
|
|
|
|
struct rte_ring *ring;
|
|
|
|
|
2018-08-20 08:44:33 +00:00
|
|
|
if (name == NULL) {
|
|
|
|
rte_exit(EXIT_FAILURE, "create ring failed, no name!\n");
|
|
|
|
}
|
2017-05-23 15:13:49 +00:00
|
|
|
|
|
|
|
if (rte_eal_process_type() == RTE_PROC_PRIMARY) {
|
2018-08-20 08:44:33 +00:00
|
|
|
ring = rte_ring_create(name, count, socket_id, flags);
|
2017-05-23 15:13:49 +00:00
|
|
|
} else {
|
2018-08-20 08:44:33 +00:00
|
|
|
ring = rte_ring_lookup(name);
|
2017-05-23 15:13:49 +00:00
|
|
|
}
|
2018-08-20 08:44:33 +00:00
|
|
|
|
|
|
|
if (ring == NULL) {
|
|
|
|
rte_exit(EXIT_FAILURE, "create ring:%s failed!\n", name);
|
|
|
|
}
|
|
|
|
|
|
|
|
return ring;
|
2017-05-23 15:13:49 +00:00
|
|
|
}
|
|
|
|
|
2017-04-21 10:43:26 +00:00
|
|
|
static int
|
2017-11-09 13:22:13 +00:00
|
|
|
init_dispatch_ring(void)
|
2017-04-21 10:43:26 +00:00
|
|
|
{
|
2017-09-25 02:10:40 +00:00
|
|
|
int j;
|
2017-04-21 10:43:26 +00:00
|
|
|
char name_buf[RTE_RING_NAMESIZE];
|
2017-09-25 02:10:40 +00:00
|
|
|
int queueid;
|
2017-04-21 10:43:26 +00:00
|
|
|
|
|
|
|
unsigned socketid = lcore_conf.socket_id;
|
|
|
|
|
|
|
|
/* Create ring according to ports actually being used. */
|
2017-09-14 11:51:07 +00:00
|
|
|
int nb_ports = ff_global_cfg.dpdk.nb_ports;
|
2017-05-17 06:35:52 +00:00
|
|
|
for (j = 0; j < nb_ports; j++) {
|
2017-09-25 02:10:40 +00:00
|
|
|
uint16_t portid = ff_global_cfg.dpdk.portid_list[j];
|
|
|
|
struct ff_port_cfg *pconf = &ff_global_cfg.dpdk.port_cfgs[portid];
|
|
|
|
int nb_queues = pconf->nb_lcores;
|
2017-11-09 13:22:13 +00:00
|
|
|
if (dispatch_ring[portid] == NULL) {
|
2017-09-25 02:10:40 +00:00
|
|
|
snprintf(name_buf, RTE_RING_NAMESIZE, "ring_ptr_p%d", portid);
|
|
|
|
|
2017-11-09 13:22:13 +00:00
|
|
|
dispatch_ring[portid] = rte_zmalloc(name_buf,
|
|
|
|
sizeof(struct rte_ring *) * nb_queues,
|
|
|
|
RTE_CACHE_LINE_SIZE);
|
|
|
|
if (dispatch_ring[portid] == NULL) {
|
2017-09-25 02:10:40 +00:00
|
|
|
rte_exit(EXIT_FAILURE, "rte_zmalloc(%s (struct rte_ring*)) "
|
2017-11-09 13:22:13 +00:00
|
|
|
"failed\n", name_buf);
|
2017-09-25 02:10:40 +00:00
|
|
|
}
|
|
|
|
}
|
2017-04-21 10:43:26 +00:00
|
|
|
|
2017-09-25 02:10:40 +00:00
|
|
|
for(queueid = 0; queueid < nb_queues; ++queueid) {
|
2017-11-09 13:22:13 +00:00
|
|
|
snprintf(name_buf, RTE_RING_NAMESIZE, "dispatch_ring_p%d_q%d",
|
|
|
|
portid, queueid);
|
|
|
|
dispatch_ring[portid][queueid] = create_ring(name_buf,
|
|
|
|
DISPATCH_RING_SIZE, socketid, RING_F_SC_DEQ);
|
2017-04-21 10:43:26 +00:00
|
|
|
|
2017-11-09 13:22:13 +00:00
|
|
|
if (dispatch_ring[portid][queueid] == NULL)
|
2017-05-23 15:13:49 +00:00
|
|
|
rte_panic("create ring:%s failed!\n", name_buf);
|
2017-05-06 13:52:25 +00:00
|
|
|
|
2017-05-23 15:13:49 +00:00
|
|
|
printf("create ring:%s success, %u ring entries are now free!\n",
|
2017-11-09 13:22:13 +00:00
|
|
|
name_buf, rte_ring_free_count(dispatch_ring[portid][queueid]));
|
2017-04-21 10:43:26 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-05-23 15:13:49 +00:00
|
|
|
static void
|
|
|
|
ff_msg_init(struct rte_mempool *mp,
|
|
|
|
__attribute__((unused)) void *opaque_arg,
|
|
|
|
void *obj, __attribute__((unused)) unsigned i)
|
|
|
|
{
|
|
|
|
struct ff_msg *msg = (struct ff_msg *)obj;
|
2017-06-06 08:52:52 +00:00
|
|
|
msg->msg_type = FF_UNKNOWN;
|
2017-05-23 15:13:49 +00:00
|
|
|
msg->buf_addr = (char *)msg + sizeof(struct ff_msg);
|
|
|
|
msg->buf_len = mp->elt_size - sizeof(struct ff_msg);
|
2020-11-21 13:27:55 +00:00
|
|
|
msg->original_buf = NULL;
|
|
|
|
msg->original_buf_len = 0;
|
2017-05-23 15:13:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
init_msg_ring(void)
|
|
|
|
{
|
2019-07-25 09:19:28 +00:00
|
|
|
uint16_t i, j;
|
2017-05-23 15:13:49 +00:00
|
|
|
uint16_t nb_procs = ff_global_cfg.dpdk.nb_procs;
|
|
|
|
unsigned socketid = lcore_conf.socket_id;
|
|
|
|
|
|
|
|
/* Create message buffer pool */
|
|
|
|
if (rte_eal_process_type() == RTE_PROC_PRIMARY) {
|
|
|
|
message_pool = rte_mempool_create(FF_MSG_POOL,
|
|
|
|
MSG_RING_SIZE * 2 * nb_procs,
|
|
|
|
MAX_MSG_BUF_SIZE, MSG_RING_SIZE / 2, 0,
|
|
|
|
NULL, NULL, ff_msg_init, NULL,
|
|
|
|
socketid, 0);
|
|
|
|
} else {
|
|
|
|
message_pool = rte_mempool_lookup(FF_MSG_POOL);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (message_pool == NULL) {
|
|
|
|
rte_panic("Create msg mempool failed\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
for(i = 0; i < nb_procs; ++i) {
|
|
|
|
snprintf(msg_ring[i].ring_name[0], RTE_RING_NAMESIZE,
|
|
|
|
"%s%u", FF_MSG_RING_IN, i);
|
|
|
|
msg_ring[i].ring[0] = create_ring(msg_ring[i].ring_name[0],
|
|
|
|
MSG_RING_SIZE, socketid, RING_F_SP_ENQ | RING_F_SC_DEQ);
|
|
|
|
if (msg_ring[i].ring[0] == NULL)
|
|
|
|
rte_panic("create ring::%s failed!\n", msg_ring[i].ring_name[0]);
|
|
|
|
|
2019-07-25 14:33:42 +00:00
|
|
|
for (j = FF_SYSCTL; j < FF_MSG_NUM; j++) {
|
2019-07-25 09:19:28 +00:00
|
|
|
snprintf(msg_ring[i].ring_name[j], RTE_RING_NAMESIZE,
|
|
|
|
"%s%u_%u", FF_MSG_RING_OUT, i, j);
|
|
|
|
msg_ring[i].ring[j] = create_ring(msg_ring[i].ring_name[j],
|
|
|
|
MSG_RING_SIZE, socketid, RING_F_SP_ENQ | RING_F_SC_DEQ);
|
|
|
|
if (msg_ring[i].ring[j] == NULL)
|
|
|
|
rte_panic("create ring::%s failed!\n", msg_ring[i].ring_name[j]);
|
|
|
|
}
|
2017-05-23 15:13:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-06-21 09:39:17 +00:00
|
|
|
#ifdef FF_KNI
|
2020-05-14 10:30:34 +00:00
|
|
|
|
|
|
|
static enum FF_KNICTL_CMD get_kni_action(const char *c){
|
|
|
|
if (!c)
|
|
|
|
return FF_KNICTL_ACTION_DEFAULT;
|
|
|
|
if (0 == strcasecmp(c, "alltokni")){
|
|
|
|
return FF_KNICTL_ACTION_ALL_TO_KNI;
|
|
|
|
} else if (0 == strcasecmp(c, "alltoff")){
|
|
|
|
return FF_KNICTL_ACTION_ALL_TO_FF;
|
|
|
|
} else if (0 == strcasecmp(c, "default")){
|
|
|
|
return FF_KNICTL_ACTION_DEFAULT;
|
|
|
|
} else {
|
|
|
|
return FF_KNICTL_ACTION_DEFAULT;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-04-21 10:43:26 +00:00
|
|
|
static int
|
|
|
|
init_kni(void)
|
|
|
|
{
|
2019-06-25 11:12:58 +00:00
|
|
|
int nb_ports = rte_eth_dev_count_avail();
|
2017-05-02 10:05:26 +00:00
|
|
|
kni_accept = 0;
|
2017-04-21 10:43:26 +00:00
|
|
|
if(strcasecmp(ff_global_cfg.kni.method, "accept") == 0)
|
2017-05-02 10:05:26 +00:00
|
|
|
kni_accept = 1;
|
2017-04-21 10:43:26 +00:00
|
|
|
|
2020-05-14 10:30:34 +00:00
|
|
|
knictl_action = get_kni_action(ff_global_cfg.kni.kni_action);
|
|
|
|
|
2017-05-02 10:05:26 +00:00
|
|
|
ff_kni_init(nb_ports, ff_global_cfg.kni.tcp_port,
|
|
|
|
ff_global_cfg.kni.udp_port);
|
2017-04-21 10:43:26 +00:00
|
|
|
|
|
|
|
unsigned socket_id = lcore_conf.socket_id;
|
|
|
|
struct rte_mempool *mbuf_pool = pktmbuf_pool[socket_id];
|
|
|
|
|
|
|
|
nb_ports = ff_global_cfg.dpdk.nb_ports;
|
|
|
|
int i, ret;
|
|
|
|
for (i = 0; i < nb_ports; i++) {
|
2017-09-14 11:51:07 +00:00
|
|
|
uint16_t port_id = ff_global_cfg.dpdk.portid_list[i];
|
2017-06-12 08:12:22 +00:00
|
|
|
ff_kni_alloc(port_id, socket_id, mbuf_pool, KNI_QUEUE_SIZE);
|
2017-04-21 10:43:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2018-06-21 09:39:17 +00:00
|
|
|
#endif
|
2017-04-21 10:43:26 +00:00
|
|
|
|
2020-11-25 03:31:04 +00:00
|
|
|
//RSS reta update will failed when enable flow isolate
|
|
|
|
#ifndef FF_FLOW_ISOLATE
|
2017-08-28 08:51:06 +00:00
|
|
|
static void
|
2018-05-15 09:49:22 +00:00
|
|
|
set_rss_table(uint16_t port_id, uint16_t reta_size, uint16_t nb_queues)
|
2017-08-28 08:51:06 +00:00
|
|
|
{
|
|
|
|
if (reta_size == 0) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
int reta_conf_size = RTE_MAX(1, reta_size / RTE_RETA_GROUP_SIZE);
|
|
|
|
struct rte_eth_rss_reta_entry64 reta_conf[reta_conf_size];
|
|
|
|
|
|
|
|
/* config HW indirection table */
|
|
|
|
unsigned i, j, hash=0;
|
|
|
|
for (i = 0; i < reta_conf_size; i++) {
|
|
|
|
reta_conf[i].mask = ~0ULL;
|
|
|
|
for (j = 0; j < RTE_RETA_GROUP_SIZE; j++) {
|
|
|
|
reta_conf[i].reta[j] = hash++ % nb_queues;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (rte_eth_dev_rss_reta_update(port_id, reta_conf, reta_size)) {
|
|
|
|
rte_exit(EXIT_FAILURE, "port[%d], failed to update rss table\n",
|
|
|
|
port_id);
|
|
|
|
}
|
|
|
|
}
|
2020-11-25 03:31:04 +00:00
|
|
|
#endif
|
2017-08-28 08:51:06 +00:00
|
|
|
|
2017-04-21 10:43:26 +00:00
|
|
|
static int
|
|
|
|
init_port_start(void)
|
|
|
|
{
|
|
|
|
int nb_ports = ff_global_cfg.dpdk.nb_ports;
|
2018-06-10 12:48:48 +00:00
|
|
|
unsigned socketid = 0;
|
|
|
|
struct rte_mempool *mbuf_pool;
|
2019-09-04 13:17:10 +00:00
|
|
|
uint16_t i, j;
|
2017-04-21 10:43:26 +00:00
|
|
|
|
|
|
|
for (i = 0; i < nb_ports; i++) {
|
2019-09-04 13:17:10 +00:00
|
|
|
uint16_t port_id, u_port_id = ff_global_cfg.dpdk.portid_list[i];
|
|
|
|
struct ff_port_cfg *pconf = &ff_global_cfg.dpdk.port_cfgs[u_port_id];
|
2017-09-14 11:51:07 +00:00
|
|
|
uint16_t nb_queues = pconf->nb_lcores;
|
2017-04-21 10:43:26 +00:00
|
|
|
|
2022-08-27 12:48:45 +00:00
|
|
|
if (pconf->nb_slaves > 0) {
|
|
|
|
rte_eth_bond_8023ad_dedicated_queues_enable(u_port_id);
|
|
|
|
}
|
2019-09-04 13:17:10 +00:00
|
|
|
for (j=0; j<=pconf->nb_slaves; j++) {
|
|
|
|
if (j < pconf->nb_slaves) {
|
|
|
|
port_id = pconf->slave_portid_list[j];
|
|
|
|
printf("To init %s's %d'st slave port[%d]\n",
|
|
|
|
ff_global_cfg.dpdk.bond_cfgs->name,
|
|
|
|
j, port_id);
|
|
|
|
} else {
|
|
|
|
port_id = u_port_id;
|
|
|
|
}
|
2019-03-14 09:17:58 +00:00
|
|
|
|
2019-09-04 13:17:10 +00:00
|
|
|
struct rte_eth_dev_info dev_info;
|
|
|
|
struct rte_eth_conf port_conf = {0};
|
|
|
|
struct rte_eth_rxconf rxq_conf;
|
|
|
|
struct rte_eth_txconf txq_conf;
|
2017-04-21 10:43:26 +00:00
|
|
|
|
2020-06-18 16:55:50 +00:00
|
|
|
int ret = rte_eth_dev_info_get(port_id, &dev_info);
|
|
|
|
if (ret != 0)
|
|
|
|
rte_exit(EXIT_FAILURE,
|
|
|
|
"Error during getting device (port %u) info: %s\n",
|
|
|
|
port_id, strerror(-ret));
|
2017-04-21 10:43:26 +00:00
|
|
|
|
2019-09-04 13:17:10 +00:00
|
|
|
if (nb_queues > dev_info.max_rx_queues) {
|
|
|
|
rte_exit(EXIT_FAILURE, "num_procs[%d] bigger than max_rx_queues[%d]\n",
|
|
|
|
nb_queues,
|
|
|
|
dev_info.max_rx_queues);
|
|
|
|
}
|
2019-06-25 11:12:58 +00:00
|
|
|
|
2019-09-04 13:17:10 +00:00
|
|
|
if (nb_queues > dev_info.max_tx_queues) {
|
|
|
|
rte_exit(EXIT_FAILURE, "num_procs[%d] bigger than max_tx_queues[%d]\n",
|
|
|
|
nb_queues,
|
|
|
|
dev_info.max_tx_queues);
|
|
|
|
}
|
2017-05-06 13:52:25 +00:00
|
|
|
|
2020-06-18 16:55:50 +00:00
|
|
|
struct rte_ether_addr addr;
|
2019-09-04 13:37:56 +00:00
|
|
|
rte_eth_macaddr_get(port_id, &addr);
|
|
|
|
printf("Port %u MAC: %02" PRIx8 " %02" PRIx8 " %02" PRIx8
|
|
|
|
" %02" PRIx8 " %02" PRIx8 " %02" PRIx8 "\n",
|
|
|
|
(unsigned)port_id,
|
|
|
|
addr.addr_bytes[0], addr.addr_bytes[1],
|
|
|
|
addr.addr_bytes[2], addr.addr_bytes[3],
|
|
|
|
addr.addr_bytes[4], addr.addr_bytes[5]);
|
|
|
|
|
|
|
|
rte_memcpy(pconf->mac,
|
2020-06-18 16:55:50 +00:00
|
|
|
addr.addr_bytes, RTE_ETHER_ADDR_LEN);
|
2019-09-04 13:37:56 +00:00
|
|
|
|
2019-09-04 13:17:10 +00:00
|
|
|
/* Set RSS mode */
|
|
|
|
uint64_t default_rss_hf = ETH_RSS_PROTO_MASK;
|
|
|
|
port_conf.rxmode.mq_mode = ETH_MQ_RX_RSS;
|
|
|
|
port_conf.rx_adv_conf.rss_conf.rss_hf = default_rss_hf;
|
|
|
|
if (dev_info.hash_key_size == 52) {
|
2020-08-29 16:55:08 +00:00
|
|
|
rsskey = default_rsskey_52bytes;
|
|
|
|
rsskey_len = 52;
|
|
|
|
}
|
|
|
|
if (ff_global_cfg.dpdk.symmetric_rss) {
|
|
|
|
printf("Use symmetric Receive-side Scaling(RSS) key\n");
|
|
|
|
rsskey = symmetric_rsskey;
|
2019-09-04 13:17:10 +00:00
|
|
|
}
|
2020-08-29 16:55:08 +00:00
|
|
|
port_conf.rx_adv_conf.rss_conf.rss_key = rsskey;
|
|
|
|
port_conf.rx_adv_conf.rss_conf.rss_key_len = rsskey_len;
|
2019-09-04 13:17:10 +00:00
|
|
|
port_conf.rx_adv_conf.rss_conf.rss_hf &= dev_info.flow_type_rss_offloads;
|
|
|
|
if (port_conf.rx_adv_conf.rss_conf.rss_hf !=
|
|
|
|
ETH_RSS_PROTO_MASK) {
|
|
|
|
printf("Port %u modified RSS hash function based on hardware support,"
|
|
|
|
"requested:%#"PRIx64" configured:%#"PRIx64"\n",
|
|
|
|
port_id, default_rss_hf,
|
|
|
|
port_conf.rx_adv_conf.rss_conf.rss_hf);
|
2017-08-09 08:52:16 +00:00
|
|
|
}
|
2017-05-06 13:52:25 +00:00
|
|
|
|
2019-09-04 13:17:10 +00:00
|
|
|
if (dev_info.tx_offload_capa & DEV_TX_OFFLOAD_MBUF_FAST_FREE) {
|
|
|
|
port_conf.txmode.offloads |=
|
|
|
|
DEV_TX_OFFLOAD_MBUF_FAST_FREE;
|
|
|
|
}
|
2017-05-06 13:52:25 +00:00
|
|
|
|
2019-09-04 13:17:10 +00:00
|
|
|
/* Set Rx VLAN stripping */
|
|
|
|
if (ff_global_cfg.dpdk.vlan_strip) {
|
|
|
|
if (dev_info.rx_offload_capa & DEV_RX_OFFLOAD_VLAN_STRIP) {
|
|
|
|
port_conf.rxmode.offloads |= DEV_RX_OFFLOAD_VLAN_STRIP;
|
|
|
|
}
|
|
|
|
}
|
2017-05-06 13:52:25 +00:00
|
|
|
|
2019-09-04 13:17:10 +00:00
|
|
|
/* Enable HW CRC stripping */
|
|
|
|
port_conf.rxmode.offloads &= ~DEV_RX_OFFLOAD_KEEP_CRC;
|
2017-05-06 13:52:25 +00:00
|
|
|
|
2019-09-04 13:17:10 +00:00
|
|
|
/* FIXME: Enable TCP LRO ?*/
|
|
|
|
#if 0
|
|
|
|
if (dev_info.rx_offload_capa & DEV_RX_OFFLOAD_TCP_LRO) {
|
|
|
|
printf("LRO is supported\n");
|
|
|
|
port_conf.rxmode.offloads |= DEV_RX_OFFLOAD_TCP_LRO;
|
|
|
|
pconf->hw_features.rx_lro = 1;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* Set Rx checksum checking */
|
|
|
|
if ((dev_info.rx_offload_capa & DEV_RX_OFFLOAD_IPV4_CKSUM) &&
|
|
|
|
(dev_info.rx_offload_capa & DEV_RX_OFFLOAD_UDP_CKSUM) &&
|
|
|
|
(dev_info.rx_offload_capa & DEV_RX_OFFLOAD_TCP_CKSUM)) {
|
|
|
|
printf("RX checksum offload supported\n");
|
|
|
|
port_conf.rxmode.offloads |= DEV_RX_OFFLOAD_CHECKSUM;
|
|
|
|
pconf->hw_features.rx_csum = 1;
|
|
|
|
}
|
2017-05-06 13:52:25 +00:00
|
|
|
|
2019-11-04 05:00:17 +00:00
|
|
|
if (ff_global_cfg.dpdk.tx_csum_offoad_skip == 0) {
|
|
|
|
if ((dev_info.tx_offload_capa & DEV_TX_OFFLOAD_IPV4_CKSUM)) {
|
|
|
|
printf("TX ip checksum offload supported\n");
|
|
|
|
port_conf.txmode.offloads |= DEV_TX_OFFLOAD_IPV4_CKSUM;
|
|
|
|
pconf->hw_features.tx_csum_ip = 1;
|
|
|
|
}
|
2017-05-06 13:52:25 +00:00
|
|
|
|
2019-11-04 05:00:17 +00:00
|
|
|
if ((dev_info.tx_offload_capa & DEV_TX_OFFLOAD_UDP_CKSUM) &&
|
|
|
|
(dev_info.tx_offload_capa & DEV_TX_OFFLOAD_TCP_CKSUM)) {
|
|
|
|
printf("TX TCP&UDP checksum offload supported\n");
|
|
|
|
port_conf.txmode.offloads |= DEV_TX_OFFLOAD_UDP_CKSUM | DEV_TX_OFFLOAD_TCP_CKSUM;
|
|
|
|
pconf->hw_features.tx_csum_l4 = 1;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
printf("TX checksum offoad is disabled\n");
|
2019-09-04 13:17:10 +00:00
|
|
|
}
|
2017-08-28 08:51:06 +00:00
|
|
|
|
2019-09-04 13:17:10 +00:00
|
|
|
if (ff_global_cfg.dpdk.tso) {
|
|
|
|
if (dev_info.tx_offload_capa & DEV_TX_OFFLOAD_TCP_TSO) {
|
|
|
|
printf("TSO is supported\n");
|
|
|
|
port_conf.txmode.offloads |= DEV_TX_OFFLOAD_TCP_TSO;
|
|
|
|
pconf->hw_features.tx_tso = 1;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
printf("TSO is disabled\n");
|
|
|
|
}
|
2017-08-28 08:51:06 +00:00
|
|
|
|
2019-09-04 13:17:10 +00:00
|
|
|
if (dev_info.reta_size) {
|
|
|
|
/* reta size must be power of 2 */
|
|
|
|
assert((dev_info.reta_size & (dev_info.reta_size - 1)) == 0);
|
2017-04-21 10:43:26 +00:00
|
|
|
|
2019-09-04 13:17:10 +00:00
|
|
|
rss_reta_size[port_id] = dev_info.reta_size;
|
|
|
|
printf("port[%d]: rss table size: %d\n", port_id,
|
|
|
|
dev_info.reta_size);
|
|
|
|
}
|
2019-03-14 09:17:58 +00:00
|
|
|
|
2019-09-04 13:17:10 +00:00
|
|
|
if (rte_eal_process_type() != RTE_PROC_PRIMARY) {
|
|
|
|
continue;
|
2018-06-10 12:48:48 +00:00
|
|
|
}
|
|
|
|
|
2020-06-18 16:55:50 +00:00
|
|
|
ret = rte_eth_dev_configure(port_id, nb_queues, nb_queues, &port_conf);
|
2019-09-04 13:17:10 +00:00
|
|
|
if (ret != 0) {
|
2017-04-21 10:43:26 +00:00
|
|
|
return ret;
|
|
|
|
}
|
2019-09-04 13:17:10 +00:00
|
|
|
|
|
|
|
static uint16_t nb_rxd = RX_QUEUE_SIZE;
|
|
|
|
static uint16_t nb_txd = TX_QUEUE_SIZE;
|
|
|
|
ret = rte_eth_dev_adjust_nb_rx_tx_desc(port_id, &nb_rxd, &nb_txd);
|
|
|
|
if (ret < 0)
|
|
|
|
printf("Could not adjust number of descriptors "
|
|
|
|
"for port%u (%d)\n", (unsigned)port_id, ret);
|
|
|
|
|
|
|
|
uint16_t q;
|
|
|
|
for (q = 0; q < nb_queues; q++) {
|
|
|
|
if (numa_on) {
|
2022-08-19 06:42:19 +00:00
|
|
|
uint16_t lcore_id = lcore_conf.port_cfgs[u_port_id].lcore_list[q];
|
2019-09-04 13:17:10 +00:00
|
|
|
socketid = rte_lcore_to_socket_id(lcore_id);
|
|
|
|
}
|
|
|
|
mbuf_pool = pktmbuf_pool[socketid];
|
|
|
|
|
|
|
|
txq_conf = dev_info.default_txconf;
|
|
|
|
txq_conf.offloads = port_conf.txmode.offloads;
|
|
|
|
ret = rte_eth_tx_queue_setup(port_id, q, nb_txd,
|
|
|
|
socketid, &txq_conf);
|
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
rxq_conf = dev_info.default_rxconf;
|
|
|
|
rxq_conf.offloads = port_conf.rxmode.offloads;
|
|
|
|
ret = rte_eth_rx_queue_setup(port_id, q, nb_rxd,
|
|
|
|
socketid, &rxq_conf, mbuf_pool);
|
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (strncmp(dev_info.driver_name, BOND_DRIVER_NAME,
|
|
|
|
strlen(dev_info.driver_name)) == 0) {
|
2019-09-04 13:37:56 +00:00
|
|
|
|
|
|
|
rte_eth_macaddr_get(port_id, &addr);
|
|
|
|
printf("Port %u MAC: %02" PRIx8 " %02" PRIx8 " %02" PRIx8
|
|
|
|
" %02" PRIx8 " %02" PRIx8 " %02" PRIx8 "\n",
|
|
|
|
(unsigned)port_id,
|
|
|
|
addr.addr_bytes[0], addr.addr_bytes[1],
|
|
|
|
addr.addr_bytes[2], addr.addr_bytes[3],
|
|
|
|
addr.addr_bytes[4], addr.addr_bytes[5]);
|
|
|
|
|
|
|
|
rte_memcpy(pconf->mac,
|
2020-06-18 16:55:50 +00:00
|
|
|
addr.addr_bytes, RTE_ETHER_ADDR_LEN);
|
2019-09-04 13:37:56 +00:00
|
|
|
|
2019-09-04 13:17:10 +00:00
|
|
|
int mode, count, x;
|
|
|
|
uint16_t slaves[RTE_MAX_ETHPORTS], len = RTE_MAX_ETHPORTS;
|
|
|
|
|
|
|
|
mode = rte_eth_bond_mode_get(port_id);
|
2019-09-04 13:37:56 +00:00
|
|
|
printf("Port %u, bond mode:%d\n", port_id, mode);
|
2019-09-04 13:17:10 +00:00
|
|
|
|
|
|
|
count = rte_eth_bond_slaves_get(port_id, slaves, len);
|
2019-09-04 13:37:56 +00:00
|
|
|
printf("Port %u, %s's slave ports count:%d\n", port_id,
|
2019-09-04 13:17:10 +00:00
|
|
|
ff_global_cfg.dpdk.bond_cfgs->name, count);
|
|
|
|
for (x=0; x<count; x++) {
|
2019-09-04 13:37:56 +00:00
|
|
|
printf("Port %u, %s's slave port[%u]\n", port_id,
|
2019-09-04 13:17:10 +00:00
|
|
|
ff_global_cfg.dpdk.bond_cfgs->name, slaves[x]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = rte_eth_dev_start(port_id);
|
2017-04-21 10:43:26 +00:00
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
|
|
|
}
|
2020-11-25 03:31:04 +00:00
|
|
|
//RSS reta update will failed when enable flow isolate
|
|
|
|
#ifndef FF_FLOW_ISOLATE
|
2019-09-04 13:17:10 +00:00
|
|
|
if (nb_queues > 1) {
|
2021-02-05 08:42:07 +00:00
|
|
|
/*
|
|
|
|
* FIXME: modify RSS set to FDIR
|
|
|
|
*/
|
2019-09-04 13:17:10 +00:00
|
|
|
set_rss_table(port_id, dev_info.reta_size, nb_queues);
|
|
|
|
}
|
2020-11-25 03:31:04 +00:00
|
|
|
#endif
|
2017-08-28 08:51:06 +00:00
|
|
|
|
2019-09-04 13:17:10 +00:00
|
|
|
/* Enable RX in promiscuous mode for the Ethernet device. */
|
|
|
|
if (ff_global_cfg.dpdk.promiscuous) {
|
2020-06-18 16:55:50 +00:00
|
|
|
ret = rte_eth_promiscuous_enable(port_id);
|
|
|
|
if (ret == 0) {
|
2019-09-04 13:17:10 +00:00
|
|
|
printf("set port %u to promiscuous mode ok\n", port_id);
|
|
|
|
} else {
|
|
|
|
printf("set port %u to promiscuous mode error\n", port_id);
|
|
|
|
}
|
2017-04-21 10:43:26 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-08-04 09:40:50 +00:00
|
|
|
if (rte_eal_process_type() == RTE_PROC_PRIMARY) {
|
|
|
|
check_all_ports_link_status();
|
|
|
|
}
|
|
|
|
|
2017-04-21 10:43:26 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
2017-08-04 09:40:50 +00:00
|
|
|
init_clock(void)
|
2017-04-21 10:43:26 +00:00
|
|
|
{
|
|
|
|
rte_timer_subsystem_init();
|
|
|
|
uint64_t hz = rte_get_timer_hz();
|
2022-09-01 11:18:12 +00:00
|
|
|
uint64_t intrs = US_PER_S / ff_global_cfg.freebsd.hz;
|
|
|
|
uint64_t tsc = (hz + US_PER_S - 1) / US_PER_S * intrs;
|
2017-04-21 10:43:26 +00:00
|
|
|
|
|
|
|
rte_timer_init(&freebsd_clock);
|
|
|
|
rte_timer_reset(&freebsd_clock, tsc, PERIODICAL,
|
2017-08-04 09:40:50 +00:00
|
|
|
rte_lcore_id(), &ff_hardclock_job, NULL);
|
|
|
|
|
|
|
|
ff_update_current_ts();
|
2017-04-21 10:43:26 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2020-11-25 03:31:04 +00:00
|
|
|
#ifdef FF_FLOW_ISOLATE
|
|
|
|
/** Print a message out of a flow error. */
|
|
|
|
static int
|
|
|
|
port_flow_complain(struct rte_flow_error *error)
|
|
|
|
{
|
|
|
|
static const char *const errstrlist[] = {
|
|
|
|
[RTE_FLOW_ERROR_TYPE_NONE] = "no error",
|
|
|
|
[RTE_FLOW_ERROR_TYPE_UNSPECIFIED] = "cause unspecified",
|
|
|
|
[RTE_FLOW_ERROR_TYPE_HANDLE] = "flow rule (handle)",
|
|
|
|
[RTE_FLOW_ERROR_TYPE_ATTR_GROUP] = "group field",
|
|
|
|
[RTE_FLOW_ERROR_TYPE_ATTR_PRIORITY] = "priority field",
|
|
|
|
[RTE_FLOW_ERROR_TYPE_ATTR_INGRESS] = "ingress field",
|
|
|
|
[RTE_FLOW_ERROR_TYPE_ATTR_EGRESS] = "egress field",
|
|
|
|
[RTE_FLOW_ERROR_TYPE_ATTR_TRANSFER] = "transfer field",
|
|
|
|
[RTE_FLOW_ERROR_TYPE_ATTR] = "attributes structure",
|
|
|
|
[RTE_FLOW_ERROR_TYPE_ITEM_NUM] = "pattern length",
|
|
|
|
[RTE_FLOW_ERROR_TYPE_ITEM_SPEC] = "item specification",
|
|
|
|
[RTE_FLOW_ERROR_TYPE_ITEM_LAST] = "item specification range",
|
|
|
|
[RTE_FLOW_ERROR_TYPE_ITEM_MASK] = "item specification mask",
|
|
|
|
[RTE_FLOW_ERROR_TYPE_ITEM] = "specific pattern item",
|
|
|
|
[RTE_FLOW_ERROR_TYPE_ACTION_NUM] = "number of actions",
|
|
|
|
[RTE_FLOW_ERROR_TYPE_ACTION_CONF] = "action configuration",
|
|
|
|
[RTE_FLOW_ERROR_TYPE_ACTION] = "specific action",
|
|
|
|
};
|
|
|
|
const char *errstr;
|
|
|
|
char buf[32];
|
|
|
|
int err = rte_errno;
|
2022-09-01 11:18:12 +00:00
|
|
|
|
2020-11-25 03:31:04 +00:00
|
|
|
if ((unsigned int)error->type >= RTE_DIM(errstrlist) ||
|
|
|
|
!errstrlist[error->type])
|
|
|
|
errstr = "unknown type";
|
|
|
|
else
|
|
|
|
errstr = errstrlist[error->type];
|
|
|
|
printf("Caught error type %d (%s): %s%s: %s\n",
|
|
|
|
error->type, errstr,
|
|
|
|
error->cause ? (snprintf(buf, sizeof(buf), "cause: %p, ",
|
|
|
|
error->cause), buf) : "",
|
|
|
|
error->message ? error->message : "(no stated reason)",
|
|
|
|
rte_strerror(err));
|
|
|
|
return -err;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
port_flow_isolate(uint16_t port_id, int set)
|
|
|
|
{
|
|
|
|
struct rte_flow_error error;
|
2022-09-01 11:18:12 +00:00
|
|
|
|
2020-11-25 03:31:04 +00:00
|
|
|
/* Poisoning to make sure PMDs update it in case of error. */
|
|
|
|
memset(&error, 0x66, sizeof(error));
|
|
|
|
if (rte_flow_isolate(port_id, set, &error))
|
|
|
|
return port_flow_complain(&error);
|
|
|
|
printf("Ingress traffic on port %u is %s to the defined flow rules\n",
|
|
|
|
port_id,
|
|
|
|
set ? "now restricted" : "not restricted anymore");
|
|
|
|
return 0;
|
|
|
|
}
|
2020-11-26 13:32:07 +00:00
|
|
|
|
|
|
|
static int
|
2020-11-27 07:20:14 +00:00
|
|
|
create_tcp_flow(uint16_t port_id, uint16_t tcp_port) {
|
2020-11-26 13:32:07 +00:00
|
|
|
struct rte_flow_attr attr = {.ingress = 1};
|
|
|
|
struct ff_port_cfg *pconf = &ff_global_cfg.dpdk.port_cfgs[port_id];
|
|
|
|
int nb_queues = pconf->nb_lcores;
|
|
|
|
uint16_t queue[RTE_MAX_QUEUES_PER_PORT];
|
|
|
|
int i = 0, j = 0;
|
|
|
|
for (i = 0, j = 0; i < nb_queues; ++i)
|
|
|
|
queue[j++] = i;
|
|
|
|
struct rte_flow_action_rss rss = {
|
|
|
|
.types = ETH_RSS_NONFRAG_IPV4_TCP,
|
|
|
|
.key_len = rsskey_len,
|
|
|
|
.key = rsskey,
|
|
|
|
.queue_num = j,
|
|
|
|
.queue = queue,
|
|
|
|
};
|
|
|
|
|
|
|
|
struct rte_eth_dev_info dev_info;
|
|
|
|
int ret = rte_eth_dev_info_get(port_id, &dev_info);
|
|
|
|
if (ret != 0)
|
|
|
|
rte_exit(EXIT_FAILURE, "Error during getting device (port %u) info: %s\n", port_id, strerror(-ret));
|
|
|
|
|
|
|
|
struct rte_flow_item pattern[3];
|
|
|
|
struct rte_flow_action action[2];
|
|
|
|
struct rte_flow_item_tcp tcp_spec;
|
|
|
|
struct rte_flow_item_tcp tcp_mask = {
|
|
|
|
.hdr = {
|
|
|
|
.src_port = RTE_BE16(0x0000),
|
|
|
|
.dst_port = RTE_BE16(0xffff),
|
|
|
|
},
|
|
|
|
};
|
|
|
|
struct rte_flow_error error;
|
|
|
|
|
|
|
|
memset(pattern, 0, sizeof(pattern));
|
|
|
|
memset(action, 0, sizeof(action));
|
|
|
|
|
|
|
|
/* set the dst ipv4 packet to the required value */
|
|
|
|
pattern[0].type = RTE_FLOW_ITEM_TYPE_IPV4;
|
|
|
|
|
|
|
|
memset(&tcp_spec, 0, sizeof(struct rte_flow_item_tcp));
|
|
|
|
tcp_spec.hdr.dst_port = rte_cpu_to_be_16(tcp_port);
|
|
|
|
pattern[1].type = RTE_FLOW_ITEM_TYPE_TCP;
|
|
|
|
pattern[1].spec = &tcp_spec;
|
|
|
|
pattern[1].mask = &tcp_mask;
|
|
|
|
|
|
|
|
/* end the pattern array */
|
|
|
|
pattern[2].type = RTE_FLOW_ITEM_TYPE_END;
|
|
|
|
|
|
|
|
/* create the action */
|
|
|
|
action[0].type = RTE_FLOW_ACTION_TYPE_RSS;
|
|
|
|
action[0].conf = &rss;
|
|
|
|
action[1].type = RTE_FLOW_ACTION_TYPE_END;
|
|
|
|
|
|
|
|
struct rte_flow *flow;
|
|
|
|
/* validate and create the flow rule */
|
|
|
|
if (!rte_flow_validate(port_id, &attr, pattern, action, &error)) {
|
|
|
|
flow = rte_flow_create(port_id, &attr, pattern, action, &error);
|
|
|
|
if (!flow) {
|
|
|
|
return port_flow_complain(&error);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
memset(pattern, 0, sizeof(pattern));
|
|
|
|
|
|
|
|
/* set the dst ipv4 packet to the required value */
|
|
|
|
pattern[0].type = RTE_FLOW_ITEM_TYPE_IPV4;
|
|
|
|
|
|
|
|
struct rte_flow_item_tcp tcp_src_mask = {
|
|
|
|
.hdr = {
|
|
|
|
.src_port = RTE_BE16(0xffff),
|
|
|
|
.dst_port = RTE_BE16(0x0000),
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
memset(&tcp_spec, 0, sizeof(struct rte_flow_item_tcp));
|
|
|
|
tcp_spec.hdr.src_port = rte_cpu_to_be_16(tcp_port);
|
|
|
|
pattern[1].type = RTE_FLOW_ITEM_TYPE_TCP;
|
|
|
|
pattern[1].spec = &tcp_spec;
|
|
|
|
pattern[1].mask = &tcp_src_mask;
|
|
|
|
|
|
|
|
/* end the pattern array */
|
|
|
|
pattern[2].type = RTE_FLOW_ITEM_TYPE_END;
|
|
|
|
|
|
|
|
/* validate and create the flow rule */
|
|
|
|
if (!rte_flow_validate(port_id, &attr, pattern, action, &error)) {
|
|
|
|
flow = rte_flow_create(port_id, &attr, pattern, action, &error);
|
|
|
|
if (!flow) {
|
|
|
|
return port_flow_complain(&error);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
2020-11-27 07:17:20 +00:00
|
|
|
init_flow(uint16_t port_id, uint16_t tcp_port) {
|
2020-11-26 13:32:07 +00:00
|
|
|
// struct ff_flow_cfg fcfg = ff_global_cfg.dpdk.flow_cfgs[0];
|
|
|
|
|
|
|
|
// int i;
|
|
|
|
// for (i = 0; i < fcfg.nb_port; i++) {
|
|
|
|
// if(!create_tcp_flow(fcfg.port_id, fcfg.tcp_ports[i])) {
|
|
|
|
// return 0;
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
|
|
|
|
if(!create_tcp_flow(port_id, tcp_port)) {
|
|
|
|
rte_exit(EXIT_FAILURE, "create tcp flow failed\n");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* ARP rule */
|
|
|
|
struct rte_flow_attr attr = {.ingress = 1};
|
|
|
|
struct rte_flow_action_queue queue = {.index = 0};
|
|
|
|
|
|
|
|
struct rte_flow_item pattern_[2];
|
|
|
|
struct rte_flow_action action[2];
|
|
|
|
struct rte_flow_item_eth eth_type = {.type = RTE_BE16(0x0806)};
|
|
|
|
struct rte_flow_item_eth eth_mask = {
|
|
|
|
.type = RTE_BE16(0xffff)
|
|
|
|
};
|
|
|
|
|
|
|
|
memset(pattern_, 0, sizeof(pattern_));
|
|
|
|
memset(action, 0, sizeof(action));
|
|
|
|
|
|
|
|
pattern_[0].type = RTE_FLOW_ITEM_TYPE_ETH;
|
|
|
|
pattern_[0].spec = ð_type;
|
|
|
|
pattern_[0].mask = ð_mask;
|
|
|
|
|
|
|
|
pattern_[1].type = RTE_FLOW_ITEM_TYPE_END;
|
|
|
|
|
|
|
|
/* create the action */
|
|
|
|
action[0].type = RTE_FLOW_ACTION_TYPE_QUEUE;
|
|
|
|
action[0].conf = &queue;
|
|
|
|
action[1].type = RTE_FLOW_ACTION_TYPE_END;
|
|
|
|
|
|
|
|
struct rte_flow *flow;
|
|
|
|
struct rte_flow_error error;
|
|
|
|
/* validate and create the flow rule */
|
|
|
|
if (!rte_flow_validate(port_id, &attr, pattern_, action, &error)) {
|
|
|
|
flow = rte_flow_create(port_id, &attr, pattern_, action, &error);
|
|
|
|
if (!flow) {
|
|
|
|
return port_flow_complain(&error);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2020-11-25 03:31:04 +00:00
|
|
|
#endif
|
|
|
|
|
2017-04-21 10:43:26 +00:00
|
|
|
int
|
|
|
|
ff_dpdk_init(int argc, char **argv)
|
|
|
|
{
|
|
|
|
if (ff_global_cfg.dpdk.nb_procs < 1 ||
|
|
|
|
ff_global_cfg.dpdk.nb_procs > RTE_MAX_LCORE ||
|
|
|
|
ff_global_cfg.dpdk.proc_id >= ff_global_cfg.dpdk.nb_procs ||
|
2017-05-15 03:27:44 +00:00
|
|
|
ff_global_cfg.dpdk.proc_id < 0) {
|
2017-04-21 10:43:26 +00:00
|
|
|
printf("param num_procs[%d] or proc_id[%d] error!\n",
|
|
|
|
ff_global_cfg.dpdk.nb_procs,
|
|
|
|
ff_global_cfg.dpdk.proc_id);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
int ret = rte_eal_init(argc, argv);
|
|
|
|
if (ret < 0) {
|
|
|
|
rte_exit(EXIT_FAILURE, "Error with EAL initialization\n");
|
|
|
|
}
|
|
|
|
|
2017-09-04 03:42:18 +00:00
|
|
|
numa_on = ff_global_cfg.dpdk.numa_on;
|
|
|
|
|
2018-08-17 17:45:11 +00:00
|
|
|
idle_sleep = ff_global_cfg.dpdk.idle_sleep;
|
2019-06-27 09:44:19 +00:00
|
|
|
pkt_tx_delay = ff_global_cfg.dpdk.pkt_tx_delay > BURST_TX_DRAIN_US ? \
|
|
|
|
BURST_TX_DRAIN_US : ff_global_cfg.dpdk.pkt_tx_delay;
|
2018-08-17 17:45:11 +00:00
|
|
|
|
2017-04-21 10:43:26 +00:00
|
|
|
init_lcore_conf();
|
|
|
|
|
|
|
|
init_mem_pool();
|
|
|
|
|
2017-11-09 13:22:13 +00:00
|
|
|
init_dispatch_ring();
|
2017-04-21 10:43:26 +00:00
|
|
|
|
2017-05-23 15:13:49 +00:00
|
|
|
init_msg_ring();
|
|
|
|
|
2018-06-21 09:39:17 +00:00
|
|
|
#ifdef FF_KNI
|
2017-04-21 10:43:26 +00:00
|
|
|
enable_kni = ff_global_cfg.kni.enable;
|
|
|
|
if (enable_kni) {
|
|
|
|
init_kni();
|
|
|
|
}
|
2018-06-21 09:39:17 +00:00
|
|
|
#endif
|
2017-04-21 10:43:26 +00:00
|
|
|
|
2019-04-01 06:54:36 +00:00
|
|
|
#ifdef FF_USE_PAGE_ARRAY
|
2019-04-01 07:42:01 +00:00
|
|
|
ff_mmap_init();
|
2019-03-29 02:42:02 +00:00
|
|
|
#endif
|
2022-09-01 11:18:12 +00:00
|
|
|
|
|
|
|
#ifdef FF_FLOW_ISOLATE
|
2020-11-25 03:31:04 +00:00
|
|
|
// run once in primary process
|
|
|
|
if (0 == lcore_conf.tx_queue_id[0]){
|
|
|
|
ret = port_flow_isolate(0, 1);
|
|
|
|
if (ret < 0)
|
|
|
|
rte_exit(EXIT_FAILURE, "init_port_isolate failed\n");
|
|
|
|
}
|
|
|
|
#endif
|
2022-09-01 11:18:12 +00:00
|
|
|
|
2017-04-21 10:43:26 +00:00
|
|
|
ret = init_port_start();
|
|
|
|
if (ret < 0) {
|
|
|
|
rte_exit(EXIT_FAILURE, "init_port_start failed\n");
|
|
|
|
}
|
|
|
|
|
2017-08-04 09:40:50 +00:00
|
|
|
init_clock();
|
2020-11-26 13:32:07 +00:00
|
|
|
#ifdef FF_FLOW_ISOLATE
|
2022-09-01 11:18:12 +00:00
|
|
|
//Only give a example usage: port_id=0, tcp_port= 80.
|
|
|
|
//Recommend:
|
2020-11-27 07:36:35 +00:00
|
|
|
//1. init_flow should replace `set_rss_table` in `init_port_start` loop, This can set all NIC's port_id_list instead only 0 device(port_id).
|
|
|
|
//2. using config options `tcp_port` replace magic number of 80
|
2020-11-27 07:17:20 +00:00
|
|
|
ret = init_flow(0, 80);
|
2020-11-26 13:32:07 +00:00
|
|
|
if (ret < 0) {
|
|
|
|
rte_exit(EXIT_FAILURE, "init_port_flow failed\n");
|
|
|
|
}
|
|
|
|
#endif
|
2017-04-21 10:43:26 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2017-05-06 13:52:25 +00:00
|
|
|
ff_veth_input(const struct ff_dpdk_if_context *ctx, struct rte_mbuf *pkt)
|
2017-04-21 10:43:26 +00:00
|
|
|
{
|
2017-05-06 13:52:25 +00:00
|
|
|
uint8_t rx_csum = ctx->hw_features.rx_csum;
|
|
|
|
if (rx_csum) {
|
|
|
|
if (pkt->ol_flags & (PKT_RX_IP_CKSUM_BAD | PKT_RX_L4_CKSUM_BAD)) {
|
2018-05-30 02:36:16 +00:00
|
|
|
rte_pktmbuf_free(pkt);
|
2017-05-06 13:52:25 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-04-21 10:43:26 +00:00
|
|
|
void *data = rte_pktmbuf_mtod(pkt, void*);
|
|
|
|
uint16_t len = rte_pktmbuf_data_len(pkt);
|
|
|
|
|
2017-05-06 13:52:25 +00:00
|
|
|
void *hdr = ff_mbuf_gethdr(pkt, pkt->pkt_len, data, len, rx_csum);
|
2017-04-21 10:43:26 +00:00
|
|
|
if (hdr == NULL) {
|
|
|
|
rte_pktmbuf_free(pkt);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-07-09 11:56:16 +00:00
|
|
|
if (pkt->ol_flags & PKT_RX_VLAN_STRIPPED) {
|
|
|
|
ff_mbuf_set_vlan_info(hdr, pkt->vlan_tci);
|
|
|
|
}
|
|
|
|
|
2017-06-14 07:25:35 +00:00
|
|
|
struct rte_mbuf *pn = pkt->next;
|
2017-04-21 10:43:26 +00:00
|
|
|
void *prev = hdr;
|
2017-06-14 07:25:35 +00:00
|
|
|
while(pn != NULL) {
|
2017-11-10 05:58:25 +00:00
|
|
|
data = rte_pktmbuf_mtod(pn, void*);
|
|
|
|
len = rte_pktmbuf_data_len(pn);
|
2017-04-21 10:43:26 +00:00
|
|
|
|
fix use after free issue in mbuf free
Two kinds of mbuf are used in f-stack: freebsd mbuf and dpdk mbuf.
freebsd mbufs are metadata used in freebsd stack, and their data
pointers (m_data) point to dpdk mbuf's data (buf_addr). And they have
their own chain, like this:
bsd_mbuf1 -> bsd_mbuf2 -> bsd_mbuf3
\ \ \
dpdk_mbuf1 -> dpdk_mbuf2 -> dpdk_mbuf3
Considering the map relationship,
- m_freem() is corresponding to rte_pktmbuf_free(), is to free the whole
chain of mbufs.
- m_free() is corresponding to rte_pktmbuf_free_seg(), is to free the
specified mbuf segment.
The current implementation in f-stack uses rte_pktmbuf_free() for
m_free(). This leads to mbufs, which are still in use, be freed
unexpectedly. For example, if the bsd_mbuf1 is trimed into zero length,
bsd will invoke m_free() to free the specified segment, however, the
whole mbuf chain is freed by calling rte_pktmbuf_free().
#0 rte_pktmbuf_free (m=0x22006fb480)
#1 in ff_dpdk_pktmbuf_free (m=0x22006fb480)
#2 in ff_mbuf_ext_free (m=0x7ffff7f82800, arg1=0x22006fb480, arg2=0x0)
#3 in mb_free_ext (m=0x7ffff7f82800)
#4 in m_free (m=0x7ffff7f82800)
#5 in sbcompress (sb=, m=0x7ffff7f82800, n=)
#6 in sbappendstream_locked (sb=, m=0x7ffff7f82800, flags=0)
The fix is straightforward. Use the correct API for segment free.
Reported-by: Yong-Hao Zou <yonghaoz1994@gmail.com>
Signed-off-by: Jianfeng Tan <henry.tjf@antgroup.com>
2020-11-27 06:45:00 +00:00
|
|
|
void *mb = ff_mbuf_get(prev, pn, data, len);
|
2017-04-21 10:43:26 +00:00
|
|
|
if (mb == NULL) {
|
|
|
|
ff_mbuf_free(hdr);
|
2017-06-14 07:25:35 +00:00
|
|
|
rte_pktmbuf_free(pkt);
|
2017-04-21 10:43:26 +00:00
|
|
|
return;
|
|
|
|
}
|
2017-06-14 07:25:35 +00:00
|
|
|
pn = pn->next;
|
2017-04-21 10:43:26 +00:00
|
|
|
prev = mb;
|
|
|
|
}
|
|
|
|
|
2017-05-06 13:52:25 +00:00
|
|
|
ff_veth_process_packet(ctx->ifp, hdr);
|
2017-04-21 10:43:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static enum FilterReturn
|
|
|
|
protocol_filter(const void *data, uint16_t len)
|
|
|
|
{
|
2020-06-18 16:55:50 +00:00
|
|
|
if(len < RTE_ETHER_ADDR_LEN)
|
2017-04-21 10:43:26 +00:00
|
|
|
return FILTER_UNKNOWN;
|
|
|
|
|
2020-06-18 16:55:50 +00:00
|
|
|
const struct rte_ether_hdr *hdr;
|
|
|
|
const struct rte_vlan_hdr *vlanhdr;
|
|
|
|
hdr = (const struct rte_ether_hdr *)data;
|
2019-07-29 06:47:11 +00:00
|
|
|
uint16_t ether_type = rte_be_to_cpu_16(hdr->ether_type);
|
2020-06-18 16:55:50 +00:00
|
|
|
data += RTE_ETHER_HDR_LEN;
|
|
|
|
len -= RTE_ETHER_HDR_LEN;
|
2019-07-09 11:56:16 +00:00
|
|
|
|
2020-06-18 16:55:50 +00:00
|
|
|
if (ether_type == RTE_ETHER_TYPE_VLAN) {
|
|
|
|
vlanhdr = (struct rte_vlan_hdr *)data;
|
2019-07-29 06:47:11 +00:00
|
|
|
ether_type = rte_be_to_cpu_16(vlanhdr->eth_proto);
|
2020-06-18 16:55:50 +00:00
|
|
|
data += sizeof(struct rte_vlan_hdr);
|
|
|
|
len -= sizeof(struct rte_vlan_hdr);
|
2019-07-09 11:56:16 +00:00
|
|
|
}
|
2017-04-21 10:43:26 +00:00
|
|
|
|
2020-06-18 16:55:50 +00:00
|
|
|
if(ether_type == RTE_ETHER_TYPE_ARP)
|
2017-04-21 10:43:26 +00:00
|
|
|
return FILTER_ARP;
|
|
|
|
|
2022-05-31 23:46:09 +00:00
|
|
|
#if (!defined(__FreeBSD__) && defined(INET6) ) || \
|
|
|
|
( defined(__FreeBSD__) && defined(INET6) && defined(FF_KNI))
|
2020-06-18 16:55:50 +00:00
|
|
|
if (ether_type == RTE_ETHER_TYPE_IPV6) {
|
2019-08-06 14:11:54 +00:00
|
|
|
return ff_kni_proto_filter(data,
|
|
|
|
len, ether_type);
|
2019-07-17 09:31:47 +00:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2018-06-21 09:39:17 +00:00
|
|
|
#ifndef FF_KNI
|
|
|
|
return FILTER_UNKNOWN;
|
|
|
|
#else
|
2017-04-21 10:43:26 +00:00
|
|
|
if (!enable_kni) {
|
|
|
|
return FILTER_UNKNOWN;
|
|
|
|
}
|
|
|
|
|
2020-06-18 16:55:50 +00:00
|
|
|
if(ether_type != RTE_ETHER_TYPE_IPV4)
|
2017-04-21 10:43:26 +00:00
|
|
|
return FILTER_UNKNOWN;
|
|
|
|
|
2019-08-06 14:11:54 +00:00
|
|
|
return ff_kni_proto_filter(data,
|
|
|
|
len, ether_type);
|
2018-06-21 09:39:17 +00:00
|
|
|
#endif
|
2017-04-21 10:43:26 +00:00
|
|
|
}
|
|
|
|
|
2017-12-01 08:55:31 +00:00
|
|
|
static inline void
|
|
|
|
pktmbuf_deep_attach(struct rte_mbuf *mi, const struct rte_mbuf *m)
|
|
|
|
{
|
|
|
|
struct rte_mbuf *md;
|
|
|
|
void *src, *dst;
|
|
|
|
|
|
|
|
dst = rte_pktmbuf_mtod(mi, void *);
|
|
|
|
src = rte_pktmbuf_mtod(m, void *);
|
|
|
|
|
|
|
|
mi->data_len = m->data_len;
|
|
|
|
rte_memcpy(dst, src, m->data_len);
|
|
|
|
|
|
|
|
mi->port = m->port;
|
|
|
|
mi->vlan_tci = m->vlan_tci;
|
|
|
|
mi->vlan_tci_outer = m->vlan_tci_outer;
|
|
|
|
mi->tx_offload = m->tx_offload;
|
|
|
|
mi->hash = m->hash;
|
|
|
|
mi->ol_flags = m->ol_flags;
|
|
|
|
mi->packet_type = m->packet_type;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* copied from rte_pktmbuf_clone */
|
|
|
|
static inline struct rte_mbuf *
|
|
|
|
pktmbuf_deep_clone(const struct rte_mbuf *md,
|
|
|
|
struct rte_mempool *mp)
|
|
|
|
{
|
|
|
|
struct rte_mbuf *mc, *mi, **prev;
|
|
|
|
uint32_t pktlen;
|
|
|
|
uint8_t nseg;
|
|
|
|
|
|
|
|
if (unlikely ((mc = rte_pktmbuf_alloc(mp)) == NULL))
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
mi = mc;
|
|
|
|
prev = &mi->next;
|
|
|
|
pktlen = md->pkt_len;
|
|
|
|
nseg = 0;
|
|
|
|
|
|
|
|
do {
|
|
|
|
nseg++;
|
|
|
|
pktmbuf_deep_attach(mi, md);
|
|
|
|
*prev = mi;
|
|
|
|
prev = &mi->next;
|
|
|
|
} while ((md = md->next) != NULL &&
|
|
|
|
(mi = rte_pktmbuf_alloc(mp)) != NULL);
|
|
|
|
|
|
|
|
*prev = NULL;
|
|
|
|
mc->nb_segs = nseg;
|
|
|
|
mc->pkt_len = pktlen;
|
|
|
|
|
|
|
|
/* Allocation of new indirect segment failed */
|
|
|
|
if (unlikely (mi == NULL)) {
|
|
|
|
rte_pktmbuf_free(mc);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
__rte_mbuf_sanity_check(mc, 1);
|
|
|
|
return mc;
|
|
|
|
}
|
|
|
|
|
2017-04-21 10:43:26 +00:00
|
|
|
static inline void
|
2018-05-15 09:49:22 +00:00
|
|
|
process_packets(uint16_t port_id, uint16_t queue_id, struct rte_mbuf **bufs,
|
2017-05-06 13:52:25 +00:00
|
|
|
uint16_t count, const struct ff_dpdk_if_context *ctx, int pkts_from_ring)
|
2017-04-21 10:43:26 +00:00
|
|
|
{
|
|
|
|
struct lcore_conf *qconf = &lcore_conf;
|
2017-11-09 13:22:13 +00:00
|
|
|
uint16_t nb_queues = qconf->nb_queue_list[port_id];
|
2017-04-21 10:43:26 +00:00
|
|
|
|
|
|
|
uint16_t i;
|
|
|
|
for (i = 0; i < count; i++) {
|
|
|
|
struct rte_mbuf *rtem = bufs[i];
|
|
|
|
|
2020-01-08 04:09:40 +00:00
|
|
|
if (unlikely( ff_global_cfg.pcap.enable)) {
|
2017-11-09 13:22:13 +00:00
|
|
|
if (!pkts_from_ring) {
|
2020-01-08 04:09:40 +00:00
|
|
|
ff_dump_packets( ff_global_cfg.pcap.save_path, rtem, ff_global_cfg.pcap.snap_len, ff_global_cfg.pcap.save_len);
|
2017-11-09 13:22:13 +00:00
|
|
|
}
|
2017-04-21 10:43:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void *data = rte_pktmbuf_mtod(rtem, void*);
|
|
|
|
uint16_t len = rte_pktmbuf_data_len(rtem);
|
|
|
|
|
2018-08-16 12:29:12 +00:00
|
|
|
if (!pkts_from_ring) {
|
2021-09-02 03:47:13 +00:00
|
|
|
ff_traffic.rx_packets += rtem->nb_segs;
|
|
|
|
ff_traffic.rx_bytes += rte_pktmbuf_pkt_len(rtem);
|
2018-08-16 12:29:12 +00:00
|
|
|
}
|
|
|
|
|
2017-11-09 13:22:13 +00:00
|
|
|
if (!pkts_from_ring && packet_dispatcher) {
|
2021-09-02 04:04:32 +00:00
|
|
|
uint64_t cur_tsc = rte_rdtsc();
|
2019-03-08 07:12:57 +00:00
|
|
|
int ret = (*packet_dispatcher)(data, &len, queue_id, nb_queues);
|
2021-09-02 04:04:32 +00:00
|
|
|
usr_cb_tsc += rte_rdtsc() - cur_tsc;
|
2019-03-08 07:12:57 +00:00
|
|
|
if (ret == FF_DISPATCH_RESPONSE) {
|
|
|
|
rte_pktmbuf_pkt_len(rtem) = rte_pktmbuf_data_len(rtem) = len;
|
2019-08-06 14:19:12 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* We have not support vlan out strip
|
|
|
|
*/
|
|
|
|
if (rtem->vlan_tci) {
|
2020-06-18 16:55:50 +00:00
|
|
|
data = rte_pktmbuf_prepend(rtem, sizeof(struct rte_vlan_hdr));
|
2019-08-06 14:19:12 +00:00
|
|
|
if (data != NULL) {
|
2020-06-18 16:55:50 +00:00
|
|
|
memmove(data, data + sizeof(struct rte_vlan_hdr), RTE_ETHER_HDR_LEN);
|
|
|
|
struct rte_ether_hdr *etherhdr = (struct rte_ether_hdr *)data;
|
|
|
|
struct rte_vlan_hdr *vlanhdr = (struct rte_vlan_hdr *)(data + RTE_ETHER_HDR_LEN);
|
2019-08-06 14:19:12 +00:00
|
|
|
vlanhdr->vlan_tci = rte_cpu_to_be_16(rtem->vlan_tci);
|
|
|
|
vlanhdr->eth_proto = etherhdr->ether_type;
|
2020-06-18 16:55:50 +00:00
|
|
|
etherhdr->ether_type = rte_cpu_to_be_16(RTE_ETHER_TYPE_VLAN);
|
2019-08-06 14:19:12 +00:00
|
|
|
}
|
|
|
|
}
|
2019-03-08 07:12:57 +00:00
|
|
|
send_single_packet(rtem, port_id);
|
|
|
|
continue;
|
|
|
|
}
|
2019-08-06 14:19:12 +00:00
|
|
|
|
2019-03-08 07:12:57 +00:00
|
|
|
if (ret == FF_DISPATCH_ERROR || ret >= nb_queues) {
|
2017-11-09 13:22:13 +00:00
|
|
|
rte_pktmbuf_free(rtem);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ret != queue_id) {
|
|
|
|
ret = rte_ring_enqueue(dispatch_ring[port_id][ret], rtem);
|
|
|
|
if (ret < 0)
|
|
|
|
rte_pktmbuf_free(rtem);
|
|
|
|
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-04-21 10:43:26 +00:00
|
|
|
enum FilterReturn filter = protocol_filter(data, len);
|
2019-07-17 09:31:47 +00:00
|
|
|
#ifdef INET6
|
|
|
|
if (filter == FILTER_ARP || filter == FILTER_NDP) {
|
|
|
|
#else
|
2017-05-02 10:05:26 +00:00
|
|
|
if (filter == FILTER_ARP) {
|
2019-07-17 09:31:47 +00:00
|
|
|
#endif
|
2017-04-21 10:43:26 +00:00
|
|
|
struct rte_mempool *mbuf_pool;
|
|
|
|
struct rte_mbuf *mbuf_clone;
|
2017-11-09 13:22:13 +00:00
|
|
|
if (!pkts_from_ring) {
|
|
|
|
uint16_t j;
|
|
|
|
for(j = 0; j < nb_queues; ++j) {
|
|
|
|
if(j == queue_id)
|
2017-04-21 10:43:26 +00:00
|
|
|
continue;
|
|
|
|
|
2017-09-04 03:42:18 +00:00
|
|
|
unsigned socket_id = 0;
|
|
|
|
if (numa_on) {
|
2017-11-09 13:22:13 +00:00
|
|
|
uint16_t lcore_id = qconf->port_cfgs[port_id].lcore_list[j];
|
2017-09-14 11:51:07 +00:00
|
|
|
socket_id = rte_lcore_to_socket_id(lcore_id);
|
2017-09-04 03:42:18 +00:00
|
|
|
}
|
|
|
|
mbuf_pool = pktmbuf_pool[socket_id];
|
2017-12-01 08:55:31 +00:00
|
|
|
mbuf_clone = pktmbuf_deep_clone(rtem, mbuf_pool);
|
2017-04-21 10:43:26 +00:00
|
|
|
if(mbuf_clone) {
|
2017-12-01 08:55:31 +00:00
|
|
|
int ret = rte_ring_enqueue(dispatch_ring[port_id][j],
|
|
|
|
mbuf_clone);
|
2017-04-21 10:43:26 +00:00
|
|
|
if (ret < 0)
|
|
|
|
rte_pktmbuf_free(mbuf_clone);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-06-21 09:39:17 +00:00
|
|
|
#ifdef FF_KNI
|
2017-04-21 10:43:26 +00:00
|
|
|
if (enable_kni && rte_eal_process_type() == RTE_PROC_PRIMARY) {
|
|
|
|
mbuf_pool = pktmbuf_pool[qconf->socket_id];
|
2017-12-01 08:55:31 +00:00
|
|
|
mbuf_clone = pktmbuf_deep_clone(rtem, mbuf_pool);
|
2017-04-21 10:43:26 +00:00
|
|
|
if(mbuf_clone) {
|
2017-07-04 03:45:35 +00:00
|
|
|
ff_kni_enqueue(port_id, mbuf_clone);
|
2017-04-21 10:43:26 +00:00
|
|
|
}
|
|
|
|
}
|
2018-06-21 09:39:17 +00:00
|
|
|
#endif
|
2017-05-06 13:52:25 +00:00
|
|
|
ff_veth_input(ctx, rtem);
|
2018-06-21 09:39:17 +00:00
|
|
|
#ifdef FF_KNI
|
2020-05-14 10:30:34 +00:00
|
|
|
} else if (enable_kni) {
|
|
|
|
if (knictl_action == FF_KNICTL_ACTION_ALL_TO_KNI){
|
|
|
|
ff_kni_enqueue(port_id, rtem);
|
|
|
|
} else if (knictl_action == FF_KNICTL_ACTION_ALL_TO_FF){
|
|
|
|
ff_veth_input(ctx, rtem);
|
|
|
|
} else if (knictl_action == FF_KNICTL_ACTION_DEFAULT){
|
|
|
|
if (enable_kni &&
|
|
|
|
((filter == FILTER_KNI && kni_accept) ||
|
|
|
|
(filter == FILTER_UNKNOWN && !kni_accept)) ) {
|
|
|
|
ff_kni_enqueue(port_id, rtem);
|
|
|
|
} else {
|
|
|
|
ff_veth_input(ctx, rtem);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
ff_veth_input(ctx, rtem);
|
|
|
|
}
|
2018-06-21 09:39:17 +00:00
|
|
|
#endif
|
2017-05-02 10:05:26 +00:00
|
|
|
} else {
|
2017-05-06 13:52:25 +00:00
|
|
|
ff_veth_input(ctx, rtem);
|
2017-04-21 10:43:26 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline int
|
2018-05-15 09:49:22 +00:00
|
|
|
process_dispatch_ring(uint16_t port_id, uint16_t queue_id,
|
2017-05-06 13:52:25 +00:00
|
|
|
struct rte_mbuf **pkts_burst, const struct ff_dpdk_if_context *ctx)
|
2017-04-21 10:43:26 +00:00
|
|
|
{
|
|
|
|
/* read packet from ring buf and to process */
|
2017-05-23 15:13:49 +00:00
|
|
|
uint16_t nb_rb;
|
2017-11-09 13:22:13 +00:00
|
|
|
nb_rb = rte_ring_dequeue_burst(dispatch_ring[port_id][queue_id],
|
2018-05-15 09:49:22 +00:00
|
|
|
(void **)pkts_burst, MAX_PKT_BURST, NULL);
|
2017-04-21 10:43:26 +00:00
|
|
|
|
2017-05-23 15:13:49 +00:00
|
|
|
if(nb_rb > 0) {
|
|
|
|
process_packets(port_id, queue_id, pkts_burst, nb_rb, ctx, 1);
|
|
|
|
}
|
|
|
|
|
2021-09-02 04:09:09 +00:00
|
|
|
return nb_rb;
|
2017-05-23 15:13:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static inline void
|
2017-11-14 08:34:53 +00:00
|
|
|
handle_sysctl_msg(struct ff_msg *msg)
|
2017-05-23 15:13:49 +00:00
|
|
|
{
|
|
|
|
int ret = ff_sysctl(msg->sysctl.name, msg->sysctl.namelen,
|
|
|
|
msg->sysctl.old, msg->sysctl.oldlenp, msg->sysctl.new,
|
|
|
|
msg->sysctl.newlen);
|
|
|
|
|
|
|
|
if (ret < 0) {
|
|
|
|
msg->result = errno;
|
|
|
|
} else {
|
|
|
|
msg->result = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-06-06 08:52:52 +00:00
|
|
|
static inline void
|
2017-11-14 08:34:53 +00:00
|
|
|
handle_ioctl_msg(struct ff_msg *msg)
|
2017-06-06 08:52:52 +00:00
|
|
|
{
|
|
|
|
int fd, ret;
|
2019-07-12 12:56:01 +00:00
|
|
|
#ifdef INET6
|
|
|
|
if (msg->msg_type == FF_IOCTL6) {
|
|
|
|
fd = ff_socket(AF_INET6, SOCK_DGRAM, 0);
|
|
|
|
} else
|
|
|
|
#endif
|
|
|
|
fd = ff_socket(AF_INET, SOCK_DGRAM, 0);
|
|
|
|
|
2017-06-06 08:52:52 +00:00
|
|
|
if (fd < 0) {
|
|
|
|
ret = -1;
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
|
2018-06-10 12:37:34 +00:00
|
|
|
ret = ff_ioctl_freebsd(fd, msg->ioctl.cmd, msg->ioctl.data);
|
2017-06-06 08:52:52 +00:00
|
|
|
|
|
|
|
ff_close(fd);
|
|
|
|
|
|
|
|
done:
|
|
|
|
if (ret < 0) {
|
|
|
|
msg->result = errno;
|
|
|
|
} else {
|
|
|
|
msg->result = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-06-14 07:25:35 +00:00
|
|
|
static inline void
|
2017-11-14 08:34:53 +00:00
|
|
|
handle_route_msg(struct ff_msg *msg)
|
2017-06-14 07:25:35 +00:00
|
|
|
{
|
2017-11-01 09:38:22 +00:00
|
|
|
int ret = ff_rtioctl(msg->route.fib, msg->route.data,
|
2017-06-14 07:25:35 +00:00
|
|
|
&msg->route.len, msg->route.maxlen);
|
2017-11-01 09:38:22 +00:00
|
|
|
if (ret < 0) {
|
|
|
|
msg->result = errno;
|
|
|
|
} else {
|
|
|
|
msg->result = 0;
|
|
|
|
}
|
2017-06-14 07:25:35 +00:00
|
|
|
}
|
|
|
|
|
2017-08-22 09:12:53 +00:00
|
|
|
static inline void
|
2017-11-14 08:34:53 +00:00
|
|
|
handle_top_msg(struct ff_msg *msg)
|
2017-08-22 09:12:53 +00:00
|
|
|
{
|
2018-08-17 09:42:21 +00:00
|
|
|
msg->top = ff_top_status;
|
2017-08-22 09:12:53 +00:00
|
|
|
msg->result = 0;
|
|
|
|
}
|
|
|
|
|
2017-11-01 09:38:22 +00:00
|
|
|
#ifdef FF_NETGRAPH
|
|
|
|
static inline void
|
2017-11-14 08:34:53 +00:00
|
|
|
handle_ngctl_msg(struct ff_msg *msg)
|
2017-11-01 09:38:22 +00:00
|
|
|
{
|
|
|
|
int ret = ff_ngctl(msg->ngctl.cmd, msg->ngctl.data);
|
|
|
|
if (ret < 0) {
|
|
|
|
msg->result = errno;
|
|
|
|
} else {
|
|
|
|
msg->result = 0;
|
|
|
|
msg->ngctl.ret = ret;
|
|
|
|
}
|
2017-11-14 08:34:53 +00:00
|
|
|
}
|
|
|
|
#endif
|
2017-11-01 09:38:22 +00:00
|
|
|
|
2017-11-14 08:34:53 +00:00
|
|
|
#ifdef FF_IPFW
|
|
|
|
static inline void
|
|
|
|
handle_ipfw_msg(struct ff_msg *msg)
|
|
|
|
{
|
|
|
|
int fd, ret;
|
|
|
|
fd = ff_socket(AF_INET, SOCK_RAW, IPPROTO_RAW);
|
|
|
|
if (fd < 0) {
|
|
|
|
ret = -1;
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (msg->ipfw.cmd) {
|
|
|
|
case FF_IPFW_GET:
|
2018-06-10 12:37:34 +00:00
|
|
|
ret = ff_getsockopt_freebsd(fd, msg->ipfw.level,
|
2017-11-14 08:34:53 +00:00
|
|
|
msg->ipfw.optname, msg->ipfw.optval,
|
|
|
|
msg->ipfw.optlen);
|
|
|
|
break;
|
|
|
|
case FF_IPFW_SET:
|
2018-06-10 12:37:34 +00:00
|
|
|
ret = ff_setsockopt_freebsd(fd, msg->ipfw.level,
|
2017-11-14 08:34:53 +00:00
|
|
|
msg->ipfw.optname, msg->ipfw.optval,
|
2022-09-01 11:18:12 +00:00
|
|
|
*(msg->ipfw.optlen));
|
2017-11-14 08:34:53 +00:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
ret = -1;
|
|
|
|
errno = ENOTSUP;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
ff_close(fd);
|
|
|
|
|
|
|
|
done:
|
|
|
|
if (ret < 0) {
|
|
|
|
msg->result = errno;
|
|
|
|
} else {
|
|
|
|
msg->result = 0;
|
|
|
|
}
|
2017-11-01 09:38:22 +00:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2018-08-16 12:29:12 +00:00
|
|
|
static inline void
|
|
|
|
handle_traffic_msg(struct ff_msg *msg)
|
|
|
|
{
|
|
|
|
msg->traffic = ff_traffic;
|
|
|
|
msg->result = 0;
|
|
|
|
}
|
|
|
|
|
2020-05-14 10:30:34 +00:00
|
|
|
#ifdef FF_KNI
|
|
|
|
static inline void
|
|
|
|
handle_knictl_msg(struct ff_msg *msg)
|
|
|
|
{
|
|
|
|
if (msg->knictl.kni_cmd == FF_KNICTL_CMD_SET){
|
|
|
|
switch (msg->knictl.kni_action){
|
|
|
|
case FF_KNICTL_ACTION_ALL_TO_FF: knictl_action = FF_KNICTL_ACTION_ALL_TO_FF; msg->result = 0; printf("new kni action: alltoff\n"); break;
|
|
|
|
case FF_KNICTL_ACTION_ALL_TO_KNI: knictl_action = FF_KNICTL_ACTION_ALL_TO_KNI; msg->result = 0; printf("new kni action: alltokni\n"); break;
|
|
|
|
case FF_KNICTL_ACTION_DEFAULT: knictl_action = FF_KNICTL_ACTION_DEFAULT; msg->result = 0; printf("new kni action: default\n"); break;
|
|
|
|
default: msg->result = -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (msg->knictl.kni_cmd == FF_KNICTL_CMD_GET){
|
|
|
|
msg->knictl.kni_action = knictl_action;
|
|
|
|
} else {
|
|
|
|
msg->result = -2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2017-05-23 15:13:49 +00:00
|
|
|
static inline void
|
2017-11-14 08:34:53 +00:00
|
|
|
handle_default_msg(struct ff_msg *msg)
|
2017-05-23 15:13:49 +00:00
|
|
|
{
|
2017-11-01 09:38:22 +00:00
|
|
|
msg->result = ENOTSUP;
|
2017-05-23 15:13:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static inline void
|
|
|
|
handle_msg(struct ff_msg *msg, uint16_t proc_id)
|
|
|
|
{
|
|
|
|
switch (msg->msg_type) {
|
|
|
|
case FF_SYSCTL:
|
2017-11-14 08:34:53 +00:00
|
|
|
handle_sysctl_msg(msg);
|
2017-05-23 15:13:49 +00:00
|
|
|
break;
|
2017-06-06 08:52:52 +00:00
|
|
|
case FF_IOCTL:
|
2019-07-12 12:56:01 +00:00
|
|
|
#ifdef INET6
|
|
|
|
case FF_IOCTL6:
|
|
|
|
#endif
|
2017-11-14 08:34:53 +00:00
|
|
|
handle_ioctl_msg(msg);
|
2017-06-06 08:52:52 +00:00
|
|
|
break;
|
2017-06-14 07:25:35 +00:00
|
|
|
case FF_ROUTE:
|
2017-11-14 08:34:53 +00:00
|
|
|
handle_route_msg(msg);
|
2017-06-14 07:25:35 +00:00
|
|
|
break;
|
2017-08-22 09:12:53 +00:00
|
|
|
case FF_TOP:
|
2017-11-14 08:34:53 +00:00
|
|
|
handle_top_msg(msg);
|
2017-08-22 09:12:53 +00:00
|
|
|
break;
|
2017-11-01 09:38:22 +00:00
|
|
|
#ifdef FF_NETGRAPH
|
|
|
|
case FF_NGCTL:
|
2017-11-14 08:34:53 +00:00
|
|
|
handle_ngctl_msg(msg);
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
#ifdef FF_IPFW
|
|
|
|
case FF_IPFW_CTL:
|
|
|
|
handle_ipfw_msg(msg);
|
2017-11-01 09:38:22 +00:00
|
|
|
break;
|
|
|
|
#endif
|
2018-08-16 12:29:12 +00:00
|
|
|
case FF_TRAFFIC:
|
|
|
|
handle_traffic_msg(msg);
|
|
|
|
break;
|
2020-05-14 10:30:34 +00:00
|
|
|
#ifdef FF_KNI
|
|
|
|
case FF_KNICTL:
|
|
|
|
handle_knictl_msg(msg);
|
|
|
|
break;
|
|
|
|
#endif
|
2017-05-23 15:13:49 +00:00
|
|
|
default:
|
2017-11-14 08:34:53 +00:00
|
|
|
handle_default_msg(msg);
|
2017-05-23 15:13:49 +00:00
|
|
|
break;
|
|
|
|
}
|
2021-01-28 14:50:22 +00:00
|
|
|
if (rte_ring_enqueue(msg_ring[proc_id].ring[msg->msg_type], msg) < 0) {
|
|
|
|
if (msg->original_buf) {
|
|
|
|
rte_free(msg->buf_addr);
|
|
|
|
msg->buf_addr = msg->original_buf;
|
|
|
|
msg->buf_len = msg->original_buf_len;
|
|
|
|
msg->original_buf = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
rte_mempool_put(message_pool, msg);
|
|
|
|
}
|
2017-05-23 15:13:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static inline int
|
2021-01-28 14:50:22 +00:00
|
|
|
process_msg_ring(uint16_t proc_id, struct rte_mbuf **pkts_burst)
|
2017-05-23 15:13:49 +00:00
|
|
|
{
|
2021-01-28 14:50:22 +00:00
|
|
|
/* read msg from ring buf and to process */
|
|
|
|
uint16_t nb_rb;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
nb_rb = rte_ring_dequeue_burst(msg_ring[proc_id].ring[0],
|
|
|
|
(void **)pkts_burst, MAX_PKT_BURST, NULL);
|
|
|
|
|
|
|
|
if (likely(nb_rb == 0))
|
|
|
|
return 0;
|
2017-05-23 15:13:49 +00:00
|
|
|
|
2021-01-28 14:50:22 +00:00
|
|
|
for (i = 0; i < nb_rb; ++i) {
|
|
|
|
handle_msg((struct ff_msg *)pkts_burst[i], proc_id);
|
2017-04-21 10:43:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Send burst of packets on an output interface */
|
|
|
|
static inline int
|
|
|
|
send_burst(struct lcore_conf *qconf, uint16_t n, uint8_t port)
|
|
|
|
{
|
|
|
|
struct rte_mbuf **m_table;
|
|
|
|
int ret;
|
|
|
|
uint16_t queueid;
|
|
|
|
|
|
|
|
queueid = qconf->tx_queue_id[port];
|
|
|
|
m_table = (struct rte_mbuf **)qconf->tx_mbufs[port].m_table;
|
|
|
|
|
2020-01-08 04:09:40 +00:00
|
|
|
if (unlikely(ff_global_cfg.pcap.enable)) {
|
2017-04-21 10:43:26 +00:00
|
|
|
uint16_t i;
|
|
|
|
for (i = 0; i < n; i++) {
|
2022-09-01 11:18:12 +00:00
|
|
|
ff_dump_packets( ff_global_cfg.pcap.save_path, m_table[i],
|
2020-01-08 13:13:12 +00:00
|
|
|
ff_global_cfg.pcap.snap_len, ff_global_cfg.pcap.save_len);
|
2017-04-21 10:43:26 +00:00
|
|
|
}
|
|
|
|
}
|
2022-09-01 11:18:12 +00:00
|
|
|
|
2017-04-21 10:43:26 +00:00
|
|
|
ret = rte_eth_tx_burst(port, queueid, m_table, n);
|
2019-03-15 08:18:17 +00:00
|
|
|
ff_traffic.tx_packets += ret;
|
|
|
|
uint16_t i;
|
|
|
|
for (i = 0; i < ret; i++) {
|
|
|
|
ff_traffic.tx_bytes += rte_pktmbuf_pkt_len(m_table[i]);
|
2019-04-01 06:54:36 +00:00
|
|
|
#ifdef FF_USE_PAGE_ARRAY
|
2019-04-01 07:42:01 +00:00
|
|
|
if (qconf->tx_mbufs[port].bsd_m_table[i])
|
|
|
|
ff_enq_tx_bsdmbuf(port, qconf->tx_mbufs[port].bsd_m_table[i], m_table[i]->nb_segs);
|
2019-04-01 06:54:36 +00:00
|
|
|
#endif
|
2022-09-01 11:18:12 +00:00
|
|
|
}
|
2019-03-15 10:07:15 +00:00
|
|
|
if (unlikely(ret < n)) {
|
|
|
|
do {
|
|
|
|
rte_pktmbuf_free(m_table[ret]);
|
2019-04-01 06:54:36 +00:00
|
|
|
#ifdef FF_USE_PAGE_ARRAY
|
|
|
|
if ( qconf->tx_mbufs[port].bsd_m_table[ret] )
|
|
|
|
ff_mbuf_free(qconf->tx_mbufs[port].bsd_m_table[ret]);
|
2019-03-29 02:42:02 +00:00
|
|
|
#endif
|
2019-03-15 10:07:15 +00:00
|
|
|
} while (++ret < n);
|
2019-03-15 08:18:17 +00:00
|
|
|
}
|
2017-04-21 10:43:26 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Enqueue a single packet, and send burst if queue is filled */
|
|
|
|
static inline int
|
|
|
|
send_single_packet(struct rte_mbuf *m, uint8_t port)
|
|
|
|
{
|
|
|
|
uint16_t len;
|
|
|
|
struct lcore_conf *qconf;
|
|
|
|
|
|
|
|
qconf = &lcore_conf;
|
|
|
|
len = qconf->tx_mbufs[port].len;
|
|
|
|
qconf->tx_mbufs[port].m_table[len] = m;
|
|
|
|
len++;
|
|
|
|
|
|
|
|
/* enough pkts to be sent */
|
|
|
|
if (unlikely(len == MAX_PKT_BURST)) {
|
|
|
|
send_burst(qconf, MAX_PKT_BURST, port);
|
|
|
|
len = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
qconf->tx_mbufs[port].len = len;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
ff_dpdk_if_send(struct ff_dpdk_if_context *ctx, void *m,
|
|
|
|
int total)
|
|
|
|
{
|
2019-04-01 06:54:36 +00:00
|
|
|
#ifdef FF_USE_PAGE_ARRAY
|
|
|
|
struct lcore_conf *qconf = &lcore_conf;
|
|
|
|
int len = 0;
|
2022-09-01 11:18:12 +00:00
|
|
|
|
2019-04-01 06:54:36 +00:00
|
|
|
len = ff_if_send_onepkt(ctx, m,total);
|
|
|
|
if (unlikely(len == MAX_PKT_BURST)) {
|
|
|
|
send_burst(qconf, MAX_PKT_BURST, ctx->port_id);
|
|
|
|
len = 0;
|
|
|
|
}
|
|
|
|
qconf->tx_mbufs[ctx->port_id].len = len;
|
|
|
|
return 0;
|
2019-03-29 02:42:02 +00:00
|
|
|
#endif
|
2017-04-21 10:43:26 +00:00
|
|
|
struct rte_mempool *mbuf_pool = pktmbuf_pool[lcore_conf.socket_id];
|
|
|
|
struct rte_mbuf *head = rte_pktmbuf_alloc(mbuf_pool);
|
|
|
|
if (head == NULL) {
|
|
|
|
ff_mbuf_free(m);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
head->pkt_len = total;
|
2017-05-06 13:52:25 +00:00
|
|
|
head->nb_segs = 0;
|
2017-04-21 10:43:26 +00:00
|
|
|
|
|
|
|
int off = 0;
|
|
|
|
struct rte_mbuf *cur = head, *prev = NULL;
|
|
|
|
while(total > 0) {
|
|
|
|
if (cur == NULL) {
|
2017-05-06 13:52:25 +00:00
|
|
|
cur = rte_pktmbuf_alloc(mbuf_pool);
|
2017-04-21 10:43:26 +00:00
|
|
|
if (cur == NULL) {
|
|
|
|
rte_pktmbuf_free(head);
|
|
|
|
ff_mbuf_free(m);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-08-20 06:54:18 +00:00
|
|
|
if (prev != NULL) {
|
|
|
|
prev->next = cur;
|
|
|
|
}
|
|
|
|
head->nb_segs++;
|
|
|
|
|
|
|
|
prev = cur;
|
2017-04-21 10:43:26 +00:00
|
|
|
void *data = rte_pktmbuf_mtod(cur, void*);
|
|
|
|
int len = total > RTE_MBUF_DEFAULT_DATAROOM ? RTE_MBUF_DEFAULT_DATAROOM : total;
|
|
|
|
int ret = ff_mbuf_copydata(m, data, off, len);
|
|
|
|
if (ret < 0) {
|
|
|
|
rte_pktmbuf_free(head);
|
|
|
|
ff_mbuf_free(m);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
cur->data_len = len;
|
|
|
|
off += len;
|
|
|
|
total -= len;
|
2017-05-06 13:52:25 +00:00
|
|
|
cur = NULL;
|
2017-04-21 10:43:26 +00:00
|
|
|
}
|
|
|
|
|
2017-05-06 13:52:25 +00:00
|
|
|
struct ff_tx_offload offload = {0};
|
|
|
|
ff_mbuf_tx_offload(m, &offload);
|
|
|
|
|
2017-11-20 09:07:10 +00:00
|
|
|
void *data = rte_pktmbuf_mtod(head, void*);
|
|
|
|
|
2017-05-06 13:52:25 +00:00
|
|
|
if (offload.ip_csum) {
|
2017-11-20 09:07:10 +00:00
|
|
|
/* ipv6 not supported yet */
|
2020-06-18 16:55:50 +00:00
|
|
|
struct rte_ipv4_hdr *iph;
|
2017-11-20 09:07:10 +00:00
|
|
|
int iph_len;
|
2020-06-18 16:55:50 +00:00
|
|
|
iph = (struct rte_ipv4_hdr *)(data + RTE_ETHER_HDR_LEN);
|
2017-11-20 09:07:10 +00:00
|
|
|
iph_len = (iph->version_ihl & 0x0f) << 2;
|
|
|
|
|
|
|
|
head->ol_flags |= PKT_TX_IP_CKSUM | PKT_TX_IPV4;
|
2020-06-18 16:55:50 +00:00
|
|
|
head->l2_len = RTE_ETHER_HDR_LEN;
|
2017-11-20 09:07:10 +00:00
|
|
|
head->l3_len = iph_len;
|
2017-05-06 13:52:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (ctx->hw_features.tx_csum_l4) {
|
2020-06-18 16:55:50 +00:00
|
|
|
struct rte_ipv4_hdr *iph;
|
2017-11-20 09:07:10 +00:00
|
|
|
int iph_len;
|
2020-06-18 16:55:50 +00:00
|
|
|
iph = (struct rte_ipv4_hdr *)(data + RTE_ETHER_HDR_LEN);
|
2017-11-20 09:07:10 +00:00
|
|
|
iph_len = (iph->version_ihl & 0x0f) << 2;
|
|
|
|
|
2017-05-06 13:52:25 +00:00
|
|
|
if (offload.tcp_csum) {
|
|
|
|
head->ol_flags |= PKT_TX_TCP_CKSUM;
|
2020-06-18 16:55:50 +00:00
|
|
|
head->l2_len = RTE_ETHER_HDR_LEN;
|
2017-11-20 09:07:10 +00:00
|
|
|
head->l3_len = iph_len;
|
2017-05-06 13:52:25 +00:00
|
|
|
}
|
|
|
|
|
2017-11-20 09:07:10 +00:00
|
|
|
/*
|
|
|
|
* TCP segmentation offload.
|
|
|
|
*
|
|
|
|
* - set the PKT_TX_TCP_SEG flag in mbuf->ol_flags (this flag
|
|
|
|
* implies PKT_TX_TCP_CKSUM)
|
|
|
|
* - set the flag PKT_TX_IPV4 or PKT_TX_IPV6
|
|
|
|
* - if it's IPv4, set the PKT_TX_IP_CKSUM flag and
|
|
|
|
* write the IP checksum to 0 in the packet
|
|
|
|
* - fill the mbuf offload information: l2_len,
|
|
|
|
* l3_len, l4_len, tso_segsz
|
|
|
|
* - calculate the pseudo header checksum without taking ip_len
|
|
|
|
* in account, and set it in the TCP header. Refer to
|
|
|
|
* rte_ipv4_phdr_cksum() and rte_ipv6_phdr_cksum() that can be
|
|
|
|
* used as helpers.
|
|
|
|
*/
|
2017-05-06 13:52:25 +00:00
|
|
|
if (offload.tso_seg_size) {
|
2020-06-18 16:55:50 +00:00
|
|
|
struct rte_tcp_hdr *tcph;
|
2017-11-20 09:07:10 +00:00
|
|
|
int tcph_len;
|
2020-06-18 16:55:50 +00:00
|
|
|
tcph = (struct rte_tcp_hdr *)((char *)iph + iph_len);
|
2017-11-20 09:07:10 +00:00
|
|
|
tcph_len = (tcph->data_off & 0xf0) >> 2;
|
|
|
|
tcph->cksum = rte_ipv4_phdr_cksum(iph, PKT_TX_TCP_SEG);
|
|
|
|
|
2017-05-06 13:52:25 +00:00
|
|
|
head->ol_flags |= PKT_TX_TCP_SEG;
|
2017-11-20 09:07:10 +00:00
|
|
|
head->l4_len = tcph_len;
|
2017-05-06 13:52:25 +00:00
|
|
|
head->tso_segsz = offload.tso_seg_size;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (offload.udp_csum) {
|
|
|
|
head->ol_flags |= PKT_TX_UDP_CKSUM;
|
2020-06-18 16:55:50 +00:00
|
|
|
head->l2_len = RTE_ETHER_HDR_LEN;
|
2017-11-20 09:07:10 +00:00
|
|
|
head->l3_len = iph_len;
|
2017-05-06 13:52:25 +00:00
|
|
|
}
|
|
|
|
}
|
2017-04-21 10:43:26 +00:00
|
|
|
|
|
|
|
ff_mbuf_free(m);
|
|
|
|
|
|
|
|
return send_single_packet(head, ctx->port_id);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
main_loop(void *arg)
|
|
|
|
{
|
|
|
|
struct loop_routine *lr = (struct loop_routine *)arg;
|
|
|
|
|
|
|
|
struct rte_mbuf *pkts_burst[MAX_PKT_BURST];
|
2018-08-17 17:45:11 +00:00
|
|
|
uint64_t prev_tsc, diff_tsc, cur_tsc, usch_tsc, div_tsc, usr_tsc, sys_tsc, end_tsc, idle_sleep_tsc;
|
2017-08-22 09:12:53 +00:00
|
|
|
int i, j, nb_rx, idle;
|
2018-05-15 09:49:22 +00:00
|
|
|
uint16_t port_id, queue_id;
|
2017-04-21 10:43:26 +00:00
|
|
|
struct lcore_conf *qconf;
|
2019-06-27 09:44:19 +00:00
|
|
|
uint64_t drain_tsc = 0;
|
2017-05-06 13:52:25 +00:00
|
|
|
struct ff_dpdk_if_context *ctx;
|
2017-04-21 10:43:26 +00:00
|
|
|
|
2019-06-27 09:44:19 +00:00
|
|
|
if (pkt_tx_delay) {
|
|
|
|
drain_tsc = (rte_get_tsc_hz() + US_PER_S - 1) / US_PER_S * pkt_tx_delay;
|
|
|
|
}
|
|
|
|
|
2017-04-21 10:43:26 +00:00
|
|
|
prev_tsc = 0;
|
2017-08-24 11:41:07 +00:00
|
|
|
usch_tsc = 0;
|
2017-04-21 10:43:26 +00:00
|
|
|
|
|
|
|
qconf = &lcore_conf;
|
|
|
|
|
|
|
|
while (1) {
|
|
|
|
cur_tsc = rte_rdtsc();
|
|
|
|
if (unlikely(freebsd_clock.expire < cur_tsc)) {
|
|
|
|
rte_timer_manage();
|
|
|
|
}
|
|
|
|
|
2017-08-22 09:12:53 +00:00
|
|
|
idle = 1;
|
|
|
|
sys_tsc = 0;
|
|
|
|
usr_tsc = 0;
|
2021-09-02 04:04:32 +00:00
|
|
|
usr_cb_tsc = 0;
|
2017-08-22 09:12:53 +00:00
|
|
|
|
2017-04-21 10:43:26 +00:00
|
|
|
/*
|
|
|
|
* TX burst queue drain
|
|
|
|
*/
|
|
|
|
diff_tsc = cur_tsc - prev_tsc;
|
2019-06-27 09:44:19 +00:00
|
|
|
if (unlikely(diff_tsc >= drain_tsc)) {
|
2017-09-22 09:35:50 +00:00
|
|
|
for (i = 0; i < qconf->nb_tx_port; i++) {
|
|
|
|
port_id = qconf->tx_port_id[i];
|
2017-04-21 10:43:26 +00:00
|
|
|
if (qconf->tx_mbufs[port_id].len == 0)
|
|
|
|
continue;
|
2017-08-22 09:12:53 +00:00
|
|
|
|
|
|
|
idle = 0;
|
2017-09-22 09:35:50 +00:00
|
|
|
|
2017-04-21 10:43:26 +00:00
|
|
|
send_burst(qconf,
|
|
|
|
qconf->tx_mbufs[port_id].len,
|
|
|
|
port_id);
|
|
|
|
qconf->tx_mbufs[port_id].len = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
prev_tsc = cur_tsc;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Read packet from RX queues
|
|
|
|
*/
|
|
|
|
for (i = 0; i < qconf->nb_rx_queue; ++i) {
|
|
|
|
port_id = qconf->rx_queue_list[i].port_id;
|
|
|
|
queue_id = qconf->rx_queue_list[i].queue_id;
|
2017-05-06 13:52:25 +00:00
|
|
|
ctx = veth_ctx[port_id];
|
2017-04-21 10:43:26 +00:00
|
|
|
|
2018-06-21 09:39:17 +00:00
|
|
|
#ifdef FF_KNI
|
2017-04-21 10:43:26 +00:00
|
|
|
if (enable_kni && rte_eal_process_type() == RTE_PROC_PRIMARY) {
|
|
|
|
ff_kni_process(port_id, queue_id, pkts_burst, MAX_PKT_BURST);
|
|
|
|
}
|
2018-06-21 09:39:17 +00:00
|
|
|
#endif
|
2017-04-21 10:43:26 +00:00
|
|
|
|
2021-09-02 04:09:09 +00:00
|
|
|
idle &= !process_dispatch_ring(port_id, queue_id, pkts_burst, ctx);
|
2017-04-21 10:43:26 +00:00
|
|
|
|
|
|
|
nb_rx = rte_eth_rx_burst(port_id, queue_id, pkts_burst,
|
|
|
|
MAX_PKT_BURST);
|
|
|
|
if (nb_rx == 0)
|
|
|
|
continue;
|
|
|
|
|
2017-08-22 09:12:53 +00:00
|
|
|
idle = 0;
|
|
|
|
|
2017-04-21 10:43:26 +00:00
|
|
|
/* Prefetch first packets */
|
|
|
|
for (j = 0; j < PREFETCH_OFFSET && j < nb_rx; j++) {
|
|
|
|
rte_prefetch0(rte_pktmbuf_mtod(
|
|
|
|
pkts_burst[j], void *));
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Prefetch and handle already prefetched packets */
|
|
|
|
for (j = 0; j < (nb_rx - PREFETCH_OFFSET); j++) {
|
|
|
|
rte_prefetch0(rte_pktmbuf_mtod(pkts_burst[
|
|
|
|
j + PREFETCH_OFFSET], void *));
|
2017-05-06 13:52:25 +00:00
|
|
|
process_packets(port_id, queue_id, &pkts_burst[j], 1, ctx, 0);
|
2017-04-21 10:43:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Handle remaining prefetched packets */
|
|
|
|
for (; j < nb_rx; j++) {
|
2017-05-06 13:52:25 +00:00
|
|
|
process_packets(port_id, queue_id, &pkts_burst[j], 1, ctx, 0);
|
2017-04-21 10:43:26 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-01-28 14:50:22 +00:00
|
|
|
process_msg_ring(qconf->proc_id, pkts_burst);
|
2017-05-23 15:13:49 +00:00
|
|
|
|
2017-08-22 09:12:53 +00:00
|
|
|
div_tsc = rte_rdtsc();
|
|
|
|
|
2019-06-27 09:44:19 +00:00
|
|
|
if (likely(lr->loop != NULL && (!idle || cur_tsc - usch_tsc >= drain_tsc))) {
|
2017-08-24 11:41:07 +00:00
|
|
|
usch_tsc = cur_tsc;
|
2017-04-21 10:43:26 +00:00
|
|
|
lr->loop(lr->arg);
|
|
|
|
}
|
2017-08-22 09:12:53 +00:00
|
|
|
|
2018-08-17 17:45:11 +00:00
|
|
|
idle_sleep_tsc = rte_rdtsc();
|
|
|
|
if (likely(idle && idle_sleep)) {
|
|
|
|
usleep(idle_sleep);
|
|
|
|
end_tsc = rte_rdtsc();
|
|
|
|
} else {
|
|
|
|
end_tsc = idle_sleep_tsc;
|
|
|
|
}
|
|
|
|
|
2021-09-02 04:04:32 +00:00
|
|
|
usr_tsc = usr_cb_tsc;
|
2017-08-24 11:41:07 +00:00
|
|
|
if (usch_tsc == cur_tsc) {
|
2021-09-02 04:04:32 +00:00
|
|
|
usr_tsc += idle_sleep_tsc - div_tsc;
|
2017-08-24 11:41:07 +00:00
|
|
|
}
|
2017-08-22 09:12:53 +00:00
|
|
|
|
|
|
|
if (!idle) {
|
2021-09-02 04:04:32 +00:00
|
|
|
sys_tsc = div_tsc - cur_tsc - usr_cb_tsc;
|
2018-08-16 12:29:12 +00:00
|
|
|
ff_top_status.sys_tsc += sys_tsc;
|
2017-08-22 09:12:53 +00:00
|
|
|
}
|
|
|
|
|
2018-08-16 12:29:12 +00:00
|
|
|
ff_top_status.usr_tsc += usr_tsc;
|
|
|
|
ff_top_status.work_tsc += end_tsc - cur_tsc;
|
|
|
|
ff_top_status.idle_tsc += end_tsc - cur_tsc - usr_tsc - sys_tsc;
|
2017-08-22 09:12:53 +00:00
|
|
|
|
2018-08-16 12:29:12 +00:00
|
|
|
ff_top_status.loops++;
|
2017-04-21 10:43:26 +00:00
|
|
|
}
|
2017-11-09 13:22:13 +00:00
|
|
|
|
|
|
|
return 0;
|
2017-04-21 10:43:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
ff_dpdk_if_up(void) {
|
|
|
|
int i;
|
2017-09-22 09:35:50 +00:00
|
|
|
struct lcore_conf *qconf = &lcore_conf;
|
|
|
|
for (i = 0; i < qconf->nb_tx_port; i++) {
|
|
|
|
uint16_t port_id = qconf->tx_port_id[i];
|
|
|
|
|
|
|
|
struct ff_port_cfg *pconf = &qconf->port_cfgs[port_id];
|
2017-09-14 11:51:07 +00:00
|
|
|
veth_ctx[port_id] = ff_veth_attach(pconf);
|
2017-04-21 10:43:26 +00:00
|
|
|
if (veth_ctx[port_id] == NULL) {
|
|
|
|
rte_exit(EXIT_FAILURE, "ff_veth_attach failed");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
ff_dpdk_run(loop_func_t loop, void *arg) {
|
2017-08-23 08:54:32 +00:00
|
|
|
struct loop_routine *lr = rte_malloc(NULL,
|
|
|
|
sizeof(struct loop_routine), 0);
|
2017-04-21 10:43:26 +00:00
|
|
|
lr->loop = loop;
|
|
|
|
lr->arg = arg;
|
2021-02-05 08:42:07 +00:00
|
|
|
rte_eal_mp_remote_launch(main_loop, lr, CALL_MAIN);
|
2017-04-21 10:43:26 +00:00
|
|
|
rte_eal_mp_wait_lcore();
|
2017-08-23 08:54:32 +00:00
|
|
|
rte_free(lr);
|
2017-04-21 10:43:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
ff_dpdk_pktmbuf_free(void *m)
|
|
|
|
{
|
fix use after free issue in mbuf free
Two kinds of mbuf are used in f-stack: freebsd mbuf and dpdk mbuf.
freebsd mbufs are metadata used in freebsd stack, and their data
pointers (m_data) point to dpdk mbuf's data (buf_addr). And they have
their own chain, like this:
bsd_mbuf1 -> bsd_mbuf2 -> bsd_mbuf3
\ \ \
dpdk_mbuf1 -> dpdk_mbuf2 -> dpdk_mbuf3
Considering the map relationship,
- m_freem() is corresponding to rte_pktmbuf_free(), is to free the whole
chain of mbufs.
- m_free() is corresponding to rte_pktmbuf_free_seg(), is to free the
specified mbuf segment.
The current implementation in f-stack uses rte_pktmbuf_free() for
m_free(). This leads to mbufs, which are still in use, be freed
unexpectedly. For example, if the bsd_mbuf1 is trimed into zero length,
bsd will invoke m_free() to free the specified segment, however, the
whole mbuf chain is freed by calling rte_pktmbuf_free().
#0 rte_pktmbuf_free (m=0x22006fb480)
#1 in ff_dpdk_pktmbuf_free (m=0x22006fb480)
#2 in ff_mbuf_ext_free (m=0x7ffff7f82800, arg1=0x22006fb480, arg2=0x0)
#3 in mb_free_ext (m=0x7ffff7f82800)
#4 in m_free (m=0x7ffff7f82800)
#5 in sbcompress (sb=, m=0x7ffff7f82800, n=)
#6 in sbappendstream_locked (sb=, m=0x7ffff7f82800, flags=0)
The fix is straightforward. Use the correct API for segment free.
Reported-by: Yong-Hao Zou <yonghaoz1994@gmail.com>
Signed-off-by: Jianfeng Tan <henry.tjf@antgroup.com>
2020-11-27 06:45:00 +00:00
|
|
|
rte_pktmbuf_free_seg((struct rte_mbuf *)m);
|
2017-04-21 10:43:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static uint32_t
|
|
|
|
toeplitz_hash(unsigned keylen, const uint8_t *key,
|
|
|
|
unsigned datalen, const uint8_t *data)
|
|
|
|
{
|
|
|
|
uint32_t hash = 0, v;
|
|
|
|
u_int i, b;
|
|
|
|
|
|
|
|
/* XXXRW: Perhaps an assertion about key length vs. data length? */
|
|
|
|
|
|
|
|
v = (key[0]<<24) + (key[1]<<16) + (key[2] <<8) + key[3];
|
|
|
|
for (i = 0; i < datalen; i++) {
|
|
|
|
for (b = 0; b < 8; b++) {
|
|
|
|
if (data[i] & (1<<(7-b)))
|
|
|
|
hash ^= v;
|
|
|
|
v <<= 1;
|
|
|
|
if ((i + 4) < keylen &&
|
|
|
|
(key[i+4] & (1<<(7-b))))
|
|
|
|
v |= 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return (hash);
|
|
|
|
}
|
|
|
|
|
2021-01-22 09:12:05 +00:00
|
|
|
int
|
|
|
|
ff_in_pcbladdr(uint16_t family, void *faddr, uint16_t fport, void *laddr)
|
|
|
|
{
|
|
|
|
int ret = 0;
|
|
|
|
uint16_t fa;
|
|
|
|
|
|
|
|
if (!pcblddr_fun)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
if (family == AF_INET)
|
|
|
|
fa = AF_INET;
|
|
|
|
else if (family == AF_INET6_FREEBSD)
|
|
|
|
fa = AF_INET6_LINUX;
|
|
|
|
else
|
|
|
|
return EADDRNOTAVAIL;
|
|
|
|
|
|
|
|
ret = (*pcblddr_fun)(fa, faddr, fport, laddr);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
ff_regist_pcblddr_fun(pcblddr_func_t func)
|
|
|
|
{
|
|
|
|
pcblddr_fun = func;
|
|
|
|
}
|
|
|
|
|
2017-04-21 10:43:26 +00:00
|
|
|
int
|
2017-08-28 08:51:06 +00:00
|
|
|
ff_rss_check(void *softc, uint32_t saddr, uint32_t daddr,
|
|
|
|
uint16_t sport, uint16_t dport)
|
2017-04-21 10:43:26 +00:00
|
|
|
{
|
|
|
|
struct lcore_conf *qconf = &lcore_conf;
|
2017-09-14 11:51:07 +00:00
|
|
|
struct ff_dpdk_if_context *ctx = ff_veth_softc_to_hostc(softc);
|
|
|
|
uint16_t nb_queues = qconf->nb_queue_list[ctx->port_id];
|
2017-04-21 10:43:26 +00:00
|
|
|
|
2017-09-22 09:35:50 +00:00
|
|
|
if (nb_queues <= 1) {
|
2017-04-21 10:43:26 +00:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2017-08-28 08:51:06 +00:00
|
|
|
uint16_t reta_size = rss_reta_size[ctx->port_id];
|
2017-09-14 11:51:07 +00:00
|
|
|
uint16_t queueid = qconf->tx_queue_id[ctx->port_id];
|
2017-08-28 08:51:06 +00:00
|
|
|
|
2017-04-21 10:43:26 +00:00
|
|
|
uint8_t data[sizeof(saddr) + sizeof(daddr) + sizeof(sport) +
|
|
|
|
sizeof(dport)];
|
|
|
|
|
|
|
|
unsigned datalen = 0;
|
|
|
|
|
|
|
|
bcopy(&saddr, &data[datalen], sizeof(saddr));
|
|
|
|
datalen += sizeof(saddr);
|
|
|
|
|
|
|
|
bcopy(&daddr, &data[datalen], sizeof(daddr));
|
|
|
|
datalen += sizeof(daddr);
|
|
|
|
|
|
|
|
bcopy(&sport, &data[datalen], sizeof(sport));
|
|
|
|
datalen += sizeof(sport);
|
|
|
|
|
|
|
|
bcopy(&dport, &data[datalen], sizeof(dport));
|
|
|
|
datalen += sizeof(dport);
|
|
|
|
|
2019-07-25 03:27:30 +00:00
|
|
|
uint32_t hash = 0;
|
2020-08-29 16:55:08 +00:00
|
|
|
hash = toeplitz_hash(rsskey_len, rsskey, datalen, data);
|
|
|
|
|
2017-09-14 11:51:07 +00:00
|
|
|
return ((hash & (reta_size - 1)) % nb_queues) == queueid;
|
2017-04-21 10:43:26 +00:00
|
|
|
}
|
2017-11-09 13:22:13 +00:00
|
|
|
|
|
|
|
void
|
|
|
|
ff_regist_packet_dispatcher(dispatch_func_t func)
|
|
|
|
{
|
|
|
|
packet_dispatcher = func;
|
|
|
|
}
|
2018-01-19 12:57:32 +00:00
|
|
|
|
|
|
|
uint64_t
|
|
|
|
ff_get_tsc_ns()
|
|
|
|
{
|
|
|
|
uint64_t cur_tsc = rte_rdtsc();
|
|
|
|
uint64_t hz = rte_get_tsc_hz();
|
|
|
|
return ((double)cur_tsc/(double)hz) * NS_PER_S;
|
|
|
|
}
|
|
|
|
|