版本:DPDK-1.8.0
原作者:张云尧 出处:http://aidaiz.com/dpdk_l2fwd/
本例中实现了相邻端口之间的相互转发。
比如一共4个端口可用,那么端口1收到数据后会转发给端口2,端口2收到数据后会转发给端口1,端口3和端口4也会相互转发。
设置环境变量
1
2
|
export RTE_SDK=/(RTE_SDK)
#DPDK的路径
export RTE_TARGET=x86_64-native-linuxapp-gcc
#DPDK的编译目标
|
进入示例目录
1
|
cd /(RTE_SDK)/example/l2wfd
|
编译
1
|
make
|
1
|
./build/l2wfd [EAL options] -- -p PORTMASK [-q NQ -T t]
|
EAL options
-p PORTMASK
PORTMASK:一个十六进制位掩码表示分配的端口数量。
-q NQ
NQ:表示分配给每个逻辑内核的收发队列数量。
-T t
t: 表示打印统计数据到屏幕上的时间间隔,默认为10秒。
1
|
./build/l2fwd -c f -n 4 -- -q 4 -p ffff
|
表示,分配给4个逻辑内核,每个内核分别有4个收发队列,而一共分配了16个端口。
1
2
3
|
ret = rte_eal_init(argc, argv);
if (ret <
0)
rte_exit(EXIT_FAILURE,
"Invalid EAL arguments\n");
|
1
2
3
|
ret = l2fwd_parse_args(argc, argv);
if (ret <
0)
rte_exit(EXIT_FAILURE,
"Invalid L2FWD arguments\n");
|
EAL参数传递已经在rte_eal_init()函数中完成了,这里主要传递“–”后面的参数。
传递参数之后,得到三个变量。
1
2
3
4
5
6
7
8
9
|
l2fwd_pktmbuf_pool =
rte_mempool_create(
"mbuf_pool", NB_MBUF,
MBUF_SIZE,
32,
sizeof(struct rte_pktmbuf_pool_private),
rte_pktmbuf_pool_init,
NULL,
rte_pktmbuf_init,
NULL,
rte_socket_id(),
0);
if (l2fwd_pktmbuf_pool ==
NULL)
rte_exit(EXIT_FAILURE,
"Cannot init mbuf pool\n");
|
1
2
3
4
5
6
7
|
//rte_eth_dev_count()函数返回端口总数
nb_ports = rte_eth_dev_count();
if (nb_ports ==
0)
rte_exit(EXIT_FAILURE,
"No Ethernet ports - bye\n");
if (nb_ports > RTE_MAX_ETHPORTS)
nb_ports = RTE_MAX_ETHPORTS;
|
1
2
3
4
5
|
for (portid =
0; portid < nb_ports; portid++) {
//跳过未分配或者不可用端口
if ((l2fwd_enabled_port_mask & (
1 << portid)) ==
0)
continue;
}
|
可用端口位掩码表示,左数第n位如果为1,表示端口n可用,如果左数第n位如果为0,表示端口n不可用。
要得到第x位为1还是0,我们的方法是将1左移x位,得到一个只在x位为1,其他位都为0的数,再与位掩码相与。结果为1,那么第x位为1,结果位0,那么第x位为0.
这里设置数据包进入端口后,转发给相邻的端口。
每两个端口为一对,相互转发。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
|
for (portid =
0; portid < nb_ports; portid++) {
if ((l2fwd_enabled_port_mask & (
1 << portid)) ==
0)
continue;
if (nb_ports_in_mask %
2) {
l2fwd_dst_ports[portid] = last_port;
l2fwd_dst_ports[last_port] = portid;
}
else
last_port = portid;
nb_ports_in_mask++;
rte_eth_dev_info_get(portid, &dev_info);
}
if (nb_ports_in_mask %
2) {
printf(
"Notice: odd number of ports in portmask.\n");
l2fwd_dst_ports[last_port] = last_port;
}
|
为每个端口分配到相应的逻辑内核
每个端口只对应一个逻辑内核
每个逻辑内核对应l2fwd_rx_queue_per_lcore个端口
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
|
for (portid =
0; portid < nb_ports; portid++) {
if ((l2fwd_enabled_port_mask & (
1 << portid)) ==
0)
continue;
//得到一个收取队列未分配满且可用的逻辑内核
while (rte_lcore_is_enabled(rx_lcore_id) ==
0 ||
lcore_queue_conf[rx_lcore_id].n_rx_port ==
l2fwd_rx_queue_per_lcore) {
rx_lcore_id++;
if (rx_lcore_id >= RTE_MAX_LCORE)
rte_exit(EXIT_FAILURE,
"Not enough cores\n");
}
if (qconf != &lcore_queue_conf[rx_lcore_id])
/* Assigned a new logical core in the loop above. */
qconf = &lcore_queue_conf[rx_lcore_id];
qconf->rx_port_list[qconf->n_rx_port] = portid;
qconf->n_rx_port++;
printf(
"Lcore %u: RX port %u\n", rx_lcore_id, (
unsigned) portid);
}
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
|
for (portid =
0; portid < nb_ports; portid++) {
if ((l2fwd_enabled_port_mask & (
1 << portid)) ==
0) {
printf(
"Skipping disabled port %u\n", (
unsigned) portid);
nb_ports_available--;
continue;
}
printf(
"Initializing port %u... ", (
unsigned) portid);
fflush(
stdout);
//初始化端口,第二个参数和第三个参数表示分配收取队列和发送队列的数量
ret = rte_eth_dev_configure(portid,
1,
1, &port_conf);
if (ret <
0)
rte_exit(EXIT_FAILURE,
"Cannot configure device: err=%d, port=%u\n",
ret, (
unsigned) portid);
//得到端口对应的mac地址,存入l2fwd_ports_eth_addr[]数组
rte_eth_macaddr_get(portid,&l2fwd_ports_eth_addr[portid]);
fflush(
stdout);
//初始化一个收取队列,nb_rxd指收取队列的大小,最大能够存储mbuf的数量
ret = rte_eth_rx_queue_setup(portid,
0, nb_rxd,
rte_eth_dev_socket_id(portid),
NULL,
l2fwd_pktmbuf_pool);
if (ret <
0)
rte_exit(EXIT_FAILURE,
"rte_eth_rx_queue_setup:err=%d, port=%u\n",
ret, (
unsigned) portid);
fflush(
stdout);
//初始化一个发送队列,nb_txd指发送队列的大小,最大能够存储mbuf的数量
ret = rte_eth_tx_queue_setup(portid,
0, nb_txd,
rte_eth_dev_socket_id(portid),
NULL);
if (ret <
0)
rte_exit(EXIT_FAILURE,
"rte_eth_tx_queue_setup:err=%d, port=%u\n",
ret, (
unsigned) portid);
//开始运行该端口
ret = rte_eth_dev_start(portid);
if (ret <
0)
rte_exit(EXIT_FAILURE,
"rte_eth_dev_start:err=%d, port=%u\n",
ret, (
unsigned) portid);
printf(
"done: \n");
rte_eth_promiscuous_enable(portid);
printf(
"Port %u, MAC address: %02X:%02X:%02X:%02X:%02X:%02X\n\n",
(
unsigned) portid,
l2fwd_ports_eth_addr[portid].addr_bytes[
0],
l2fwd_ports_eth_addr[portid].addr_bytes[
1],
l2fwd_ports_eth_addr[portid].addr_bytes[
2],
l2fwd_ports_eth_addr[portid].addr_bytes[
3],
l2fwd_ports_eth_addr[portid].addr_bytes[
4],
l2fwd_ports_eth_addr[portid].addr_bytes[
5]);
//初始化端口的统计数据
memset(&port_statistics,
0,
sizeof(port_statistics));
}
|
1
|
check_all_ports_link_status(nb_ports, l2fwd_enabled_port_mask);
|
1
2
3
4
5
|
rte_eal_mp_remote_launch(l2fwd_launch_one_lcore,
NULL, CALL_MASTER);
RTE_LCORE_FOREACH_SLAVE(lcore_id) {
if (rte_eal_wait_lcore(lcore_id) <
0)
return
-1;
}
|
收包
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
|
for (i =
0; i < qconf->n_rx_port; i++) {
portid = qconf->rx_port_list[i];
//收包,一次最多收取MAX_PKT_BURST个数据包
nb_rx = rte_eth_rx_burst((
uint8_t) portid,
0,
pkts_burst, MAX_PKT_BURST);
//更新统计数据
port_statistics[portid].rx += nb_rx;
for (j =
0; j < nb_rx; j++) {
m = pkts_burst[j];
rte_prefetch0(rte_pktmbuf_mtod(m,
void *));
//转发
l2fwd_simple_forward(m, portid);
}
}
|
转发
替换源MAC地址和目的MAC地址
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
|
static void
l2fwd_simple_forward
(struct rte_mbuf *m, unsigned portid)
{
struct ether_hdr *eth;
void *tmp;
unsigned dst_port;
dst_port = l2fwd_dst_ports[portid];
eth = rte_pktmbuf_mtod(m, struct ether_hdr *);
//目的地址
/* 02:00:00:00:00:xx */
tmp = ð->d_addr.addr_bytes[
0];
*((
uint64_t *)tmp) =
0x000000000002 + ((
uint64_t)dst_port <<
40);
//源地址
ether_addr_copy(&l2fwd_ports_eth_addr[dst_port], ð->s_addr);
l2fwd_send_packet(m, (
uint8_t) dst_port);
}
|
将数据包推送至发送队列,如果发送队列存够MAX_PKT_BURST,即每次最大收取包的数量,就会发包
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
|
static int
l2fwd_send_packet
(struct rte_mbuf *m, uint8_t port)
{
unsigned lcore_id, len;
struct lcore_queue_conf *qconf;
lcore_id = rte_lcore_id();
qconf = &lcore_queue_conf[lcore_id];
len = qconf->tx_mbufs[port].len;
qconf->tx_mbufs[port].m_table[len] = m;
len++;
//当发包队列存够MAX_PKT_BURST,发包
if (unlikely(len == MAX_PKT_BURST)) {
l2fwd_send_burst(qconf, MAX_PKT_BURST, port);
len =
0;
}
qconf->tx_mbufs[port].len = len;
return
0;
}
|
每隔一定时间也会发包
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
|
//上次收包时间和这次收包时间差
diff_tsc = cur_tsc - prev_tsc;
//如果时间差大于我们设定的阈值,这里是100us
if (unlikely(diff_tsc > drain_tsc)) {
for (portid =
0; portid < RTE_MAX_ETHPORTS; portid++) {
if (qconf->tx_mbufs[portid].len ==
0)
continue;
//发包
l2fwd_send_burst(&lcore_queue_conf[lcore_id],
qconf->tx_mbufs[portid].len,
(
uint8_t) portid);
qconf->tx_mbufs[portid].len =
0;
}
if (timer_period >
0) {
timer_tsc += diff_tsc;
//如果累积时间超过我们设定的阈值,就打印出统计数据,默认是10s
if (unlikely(timer_tsc >= (
uint64_t) timer_period)) {
//打印数据在发生在主逻辑内核上
if (lcore_id == rte_get_master_lcore()) {
//打印统计数据
print_stats();
//累积时间置零
timer_tsc =
0;
}
}
}
prev_tsc = cur_tsc;
}
|
这两种情况都会产生发包,无论是发送队列存够阈值MAX_PKT_BURST,或者,时间差超过阈值brain_tsc,都会把发送队列上MAX_PKT_BURST个数据包推送出去,如果不足MAX_PKT_BURST,则把发送队列上全部数据包推送出去。
发包函数
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
|
static int
l2fwd_send_burst
(struct lcore_queue_conf *qconf, unsigned n, uint8_t port)
{
struct rte_mbuf **m_table;
unsigned ret;
unsigned queueid =
0;
m_table = (struct rte_mbuf **)qconf->tx_mbufs[port].m_table;
//发包
ret = rte_eth_tx_burst(port, (
uint16_t) queueid, m_table, (
uint16_t) n);
//更新统计数据
port_statistics[port].tx += ret;
//丢包
if (unlikely(ret < n)) {
//更新统计数据
port_statistics[port].dropped += (n - ret);
do {
//把丢包部分free掉
rte_pktmbuf_free(m_table[ret]);
}
while (++ret < n);
}
return
0;
}
|
在函数rte_eth_tx_burst()中: