xref: /dpdk/examples/server_node_efd/server/init.c (revision 60efb44f)
1ed2a80fdSPablo de Lara /*-
2ed2a80fdSPablo de Lara  *   BSD LICENSE
3ed2a80fdSPablo de Lara  *
4ed2a80fdSPablo de Lara  *   Copyright(c) 2016-2017 Intel Corporation. All rights reserved.
5ed2a80fdSPablo de Lara  *   All rights reserved.
6ed2a80fdSPablo de Lara  *
7ed2a80fdSPablo de Lara  *   Redistribution and use in source and binary forms, with or without
8ed2a80fdSPablo de Lara  *   modification, are permitted provided that the following conditions
9ed2a80fdSPablo de Lara  *   are met:
10ed2a80fdSPablo de Lara  *
11ed2a80fdSPablo de Lara  *     * Redistributions of source code must retain the above copyright
12ed2a80fdSPablo de Lara  *       notice, this list of conditions and the following disclaimer.
13ed2a80fdSPablo de Lara  *     * Redistributions in binary form must reproduce the above copyright
14ed2a80fdSPablo de Lara  *       notice, this list of conditions and the following disclaimer in
15ed2a80fdSPablo de Lara  *       the documentation and/or other materials provided with the
16ed2a80fdSPablo de Lara  *       distribution.
17ed2a80fdSPablo de Lara  *     * Neither the name of Intel Corporation nor the names of its
18ed2a80fdSPablo de Lara  *       contributors may be used to endorse or promote products derived
19ed2a80fdSPablo de Lara  *       from this software without specific prior written permission.
20ed2a80fdSPablo de Lara  *
21ed2a80fdSPablo de Lara  *   THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
22ed2a80fdSPablo de Lara  *   "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
23ed2a80fdSPablo de Lara  *   LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
24ed2a80fdSPablo de Lara  *   A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
25ed2a80fdSPablo de Lara  *   OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
26ed2a80fdSPablo de Lara  *   SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
27ed2a80fdSPablo de Lara  *   LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
28ed2a80fdSPablo de Lara  *   DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
29ed2a80fdSPablo de Lara  *   THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
30ed2a80fdSPablo de Lara  *   (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
31ed2a80fdSPablo de Lara  *   OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
32ed2a80fdSPablo de Lara  */
33ed2a80fdSPablo de Lara 
34ed2a80fdSPablo de Lara #include <stdint.h>
35ed2a80fdSPablo de Lara #include <stdio.h>
36ed2a80fdSPablo de Lara #include <string.h>
37ed2a80fdSPablo de Lara #include <sys/queue.h>
38ed2a80fdSPablo de Lara #include <errno.h>
39ed2a80fdSPablo de Lara #include <stdarg.h>
40ed2a80fdSPablo de Lara #include <inttypes.h>
41ed2a80fdSPablo de Lara 
42ed2a80fdSPablo de Lara #include <rte_common.h>
43ed2a80fdSPablo de Lara #include <rte_memory.h>
44ed2a80fdSPablo de Lara #include <rte_memzone.h>
45ed2a80fdSPablo de Lara #include <rte_eal.h>
46ed2a80fdSPablo de Lara #include <rte_byteorder.h>
47ed2a80fdSPablo de Lara #include <rte_atomic.h>
48ed2a80fdSPablo de Lara #include <rte_launch.h>
49ed2a80fdSPablo de Lara #include <rte_per_lcore.h>
50ed2a80fdSPablo de Lara #include <rte_lcore.h>
51ed2a80fdSPablo de Lara #include <rte_branch_prediction.h>
52ed2a80fdSPablo de Lara #include <rte_debug.h>
53ed2a80fdSPablo de Lara #include <rte_ring.h>
54ed2a80fdSPablo de Lara #include <rte_log.h>
55ed2a80fdSPablo de Lara #include <rte_mempool.h>
56ed2a80fdSPablo de Lara #include <rte_memcpy.h>
57ed2a80fdSPablo de Lara #include <rte_mbuf.h>
58ed2a80fdSPablo de Lara #include <rte_interrupts.h>
59ed2a80fdSPablo de Lara #include <rte_pci.h>
60ed2a80fdSPablo de Lara #include <rte_ether.h>
61ed2a80fdSPablo de Lara #include <rte_ethdev.h>
62ed2a80fdSPablo de Lara #include <rte_malloc.h>
63ed2a80fdSPablo de Lara #include <rte_string_fns.h>
64ed2a80fdSPablo de Lara #include <rte_cycles.h>
65ed2a80fdSPablo de Lara #include <rte_efd.h>
66ed2a80fdSPablo de Lara #include <rte_hash.h>
67ed2a80fdSPablo de Lara 
68ed2a80fdSPablo de Lara #include "common.h"
69ed2a80fdSPablo de Lara #include "args.h"
70ed2a80fdSPablo de Lara #include "init.h"
71ed2a80fdSPablo de Lara 
72ed2a80fdSPablo de Lara #define MBUFS_PER_NODE 1536
73ed2a80fdSPablo de Lara #define MBUFS_PER_PORT 1536
74ed2a80fdSPablo de Lara #define MBUF_CACHE_SIZE 512
75ed2a80fdSPablo de Lara 
76ed2a80fdSPablo de Lara #define RTE_MP_RX_DESC_DEFAULT 512
77ed2a80fdSPablo de Lara #define RTE_MP_TX_DESC_DEFAULT 512
78ed2a80fdSPablo de Lara #define NODE_QUEUE_RINGSIZE 128
79ed2a80fdSPablo de Lara 
80ed2a80fdSPablo de Lara #define NO_FLAGS 0
81ed2a80fdSPablo de Lara 
82ed2a80fdSPablo de Lara /* The mbuf pool for packet rx */
83ed2a80fdSPablo de Lara struct rte_mempool *pktmbuf_pool;
84ed2a80fdSPablo de Lara 
85ed2a80fdSPablo de Lara /* array of info/queues for nodes */
86ed2a80fdSPablo de Lara struct node *nodes;
87ed2a80fdSPablo de Lara 
88ed2a80fdSPablo de Lara /* EFD table */
89ed2a80fdSPablo de Lara struct rte_efd_table *efd_table;
90ed2a80fdSPablo de Lara 
91ed2a80fdSPablo de Lara /* Shared info between server and nodes */
92ed2a80fdSPablo de Lara struct shared_info *info;
93ed2a80fdSPablo de Lara 
94ed2a80fdSPablo de Lara /**
95ed2a80fdSPablo de Lara  * Initialise the mbuf pool for packet reception for the NIC, and any other
96ed2a80fdSPablo de Lara  * buffer pools needed by the app - currently none.
97ed2a80fdSPablo de Lara  */
98ed2a80fdSPablo de Lara static int
99ed2a80fdSPablo de Lara init_mbuf_pools(void)
100ed2a80fdSPablo de Lara {
101ed2a80fdSPablo de Lara 	const unsigned int num_mbufs = (num_nodes * MBUFS_PER_NODE) +
102ed2a80fdSPablo de Lara 			(info->num_ports * MBUFS_PER_PORT);
103ed2a80fdSPablo de Lara 
104ed2a80fdSPablo de Lara 	/*
105ed2a80fdSPablo de Lara 	 * Don't pass single-producer/single-consumer flags to mbuf create as it
106ed2a80fdSPablo de Lara 	 * seems faster to use a cache instead
107ed2a80fdSPablo de Lara 	 */
108ed2a80fdSPablo de Lara 	printf("Creating mbuf pool '%s' [%u mbufs] ...\n",
109ed2a80fdSPablo de Lara 			PKTMBUF_POOL_NAME, num_mbufs);
110ed2a80fdSPablo de Lara 	pktmbuf_pool = rte_pktmbuf_pool_create(PKTMBUF_POOL_NAME, num_mbufs,
111ed2a80fdSPablo de Lara 		MBUF_CACHE_SIZE, 0, RTE_MBUF_DEFAULT_BUF_SIZE, rte_socket_id());
112ed2a80fdSPablo de Lara 
113ed2a80fdSPablo de Lara 	return pktmbuf_pool == NULL; /* 0  on success */
114ed2a80fdSPablo de Lara }
115ed2a80fdSPablo de Lara 
116ed2a80fdSPablo de Lara /**
117ed2a80fdSPablo de Lara  * Initialise an individual port:
118ed2a80fdSPablo de Lara  * - configure number of rx and tx rings
119ed2a80fdSPablo de Lara  * - set up each rx ring, to pull from the main mbuf pool
120ed2a80fdSPablo de Lara  * - set up each tx ring
121ed2a80fdSPablo de Lara  * - start the port and report its status to stdout
122ed2a80fdSPablo de Lara  */
123ed2a80fdSPablo de Lara static int
124ed2a80fdSPablo de Lara init_port(uint8_t port_num)
125ed2a80fdSPablo de Lara {
126ed2a80fdSPablo de Lara 	/* for port configuration all features are off by default */
127ed2a80fdSPablo de Lara 	const struct rte_eth_conf port_conf = {
128ed2a80fdSPablo de Lara 		.rxmode = {
129ed2a80fdSPablo de Lara 			.mq_mode = ETH_MQ_RX_RSS
130ed2a80fdSPablo de Lara 		}
131ed2a80fdSPablo de Lara 	};
132ed2a80fdSPablo de Lara 	const uint16_t rx_rings = 1, tx_rings = num_nodes;
133*60efb44fSRoman Zhukov 	uint16_t rx_ring_size = RTE_MP_RX_DESC_DEFAULT;
134*60efb44fSRoman Zhukov 	uint16_t tx_ring_size = RTE_MP_TX_DESC_DEFAULT;
135ed2a80fdSPablo de Lara 
136ed2a80fdSPablo de Lara 	uint16_t q;
137ed2a80fdSPablo de Lara 	int retval;
138ed2a80fdSPablo de Lara 
139ed2a80fdSPablo de Lara 	printf("Port %u init ... ", (unsigned int)port_num);
140ed2a80fdSPablo de Lara 	fflush(stdout);
141ed2a80fdSPablo de Lara 
142ed2a80fdSPablo de Lara 	/*
143ed2a80fdSPablo de Lara 	 * Standard DPDK port initialisation - config port, then set up
144ed2a80fdSPablo de Lara 	 * rx and tx rings.
145ed2a80fdSPablo de Lara 	 */
146ed2a80fdSPablo de Lara 	retval = rte_eth_dev_configure(port_num, rx_rings, tx_rings, &port_conf);
147ed2a80fdSPablo de Lara 	if (retval != 0)
148ed2a80fdSPablo de Lara 		return retval;
149ed2a80fdSPablo de Lara 
150*60efb44fSRoman Zhukov 	retval = rte_eth_dev_adjust_nb_rx_tx_desc(port_num, &rx_ring_size,
151*60efb44fSRoman Zhukov 			&tx_ring_size);
152*60efb44fSRoman Zhukov 	if (retval != 0)
153*60efb44fSRoman Zhukov 		return retval;
154*60efb44fSRoman Zhukov 
155ed2a80fdSPablo de Lara 	for (q = 0; q < rx_rings; q++) {
156ed2a80fdSPablo de Lara 		retval = rte_eth_rx_queue_setup(port_num, q, rx_ring_size,
157ed2a80fdSPablo de Lara 				rte_eth_dev_socket_id(port_num),
158ed2a80fdSPablo de Lara 				NULL, pktmbuf_pool);
159ed2a80fdSPablo de Lara 		if (retval < 0)
160ed2a80fdSPablo de Lara 			return retval;
161ed2a80fdSPablo de Lara 	}
162ed2a80fdSPablo de Lara 
163ed2a80fdSPablo de Lara 	for (q = 0; q < tx_rings; q++) {
164ed2a80fdSPablo de Lara 		retval = rte_eth_tx_queue_setup(port_num, q, tx_ring_size,
165ed2a80fdSPablo de Lara 				rte_eth_dev_socket_id(port_num),
166ed2a80fdSPablo de Lara 				NULL);
167ed2a80fdSPablo de Lara 		if (retval < 0)
168ed2a80fdSPablo de Lara 			return retval;
169ed2a80fdSPablo de Lara 	}
170ed2a80fdSPablo de Lara 
171ed2a80fdSPablo de Lara 	rte_eth_promiscuous_enable(port_num);
172ed2a80fdSPablo de Lara 
173ed2a80fdSPablo de Lara 	retval = rte_eth_dev_start(port_num);
174ed2a80fdSPablo de Lara 	if (retval < 0)
175ed2a80fdSPablo de Lara 		return retval;
176ed2a80fdSPablo de Lara 
177ed2a80fdSPablo de Lara 	printf("done:\n");
178ed2a80fdSPablo de Lara 
179ed2a80fdSPablo de Lara 	return 0;
180ed2a80fdSPablo de Lara }
181ed2a80fdSPablo de Lara 
182ed2a80fdSPablo de Lara /**
183ed2a80fdSPablo de Lara  * Set up the DPDK rings which will be used to pass packets, via
184ed2a80fdSPablo de Lara  * pointers, between the multi-process server and node processes.
185ed2a80fdSPablo de Lara  * Each node needs one RX queue.
186ed2a80fdSPablo de Lara  */
187ed2a80fdSPablo de Lara static int
188ed2a80fdSPablo de Lara init_shm_rings(void)
189ed2a80fdSPablo de Lara {
190ed2a80fdSPablo de Lara 	unsigned int i;
191ed2a80fdSPablo de Lara 	unsigned int socket_id;
192ed2a80fdSPablo de Lara 	const char *q_name;
193ed2a80fdSPablo de Lara 	const unsigned int ringsize = NODE_QUEUE_RINGSIZE;
194ed2a80fdSPablo de Lara 
195ed2a80fdSPablo de Lara 	nodes = rte_malloc("node details",
196ed2a80fdSPablo de Lara 		sizeof(*nodes) * num_nodes, 0);
197ed2a80fdSPablo de Lara 	if (nodes == NULL)
198ed2a80fdSPablo de Lara 		rte_exit(EXIT_FAILURE, "Cannot allocate memory for "
199ed2a80fdSPablo de Lara 				"node program details\n");
200ed2a80fdSPablo de Lara 
201ed2a80fdSPablo de Lara 	for (i = 0; i < num_nodes; i++) {
202ed2a80fdSPablo de Lara 		/* Create an RX queue for each node */
203ed2a80fdSPablo de Lara 		socket_id = rte_socket_id();
204ed2a80fdSPablo de Lara 		q_name = get_rx_queue_name(i);
205ed2a80fdSPablo de Lara 		nodes[i].rx_q = rte_ring_create(q_name,
206ed2a80fdSPablo de Lara 				ringsize, socket_id,
207ed2a80fdSPablo de Lara 				RING_F_SP_ENQ | RING_F_SC_DEQ);
208ed2a80fdSPablo de Lara 		if (nodes[i].rx_q == NULL)
209ed2a80fdSPablo de Lara 			rte_exit(EXIT_FAILURE, "Cannot create rx ring queue "
210ed2a80fdSPablo de Lara 					"for node %u\n", i);
211ed2a80fdSPablo de Lara 	}
212ed2a80fdSPablo de Lara 	return 0;
213ed2a80fdSPablo de Lara }
214ed2a80fdSPablo de Lara 
215ed2a80fdSPablo de Lara /*
216ed2a80fdSPablo de Lara  * Create EFD table which will contain all the flows
217ed2a80fdSPablo de Lara  * that will be distributed among the nodes
218ed2a80fdSPablo de Lara  */
219ed2a80fdSPablo de Lara static void
220ed2a80fdSPablo de Lara create_efd_table(void)
221ed2a80fdSPablo de Lara {
222ed2a80fdSPablo de Lara 	uint8_t socket_id = rte_socket_id();
223ed2a80fdSPablo de Lara 
224ed2a80fdSPablo de Lara 	/* create table */
225ed2a80fdSPablo de Lara 	efd_table = rte_efd_create("flow table", num_flows * 2, sizeof(uint32_t),
226ed2a80fdSPablo de Lara 			1 << socket_id,	socket_id);
227ed2a80fdSPablo de Lara 
228ed2a80fdSPablo de Lara 	if (efd_table == NULL)
229ed2a80fdSPablo de Lara 		rte_exit(EXIT_FAILURE, "Problem creating the flow table\n");
230ed2a80fdSPablo de Lara }
231ed2a80fdSPablo de Lara 
232ed2a80fdSPablo de Lara static void
233ed2a80fdSPablo de Lara populate_efd_table(void)
234ed2a80fdSPablo de Lara {
235ed2a80fdSPablo de Lara 	unsigned int i;
236ed2a80fdSPablo de Lara 	int32_t ret;
237ed2a80fdSPablo de Lara 	uint32_t ip_dst;
238ed2a80fdSPablo de Lara 	uint8_t socket_id = rte_socket_id();
239ed2a80fdSPablo de Lara 	uint64_t node_id;
240ed2a80fdSPablo de Lara 
241ed2a80fdSPablo de Lara 	/* Add flows in table */
242ed2a80fdSPablo de Lara 	for (i = 0; i < num_flows; i++) {
243ed2a80fdSPablo de Lara 		node_id = i % num_nodes;
244ed2a80fdSPablo de Lara 
245ed2a80fdSPablo de Lara 		ip_dst = rte_cpu_to_be_32(i);
246ed2a80fdSPablo de Lara 		ret = rte_efd_update(efd_table, socket_id,
247ed2a80fdSPablo de Lara 				(void *)&ip_dst, (efd_value_t)node_id);
248ed2a80fdSPablo de Lara 		if (ret < 0)
249ed2a80fdSPablo de Lara 			rte_exit(EXIT_FAILURE, "Unable to add entry %u in "
250ed2a80fdSPablo de Lara 					"EFD table\n", i);
251ed2a80fdSPablo de Lara 	}
252ed2a80fdSPablo de Lara 
253ed2a80fdSPablo de Lara 	printf("EFD table: Adding 0x%x keys\n", num_flows);
254ed2a80fdSPablo de Lara }
255ed2a80fdSPablo de Lara 
256ed2a80fdSPablo de Lara /* Check the link status of all ports in up to 9s, and print them finally */
257ed2a80fdSPablo de Lara static void
258ed2a80fdSPablo de Lara check_all_ports_link_status(uint8_t port_num, uint32_t port_mask)
259ed2a80fdSPablo de Lara {
260ed2a80fdSPablo de Lara #define CHECK_INTERVAL 100 /* 100ms */
261ed2a80fdSPablo de Lara #define MAX_CHECK_TIME 90 /* 9s (90 * 100ms) in total */
262ed2a80fdSPablo de Lara 	uint8_t portid, count, all_ports_up, print_flag = 0;
263ed2a80fdSPablo de Lara 	struct rte_eth_link link;
264ed2a80fdSPablo de Lara 
265ed2a80fdSPablo de Lara 	printf("\nChecking link status");
266ed2a80fdSPablo de Lara 	fflush(stdout);
267ed2a80fdSPablo de Lara 	for (count = 0; count <= MAX_CHECK_TIME; count++) {
268ed2a80fdSPablo de Lara 		all_ports_up = 1;
269ed2a80fdSPablo de Lara 		for (portid = 0; portid < port_num; portid++) {
270ed2a80fdSPablo de Lara 			if ((port_mask & (1 << info->id[portid])) == 0)
271ed2a80fdSPablo de Lara 				continue;
272ed2a80fdSPablo de Lara 			memset(&link, 0, sizeof(link));
273ed2a80fdSPablo de Lara 			rte_eth_link_get_nowait(info->id[portid], &link);
274ed2a80fdSPablo de Lara 			/* print link status if flag set */
275ed2a80fdSPablo de Lara 			if (print_flag == 1) {
276ed2a80fdSPablo de Lara 				if (link.link_status)
277ed2a80fdSPablo de Lara 					printf("Port %d Link Up - speed %u "
278ed2a80fdSPablo de Lara 						"Mbps - %s\n", info->id[portid],
279ed2a80fdSPablo de Lara 						(unsigned int)link.link_speed,
280ed2a80fdSPablo de Lara 				(link.link_duplex == ETH_LINK_FULL_DUPLEX) ?
281ed2a80fdSPablo de Lara 					("full-duplex") : ("half-duplex\n"));
282ed2a80fdSPablo de Lara 				else
283ed2a80fdSPablo de Lara 					printf("Port %d Link Down\n",
284ed2a80fdSPablo de Lara 						(uint8_t)info->id[portid]);
285ed2a80fdSPablo de Lara 				continue;
286ed2a80fdSPablo de Lara 			}
287ed2a80fdSPablo de Lara 			/* clear all_ports_up flag if any link down */
288ed2a80fdSPablo de Lara 			if (link.link_status == ETH_LINK_DOWN) {
289ed2a80fdSPablo de Lara 				all_ports_up = 0;
290ed2a80fdSPablo de Lara 				break;
291ed2a80fdSPablo de Lara 			}
292ed2a80fdSPablo de Lara 		}
293ed2a80fdSPablo de Lara 		/* after finally printing all link status, get out */
294ed2a80fdSPablo de Lara 		if (print_flag == 1)
295ed2a80fdSPablo de Lara 			break;
296ed2a80fdSPablo de Lara 
297ed2a80fdSPablo de Lara 		if (all_ports_up == 0) {
298ed2a80fdSPablo de Lara 			printf(".");
299ed2a80fdSPablo de Lara 			fflush(stdout);
300ed2a80fdSPablo de Lara 			rte_delay_ms(CHECK_INTERVAL);
301ed2a80fdSPablo de Lara 		}
302ed2a80fdSPablo de Lara 
303ed2a80fdSPablo de Lara 		/* set the print_flag if all ports up or timeout */
304ed2a80fdSPablo de Lara 		if (all_ports_up == 1 || count == (MAX_CHECK_TIME - 1)) {
305ed2a80fdSPablo de Lara 			print_flag = 1;
306ed2a80fdSPablo de Lara 			printf("done\n");
307ed2a80fdSPablo de Lara 		}
308ed2a80fdSPablo de Lara 	}
309ed2a80fdSPablo de Lara }
310ed2a80fdSPablo de Lara 
311ed2a80fdSPablo de Lara /**
312ed2a80fdSPablo de Lara  * Main init function for the multi-process server app,
313ed2a80fdSPablo de Lara  * calls subfunctions to do each stage of the initialisation.
314ed2a80fdSPablo de Lara  */
315ed2a80fdSPablo de Lara int
316ed2a80fdSPablo de Lara init(int argc, char *argv[])
317ed2a80fdSPablo de Lara {
318ed2a80fdSPablo de Lara 	int retval;
319ed2a80fdSPablo de Lara 	const struct rte_memzone *mz;
320ed2a80fdSPablo de Lara 	uint8_t i, total_ports;
321ed2a80fdSPablo de Lara 
322ed2a80fdSPablo de Lara 	/* init EAL, parsing EAL args */
323ed2a80fdSPablo de Lara 	retval = rte_eal_init(argc, argv);
324ed2a80fdSPablo de Lara 	if (retval < 0)
325ed2a80fdSPablo de Lara 		return -1;
326ed2a80fdSPablo de Lara 	argc -= retval;
327ed2a80fdSPablo de Lara 	argv += retval;
328ed2a80fdSPablo de Lara 
329ed2a80fdSPablo de Lara 	/* get total number of ports */
330ed2a80fdSPablo de Lara 	total_ports = rte_eth_dev_count();
331ed2a80fdSPablo de Lara 
332ed2a80fdSPablo de Lara 	/* set up array for port data */
333ed2a80fdSPablo de Lara 	mz = rte_memzone_reserve(MZ_SHARED_INFO, sizeof(*info),
334ed2a80fdSPablo de Lara 				rte_socket_id(), NO_FLAGS);
335ed2a80fdSPablo de Lara 	if (mz == NULL)
336ed2a80fdSPablo de Lara 		rte_exit(EXIT_FAILURE, "Cannot reserve memory zone "
337ed2a80fdSPablo de Lara 				"for port information\n");
338ed2a80fdSPablo de Lara 	memset(mz->addr, 0, sizeof(*info));
339ed2a80fdSPablo de Lara 	info = mz->addr;
340ed2a80fdSPablo de Lara 
341ed2a80fdSPablo de Lara 	/* parse additional, application arguments */
342ed2a80fdSPablo de Lara 	retval = parse_app_args(total_ports, argc, argv);
343ed2a80fdSPablo de Lara 	if (retval != 0)
344ed2a80fdSPablo de Lara 		return -1;
345ed2a80fdSPablo de Lara 
346ed2a80fdSPablo de Lara 	/* initialise mbuf pools */
347ed2a80fdSPablo de Lara 	retval = init_mbuf_pools();
348ed2a80fdSPablo de Lara 	if (retval != 0)
349ed2a80fdSPablo de Lara 		rte_exit(EXIT_FAILURE, "Cannot create needed mbuf pools\n");
350ed2a80fdSPablo de Lara 
351ed2a80fdSPablo de Lara 	/* now initialise the ports we will use */
352ed2a80fdSPablo de Lara 	for (i = 0; i < info->num_ports; i++) {
353ed2a80fdSPablo de Lara 		retval = init_port(info->id[i]);
354ed2a80fdSPablo de Lara 		if (retval != 0)
355ed2a80fdSPablo de Lara 			rte_exit(EXIT_FAILURE, "Cannot initialise port %u\n",
356ed2a80fdSPablo de Lara 					(unsigned int) i);
357ed2a80fdSPablo de Lara 	}
358ed2a80fdSPablo de Lara 
359ed2a80fdSPablo de Lara 	check_all_ports_link_status(info->num_ports, (~0x0));
360ed2a80fdSPablo de Lara 
361ed2a80fdSPablo de Lara 	/* initialise the node queues/rings for inter-eu comms */
362ed2a80fdSPablo de Lara 	init_shm_rings();
363ed2a80fdSPablo de Lara 
364ed2a80fdSPablo de Lara 	/* Create the EFD table */
365ed2a80fdSPablo de Lara 	create_efd_table();
366ed2a80fdSPablo de Lara 
367ed2a80fdSPablo de Lara 	/* Populate the EFD table */
368ed2a80fdSPablo de Lara 	populate_efd_table();
369ed2a80fdSPablo de Lara 
370ed2a80fdSPablo de Lara 	/* Share the total number of nodes */
371ed2a80fdSPablo de Lara 	info->num_nodes = num_nodes;
372ed2a80fdSPablo de Lara 
373ed2a80fdSPablo de Lara 	/* Share the total number of flows */
374ed2a80fdSPablo de Lara 	info->num_flows = num_flows;
375ed2a80fdSPablo de Lara 	return 0;
376ed2a80fdSPablo de Lara }
377