xref: /f-stack/dpdk/lib/librte_graph/node.c (revision 2d9fd380)
1*2d9fd380Sjfb8856606 /* SPDX-License-Identifier: BSD-3-Clause
2*2d9fd380Sjfb8856606  * Copyright(C) 2020 Marvell International Ltd.
3*2d9fd380Sjfb8856606  */
4*2d9fd380Sjfb8856606 
5*2d9fd380Sjfb8856606 #include <stdbool.h>
6*2d9fd380Sjfb8856606 #include <stdio.h>
7*2d9fd380Sjfb8856606 #include <string.h>
8*2d9fd380Sjfb8856606 
9*2d9fd380Sjfb8856606 #include <rte_common.h>
10*2d9fd380Sjfb8856606 #include <rte_debug.h>
11*2d9fd380Sjfb8856606 #include <rte_eal.h>
12*2d9fd380Sjfb8856606 #include <rte_errno.h>
13*2d9fd380Sjfb8856606 #include <rte_string_fns.h>
14*2d9fd380Sjfb8856606 
15*2d9fd380Sjfb8856606 #include "graph_private.h"
16*2d9fd380Sjfb8856606 
17*2d9fd380Sjfb8856606 static struct node_head node_list = STAILQ_HEAD_INITIALIZER(node_list);
18*2d9fd380Sjfb8856606 static rte_node_t node_id;
19*2d9fd380Sjfb8856606 
20*2d9fd380Sjfb8856606 #define NODE_ID_CHECK(id) ID_CHECK(id, node_id)
21*2d9fd380Sjfb8856606 
22*2d9fd380Sjfb8856606 /* Private functions */
23*2d9fd380Sjfb8856606 struct node_head *
node_list_head_get(void)24*2d9fd380Sjfb8856606 node_list_head_get(void)
25*2d9fd380Sjfb8856606 {
26*2d9fd380Sjfb8856606 	return &node_list;
27*2d9fd380Sjfb8856606 }
28*2d9fd380Sjfb8856606 
29*2d9fd380Sjfb8856606 struct node *
node_from_name(const char * name)30*2d9fd380Sjfb8856606 node_from_name(const char *name)
31*2d9fd380Sjfb8856606 {
32*2d9fd380Sjfb8856606 	struct node *node;
33*2d9fd380Sjfb8856606 
34*2d9fd380Sjfb8856606 	STAILQ_FOREACH(node, &node_list, next)
35*2d9fd380Sjfb8856606 		if (strncmp(node->name, name, RTE_NODE_NAMESIZE) == 0)
36*2d9fd380Sjfb8856606 			return node;
37*2d9fd380Sjfb8856606 
38*2d9fd380Sjfb8856606 	return NULL;
39*2d9fd380Sjfb8856606 }
40*2d9fd380Sjfb8856606 
41*2d9fd380Sjfb8856606 static bool
node_has_duplicate_entry(const char * name)42*2d9fd380Sjfb8856606 node_has_duplicate_entry(const char *name)
43*2d9fd380Sjfb8856606 {
44*2d9fd380Sjfb8856606 	struct node *node;
45*2d9fd380Sjfb8856606 
46*2d9fd380Sjfb8856606 	/* Is duplicate name registered */
47*2d9fd380Sjfb8856606 	STAILQ_FOREACH(node, &node_list, next) {
48*2d9fd380Sjfb8856606 		if (strncmp(node->name, name, RTE_NODE_NAMESIZE) == 0) {
49*2d9fd380Sjfb8856606 			rte_errno = EEXIST;
50*2d9fd380Sjfb8856606 			return 1;
51*2d9fd380Sjfb8856606 		}
52*2d9fd380Sjfb8856606 	}
53*2d9fd380Sjfb8856606 	return 0;
54*2d9fd380Sjfb8856606 }
55*2d9fd380Sjfb8856606 
56*2d9fd380Sjfb8856606 /* Public functions */
57*2d9fd380Sjfb8856606 rte_node_t
__rte_node_register(const struct rte_node_register * reg)58*2d9fd380Sjfb8856606 __rte_node_register(const struct rte_node_register *reg)
59*2d9fd380Sjfb8856606 {
60*2d9fd380Sjfb8856606 	struct node *node;
61*2d9fd380Sjfb8856606 	rte_edge_t i;
62*2d9fd380Sjfb8856606 	size_t sz;
63*2d9fd380Sjfb8856606 
64*2d9fd380Sjfb8856606 	/* Limit Node specific metadata to one cacheline on 64B CL machine */
65*2d9fd380Sjfb8856606 	RTE_BUILD_BUG_ON((offsetof(struct rte_node, nodes) -
66*2d9fd380Sjfb8856606 			  offsetof(struct rte_node, ctx)) !=
67*2d9fd380Sjfb8856606 			 RTE_CACHE_LINE_MIN_SIZE);
68*2d9fd380Sjfb8856606 
69*2d9fd380Sjfb8856606 	graph_spinlock_lock();
70*2d9fd380Sjfb8856606 
71*2d9fd380Sjfb8856606 	/* Check sanity */
72*2d9fd380Sjfb8856606 	if (reg == NULL || reg->process == NULL) {
73*2d9fd380Sjfb8856606 		rte_errno = EINVAL;
74*2d9fd380Sjfb8856606 		goto fail;
75*2d9fd380Sjfb8856606 	}
76*2d9fd380Sjfb8856606 
77*2d9fd380Sjfb8856606 	/* Check for duplicate name */
78*2d9fd380Sjfb8856606 	if (node_has_duplicate_entry(reg->name))
79*2d9fd380Sjfb8856606 		goto fail;
80*2d9fd380Sjfb8856606 
81*2d9fd380Sjfb8856606 	sz = sizeof(struct node) + (reg->nb_edges * RTE_NODE_NAMESIZE);
82*2d9fd380Sjfb8856606 	node = calloc(1, sz);
83*2d9fd380Sjfb8856606 	if (node == NULL) {
84*2d9fd380Sjfb8856606 		rte_errno = ENOMEM;
85*2d9fd380Sjfb8856606 		goto fail;
86*2d9fd380Sjfb8856606 	}
87*2d9fd380Sjfb8856606 
88*2d9fd380Sjfb8856606 	/* Initialize the node */
89*2d9fd380Sjfb8856606 	if (rte_strscpy(node->name, reg->name, RTE_NODE_NAMESIZE) < 0) {
90*2d9fd380Sjfb8856606 		rte_errno = E2BIG;
91*2d9fd380Sjfb8856606 		goto free;
92*2d9fd380Sjfb8856606 	}
93*2d9fd380Sjfb8856606 	node->flags = reg->flags;
94*2d9fd380Sjfb8856606 	node->process = reg->process;
95*2d9fd380Sjfb8856606 	node->init = reg->init;
96*2d9fd380Sjfb8856606 	node->fini = reg->fini;
97*2d9fd380Sjfb8856606 	node->nb_edges = reg->nb_edges;
98*2d9fd380Sjfb8856606 	node->parent_id = reg->parent_id;
99*2d9fd380Sjfb8856606 	for (i = 0; i < reg->nb_edges; i++) {
100*2d9fd380Sjfb8856606 		if (rte_strscpy(node->next_nodes[i], reg->next_nodes[i],
101*2d9fd380Sjfb8856606 				RTE_NODE_NAMESIZE) < 0) {
102*2d9fd380Sjfb8856606 			rte_errno = E2BIG;
103*2d9fd380Sjfb8856606 			goto free;
104*2d9fd380Sjfb8856606 		}
105*2d9fd380Sjfb8856606 	}
106*2d9fd380Sjfb8856606 
107*2d9fd380Sjfb8856606 	node->id = node_id++;
108*2d9fd380Sjfb8856606 
109*2d9fd380Sjfb8856606 	/* Add the node at tail */
110*2d9fd380Sjfb8856606 	STAILQ_INSERT_TAIL(&node_list, node, next);
111*2d9fd380Sjfb8856606 	graph_spinlock_unlock();
112*2d9fd380Sjfb8856606 
113*2d9fd380Sjfb8856606 	return node->id;
114*2d9fd380Sjfb8856606 free:
115*2d9fd380Sjfb8856606 	free(node);
116*2d9fd380Sjfb8856606 fail:
117*2d9fd380Sjfb8856606 	graph_spinlock_unlock();
118*2d9fd380Sjfb8856606 	return RTE_NODE_ID_INVALID;
119*2d9fd380Sjfb8856606 }
120*2d9fd380Sjfb8856606 
121*2d9fd380Sjfb8856606 static int
clone_name(struct rte_node_register * reg,struct node * node,const char * name)122*2d9fd380Sjfb8856606 clone_name(struct rte_node_register *reg, struct node *node, const char *name)
123*2d9fd380Sjfb8856606 {
124*2d9fd380Sjfb8856606 	ssize_t sz, rc;
125*2d9fd380Sjfb8856606 
126*2d9fd380Sjfb8856606 #define SZ RTE_NODE_NAMESIZE
127*2d9fd380Sjfb8856606 	rc = rte_strscpy(reg->name, node->name, SZ);
128*2d9fd380Sjfb8856606 	if (rc < 0)
129*2d9fd380Sjfb8856606 		goto fail;
130*2d9fd380Sjfb8856606 	sz = rc;
131*2d9fd380Sjfb8856606 	rc = rte_strscpy(reg->name + sz, "-", RTE_MAX((int16_t)(SZ - sz), 0));
132*2d9fd380Sjfb8856606 	if (rc < 0)
133*2d9fd380Sjfb8856606 		goto fail;
134*2d9fd380Sjfb8856606 	sz += rc;
135*2d9fd380Sjfb8856606 	sz = rte_strscpy(reg->name + sz, name, RTE_MAX((int16_t)(SZ - sz), 0));
136*2d9fd380Sjfb8856606 	if (sz < 0)
137*2d9fd380Sjfb8856606 		goto fail;
138*2d9fd380Sjfb8856606 
139*2d9fd380Sjfb8856606 	return 0;
140*2d9fd380Sjfb8856606 fail:
141*2d9fd380Sjfb8856606 	rte_errno = E2BIG;
142*2d9fd380Sjfb8856606 	return -rte_errno;
143*2d9fd380Sjfb8856606 }
144*2d9fd380Sjfb8856606 
145*2d9fd380Sjfb8856606 static rte_node_t
node_clone(struct node * node,const char * name)146*2d9fd380Sjfb8856606 node_clone(struct node *node, const char *name)
147*2d9fd380Sjfb8856606 {
148*2d9fd380Sjfb8856606 	rte_node_t rc = RTE_NODE_ID_INVALID;
149*2d9fd380Sjfb8856606 	struct rte_node_register *reg;
150*2d9fd380Sjfb8856606 	rte_edge_t i;
151*2d9fd380Sjfb8856606 
152*2d9fd380Sjfb8856606 	/* Don't allow to clone a node from a cloned node */
153*2d9fd380Sjfb8856606 	if (node->parent_id != RTE_NODE_ID_INVALID) {
154*2d9fd380Sjfb8856606 		rte_errno = EEXIST;
155*2d9fd380Sjfb8856606 		goto fail;
156*2d9fd380Sjfb8856606 	}
157*2d9fd380Sjfb8856606 
158*2d9fd380Sjfb8856606 	/* Check for duplicate name */
159*2d9fd380Sjfb8856606 	if (node_has_duplicate_entry(name))
160*2d9fd380Sjfb8856606 		goto fail;
161*2d9fd380Sjfb8856606 
162*2d9fd380Sjfb8856606 	reg = calloc(1, sizeof(*reg) + (sizeof(char *) * node->nb_edges));
163*2d9fd380Sjfb8856606 	if (reg == NULL) {
164*2d9fd380Sjfb8856606 		rte_errno = ENOMEM;
165*2d9fd380Sjfb8856606 		goto fail;
166*2d9fd380Sjfb8856606 	}
167*2d9fd380Sjfb8856606 
168*2d9fd380Sjfb8856606 	/* Clone the source node */
169*2d9fd380Sjfb8856606 	reg->flags = node->flags;
170*2d9fd380Sjfb8856606 	reg->process = node->process;
171*2d9fd380Sjfb8856606 	reg->init = node->init;
172*2d9fd380Sjfb8856606 	reg->fini = node->fini;
173*2d9fd380Sjfb8856606 	reg->nb_edges = node->nb_edges;
174*2d9fd380Sjfb8856606 	reg->parent_id = node->id;
175*2d9fd380Sjfb8856606 
176*2d9fd380Sjfb8856606 	for (i = 0; i < node->nb_edges; i++)
177*2d9fd380Sjfb8856606 		reg->next_nodes[i] = node->next_nodes[i];
178*2d9fd380Sjfb8856606 
179*2d9fd380Sjfb8856606 	/* Naming ceremony of the new node. name is node->name + "-" + name */
180*2d9fd380Sjfb8856606 	if (clone_name(reg, node, name))
181*2d9fd380Sjfb8856606 		goto free;
182*2d9fd380Sjfb8856606 
183*2d9fd380Sjfb8856606 	rc = __rte_node_register(reg);
184*2d9fd380Sjfb8856606 free:
185*2d9fd380Sjfb8856606 	free(reg);
186*2d9fd380Sjfb8856606 fail:
187*2d9fd380Sjfb8856606 	return rc;
188*2d9fd380Sjfb8856606 }
189*2d9fd380Sjfb8856606 
190*2d9fd380Sjfb8856606 rte_node_t
rte_node_clone(rte_node_t id,const char * name)191*2d9fd380Sjfb8856606 rte_node_clone(rte_node_t id, const char *name)
192*2d9fd380Sjfb8856606 {
193*2d9fd380Sjfb8856606 	struct node *node;
194*2d9fd380Sjfb8856606 
195*2d9fd380Sjfb8856606 	NODE_ID_CHECK(id);
196*2d9fd380Sjfb8856606 	STAILQ_FOREACH(node, &node_list, next)
197*2d9fd380Sjfb8856606 		if (node->id == id)
198*2d9fd380Sjfb8856606 			return node_clone(node, name);
199*2d9fd380Sjfb8856606 
200*2d9fd380Sjfb8856606 fail:
201*2d9fd380Sjfb8856606 	return RTE_NODE_ID_INVALID;
202*2d9fd380Sjfb8856606 }
203*2d9fd380Sjfb8856606 
204*2d9fd380Sjfb8856606 rte_node_t
rte_node_from_name(const char * name)205*2d9fd380Sjfb8856606 rte_node_from_name(const char *name)
206*2d9fd380Sjfb8856606 {
207*2d9fd380Sjfb8856606 	struct node *node;
208*2d9fd380Sjfb8856606 
209*2d9fd380Sjfb8856606 	STAILQ_FOREACH(node, &node_list, next)
210*2d9fd380Sjfb8856606 		if (strncmp(node->name, name, RTE_NODE_NAMESIZE) == 0)
211*2d9fd380Sjfb8856606 			return node->id;
212*2d9fd380Sjfb8856606 
213*2d9fd380Sjfb8856606 	return RTE_NODE_ID_INVALID;
214*2d9fd380Sjfb8856606 }
215*2d9fd380Sjfb8856606 
216*2d9fd380Sjfb8856606 char *
rte_node_id_to_name(rte_node_t id)217*2d9fd380Sjfb8856606 rte_node_id_to_name(rte_node_t id)
218*2d9fd380Sjfb8856606 {
219*2d9fd380Sjfb8856606 	struct node *node;
220*2d9fd380Sjfb8856606 
221*2d9fd380Sjfb8856606 	NODE_ID_CHECK(id);
222*2d9fd380Sjfb8856606 	STAILQ_FOREACH(node, &node_list, next)
223*2d9fd380Sjfb8856606 		if (node->id == id)
224*2d9fd380Sjfb8856606 			return node->name;
225*2d9fd380Sjfb8856606 
226*2d9fd380Sjfb8856606 fail:
227*2d9fd380Sjfb8856606 	return NULL;
228*2d9fd380Sjfb8856606 }
229*2d9fd380Sjfb8856606 
230*2d9fd380Sjfb8856606 rte_edge_t
rte_node_edge_count(rte_node_t id)231*2d9fd380Sjfb8856606 rte_node_edge_count(rte_node_t id)
232*2d9fd380Sjfb8856606 {
233*2d9fd380Sjfb8856606 	struct node *node;
234*2d9fd380Sjfb8856606 
235*2d9fd380Sjfb8856606 	NODE_ID_CHECK(id);
236*2d9fd380Sjfb8856606 	STAILQ_FOREACH(node, &node_list, next)
237*2d9fd380Sjfb8856606 		if (node->id == id)
238*2d9fd380Sjfb8856606 			return node->nb_edges;
239*2d9fd380Sjfb8856606 fail:
240*2d9fd380Sjfb8856606 	return RTE_EDGE_ID_INVALID;
241*2d9fd380Sjfb8856606 }
242*2d9fd380Sjfb8856606 
243*2d9fd380Sjfb8856606 static rte_edge_t
edge_update(struct node * node,struct node * prev,rte_edge_t from,const char ** next_nodes,rte_edge_t nb_edges)244*2d9fd380Sjfb8856606 edge_update(struct node *node, struct node *prev, rte_edge_t from,
245*2d9fd380Sjfb8856606 	    const char **next_nodes, rte_edge_t nb_edges)
246*2d9fd380Sjfb8856606 {
247*2d9fd380Sjfb8856606 	rte_edge_t i, max_edges, count = 0;
248*2d9fd380Sjfb8856606 	struct node *new_node;
249*2d9fd380Sjfb8856606 	bool need_realloc;
250*2d9fd380Sjfb8856606 	size_t sz;
251*2d9fd380Sjfb8856606 
252*2d9fd380Sjfb8856606 	if (from == RTE_EDGE_ID_INVALID)
253*2d9fd380Sjfb8856606 		from = node->nb_edges;
254*2d9fd380Sjfb8856606 
255*2d9fd380Sjfb8856606 	/* Don't create hole in next_nodes[] list */
256*2d9fd380Sjfb8856606 	if (from > node->nb_edges) {
257*2d9fd380Sjfb8856606 		rte_errno = ENOMEM;
258*2d9fd380Sjfb8856606 		goto fail;
259*2d9fd380Sjfb8856606 	}
260*2d9fd380Sjfb8856606 
261*2d9fd380Sjfb8856606 	/* Remove me from list */
262*2d9fd380Sjfb8856606 	STAILQ_REMOVE(&node_list, node, node, next);
263*2d9fd380Sjfb8856606 
264*2d9fd380Sjfb8856606 	/* Allocate the storage space for new node if required */
265*2d9fd380Sjfb8856606 	max_edges = from + nb_edges;
266*2d9fd380Sjfb8856606 	need_realloc = max_edges > node->nb_edges;
267*2d9fd380Sjfb8856606 	if (need_realloc) {
268*2d9fd380Sjfb8856606 		sz = sizeof(struct node) + (max_edges * RTE_NODE_NAMESIZE);
269*2d9fd380Sjfb8856606 		new_node = realloc(node, sz);
270*2d9fd380Sjfb8856606 		if (new_node == NULL) {
271*2d9fd380Sjfb8856606 			rte_errno = ENOMEM;
272*2d9fd380Sjfb8856606 			goto restore;
273*2d9fd380Sjfb8856606 		} else {
274*2d9fd380Sjfb8856606 			node = new_node;
275*2d9fd380Sjfb8856606 		}
276*2d9fd380Sjfb8856606 	}
277*2d9fd380Sjfb8856606 
278*2d9fd380Sjfb8856606 	/* Update the new nodes name */
279*2d9fd380Sjfb8856606 	for (i = from; i < max_edges; i++, count++) {
280*2d9fd380Sjfb8856606 		if (rte_strscpy(node->next_nodes[i], next_nodes[count],
281*2d9fd380Sjfb8856606 				RTE_NODE_NAMESIZE) < 0) {
282*2d9fd380Sjfb8856606 			rte_errno = E2BIG;
283*2d9fd380Sjfb8856606 			goto restore;
284*2d9fd380Sjfb8856606 		}
285*2d9fd380Sjfb8856606 	}
286*2d9fd380Sjfb8856606 restore:
287*2d9fd380Sjfb8856606 	/* Update the linked list to point new node address in prev node */
288*2d9fd380Sjfb8856606 	if (prev)
289*2d9fd380Sjfb8856606 		STAILQ_INSERT_AFTER(&node_list, prev, node, next);
290*2d9fd380Sjfb8856606 	else
291*2d9fd380Sjfb8856606 		STAILQ_INSERT_HEAD(&node_list, node, next);
292*2d9fd380Sjfb8856606 
293*2d9fd380Sjfb8856606 	if (need_realloc)
294*2d9fd380Sjfb8856606 		node->nb_edges = max_edges;
295*2d9fd380Sjfb8856606 
296*2d9fd380Sjfb8856606 fail:
297*2d9fd380Sjfb8856606 	return count;
298*2d9fd380Sjfb8856606 }
299*2d9fd380Sjfb8856606 
300*2d9fd380Sjfb8856606 rte_edge_t
rte_node_edge_shrink(rte_node_t id,rte_edge_t size)301*2d9fd380Sjfb8856606 rte_node_edge_shrink(rte_node_t id, rte_edge_t size)
302*2d9fd380Sjfb8856606 {
303*2d9fd380Sjfb8856606 	rte_edge_t rc = RTE_EDGE_ID_INVALID;
304*2d9fd380Sjfb8856606 	struct node *node;
305*2d9fd380Sjfb8856606 
306*2d9fd380Sjfb8856606 	NODE_ID_CHECK(id);
307*2d9fd380Sjfb8856606 	graph_spinlock_lock();
308*2d9fd380Sjfb8856606 
309*2d9fd380Sjfb8856606 	STAILQ_FOREACH(node, &node_list, next) {
310*2d9fd380Sjfb8856606 		if (node->id == id) {
311*2d9fd380Sjfb8856606 			if (node->nb_edges < size) {
312*2d9fd380Sjfb8856606 				rte_errno = E2BIG;
313*2d9fd380Sjfb8856606 				goto fail;
314*2d9fd380Sjfb8856606 			}
315*2d9fd380Sjfb8856606 			node->nb_edges = size;
316*2d9fd380Sjfb8856606 			rc = size;
317*2d9fd380Sjfb8856606 			break;
318*2d9fd380Sjfb8856606 		}
319*2d9fd380Sjfb8856606 	}
320*2d9fd380Sjfb8856606 
321*2d9fd380Sjfb8856606 fail:
322*2d9fd380Sjfb8856606 	graph_spinlock_unlock();
323*2d9fd380Sjfb8856606 	return rc;
324*2d9fd380Sjfb8856606 }
325*2d9fd380Sjfb8856606 
326*2d9fd380Sjfb8856606 rte_edge_t
rte_node_edge_update(rte_node_t id,rte_edge_t from,const char ** next_nodes,uint16_t nb_edges)327*2d9fd380Sjfb8856606 rte_node_edge_update(rte_node_t id, rte_edge_t from, const char **next_nodes,
328*2d9fd380Sjfb8856606 		     uint16_t nb_edges)
329*2d9fd380Sjfb8856606 {
330*2d9fd380Sjfb8856606 	rte_edge_t rc = RTE_EDGE_ID_INVALID;
331*2d9fd380Sjfb8856606 	struct node *n, *prev;
332*2d9fd380Sjfb8856606 
333*2d9fd380Sjfb8856606 	NODE_ID_CHECK(id);
334*2d9fd380Sjfb8856606 	graph_spinlock_lock();
335*2d9fd380Sjfb8856606 
336*2d9fd380Sjfb8856606 	prev = NULL;
337*2d9fd380Sjfb8856606 	STAILQ_FOREACH(n, &node_list, next) {
338*2d9fd380Sjfb8856606 		if (n->id == id) {
339*2d9fd380Sjfb8856606 			rc = edge_update(n, prev, from, next_nodes, nb_edges);
340*2d9fd380Sjfb8856606 			break;
341*2d9fd380Sjfb8856606 		}
342*2d9fd380Sjfb8856606 		prev = n;
343*2d9fd380Sjfb8856606 	}
344*2d9fd380Sjfb8856606 
345*2d9fd380Sjfb8856606 	graph_spinlock_unlock();
346*2d9fd380Sjfb8856606 fail:
347*2d9fd380Sjfb8856606 	return rc;
348*2d9fd380Sjfb8856606 }
349*2d9fd380Sjfb8856606 
350*2d9fd380Sjfb8856606 static rte_node_t
node_copy_edges(struct node * node,char * next_nodes[])351*2d9fd380Sjfb8856606 node_copy_edges(struct node *node, char *next_nodes[])
352*2d9fd380Sjfb8856606 {
353*2d9fd380Sjfb8856606 	rte_edge_t i;
354*2d9fd380Sjfb8856606 
355*2d9fd380Sjfb8856606 	for (i = 0; i < node->nb_edges; i++)
356*2d9fd380Sjfb8856606 		next_nodes[i] = node->next_nodes[i];
357*2d9fd380Sjfb8856606 
358*2d9fd380Sjfb8856606 	return i;
359*2d9fd380Sjfb8856606 }
360*2d9fd380Sjfb8856606 
361*2d9fd380Sjfb8856606 rte_node_t
rte_node_edge_get(rte_node_t id,char * next_nodes[])362*2d9fd380Sjfb8856606 rte_node_edge_get(rte_node_t id, char *next_nodes[])
363*2d9fd380Sjfb8856606 {
364*2d9fd380Sjfb8856606 	rte_node_t rc = RTE_NODE_ID_INVALID;
365*2d9fd380Sjfb8856606 	struct node *node;
366*2d9fd380Sjfb8856606 
367*2d9fd380Sjfb8856606 	NODE_ID_CHECK(id);
368*2d9fd380Sjfb8856606 	graph_spinlock_lock();
369*2d9fd380Sjfb8856606 
370*2d9fd380Sjfb8856606 	STAILQ_FOREACH(node, &node_list, next) {
371*2d9fd380Sjfb8856606 		if (node->id == id) {
372*2d9fd380Sjfb8856606 			if (next_nodes == NULL)
373*2d9fd380Sjfb8856606 				rc = sizeof(char *) * node->nb_edges;
374*2d9fd380Sjfb8856606 			else
375*2d9fd380Sjfb8856606 				rc = node_copy_edges(node, next_nodes);
376*2d9fd380Sjfb8856606 			break;
377*2d9fd380Sjfb8856606 		}
378*2d9fd380Sjfb8856606 	}
379*2d9fd380Sjfb8856606 
380*2d9fd380Sjfb8856606 	graph_spinlock_unlock();
381*2d9fd380Sjfb8856606 fail:
382*2d9fd380Sjfb8856606 	return rc;
383*2d9fd380Sjfb8856606 }
384*2d9fd380Sjfb8856606 
385*2d9fd380Sjfb8856606 static void
node_scan_dump(FILE * f,rte_node_t id,bool all)386*2d9fd380Sjfb8856606 node_scan_dump(FILE *f, rte_node_t id, bool all)
387*2d9fd380Sjfb8856606 {
388*2d9fd380Sjfb8856606 	struct node *node;
389*2d9fd380Sjfb8856606 
390*2d9fd380Sjfb8856606 	RTE_ASSERT(f != NULL);
391*2d9fd380Sjfb8856606 	NODE_ID_CHECK(id);
392*2d9fd380Sjfb8856606 
393*2d9fd380Sjfb8856606 	STAILQ_FOREACH(node, &node_list, next) {
394*2d9fd380Sjfb8856606 		if (all == true) {
395*2d9fd380Sjfb8856606 			node_dump(f, node);
396*2d9fd380Sjfb8856606 		} else if (node->id == id) {
397*2d9fd380Sjfb8856606 			node_dump(f, node);
398*2d9fd380Sjfb8856606 			return;
399*2d9fd380Sjfb8856606 		}
400*2d9fd380Sjfb8856606 	}
401*2d9fd380Sjfb8856606 fail:
402*2d9fd380Sjfb8856606 	return;
403*2d9fd380Sjfb8856606 }
404*2d9fd380Sjfb8856606 
405*2d9fd380Sjfb8856606 void
rte_node_dump(FILE * f,rte_node_t id)406*2d9fd380Sjfb8856606 rte_node_dump(FILE *f, rte_node_t id)
407*2d9fd380Sjfb8856606 {
408*2d9fd380Sjfb8856606 	node_scan_dump(f, id, false);
409*2d9fd380Sjfb8856606 }
410*2d9fd380Sjfb8856606 
411*2d9fd380Sjfb8856606 void
rte_node_list_dump(FILE * f)412*2d9fd380Sjfb8856606 rte_node_list_dump(FILE *f)
413*2d9fd380Sjfb8856606 {
414*2d9fd380Sjfb8856606 	node_scan_dump(f, 0, true);
415*2d9fd380Sjfb8856606 }
416*2d9fd380Sjfb8856606 
417*2d9fd380Sjfb8856606 rte_node_t
rte_node_max_count(void)418*2d9fd380Sjfb8856606 rte_node_max_count(void)
419*2d9fd380Sjfb8856606 {
420*2d9fd380Sjfb8856606 	return node_id;
421*2d9fd380Sjfb8856606 }
422