xref: /freebsd/sys/netgraph/ng_base.c (revision 262e143bd46171a6415a5b28af260a5efa2a3db8)
1 /*
2  * ng_base.c
3  */
4 
5 /*-
6  * Copyright (c) 1996-1999 Whistle Communications, Inc.
7  * All rights reserved.
8  *
9  * Subject to the following obligations and disclaimer of warranty, use and
10  * redistribution of this software, in source or object code forms, with or
11  * without modifications are expressly permitted by Whistle Communications;
12  * provided, however, that:
13  * 1. Any and all reproductions of the source or object code must include the
14  *    copyright notice above and the following disclaimer of warranties; and
15  * 2. No rights are granted, in any manner or form, to use Whistle
16  *    Communications, Inc. trademarks, including the mark "WHISTLE
17  *    COMMUNICATIONS" on advertising, endorsements, or otherwise except as
18  *    such appears in the above copyright notice or in the software.
19  *
20  * THIS SOFTWARE IS BEING PROVIDED BY WHISTLE COMMUNICATIONS "AS IS", AND
21  * TO THE MAXIMUM EXTENT PERMITTED BY LAW, WHISTLE COMMUNICATIONS MAKES NO
22  * REPRESENTATIONS OR WARRANTIES, EXPRESS OR IMPLIED, REGARDING THIS SOFTWARE,
23  * INCLUDING WITHOUT LIMITATION, ANY AND ALL IMPLIED WARRANTIES OF
24  * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE, OR NON-INFRINGEMENT.
25  * WHISTLE COMMUNICATIONS DOES NOT WARRANT, GUARANTEE, OR MAKE ANY
26  * REPRESENTATIONS REGARDING THE USE OF, OR THE RESULTS OF THE USE OF THIS
27  * SOFTWARE IN TERMS OF ITS CORRECTNESS, ACCURACY, RELIABILITY OR OTHERWISE.
28  * IN NO EVENT SHALL WHISTLE COMMUNICATIONS BE LIABLE FOR ANY DAMAGES
29  * RESULTING FROM OR ARISING OUT OF ANY USE OF THIS SOFTWARE, INCLUDING
30  * WITHOUT LIMITATION, ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY,
31  * PUNITIVE, OR CONSEQUENTIAL DAMAGES, PROCUREMENT OF SUBSTITUTE GOODS OR
32  * SERVICES, LOSS OF USE, DATA OR PROFITS, HOWEVER CAUSED AND UNDER ANY
33  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
34  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
35  * THIS SOFTWARE, EVEN IF WHISTLE COMMUNICATIONS IS ADVISED OF THE POSSIBILITY
36  * OF SUCH DAMAGE.
37  *
38  * Authors: Julian Elischer <julian@freebsd.org>
39  *          Archie Cobbs <archie@freebsd.org>
40  *
41  * $FreeBSD$
42  * $Whistle: ng_base.c,v 1.39 1999/01/28 23:54:53 julian Exp $
43  */
44 
45 /*
46  * This file implements the base netgraph code.
47  */
48 
49 #include <sys/param.h>
50 #include <sys/systm.h>
51 #include <sys/ctype.h>
52 #include <sys/errno.h>
53 #include <sys/kdb.h>
54 #include <sys/kernel.h>
55 #include <sys/limits.h>
56 #include <sys/malloc.h>
57 #include <sys/mbuf.h>
58 #include <sys/queue.h>
59 #include <sys/sysctl.h>
60 #include <sys/syslog.h>
61 
62 #include <net/netisr.h>
63 
64 #include <netgraph/ng_message.h>
65 #include <netgraph/netgraph.h>
66 #include <netgraph/ng_parse.h>
67 
68 MODULE_VERSION(netgraph, NG_ABI_VERSION);
69 
70 /* List of all active nodes */
71 static LIST_HEAD(, ng_node) ng_nodelist;
72 static struct mtx	ng_nodelist_mtx;
73 
74 /* Mutex to protect topology events. */
75 static struct mtx	ng_topo_mtx;
76 
77 #ifdef	NETGRAPH_DEBUG
78 static struct mtx	ngq_mtx;	/* protects the queue item list */
79 
80 static SLIST_HEAD(, ng_node) ng_allnodes;
81 static LIST_HEAD(, ng_node) ng_freenodes; /* in debug, we never free() them */
82 static SLIST_HEAD(, ng_hook) ng_allhooks;
83 static LIST_HEAD(, ng_hook) ng_freehooks; /* in debug, we never free() them */
84 
85 static void ng_dumpitems(void);
86 static void ng_dumpnodes(void);
87 static void ng_dumphooks(void);
88 
89 #endif	/* NETGRAPH_DEBUG */
90 /*
91  * DEAD versions of the structures.
92  * In order to avoid races, it is sometimes neccesary to point
93  * at SOMETHING even though theoretically, the current entity is
94  * INVALID. Use these to avoid these races.
95  */
96 struct ng_type ng_deadtype = {
97 	NG_ABI_VERSION,
98 	"dead",
99 	NULL,	/* modevent */
100 	NULL,	/* constructor */
101 	NULL,	/* rcvmsg */
102 	NULL,	/* shutdown */
103 	NULL,	/* newhook */
104 	NULL,	/* findhook */
105 	NULL,	/* connect */
106 	NULL,	/* rcvdata */
107 	NULL,	/* disconnect */
108 	NULL, 	/* cmdlist */
109 };
110 
111 struct ng_node ng_deadnode = {
112 	"dead",
113 	&ng_deadtype,
114 	NGF_INVALID,
115 	1,	/* refs */
116 	0,	/* numhooks */
117 	NULL,	/* private */
118 	0,	/* ID */
119 	LIST_HEAD_INITIALIZER(ng_deadnode.hooks),
120 	{},	/* all_nodes list entry */
121 	{},	/* id hashtable list entry */
122 	{},	/* workqueue entry */
123 	{	0,
124 		{}, /* should never use! (should hang) */
125 		NULL,
126 		&ng_deadnode.nd_input_queue.queue,
127 		&ng_deadnode
128 	},
129 #ifdef	NETGRAPH_DEBUG
130 	ND_MAGIC,
131 	__FILE__,
132 	__LINE__,
133 	{NULL}
134 #endif	/* NETGRAPH_DEBUG */
135 };
136 
137 struct ng_hook ng_deadhook = {
138 	"dead",
139 	NULL,		/* private */
140 	HK_INVALID | HK_DEAD,
141 	1,		/* refs always >= 1 */
142 	0,		/* undefined data link type */
143 	&ng_deadhook,	/* Peer is self */
144 	&ng_deadnode,	/* attached to deadnode */
145 	{},		/* hooks list */
146 	NULL,		/* override rcvmsg() */
147 	NULL,		/* override rcvdata() */
148 #ifdef	NETGRAPH_DEBUG
149 	HK_MAGIC,
150 	__FILE__,
151 	__LINE__,
152 	{NULL}
153 #endif	/* NETGRAPH_DEBUG */
154 };
155 
156 /*
157  * END DEAD STRUCTURES
158  */
159 /* List nodes with unallocated work */
160 static TAILQ_HEAD(, ng_node) ng_worklist = TAILQ_HEAD_INITIALIZER(ng_worklist);
161 static struct mtx	ng_worklist_mtx;   /* MUST LOCK NODE FIRST */
162 
163 /* List of installed types */
164 static LIST_HEAD(, ng_type) ng_typelist;
165 static struct mtx	ng_typelist_mtx;
166 
167 /* Hash related definitions */
168 /* XXX Don't need to initialise them because it's a LIST */
169 #define NG_ID_HASH_SIZE 32 /* most systems wont need even this many */
170 static LIST_HEAD(, ng_node) ng_ID_hash[NG_ID_HASH_SIZE];
171 static struct mtx	ng_idhash_mtx;
172 /* Method to find a node.. used twice so do it here */
173 #define NG_IDHASH_FN(ID) ((ID) % (NG_ID_HASH_SIZE))
174 #define NG_IDHASH_FIND(ID, node)					\
175 	do { 								\
176 		mtx_assert(&ng_idhash_mtx, MA_OWNED);			\
177 		LIST_FOREACH(node, &ng_ID_hash[NG_IDHASH_FN(ID)],	\
178 						nd_idnodes) {		\
179 			if (NG_NODE_IS_VALID(node)			\
180 			&& (NG_NODE_ID(node) == ID)) {			\
181 				break;					\
182 			}						\
183 		}							\
184 	} while (0)
185 
186 
187 /* Internal functions */
188 static int	ng_add_hook(node_p node, const char *name, hook_p * hookp);
189 static int	ng_generic_msg(node_p here, item_p item, hook_p lasthook);
190 static ng_ID_t	ng_decodeidname(const char *name);
191 static int	ngb_mod_event(module_t mod, int event, void *data);
192 static void	ng_worklist_remove(node_p node);
193 static void	ngintr(void);
194 static int	ng_apply_item(node_p node, item_p item, int rw);
195 static void	ng_flush_input_queue(struct ng_queue * ngq);
196 static void	ng_setisr(node_p node);
197 static node_p	ng_ID2noderef(ng_ID_t ID);
198 static int	ng_con_nodes(node_p node, const char *name, node_p node2,
199 							const char *name2);
200 static void	ng_con_part2(node_p node, hook_p hook, void *arg1, int arg2);
201 static void	ng_con_part3(node_p node, hook_p hook, void *arg1, int arg2);
202 static int	ng_mkpeer(node_p node, const char *name,
203 						const char *name2, char *type);
204 
205 /* Imported, these used to be externally visible, some may go back. */
206 void	ng_destroy_hook(hook_p hook);
207 node_p	ng_name2noderef(node_p node, const char *name);
208 int	ng_path2noderef(node_p here, const char *path,
209 	node_p *dest, hook_p *lasthook);
210 int	ng_make_node(const char *type, node_p *nodepp);
211 int	ng_path_parse(char *addr, char **node, char **path, char **hook);
212 void	ng_rmnode(node_p node, hook_p dummy1, void *dummy2, int dummy3);
213 void	ng_unname(node_p node);
214 
215 
216 /* Our own netgraph malloc type */
217 MALLOC_DEFINE(M_NETGRAPH, "netgraph", "netgraph structures and ctrl messages");
218 MALLOC_DEFINE(M_NETGRAPH_HOOK, "netgraph_hook", "netgraph hook structures");
219 MALLOC_DEFINE(M_NETGRAPH_NODE, "netgraph_node", "netgraph node structures");
220 MALLOC_DEFINE(M_NETGRAPH_ITEM, "netgraph_item", "netgraph item structures");
221 MALLOC_DEFINE(M_NETGRAPH_MSG, "netgraph_msg", "netgraph name storage");
222 
223 /* Should not be visible outside this file */
224 
225 #define _NG_ALLOC_HOOK(hook) \
226 	MALLOC(hook, hook_p, sizeof(*hook), M_NETGRAPH_HOOK, M_NOWAIT | M_ZERO)
227 #define _NG_ALLOC_NODE(node) \
228 	MALLOC(node, node_p, sizeof(*node), M_NETGRAPH_NODE, M_NOWAIT | M_ZERO)
229 
230 #ifdef NETGRAPH_DEBUG /*----------------------------------------------*/
231 /*
232  * In debug mode:
233  * In an attempt to help track reference count screwups
234  * we do not free objects back to the malloc system, but keep them
235  * in a local cache where we can examine them and keep information safely
236  * after they have been freed.
237  * We use this scheme for nodes and hooks, and to some extent for items.
238  */
239 static __inline hook_p
240 ng_alloc_hook(void)
241 {
242 	hook_p hook;
243 	SLIST_ENTRY(ng_hook) temp;
244 	mtx_lock(&ng_nodelist_mtx);
245 	hook = LIST_FIRST(&ng_freehooks);
246 	if (hook) {
247 		LIST_REMOVE(hook, hk_hooks);
248 		bcopy(&hook->hk_all, &temp, sizeof(temp));
249 		bzero(hook, sizeof(struct ng_hook));
250 		bcopy(&temp, &hook->hk_all, sizeof(temp));
251 		mtx_unlock(&ng_nodelist_mtx);
252 		hook->hk_magic = HK_MAGIC;
253 	} else {
254 		mtx_unlock(&ng_nodelist_mtx);
255 		_NG_ALLOC_HOOK(hook);
256 		if (hook) {
257 			hook->hk_magic = HK_MAGIC;
258 			mtx_lock(&ng_nodelist_mtx);
259 			SLIST_INSERT_HEAD(&ng_allhooks, hook, hk_all);
260 			mtx_unlock(&ng_nodelist_mtx);
261 		}
262 	}
263 	return (hook);
264 }
265 
266 static __inline node_p
267 ng_alloc_node(void)
268 {
269 	node_p node;
270 	SLIST_ENTRY(ng_node) temp;
271 	mtx_lock(&ng_nodelist_mtx);
272 	node = LIST_FIRST(&ng_freenodes);
273 	if (node) {
274 		LIST_REMOVE(node, nd_nodes);
275 		bcopy(&node->nd_all, &temp, sizeof(temp));
276 		bzero(node, sizeof(struct ng_node));
277 		bcopy(&temp, &node->nd_all, sizeof(temp));
278 		mtx_unlock(&ng_nodelist_mtx);
279 		node->nd_magic = ND_MAGIC;
280 	} else {
281 		mtx_unlock(&ng_nodelist_mtx);
282 		_NG_ALLOC_NODE(node);
283 		if (node) {
284 			node->nd_magic = ND_MAGIC;
285 			mtx_lock(&ng_nodelist_mtx);
286 			SLIST_INSERT_HEAD(&ng_allnodes, node, nd_all);
287 			mtx_unlock(&ng_nodelist_mtx);
288 		}
289 	}
290 	return (node);
291 }
292 
293 #define NG_ALLOC_HOOK(hook) do { (hook) = ng_alloc_hook(); } while (0)
294 #define NG_ALLOC_NODE(node) do { (node) = ng_alloc_node(); } while (0)
295 
296 
297 #define NG_FREE_HOOK(hook)						\
298 	do {								\
299 		mtx_lock(&ng_nodelist_mtx);			\
300 		LIST_INSERT_HEAD(&ng_freehooks, hook, hk_hooks);	\
301 		hook->hk_magic = 0;					\
302 		mtx_unlock(&ng_nodelist_mtx);			\
303 	} while (0)
304 
305 #define NG_FREE_NODE(node)						\
306 	do {								\
307 		mtx_lock(&ng_nodelist_mtx);			\
308 		LIST_INSERT_HEAD(&ng_freenodes, node, nd_nodes);	\
309 		node->nd_magic = 0;					\
310 		mtx_unlock(&ng_nodelist_mtx);			\
311 	} while (0)
312 
313 #else /* NETGRAPH_DEBUG */ /*----------------------------------------------*/
314 
315 #define NG_ALLOC_HOOK(hook) _NG_ALLOC_HOOK(hook)
316 #define NG_ALLOC_NODE(node) _NG_ALLOC_NODE(node)
317 
318 #define NG_FREE_HOOK(hook) do { FREE((hook), M_NETGRAPH_HOOK); } while (0)
319 #define NG_FREE_NODE(node) do { FREE((node), M_NETGRAPH_NODE); } while (0)
320 
321 #endif /* NETGRAPH_DEBUG */ /*----------------------------------------------*/
322 
323 /* Set this to kdb_enter("X") to catch all errors as they occur */
324 #ifndef TRAP_ERROR
325 #define TRAP_ERROR()
326 #endif
327 
328 static	ng_ID_t nextID = 1;
329 
330 #ifdef INVARIANTS
331 #define CHECK_DATA_MBUF(m)	do {					\
332 		struct mbuf *n;						\
333 		int total;						\
334 									\
335 		M_ASSERTPKTHDR(m);					\
336 		for (total = 0, n = (m); n != NULL; n = n->m_next) {	\
337 			total += n->m_len;				\
338 			if (n->m_nextpkt != NULL)			\
339 				panic("%s: m_nextpkt", __func__);	\
340 		}							\
341 									\
342 		if ((m)->m_pkthdr.len != total) {			\
343 			panic("%s: %d != %d",				\
344 			    __func__, (m)->m_pkthdr.len, total);	\
345 		}							\
346 	} while (0)
347 #else
348 #define CHECK_DATA_MBUF(m)
349 #endif
350 
351 
352 /************************************************************************
353 	Parse type definitions for generic messages
354 ************************************************************************/
355 
356 /* Handy structure parse type defining macro */
357 #define DEFINE_PARSE_STRUCT_TYPE(lo, up, args)				\
358 static const struct ng_parse_struct_field				\
359 	ng_ ## lo ## _type_fields[] = NG_GENERIC_ ## up ## _INFO args;	\
360 static const struct ng_parse_type ng_generic_ ## lo ## _type = {	\
361 	&ng_parse_struct_type,						\
362 	&ng_ ## lo ## _type_fields					\
363 }
364 
365 DEFINE_PARSE_STRUCT_TYPE(mkpeer, MKPEER, ());
366 DEFINE_PARSE_STRUCT_TYPE(connect, CONNECT, ());
367 DEFINE_PARSE_STRUCT_TYPE(name, NAME, ());
368 DEFINE_PARSE_STRUCT_TYPE(rmhook, RMHOOK, ());
369 DEFINE_PARSE_STRUCT_TYPE(nodeinfo, NODEINFO, ());
370 DEFINE_PARSE_STRUCT_TYPE(typeinfo, TYPEINFO, ());
371 DEFINE_PARSE_STRUCT_TYPE(linkinfo, LINKINFO, (&ng_generic_nodeinfo_type));
372 
373 /* Get length of an array when the length is stored as a 32 bit
374    value immediately preceding the array -- as with struct namelist
375    and struct typelist. */
376 static int
377 ng_generic_list_getLength(const struct ng_parse_type *type,
378 	const u_char *start, const u_char *buf)
379 {
380 	return *((const u_int32_t *)(buf - 4));
381 }
382 
383 /* Get length of the array of struct linkinfo inside a struct hooklist */
384 static int
385 ng_generic_linkinfo_getLength(const struct ng_parse_type *type,
386 	const u_char *start, const u_char *buf)
387 {
388 	const struct hooklist *hl = (const struct hooklist *)start;
389 
390 	return hl->nodeinfo.hooks;
391 }
392 
393 /* Array type for a variable length array of struct namelist */
394 static const struct ng_parse_array_info ng_nodeinfoarray_type_info = {
395 	&ng_generic_nodeinfo_type,
396 	&ng_generic_list_getLength
397 };
398 static const struct ng_parse_type ng_generic_nodeinfoarray_type = {
399 	&ng_parse_array_type,
400 	&ng_nodeinfoarray_type_info
401 };
402 
403 /* Array type for a variable length array of struct typelist */
404 static const struct ng_parse_array_info ng_typeinfoarray_type_info = {
405 	&ng_generic_typeinfo_type,
406 	&ng_generic_list_getLength
407 };
408 static const struct ng_parse_type ng_generic_typeinfoarray_type = {
409 	&ng_parse_array_type,
410 	&ng_typeinfoarray_type_info
411 };
412 
413 /* Array type for array of struct linkinfo in struct hooklist */
414 static const struct ng_parse_array_info ng_generic_linkinfo_array_type_info = {
415 	&ng_generic_linkinfo_type,
416 	&ng_generic_linkinfo_getLength
417 };
418 static const struct ng_parse_type ng_generic_linkinfo_array_type = {
419 	&ng_parse_array_type,
420 	&ng_generic_linkinfo_array_type_info
421 };
422 
423 DEFINE_PARSE_STRUCT_TYPE(typelist, TYPELIST, (&ng_generic_nodeinfoarray_type));
424 DEFINE_PARSE_STRUCT_TYPE(hooklist, HOOKLIST,
425 	(&ng_generic_nodeinfo_type, &ng_generic_linkinfo_array_type));
426 DEFINE_PARSE_STRUCT_TYPE(listnodes, LISTNODES,
427 	(&ng_generic_nodeinfoarray_type));
428 
429 /* List of commands and how to convert arguments to/from ASCII */
430 static const struct ng_cmdlist ng_generic_cmds[] = {
431 	{
432 	  NGM_GENERIC_COOKIE,
433 	  NGM_SHUTDOWN,
434 	  "shutdown",
435 	  NULL,
436 	  NULL
437 	},
438 	{
439 	  NGM_GENERIC_COOKIE,
440 	  NGM_MKPEER,
441 	  "mkpeer",
442 	  &ng_generic_mkpeer_type,
443 	  NULL
444 	},
445 	{
446 	  NGM_GENERIC_COOKIE,
447 	  NGM_CONNECT,
448 	  "connect",
449 	  &ng_generic_connect_type,
450 	  NULL
451 	},
452 	{
453 	  NGM_GENERIC_COOKIE,
454 	  NGM_NAME,
455 	  "name",
456 	  &ng_generic_name_type,
457 	  NULL
458 	},
459 	{
460 	  NGM_GENERIC_COOKIE,
461 	  NGM_RMHOOK,
462 	  "rmhook",
463 	  &ng_generic_rmhook_type,
464 	  NULL
465 	},
466 	{
467 	  NGM_GENERIC_COOKIE,
468 	  NGM_NODEINFO,
469 	  "nodeinfo",
470 	  NULL,
471 	  &ng_generic_nodeinfo_type
472 	},
473 	{
474 	  NGM_GENERIC_COOKIE,
475 	  NGM_LISTHOOKS,
476 	  "listhooks",
477 	  NULL,
478 	  &ng_generic_hooklist_type
479 	},
480 	{
481 	  NGM_GENERIC_COOKIE,
482 	  NGM_LISTNAMES,
483 	  "listnames",
484 	  NULL,
485 	  &ng_generic_listnodes_type	/* same as NGM_LISTNODES */
486 	},
487 	{
488 	  NGM_GENERIC_COOKIE,
489 	  NGM_LISTNODES,
490 	  "listnodes",
491 	  NULL,
492 	  &ng_generic_listnodes_type
493 	},
494 	{
495 	  NGM_GENERIC_COOKIE,
496 	  NGM_LISTTYPES,
497 	  "listtypes",
498 	  NULL,
499 	  &ng_generic_typeinfo_type
500 	},
501 	{
502 	  NGM_GENERIC_COOKIE,
503 	  NGM_TEXT_CONFIG,
504 	  "textconfig",
505 	  NULL,
506 	  &ng_parse_string_type
507 	},
508 	{
509 	  NGM_GENERIC_COOKIE,
510 	  NGM_TEXT_STATUS,
511 	  "textstatus",
512 	  NULL,
513 	  &ng_parse_string_type
514 	},
515 	{
516 	  NGM_GENERIC_COOKIE,
517 	  NGM_ASCII2BINARY,
518 	  "ascii2binary",
519 	  &ng_parse_ng_mesg_type,
520 	  &ng_parse_ng_mesg_type
521 	},
522 	{
523 	  NGM_GENERIC_COOKIE,
524 	  NGM_BINARY2ASCII,
525 	  "binary2ascii",
526 	  &ng_parse_ng_mesg_type,
527 	  &ng_parse_ng_mesg_type
528 	},
529 	{ 0 }
530 };
531 
532 /************************************************************************
533 			Node routines
534 ************************************************************************/
535 
536 /*
537  * Instantiate a node of the requested type
538  */
539 int
540 ng_make_node(const char *typename, node_p *nodepp)
541 {
542 	struct ng_type *type;
543 	int	error;
544 
545 	/* Check that the type makes sense */
546 	if (typename == NULL) {
547 		TRAP_ERROR();
548 		return (EINVAL);
549 	}
550 
551 	/* Locate the node type. If we fail we return. Do not try to load
552 	 * module.
553 	 */
554 	if ((type = ng_findtype(typename)) == NULL)
555 		return (ENXIO);
556 
557 	/*
558 	 * If we have a constructor, then make the node and
559 	 * call the constructor to do type specific initialisation.
560 	 */
561 	if (type->constructor != NULL) {
562 		if ((error = ng_make_node_common(type, nodepp)) == 0) {
563 			if ((error = ((*type->constructor)(*nodepp)) != 0)) {
564 				NG_NODE_UNREF(*nodepp);
565 			}
566 		}
567 	} else {
568 		/*
569 		 * Node has no constructor. We cannot ask for one
570 		 * to be made. It must be brought into existance by
571 		 * some external agency. The external agency should
572 		 * call ng_make_node_common() directly to get the
573 		 * netgraph part initialised.
574 		 */
575 		TRAP_ERROR();
576 		error = EINVAL;
577 	}
578 	return (error);
579 }
580 
581 /*
582  * Generic node creation. Called by node initialisation for externally
583  * instantiated nodes (e.g. hardware, sockets, etc ).
584  * The returned node has a reference count of 1.
585  */
586 int
587 ng_make_node_common(struct ng_type *type, node_p *nodepp)
588 {
589 	node_p node;
590 
591 	/* Require the node type to have been already installed */
592 	if (ng_findtype(type->name) == NULL) {
593 		TRAP_ERROR();
594 		return (EINVAL);
595 	}
596 
597 	/* Make a node and try attach it to the type */
598 	NG_ALLOC_NODE(node);
599 	if (node == NULL) {
600 		TRAP_ERROR();
601 		return (ENOMEM);
602 	}
603 	node->nd_type = type;
604 	NG_NODE_REF(node);				/* note reference */
605 	type->refs++;
606 
607 	mtx_init(&node->nd_input_queue.q_mtx, "ng_node", NULL, MTX_SPIN);
608 	node->nd_input_queue.queue = NULL;
609 	node->nd_input_queue.last = &node->nd_input_queue.queue;
610 	node->nd_input_queue.q_flags = 0;
611 	node->nd_input_queue.q_node = node;
612 
613 	/* Initialize hook list for new node */
614 	LIST_INIT(&node->nd_hooks);
615 
616 	/* Link us into the node linked list */
617 	mtx_lock(&ng_nodelist_mtx);
618 	LIST_INSERT_HEAD(&ng_nodelist, node, nd_nodes);
619 	mtx_unlock(&ng_nodelist_mtx);
620 
621 
622 	/* get an ID and put us in the hash chain */
623 	mtx_lock(&ng_idhash_mtx);
624 	for (;;) { /* wrap protection, even if silly */
625 		node_p node2 = NULL;
626 		node->nd_ID = nextID++; /* 137/second for 1 year before wrap */
627 
628 		/* Is there a problem with the new number? */
629 		NG_IDHASH_FIND(node->nd_ID, node2); /* already taken? */
630 		if ((node->nd_ID != 0) && (node2 == NULL)) {
631 			break;
632 		}
633 	}
634 	LIST_INSERT_HEAD(&ng_ID_hash[NG_IDHASH_FN(node->nd_ID)],
635 							node, nd_idnodes);
636 	mtx_unlock(&ng_idhash_mtx);
637 
638 	/* Done */
639 	*nodepp = node;
640 	return (0);
641 }
642 
643 /*
644  * Forceably start the shutdown process on a node. Either call
645  * it's shutdown method, or do the default shutdown if there is
646  * no type-specific method.
647  *
648  * We can only be called form a shutdown message, so we know we have
649  * a writer lock, and therefore exclusive access. It also means
650  * that we should not be on the work queue, but we check anyhow.
651  *
652  * Persistent node types must have a type-specific method which
653  * Allocates a new node in which case, this one is irretrievably going away,
654  * or cleans up anything it needs, and just makes the node valid again,
655  * in which case we allow the node to survive.
656  *
657  * XXX We need to think of how to tell a persistant node that we
658  * REALLY need to go away because the hardware has gone or we
659  * are rebooting.... etc.
660  */
661 void
662 ng_rmnode(node_p node, hook_p dummy1, void *dummy2, int dummy3)
663 {
664 	hook_p hook;
665 
666 	/* Check if it's already shutting down */
667 	if ((node->nd_flags & NGF_CLOSING) != 0)
668 		return;
669 
670 	if (node == &ng_deadnode) {
671 		printf ("shutdown called on deadnode\n");
672 		return;
673 	}
674 
675 	/* Add an extra reference so it doesn't go away during this */
676 	NG_NODE_REF(node);
677 
678 	/*
679 	 * Mark it invalid so any newcomers know not to try use it
680 	 * Also add our own mark so we can't recurse
681 	 * note that NGF_INVALID does not do this as it's also set during
682 	 * creation
683 	 */
684 	node->nd_flags |= NGF_INVALID|NGF_CLOSING;
685 
686 	/* If node has its pre-shutdown method, then call it first*/
687 	if (node->nd_type && node->nd_type->close)
688 		(*node->nd_type->close)(node);
689 
690 	/* Notify all remaining connected nodes to disconnect */
691 	while ((hook = LIST_FIRST(&node->nd_hooks)) != NULL)
692 		ng_destroy_hook(hook);
693 
694 	/*
695 	 * Drain the input queue forceably.
696 	 * it has no hooks so what's it going to do, bleed on someone?
697 	 * Theoretically we came here from a queue entry that was added
698 	 * Just before the queue was closed, so it should be empty anyway.
699 	 * Also removes us from worklist if needed.
700 	 */
701 	ng_flush_input_queue(&node->nd_input_queue);
702 
703 	/* Ask the type if it has anything to do in this case */
704 	if (node->nd_type && node->nd_type->shutdown) {
705 		(*node->nd_type->shutdown)(node);
706 		if (NG_NODE_IS_VALID(node)) {
707 			/*
708 			 * Well, blow me down if the node code hasn't declared
709 			 * that it doesn't want to die.
710 			 * Presumably it is a persistant node.
711 			 * If we REALLY want it to go away,
712 			 *  e.g. hardware going away,
713 			 * Our caller should set NGF_REALLY_DIE in nd_flags.
714 			 */
715 			node->nd_flags &= ~(NGF_INVALID|NGF_CLOSING);
716 			NG_NODE_UNREF(node); /* Assume they still have theirs */
717 			return;
718 		}
719 	} else {				/* do the default thing */
720 		NG_NODE_UNREF(node);
721 	}
722 
723 	ng_unname(node); /* basically a NOP these days */
724 
725 	/*
726 	 * Remove extra reference, possibly the last
727 	 * Possible other holders of references may include
728 	 * timeout callouts, but theoretically the node's supposed to
729 	 * have cancelled them. Possibly hardware dependencies may
730 	 * force a driver to 'linger' with a reference.
731 	 */
732 	NG_NODE_UNREF(node);
733 }
734 
735 #ifdef	NETGRAPH_DEBUG
736 void
737 ng_ref_node(node_p node)
738 {
739 	_NG_NODE_REF(node);
740 }
741 #endif
742 
743 /*
744  * Remove a reference to the node, possibly the last.
745  * deadnode always acts as it it were the last.
746  */
747 int
748 ng_unref_node(node_p node)
749 {
750 	int v;
751 
752 	if (node == &ng_deadnode) {
753 		return (0);
754 	}
755 
756 	do {
757 		v = node->nd_refs - 1;
758 	} while (! atomic_cmpset_int(&node->nd_refs, v + 1, v));
759 
760 	if (v == 0) { /* we were the last */
761 
762 		mtx_lock(&ng_nodelist_mtx);
763 		node->nd_type->refs--; /* XXX maybe should get types lock? */
764 		LIST_REMOVE(node, nd_nodes);
765 		mtx_unlock(&ng_nodelist_mtx);
766 
767 		mtx_lock(&ng_idhash_mtx);
768 		LIST_REMOVE(node, nd_idnodes);
769 		mtx_unlock(&ng_idhash_mtx);
770 
771 		mtx_destroy(&node->nd_input_queue.q_mtx);
772 		NG_FREE_NODE(node);
773 	}
774 	return (v);
775 }
776 
777 /************************************************************************
778 			Node ID handling
779 ************************************************************************/
780 static node_p
781 ng_ID2noderef(ng_ID_t ID)
782 {
783 	node_p node;
784 	mtx_lock(&ng_idhash_mtx);
785 	NG_IDHASH_FIND(ID, node);
786 	if(node)
787 		NG_NODE_REF(node);
788 	mtx_unlock(&ng_idhash_mtx);
789 	return(node);
790 }
791 
792 ng_ID_t
793 ng_node2ID(node_p node)
794 {
795 	return (node ? NG_NODE_ID(node) : 0);
796 }
797 
798 /************************************************************************
799 			Node name handling
800 ************************************************************************/
801 
802 /*
803  * Assign a node a name. Once assigned, the name cannot be changed.
804  */
805 int
806 ng_name_node(node_p node, const char *name)
807 {
808 	int i;
809 	node_p node2;
810 
811 	/* Check the name is valid */
812 	for (i = 0; i < NG_NODESIZ; i++) {
813 		if (name[i] == '\0' || name[i] == '.' || name[i] == ':')
814 			break;
815 	}
816 	if (i == 0 || name[i] != '\0') {
817 		TRAP_ERROR();
818 		return (EINVAL);
819 	}
820 	if (ng_decodeidname(name) != 0) { /* valid IDs not allowed here */
821 		TRAP_ERROR();
822 		return (EINVAL);
823 	}
824 
825 	/* Check the name isn't already being used */
826 	if ((node2 = ng_name2noderef(node, name)) != NULL) {
827 		NG_NODE_UNREF(node2);
828 		TRAP_ERROR();
829 		return (EADDRINUSE);
830 	}
831 
832 	/* copy it */
833 	strlcpy(NG_NODE_NAME(node), name, NG_NODESIZ);
834 
835 	return (0);
836 }
837 
838 /*
839  * Find a node by absolute name. The name should NOT end with ':'
840  * The name "." means "this node" and "[xxx]" means "the node
841  * with ID (ie, at address) xxx".
842  *
843  * Returns the node if found, else NULL.
844  * Eventually should add something faster than a sequential search.
845  * Note it aquires a reference on the node so you can be sure it's still there.
846  */
847 node_p
848 ng_name2noderef(node_p here, const char *name)
849 {
850 	node_p node;
851 	ng_ID_t temp;
852 
853 	/* "." means "this node" */
854 	if (strcmp(name, ".") == 0) {
855 		NG_NODE_REF(here);
856 		return(here);
857 	}
858 
859 	/* Check for name-by-ID */
860 	if ((temp = ng_decodeidname(name)) != 0) {
861 		return (ng_ID2noderef(temp));
862 	}
863 
864 	/* Find node by name */
865 	mtx_lock(&ng_nodelist_mtx);
866 	LIST_FOREACH(node, &ng_nodelist, nd_nodes) {
867 		if (NG_NODE_IS_VALID(node)
868 		&& NG_NODE_HAS_NAME(node)
869 		&& (strcmp(NG_NODE_NAME(node), name) == 0)) {
870 			break;
871 		}
872 	}
873 	if (node)
874 		NG_NODE_REF(node);
875 	mtx_unlock(&ng_nodelist_mtx);
876 	return (node);
877 }
878 
879 /*
880  * Decode an ID name, eg. "[f03034de]". Returns 0 if the
881  * string is not valid, otherwise returns the value.
882  */
883 static ng_ID_t
884 ng_decodeidname(const char *name)
885 {
886 	const int len = strlen(name);
887 	char *eptr;
888 	u_long val;
889 
890 	/* Check for proper length, brackets, no leading junk */
891 	if ((len < 3)
892 	|| (name[0] != '[')
893 	|| (name[len - 1] != ']')
894 	|| (!isxdigit(name[1]))) {
895 		return ((ng_ID_t)0);
896 	}
897 
898 	/* Decode number */
899 	val = strtoul(name + 1, &eptr, 16);
900 	if ((eptr - name != len - 1)
901 	|| (val == ULONG_MAX)
902 	|| (val == 0)) {
903 		return ((ng_ID_t)0);
904 	}
905 	return (ng_ID_t)val;
906 }
907 
908 /*
909  * Remove a name from a node. This should only be called
910  * when shutting down and removing the node.
911  * IF we allow name changing this may be more resurected.
912  */
913 void
914 ng_unname(node_p node)
915 {
916 }
917 
918 /************************************************************************
919 			Hook routines
920  Names are not optional. Hooks are always connected, except for a
921  brief moment within these routines. On invalidation or during creation
922  they are connected to the 'dead' hook.
923 ************************************************************************/
924 
925 /*
926  * Remove a hook reference
927  */
928 void
929 ng_unref_hook(hook_p hook)
930 {
931 	int v;
932 
933 	if (hook == &ng_deadhook) {
934 		return;
935 	}
936 	do {
937 		v = hook->hk_refs;
938 	} while (! atomic_cmpset_int(&hook->hk_refs, v, v - 1));
939 
940 	if (v == 1) { /* we were the last */
941 		if (_NG_HOOK_NODE(hook)) { /* it'll probably be ng_deadnode */
942 			_NG_NODE_UNREF((_NG_HOOK_NODE(hook)));
943 			hook->hk_node = NULL;
944 		}
945 		NG_FREE_HOOK(hook);
946 	}
947 }
948 
949 /*
950  * Add an unconnected hook to a node. Only used internally.
951  * Assumes node is locked. (XXX not yet true )
952  */
953 static int
954 ng_add_hook(node_p node, const char *name, hook_p *hookp)
955 {
956 	hook_p hook;
957 	int error = 0;
958 
959 	/* Check that the given name is good */
960 	if (name == NULL) {
961 		TRAP_ERROR();
962 		return (EINVAL);
963 	}
964 	if (ng_findhook(node, name) != NULL) {
965 		TRAP_ERROR();
966 		return (EEXIST);
967 	}
968 
969 	/* Allocate the hook and link it up */
970 	NG_ALLOC_HOOK(hook);
971 	if (hook == NULL) {
972 		TRAP_ERROR();
973 		return (ENOMEM);
974 	}
975 	hook->hk_refs = 1;		/* add a reference for us to return */
976 	hook->hk_flags = HK_INVALID;
977 	hook->hk_peer = &ng_deadhook;	/* start off this way */
978 	hook->hk_node = node;
979 	NG_NODE_REF(node);		/* each hook counts as a reference */
980 
981 	/* Set hook name */
982 	strlcpy(NG_HOOK_NAME(hook), name, NG_HOOKSIZ);
983 
984 	/*
985 	 * Check if the node type code has something to say about it
986 	 * If it fails, the unref of the hook will also unref the node.
987 	 */
988 	if (node->nd_type->newhook != NULL) {
989 		if ((error = (*node->nd_type->newhook)(node, hook, name))) {
990 			NG_HOOK_UNREF(hook);	/* this frees the hook */
991 			return (error);
992 		}
993 	}
994 	/*
995 	 * The 'type' agrees so far, so go ahead and link it in.
996 	 * We'll ask again later when we actually connect the hooks.
997 	 */
998 	LIST_INSERT_HEAD(&node->nd_hooks, hook, hk_hooks);
999 	node->nd_numhooks++;
1000 	NG_HOOK_REF(hook);	/* one for the node */
1001 
1002 	if (hookp)
1003 		*hookp = hook;
1004 	return (0);
1005 }
1006 
1007 /*
1008  * Find a hook
1009  *
1010  * Node types may supply their own optimized routines for finding
1011  * hooks.  If none is supplied, we just do a linear search.
1012  * XXX Possibly we should add a reference to the hook?
1013  */
1014 hook_p
1015 ng_findhook(node_p node, const char *name)
1016 {
1017 	hook_p hook;
1018 
1019 	if (node->nd_type->findhook != NULL)
1020 		return (*node->nd_type->findhook)(node, name);
1021 	LIST_FOREACH(hook, &node->nd_hooks, hk_hooks) {
1022 		if (NG_HOOK_IS_VALID(hook)
1023 		&& (strcmp(NG_HOOK_NAME(hook), name) == 0))
1024 			return (hook);
1025 	}
1026 	return (NULL);
1027 }
1028 
1029 /*
1030  * Destroy a hook
1031  *
1032  * As hooks are always attached, this really destroys two hooks.
1033  * The one given, and the one attached to it. Disconnect the hooks
1034  * from each other first. We reconnect the peer hook to the 'dead'
1035  * hook so that it can still exist after we depart. We then
1036  * send the peer its own destroy message. This ensures that we only
1037  * interact with the peer's structures when it is locked processing that
1038  * message. We hold a reference to the peer hook so we are guaranteed that
1039  * the peer hook and node are still going to exist until
1040  * we are finished there as the hook holds a ref on the node.
1041  * We run this same code again on the peer hook, but that time it is already
1042  * attached to the 'dead' hook.
1043  *
1044  * This routine is called at all stages of hook creation
1045  * on error detection and must be able to handle any such stage.
1046  */
1047 void
1048 ng_destroy_hook(hook_p hook)
1049 {
1050 	hook_p peer;
1051 	node_p node;
1052 
1053 	if (hook == &ng_deadhook) {	/* better safe than sorry */
1054 		printf("ng_destroy_hook called on deadhook\n");
1055 		return;
1056 	}
1057 
1058 	/*
1059 	 * Protect divorce process with mutex, to avoid races on
1060 	 * simultaneous disconnect.
1061 	 */
1062 	mtx_lock(&ng_topo_mtx);
1063 
1064 	hook->hk_flags |= HK_INVALID;
1065 
1066 	peer = NG_HOOK_PEER(hook);
1067 	node = NG_HOOK_NODE(hook);
1068 
1069 	if (peer && (peer != &ng_deadhook)) {
1070 		/*
1071 		 * Set the peer to point to ng_deadhook
1072 		 * from this moment on we are effectively independent it.
1073 		 * send it an rmhook message of it's own.
1074 		 */
1075 		peer->hk_peer = &ng_deadhook;	/* They no longer know us */
1076 		hook->hk_peer = &ng_deadhook;	/* Nor us, them */
1077 		if (NG_HOOK_NODE(peer) == &ng_deadnode) {
1078 			/*
1079 			 * If it's already divorced from a node,
1080 			 * just free it.
1081 			 */
1082 			mtx_unlock(&ng_topo_mtx);
1083 		} else {
1084 			mtx_unlock(&ng_topo_mtx);
1085 			ng_rmhook_self(peer); 	/* Send it a surprise */
1086 		}
1087 		NG_HOOK_UNREF(peer);		/* account for peer link */
1088 		NG_HOOK_UNREF(hook);		/* account for peer link */
1089 	} else
1090 		mtx_unlock(&ng_topo_mtx);
1091 
1092 	mtx_assert(&ng_topo_mtx, MA_NOTOWNED);
1093 
1094 	/*
1095 	 * Remove the hook from the node's list to avoid possible recursion
1096 	 * in case the disconnection results in node shutdown.
1097 	 */
1098 	if (node == &ng_deadnode) { /* happens if called from ng_con_nodes() */
1099 		return;
1100 	}
1101 	LIST_REMOVE(hook, hk_hooks);
1102 	node->nd_numhooks--;
1103 	if (node->nd_type->disconnect) {
1104 		/*
1105 		 * The type handler may elect to destroy the node so don't
1106 		 * trust its existance after this point. (except
1107 		 * that we still hold a reference on it. (which we
1108 		 * inherrited from the hook we are destroying)
1109 		 */
1110 		(*node->nd_type->disconnect) (hook);
1111 	}
1112 
1113 	/*
1114 	 * Note that because we will point to ng_deadnode, the original node
1115 	 * is not decremented automatically so we do that manually.
1116 	 */
1117 	_NG_HOOK_NODE(hook) = &ng_deadnode;
1118 	NG_NODE_UNREF(node);	/* We no longer point to it so adjust count */
1119 	NG_HOOK_UNREF(hook);	/* Account for linkage (in list) to node */
1120 }
1121 
1122 /*
1123  * Take two hooks on a node and merge the connection so that the given node
1124  * is effectively bypassed.
1125  */
1126 int
1127 ng_bypass(hook_p hook1, hook_p hook2)
1128 {
1129 	if (hook1->hk_node != hook2->hk_node) {
1130 		TRAP_ERROR();
1131 		return (EINVAL);
1132 	}
1133 	hook1->hk_peer->hk_peer = hook2->hk_peer;
1134 	hook2->hk_peer->hk_peer = hook1->hk_peer;
1135 
1136 	hook1->hk_peer = &ng_deadhook;
1137 	hook2->hk_peer = &ng_deadhook;
1138 
1139 	/* XXX If we ever cache methods on hooks update them as well */
1140 	ng_destroy_hook(hook1);
1141 	ng_destroy_hook(hook2);
1142 	return (0);
1143 }
1144 
1145 /*
1146  * Install a new netgraph type
1147  */
1148 int
1149 ng_newtype(struct ng_type *tp)
1150 {
1151 	const size_t namelen = strlen(tp->name);
1152 
1153 	/* Check version and type name fields */
1154 	if ((tp->version != NG_ABI_VERSION)
1155 	|| (namelen == 0)
1156 	|| (namelen >= NG_TYPESIZ)) {
1157 		TRAP_ERROR();
1158 		if (tp->version != NG_ABI_VERSION) {
1159 			printf("Netgraph: Node type rejected. ABI mismatch. Suggest recompile\n");
1160 		}
1161 		return (EINVAL);
1162 	}
1163 
1164 	/* Check for name collision */
1165 	if (ng_findtype(tp->name) != NULL) {
1166 		TRAP_ERROR();
1167 		return (EEXIST);
1168 	}
1169 
1170 
1171 	/* Link in new type */
1172 	mtx_lock(&ng_typelist_mtx);
1173 	LIST_INSERT_HEAD(&ng_typelist, tp, types);
1174 	tp->refs = 1;	/* first ref is linked list */
1175 	mtx_unlock(&ng_typelist_mtx);
1176 	return (0);
1177 }
1178 
1179 /*
1180  * unlink a netgraph type
1181  * If no examples exist
1182  */
1183 int
1184 ng_rmtype(struct ng_type *tp)
1185 {
1186 	/* Check for name collision */
1187 	if (tp->refs != 1) {
1188 		TRAP_ERROR();
1189 		return (EBUSY);
1190 	}
1191 
1192 	/* Unlink type */
1193 	mtx_lock(&ng_typelist_mtx);
1194 	LIST_REMOVE(tp, types);
1195 	mtx_unlock(&ng_typelist_mtx);
1196 	return (0);
1197 }
1198 
1199 /*
1200  * Look for a type of the name given
1201  */
1202 struct ng_type *
1203 ng_findtype(const char *typename)
1204 {
1205 	struct ng_type *type;
1206 
1207 	mtx_lock(&ng_typelist_mtx);
1208 	LIST_FOREACH(type, &ng_typelist, types) {
1209 		if (strcmp(type->name, typename) == 0)
1210 			break;
1211 	}
1212 	mtx_unlock(&ng_typelist_mtx);
1213 	return (type);
1214 }
1215 
1216 /************************************************************************
1217 			Composite routines
1218 ************************************************************************/
1219 /*
1220  * Connect two nodes using the specified hooks, using queued functions.
1221  */
1222 static void
1223 ng_con_part3(node_p node, hook_p hook, void *arg1, int arg2)
1224 {
1225 
1226 	/*
1227 	 * When we run, we know that the node 'node' is locked for us.
1228 	 * Our caller has a reference on the hook.
1229 	 * Our caller has a reference on the node.
1230 	 * (In this case our caller is ng_apply_item() ).
1231 	 * The peer hook has a reference on the hook.
1232 	 * We are all set up except for the final call to the node, and
1233 	 * the clearing of the INVALID flag.
1234 	 */
1235 	if (NG_HOOK_NODE(hook) == &ng_deadnode) {
1236 		/*
1237 		 * The node must have been freed again since we last visited
1238 		 * here. ng_destry_hook() has this effect but nothing else does.
1239 		 * We should just release our references and
1240 		 * free anything we can think of.
1241 		 * Since we know it's been destroyed, and it's our caller
1242 		 * that holds the references, just return.
1243 		 */
1244 		return ;
1245 	}
1246 	if (hook->hk_node->nd_type->connect) {
1247 		if ((*hook->hk_node->nd_type->connect) (hook)) {
1248 			ng_destroy_hook(hook);	/* also zaps peer */
1249 			printf("failed in ng_con_part3()\n");
1250 			return ;
1251 		}
1252 	}
1253 	/*
1254 	 *  XXX this is wrong for SMP. Possibly we need
1255 	 * to separate out 'create' and 'invalid' flags.
1256 	 * should only set flags on hooks we have locked under our node.
1257 	 */
1258 	hook->hk_flags &= ~HK_INVALID;
1259 	return ;
1260 }
1261 
1262 static void
1263 ng_con_part2(node_p node, hook_p hook, void *arg1, int arg2)
1264 {
1265 	hook_p peer;
1266 
1267 	/*
1268 	 * When we run, we know that the node 'node' is locked for us.
1269 	 * Our caller has a reference on the hook.
1270 	 * Our caller has a reference on the node.
1271 	 * (In this case our caller is ng_apply_item() ).
1272 	 * The peer hook has a reference on the hook.
1273 	 * our node pointer points to the 'dead' node.
1274 	 * First check the hook name is unique.
1275 	 * Should not happen because we checked before queueing this.
1276 	 */
1277 	if (ng_findhook(node, NG_HOOK_NAME(hook)) != NULL) {
1278 		TRAP_ERROR();
1279 		ng_destroy_hook(hook); /* should destroy peer too */
1280 		printf("failed in ng_con_part2()\n");
1281 		return ;
1282 	}
1283 	/*
1284 	 * Check if the node type code has something to say about it
1285 	 * If it fails, the unref of the hook will also unref the attached node,
1286 	 * however since that node is 'ng_deadnode' this will do nothing.
1287 	 * The peer hook will also be destroyed.
1288 	 */
1289 	if (node->nd_type->newhook != NULL) {
1290 		if ((*node->nd_type->newhook)(node, hook, hook->hk_name)) {
1291 			ng_destroy_hook(hook); /* should destroy peer too */
1292 			printf("failed in ng_con_part2()\n");
1293 			return ;
1294 		}
1295 	}
1296 
1297 	/*
1298 	 * The 'type' agrees so far, so go ahead and link it in.
1299 	 * We'll ask again later when we actually connect the hooks.
1300 	 */
1301 	hook->hk_node = node;		/* just overwrite ng_deadnode */
1302 	NG_NODE_REF(node);		/* each hook counts as a reference */
1303 	LIST_INSERT_HEAD(&node->nd_hooks, hook, hk_hooks);
1304 	node->nd_numhooks++;
1305 	NG_HOOK_REF(hook);	/* one for the node */
1306 
1307 	/*
1308 	 * We now have a symetrical situation, where both hooks have been
1309 	 * linked to their nodes, the newhook methods have been called
1310 	 * And the references are all correct. The hooks are still marked
1311 	 * as invalid, as we have not called the 'connect' methods
1312 	 * yet.
1313 	 * We can call the local one immediatly as we have the
1314 	 * node locked, but we need to queue the remote one.
1315 	 */
1316 	if (hook->hk_node->nd_type->connect) {
1317 		if ((*hook->hk_node->nd_type->connect) (hook)) {
1318 			ng_destroy_hook(hook);	/* also zaps peer */
1319 			printf("failed in ng_con_part2(A)\n");
1320 			return ;
1321 		}
1322 	}
1323 
1324 	/*
1325 	 * Acquire topo mutex to avoid race with ng_destroy_hook().
1326 	 */
1327 	mtx_lock(&ng_topo_mtx);
1328 	peer = hook->hk_peer;
1329 	if (peer == &ng_deadhook) {
1330 		mtx_unlock(&ng_topo_mtx);
1331 		printf("failed in ng_con_part2(B)\n");
1332 		ng_destroy_hook(hook);
1333 		return ;
1334 	}
1335 	mtx_unlock(&ng_topo_mtx);
1336 
1337 	if (ng_send_fn(peer->hk_node, peer, &ng_con_part3, arg1, arg2)) {
1338 		printf("failed in ng_con_part2(C)\n");
1339 		ng_destroy_hook(hook);	/* also zaps peer */
1340 		return ;
1341 	}
1342 	hook->hk_flags &= ~HK_INVALID; /* need both to be able to work */
1343 	return ;
1344 }
1345 
1346 /*
1347  * Connect this node with another node. We assume that this node is
1348  * currently locked, as we are only called from an NGM_CONNECT message.
1349  */
1350 static int
1351 ng_con_nodes(node_p node, const char *name, node_p node2, const char *name2)
1352 {
1353 	int	error;
1354 	hook_p	hook;
1355 	hook_p	hook2;
1356 
1357 	if (ng_findhook(node2, name2) != NULL) {
1358 		return(EEXIST);
1359 	}
1360 	if ((error = ng_add_hook(node, name, &hook)))  /* gives us a ref */
1361 		return (error);
1362 	/* Allocate the other hook and link it up */
1363 	NG_ALLOC_HOOK(hook2);
1364 	if (hook2 == NULL) {
1365 		TRAP_ERROR();
1366 		ng_destroy_hook(hook);	/* XXX check ref counts so far */
1367 		NG_HOOK_UNREF(hook);	/* including our ref */
1368 		return (ENOMEM);
1369 	}
1370 	hook2->hk_refs = 1;		/* start with a reference for us. */
1371 	hook2->hk_flags = HK_INVALID;
1372 	hook2->hk_peer = hook;		/* Link the two together */
1373 	hook->hk_peer = hook2;
1374 	NG_HOOK_REF(hook);		/* Add a ref for the peer to each*/
1375 	NG_HOOK_REF(hook2);
1376 	hook2->hk_node = &ng_deadnode;
1377 	strlcpy(NG_HOOK_NAME(hook2), name2, NG_HOOKSIZ);
1378 
1379 	/*
1380 	 * Queue the function above.
1381 	 * Procesing continues in that function in the lock context of
1382 	 * the other node.
1383 	 */
1384 	ng_send_fn(node2, hook2, &ng_con_part2, NULL, 0);
1385 
1386 	NG_HOOK_UNREF(hook);		/* Let each hook go if it wants to */
1387 	NG_HOOK_UNREF(hook2);
1388 	return (0);
1389 }
1390 
1391 /*
1392  * Make a peer and connect.
1393  * We assume that the local node is locked.
1394  * The new node probably doesn't need a lock until
1395  * it has a hook, because it cannot really have any work until then,
1396  * but we should think about it a bit more.
1397  *
1398  * The problem may come if the other node also fires up
1399  * some hardware or a timer or some other source of activation,
1400  * also it may already get a command msg via it's ID.
1401  *
1402  * We could use the same method as ng_con_nodes() but we'd have
1403  * to add ability to remove the node when failing. (Not hard, just
1404  * make arg1 point to the node to remove).
1405  * Unless of course we just ignore failure to connect and leave
1406  * an unconnected node?
1407  */
1408 static int
1409 ng_mkpeer(node_p node, const char *name, const char *name2, char *type)
1410 {
1411 	node_p	node2;
1412 	hook_p	hook1, hook2;
1413 	int	error;
1414 
1415 	if ((error = ng_make_node(type, &node2))) {
1416 		return (error);
1417 	}
1418 
1419 	if ((error = ng_add_hook(node, name, &hook1))) { /* gives us a ref */
1420 		ng_rmnode(node2, NULL, NULL, 0);
1421 		return (error);
1422 	}
1423 
1424 	if ((error = ng_add_hook(node2, name2, &hook2))) {
1425 		ng_rmnode(node2, NULL, NULL, 0);
1426 		ng_destroy_hook(hook1);
1427 		NG_HOOK_UNREF(hook1);
1428 		return (error);
1429 	}
1430 
1431 	/*
1432 	 * Actually link the two hooks together.
1433 	 */
1434 	hook1->hk_peer = hook2;
1435 	hook2->hk_peer = hook1;
1436 
1437 	/* Each hook is referenced by the other */
1438 	NG_HOOK_REF(hook1);
1439 	NG_HOOK_REF(hook2);
1440 
1441 	/* Give each node the opportunity to veto the pending connection */
1442 	if (hook1->hk_node->nd_type->connect) {
1443 		error = (*hook1->hk_node->nd_type->connect) (hook1);
1444 	}
1445 
1446 	if ((error == 0) && hook2->hk_node->nd_type->connect) {
1447 		error = (*hook2->hk_node->nd_type->connect) (hook2);
1448 
1449 	}
1450 
1451 	/*
1452 	 * drop the references we were holding on the two hooks.
1453 	 */
1454 	if (error) {
1455 		ng_destroy_hook(hook2);	/* also zaps hook1 */
1456 		ng_rmnode(node2, NULL, NULL, 0);
1457 	} else {
1458 		/* As a last act, allow the hooks to be used */
1459 		hook1->hk_flags &= ~HK_INVALID;
1460 		hook2->hk_flags &= ~HK_INVALID;
1461 	}
1462 	NG_HOOK_UNREF(hook1);
1463 	NG_HOOK_UNREF(hook2);
1464 	return (error);
1465 }
1466 
1467 /************************************************************************
1468 		Utility routines to send self messages
1469 ************************************************************************/
1470 
1471 /* Shut this node down as soon as everyone is clear of it */
1472 /* Should add arg "immediatly" to jump the queue */
1473 int
1474 ng_rmnode_self(node_p node)
1475 {
1476 	int		error;
1477 
1478 	if (node == &ng_deadnode)
1479 		return (0);
1480 	node->nd_flags |= NGF_INVALID;
1481 	if (node->nd_flags & NGF_CLOSING)
1482 		return (0);
1483 
1484 	error = ng_send_fn(node, NULL, &ng_rmnode, NULL, 0);
1485 	return (error);
1486 }
1487 
1488 static void
1489 ng_rmhook_part2(node_p node, hook_p hook, void *arg1, int arg2)
1490 {
1491 	ng_destroy_hook(hook);
1492 	return ;
1493 }
1494 
1495 int
1496 ng_rmhook_self(hook_p hook)
1497 {
1498 	int		error;
1499 	node_p node = NG_HOOK_NODE(hook);
1500 
1501 	if (node == &ng_deadnode)
1502 		return (0);
1503 
1504 	error = ng_send_fn(node, hook, &ng_rmhook_part2, NULL, 0);
1505 	return (error);
1506 }
1507 
1508 /***********************************************************************
1509  * Parse and verify a string of the form:  <NODE:><PATH>
1510  *
1511  * Such a string can refer to a specific node or a specific hook
1512  * on a specific node, depending on how you look at it. In the
1513  * latter case, the PATH component must not end in a dot.
1514  *
1515  * Both <NODE:> and <PATH> are optional. The <PATH> is a string
1516  * of hook names separated by dots. This breaks out the original
1517  * string, setting *nodep to "NODE" (or NULL if none) and *pathp
1518  * to "PATH" (or NULL if degenerate). Also, *hookp will point to
1519  * the final hook component of <PATH>, if any, otherwise NULL.
1520  *
1521  * This returns -1 if the path is malformed. The char ** are optional.
1522  ***********************************************************************/
1523 int
1524 ng_path_parse(char *addr, char **nodep, char **pathp, char **hookp)
1525 {
1526 	char	*node, *path, *hook;
1527 	int	k;
1528 
1529 	/*
1530 	 * Extract absolute NODE, if any
1531 	 */
1532 	for (path = addr; *path && *path != ':'; path++);
1533 	if (*path) {
1534 		node = addr;	/* Here's the NODE */
1535 		*path++ = '\0';	/* Here's the PATH */
1536 
1537 		/* Node name must not be empty */
1538 		if (!*node)
1539 			return -1;
1540 
1541 		/* A name of "." is OK; otherwise '.' not allowed */
1542 		if (strcmp(node, ".") != 0) {
1543 			for (k = 0; node[k]; k++)
1544 				if (node[k] == '.')
1545 					return -1;
1546 		}
1547 	} else {
1548 		node = NULL;	/* No absolute NODE */
1549 		path = addr;	/* Here's the PATH */
1550 	}
1551 
1552 	/* Snoop for illegal characters in PATH */
1553 	for (k = 0; path[k]; k++)
1554 		if (path[k] == ':')
1555 			return -1;
1556 
1557 	/* Check for no repeated dots in PATH */
1558 	for (k = 0; path[k]; k++)
1559 		if (path[k] == '.' && path[k + 1] == '.')
1560 			return -1;
1561 
1562 	/* Remove extra (degenerate) dots from beginning or end of PATH */
1563 	if (path[0] == '.')
1564 		path++;
1565 	if (*path && path[strlen(path) - 1] == '.')
1566 		path[strlen(path) - 1] = 0;
1567 
1568 	/* If PATH has a dot, then we're not talking about a hook */
1569 	if (*path) {
1570 		for (hook = path, k = 0; path[k]; k++)
1571 			if (path[k] == '.') {
1572 				hook = NULL;
1573 				break;
1574 			}
1575 	} else
1576 		path = hook = NULL;
1577 
1578 	/* Done */
1579 	if (nodep)
1580 		*nodep = node;
1581 	if (pathp)
1582 		*pathp = path;
1583 	if (hookp)
1584 		*hookp = hook;
1585 	return (0);
1586 }
1587 
1588 /*
1589  * Given a path, which may be absolute or relative, and a starting node,
1590  * return the destination node.
1591  */
1592 int
1593 ng_path2noderef(node_p here, const char *address,
1594 				node_p *destp, hook_p *lasthook)
1595 {
1596 	char    fullpath[NG_PATHSIZ];
1597 	char   *nodename, *path, pbuf[2];
1598 	node_p  node, oldnode;
1599 	char   *cp;
1600 	hook_p hook = NULL;
1601 
1602 	/* Initialize */
1603 	if (destp == NULL) {
1604 		TRAP_ERROR();
1605 		return EINVAL;
1606 	}
1607 	*destp = NULL;
1608 
1609 	/* Make a writable copy of address for ng_path_parse() */
1610 	strncpy(fullpath, address, sizeof(fullpath) - 1);
1611 	fullpath[sizeof(fullpath) - 1] = '\0';
1612 
1613 	/* Parse out node and sequence of hooks */
1614 	if (ng_path_parse(fullpath, &nodename, &path, NULL) < 0) {
1615 		TRAP_ERROR();
1616 		return EINVAL;
1617 	}
1618 	if (path == NULL) {
1619 		pbuf[0] = '.';	/* Needs to be writable */
1620 		pbuf[1] = '\0';
1621 		path = pbuf;
1622 	}
1623 
1624 	/*
1625 	 * For an absolute address, jump to the starting node.
1626 	 * Note that this holds a reference on the node for us.
1627 	 * Don't forget to drop the reference if we don't need it.
1628 	 */
1629 	if (nodename) {
1630 		node = ng_name2noderef(here, nodename);
1631 		if (node == NULL) {
1632 			TRAP_ERROR();
1633 			return (ENOENT);
1634 		}
1635 	} else {
1636 		if (here == NULL) {
1637 			TRAP_ERROR();
1638 			return (EINVAL);
1639 		}
1640 		node = here;
1641 		NG_NODE_REF(node);
1642 	}
1643 
1644 	/*
1645 	 * Now follow the sequence of hooks
1646 	 * XXX
1647 	 * We actually cannot guarantee that the sequence
1648 	 * is not being demolished as we crawl along it
1649 	 * without extra-ordinary locking etc.
1650 	 * So this is a bit dodgy to say the least.
1651 	 * We can probably hold up some things by holding
1652 	 * the nodelist mutex for the time of this
1653 	 * crawl if we wanted.. At least that way we wouldn't have to
1654 	 * worry about the nodes dissappearing, but the hooks would still
1655 	 * be a problem.
1656 	 */
1657 	for (cp = path; node != NULL && *cp != '\0'; ) {
1658 		char *segment;
1659 
1660 		/*
1661 		 * Break out the next path segment. Replace the dot we just
1662 		 * found with a NUL; "cp" points to the next segment (or the
1663 		 * NUL at the end).
1664 		 */
1665 		for (segment = cp; *cp != '\0'; cp++) {
1666 			if (*cp == '.') {
1667 				*cp++ = '\0';
1668 				break;
1669 			}
1670 		}
1671 
1672 		/* Empty segment */
1673 		if (*segment == '\0')
1674 			continue;
1675 
1676 		/* We have a segment, so look for a hook by that name */
1677 		hook = ng_findhook(node, segment);
1678 
1679 		/* Can't get there from here... */
1680 		if (hook == NULL
1681 		    || NG_HOOK_PEER(hook) == NULL
1682 		    || NG_HOOK_NOT_VALID(hook)
1683 		    || NG_HOOK_NOT_VALID(NG_HOOK_PEER(hook))) {
1684 			TRAP_ERROR();
1685 			NG_NODE_UNREF(node);
1686 #if 0
1687 			printf("hooknotvalid %s %s %d %d %d %d ",
1688 					path,
1689 					segment,
1690 					hook == NULL,
1691 					NG_HOOK_PEER(hook) == NULL,
1692 					NG_HOOK_NOT_VALID(hook),
1693 					NG_HOOK_NOT_VALID(NG_HOOK_PEER(hook)));
1694 #endif
1695 			return (ENOENT);
1696 		}
1697 
1698 		/*
1699 		 * Hop on over to the next node
1700 		 * XXX
1701 		 * Big race conditions here as hooks and nodes go away
1702 		 * *** Idea.. store an ng_ID_t in each hook and use that
1703 		 * instead of the direct hook in this crawl?
1704 		 */
1705 		oldnode = node;
1706 		if ((node = NG_PEER_NODE(hook)))
1707 			NG_NODE_REF(node);	/* XXX RACE */
1708 		NG_NODE_UNREF(oldnode);	/* XXX another race */
1709 		if (NG_NODE_NOT_VALID(node)) {
1710 			NG_NODE_UNREF(node);	/* XXX more races */
1711 			node = NULL;
1712 		}
1713 	}
1714 
1715 	/* If node somehow missing, fail here (probably this is not needed) */
1716 	if (node == NULL) {
1717 		TRAP_ERROR();
1718 		return (ENXIO);
1719 	}
1720 
1721 	/* Done */
1722 	*destp = node;
1723 	if (lasthook != NULL)
1724 		*lasthook = (hook ? NG_HOOK_PEER(hook) : NULL);
1725 	return (0);
1726 }
1727 
1728 /***************************************************************\
1729 * Input queue handling.
1730 * All activities are submitted to the node via the input queue
1731 * which implements a multiple-reader/single-writer gate.
1732 * Items which cannot be handled immeditly are queued.
1733 *
1734 * read-write queue locking inline functions			*
1735 \***************************************************************/
1736 
1737 static __inline item_p ng_dequeue(struct ng_queue * ngq, int *rw);
1738 static __inline item_p ng_acquire_read(struct ng_queue * ngq,
1739 					item_p  item);
1740 static __inline item_p ng_acquire_write(struct ng_queue * ngq,
1741 					item_p  item);
1742 static __inline void	ng_leave_read(struct ng_queue * ngq);
1743 static __inline void	ng_leave_write(struct ng_queue * ngq);
1744 static __inline void	ng_queue_rw(struct ng_queue * ngq,
1745 					item_p  item, int rw);
1746 
1747 /*
1748  * Definition of the bits fields in the ng_queue flag word.
1749  * Defined here rather than in netgraph.h because no-one should fiddle
1750  * with them.
1751  *
1752  * The ordering here may be important! don't shuffle these.
1753  */
1754 /*-
1755  Safety Barrier--------+ (adjustable to suit taste) (not used yet)
1756                        |
1757                        V
1758 +-------+-------+-------+-------+-------+-------+-------+-------+
1759   | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | |
1760   | |A|c|t|i|v|e| |R|e|a|d|e|r| |C|o|u|n|t| | | | | | | | | |P|A|
1761   | | | | | | | | | | | | | | | | | | | | | | | | | | | | | |O|W|
1762 +-------+-------+-------+-------+-------+-------+-------+-------+
1763   \___________________________ ____________________________/ | |
1764                             V                                | |
1765                   [active reader count]                      | |
1766                                                              | |
1767             Operation Pending -------------------------------+ |
1768                                                                |
1769           Active Writer ---------------------------------------+
1770 
1771 
1772 */
1773 #define WRITER_ACTIVE	0x00000001
1774 #define OP_PENDING	0x00000002
1775 #define READER_INCREMENT 0x00000004
1776 #define READER_MASK	0xfffffffc	/* Not valid if WRITER_ACTIVE is set */
1777 #define SAFETY_BARRIER	0x00100000	/* 128K items queued should be enough */
1778 
1779 /* Defines of more elaborate states on the queue */
1780 /* Mask of bits a new read cares about */
1781 #define NGQ_RMASK	(WRITER_ACTIVE|OP_PENDING)
1782 
1783 /* Mask of bits a new write cares about */
1784 #define NGQ_WMASK	(NGQ_RMASK|READER_MASK)
1785 
1786 /* Test to decide if there is something on the queue. */
1787 #define QUEUE_ACTIVE(QP) ((QP)->q_flags & OP_PENDING)
1788 
1789 /* How to decide what the next queued item is. */
1790 #define HEAD_IS_READER(QP)  NGI_QUEUED_READER((QP)->queue)
1791 #define HEAD_IS_WRITER(QP)  NGI_QUEUED_WRITER((QP)->queue) /* notused */
1792 
1793 /* Read the status to decide if the next item on the queue can now run. */
1794 #define QUEUED_READER_CAN_PROCEED(QP)			\
1795 		(((QP)->q_flags & (NGQ_RMASK & ~OP_PENDING)) == 0)
1796 #define QUEUED_WRITER_CAN_PROCEED(QP)			\
1797 		(((QP)->q_flags & (NGQ_WMASK & ~OP_PENDING)) == 0)
1798 
1799 /* Is there a chance of getting ANY work off the queue? */
1800 #define NEXT_QUEUED_ITEM_CAN_PROCEED(QP)				\
1801 	(QUEUE_ACTIVE(QP) && 						\
1802 	((HEAD_IS_READER(QP)) ? QUEUED_READER_CAN_PROCEED(QP) :		\
1803 				QUEUED_WRITER_CAN_PROCEED(QP)))
1804 
1805 
1806 #define NGQRW_R 0
1807 #define NGQRW_W 1
1808 
1809 /*
1810  * Taking into account the current state of the queue and node, possibly take
1811  * the next entry off the queue and return it. Return NULL if there was
1812  * nothing we could return, either because there really was nothing there, or
1813  * because the node was in a state where it cannot yet process the next item
1814  * on the queue.
1815  *
1816  * This MUST MUST MUST be called with the mutex held.
1817  */
1818 static __inline item_p
1819 ng_dequeue(struct ng_queue *ngq, int *rw)
1820 {
1821 	item_p item;
1822 	u_int		add_arg;
1823 
1824 	mtx_assert(&ngq->q_mtx, MA_OWNED);
1825 	/*
1826 	 * If there is nothing queued, then just return.
1827 	 * No point in continuing.
1828 	 * XXXGL: assert this?
1829 	 */
1830 	if (!QUEUE_ACTIVE(ngq)) {
1831 		return (NULL);
1832 	}
1833 
1834 	/*
1835 	 * From here, we can assume there is a head item.
1836 	 * We need to find out what it is and if it can be dequeued, given
1837 	 * the current state of the node.
1838 	 */
1839 	if (HEAD_IS_READER(ngq)) {
1840 		if (!QUEUED_READER_CAN_PROCEED(ngq)) {
1841 			/*
1842 			 * It's a reader but we can't use it.
1843 			 * We are stalled so make sure we don't
1844 			 * get called again until something changes.
1845 			 */
1846 			ng_worklist_remove(ngq->q_node);
1847 			return (NULL);
1848 		}
1849 		/*
1850 		 * Head of queue is a reader and we have no write active.
1851 		 * We don't care how many readers are already active.
1852 		 * Add the correct increment for the reader count.
1853 		 */
1854 		add_arg = READER_INCREMENT;
1855 		*rw = NGQRW_R;
1856 	} else if (QUEUED_WRITER_CAN_PROCEED(ngq)) {
1857 		/*
1858 		 * There is a pending write, no readers and no active writer.
1859 		 * This means we can go ahead with the pending writer. Note
1860 		 * the fact that we now have a writer, ready for when we take
1861 		 * it off the queue.
1862 		 *
1863 		 * We don't need to worry about a possible collision with the
1864 		 * fasttrack reader.
1865 		 *
1866 		 * The fasttrack thread may take a long time to discover that we
1867 		 * are running so we would have an inconsistent state in the
1868 		 * flags for a while. Since we ignore the reader count
1869 		 * entirely when the WRITER_ACTIVE flag is set, this should
1870 		 * not matter (in fact it is defined that way). If it tests
1871 		 * the flag before this operation, the OP_PENDING flag
1872 		 * will make it fail, and if it tests it later, the
1873 		 * WRITER_ACTIVE flag will do the same. If it is SO slow that
1874 		 * we have actually completed the operation, and neither flag
1875 		 * is set by the time that it tests the flags, then it is
1876 		 * actually ok for it to continue. If it completes and we've
1877 		 * finished and the read pending is set it still fails.
1878 		 *
1879 		 * So we can just ignore it,  as long as we can ensure that the
1880 		 * transition from WRITE_PENDING state to the WRITER_ACTIVE
1881 		 * state is atomic.
1882 		 *
1883 		 * After failing, first it will be held back by the mutex, then
1884 		 * when it can proceed, it will queue its request, then it
1885 		 * would arrive at this function. Usually it will have to
1886 		 * leave empty handed because the ACTIVE WRITER bit will be
1887 		 * set.
1888 		 *
1889 		 * Adjust the flags for the new active writer.
1890 		 */
1891 		add_arg = WRITER_ACTIVE;
1892 		*rw = NGQRW_W;
1893 		/*
1894 		 * We want to write "active writer, no readers " Now go make
1895 		 * it true. In fact there may be a number in the readers
1896 		 * count but we know it is not true and will be fixed soon.
1897 		 * We will fix the flags for the next pending entry in a
1898 		 * moment.
1899 		 */
1900 	} else {
1901 		/*
1902 		 * We can't dequeue anything.. return and say so. Probably we
1903 		 * have a write pending and the readers count is non zero. If
1904 		 * we got here because a reader hit us just at the wrong
1905 		 * moment with the fasttrack code, and put us in a strange
1906 		 * state, then it will be coming through in just a moment,
1907 		 * (just as soon as we release the mutex) and keep things
1908 		 * moving.
1909 		 * Make sure we remove ourselves from the work queue. It
1910 		 * would be a waste of effort to do all this again.
1911 		 */
1912 		ng_worklist_remove(ngq->q_node);
1913 		return (NULL);
1914 	}
1915 
1916 	/*
1917 	 * Now we dequeue the request (whatever it may be) and correct the
1918 	 * pending flags and the next and last pointers.
1919 	 */
1920 	item = ngq->queue;
1921 	ngq->queue = item->el_next;
1922 	if (ngq->last == &(item->el_next)) {
1923 		/*
1924 		 * that was the last entry in the queue so set the 'last
1925 		 * pointer up correctly and make sure the pending flag is
1926 		 * clear.
1927 		 */
1928 		add_arg += -OP_PENDING;
1929 		ngq->last = &(ngq->queue);
1930 		/*
1931 		 * Whatever flag was set will be cleared and
1932 		 * the new acive field will be set by the add as well,
1933 		 * so we don't need to change add_arg.
1934 		 * But we know we don't need to be on the work list.
1935 		 */
1936 		atomic_add_long(&ngq->q_flags, add_arg);
1937 		ng_worklist_remove(ngq->q_node);
1938 	} else {
1939 		/*
1940 		 * Since there is still something on the queue
1941 		 * we don't need to change the PENDING flag.
1942 		 */
1943 		atomic_add_long(&ngq->q_flags, add_arg);
1944 		/*
1945 		 * If we see more doable work, make sure we are
1946 		 * on the work queue.
1947 		 */
1948 		if (NEXT_QUEUED_ITEM_CAN_PROCEED(ngq)) {
1949 			ng_setisr(ngq->q_node);
1950 		}
1951 	}
1952 	return (item);
1953 }
1954 
1955 /*
1956  * Queue a packet to be picked up by someone else.
1957  * We really don't care who, but we can't or don't want to hang around
1958  * to process it ourselves. We are probably an interrupt routine..
1959  * If the queue could be run, flag the netisr handler to start.
1960  */
1961 static __inline void
1962 ng_queue_rw(struct ng_queue * ngq, item_p  item, int rw)
1963 {
1964 	mtx_assert(&ngq->q_mtx, MA_OWNED);
1965 
1966 	if (rw == NGQRW_W)
1967 		NGI_SET_WRITER(item);
1968 	else
1969 		NGI_SET_READER(item);
1970 	item->el_next = NULL;	/* maybe not needed */
1971 	*ngq->last = item;
1972 	/*
1973 	 * If it was the first item in the queue then we need to
1974 	 * set the last pointer and the type flags.
1975 	 */
1976 	if (ngq->last == &(ngq->queue))
1977 		atomic_add_long(&ngq->q_flags, OP_PENDING);
1978 
1979 	ngq->last = &(item->el_next);
1980 	/*
1981 	 * We can take the worklist lock with the node locked
1982 	 * BUT NOT THE REVERSE!
1983 	 */
1984 	if (NEXT_QUEUED_ITEM_CAN_PROCEED(ngq))
1985 		ng_setisr(ngq->q_node);
1986 }
1987 
1988 
1989 /*
1990  * This function 'cheats' in that it first tries to 'grab' the use of the
1991  * node, without going through the mutex. We can do this becasue of the
1992  * semantics of the lock. The semantics include a clause that says that the
1993  * value of the readers count is invalid if the WRITER_ACTIVE flag is set. It
1994  * also says that the WRITER_ACTIVE flag cannot be set if the readers count
1995  * is not zero. Note that this talks about what is valid to SET the
1996  * WRITER_ACTIVE flag, because from the moment it is set, the value if the
1997  * reader count is immaterial, and not valid. The two 'pending' flags have a
1998  * similar effect, in that If they are orthogonal to the two active fields in
1999  * how they are set, but if either is set, the attempted 'grab' need to be
2000  * backed out because there is earlier work, and we maintain ordering in the
2001  * queue. The result of this is that the reader request can try obtain use of
2002  * the node with only a single atomic addition, and without any of the mutex
2003  * overhead. If this fails the operation degenerates to the same as for other
2004  * cases.
2005  *
2006  */
2007 static __inline item_p
2008 ng_acquire_read(struct ng_queue *ngq, item_p item)
2009 {
2010 	KASSERT(ngq != &ng_deadnode.nd_input_queue,
2011 	    ("%s: working on deadnode", __func__));
2012 
2013 	/* ######### Hack alert ######### */
2014 	atomic_add_long(&ngq->q_flags, READER_INCREMENT);
2015 	if ((ngq->q_flags & NGQ_RMASK) == 0) {
2016 		/* Successfully grabbed node */
2017 		return (item);
2018 	}
2019 	/* undo the damage if we didn't succeed */
2020 	atomic_subtract_long(&ngq->q_flags, READER_INCREMENT);
2021 
2022 	/* ######### End Hack alert ######### */
2023 	mtx_lock_spin((&ngq->q_mtx));
2024 	/*
2025 	 * Try again. Another processor (or interrupt for that matter) may
2026 	 * have removed the last queued item that was stopping us from
2027 	 * running, between the previous test, and the moment that we took
2028 	 * the mutex. (Or maybe a writer completed.)
2029 	 * Even if another fast-track reader hits during this period
2030 	 * we don't care as multiple readers is OK.
2031 	 */
2032 	if ((ngq->q_flags & NGQ_RMASK) == 0) {
2033 		atomic_add_long(&ngq->q_flags, READER_INCREMENT);
2034 		mtx_unlock_spin((&ngq->q_mtx));
2035 		return (item);
2036 	}
2037 
2038 	/*
2039 	 * and queue the request for later.
2040 	 */
2041 	ng_queue_rw(ngq, item, NGQRW_R);
2042 	mtx_unlock_spin(&(ngq->q_mtx));
2043 
2044 	return (NULL);
2045 }
2046 
2047 static __inline item_p
2048 ng_acquire_write(struct ng_queue *ngq, item_p item)
2049 {
2050 	KASSERT(ngq != &ng_deadnode.nd_input_queue,
2051 	    ("%s: working on deadnode", __func__));
2052 
2053 restart:
2054 	mtx_lock_spin(&(ngq->q_mtx));
2055 	/*
2056 	 * If there are no readers, no writer, and no pending packets, then
2057 	 * we can just go ahead. In all other situations we need to queue the
2058 	 * request
2059 	 */
2060 	if ((ngq->q_flags & NGQ_WMASK) == 0) {
2061 		/* collision could happen *HERE* */
2062 		atomic_add_long(&ngq->q_flags, WRITER_ACTIVE);
2063 		mtx_unlock_spin((&ngq->q_mtx));
2064 		if (ngq->q_flags & READER_MASK) {
2065 			/* Collision with fast-track reader */
2066 			atomic_subtract_long(&ngq->q_flags, WRITER_ACTIVE);
2067 			goto restart;
2068 		}
2069 		return (item);
2070 	}
2071 
2072 	/*
2073 	 * and queue the request for later.
2074 	 */
2075 	ng_queue_rw(ngq, item, NGQRW_W);
2076 	mtx_unlock_spin(&(ngq->q_mtx));
2077 
2078 	return (NULL);
2079 }
2080 
2081 static __inline void
2082 ng_leave_read(struct ng_queue *ngq)
2083 {
2084 	atomic_subtract_long(&ngq->q_flags, READER_INCREMENT);
2085 }
2086 
2087 static __inline void
2088 ng_leave_write(struct ng_queue *ngq)
2089 {
2090 	atomic_subtract_long(&ngq->q_flags, WRITER_ACTIVE);
2091 }
2092 
2093 static void
2094 ng_flush_input_queue(struct ng_queue * ngq)
2095 {
2096 	item_p item;
2097 
2098 	mtx_lock_spin(&ngq->q_mtx);
2099 	while (ngq->queue) {
2100 		item = ngq->queue;
2101 		ngq->queue = item->el_next;
2102 		if (ngq->last == &(item->el_next)) {
2103 			ngq->last = &(ngq->queue);
2104 			atomic_add_long(&ngq->q_flags, -OP_PENDING);
2105 		}
2106 		mtx_unlock_spin(&ngq->q_mtx);
2107 
2108 		/* If the item is supplying a callback, call it with an error */
2109 		if (item->apply != NULL) {
2110 			(item->apply)(item->context, ENOENT);
2111 			item->apply = NULL;
2112 		}
2113 		NG_FREE_ITEM(item);
2114 		mtx_lock_spin(&ngq->q_mtx);
2115 	}
2116 	/*
2117 	 * Take us off the work queue if we are there.
2118 	 * We definatly have no work to be done.
2119 	 */
2120 	ng_worklist_remove(ngq->q_node);
2121 	mtx_unlock_spin(&ngq->q_mtx);
2122 }
2123 
2124 /***********************************************************************
2125 * Externally visible method for sending or queueing messages or data.
2126 ***********************************************************************/
2127 
2128 /*
2129  * The module code should have filled out the item correctly by this stage:
2130  * Common:
2131  *    reference to destination node.
2132  *    Reference to destination rcv hook if relevant.
2133  * Data:
2134  *    pointer to mbuf
2135  * Control_Message:
2136  *    pointer to msg.
2137  *    ID of original sender node. (return address)
2138  * Function:
2139  *    Function pointer
2140  *    void * argument
2141  *    integer argument
2142  *
2143  * The nodes have several routines and macros to help with this task:
2144  */
2145 
2146 int
2147 ng_snd_item(item_p item, int flags)
2148 {
2149 	hook_p hook = NGI_HOOK(item);
2150 	node_p node = NGI_NODE(item);
2151 	int queue, rw;
2152 	struct ng_queue * ngq = &node->nd_input_queue;
2153 	int error = 0;
2154 
2155 #ifdef	NETGRAPH_DEBUG
2156 	_ngi_check(item, __FILE__, __LINE__);
2157 #endif
2158 
2159 	queue = (flags & NG_QUEUE) ? 1 : 0;
2160 
2161 	if (item == NULL) {
2162 		TRAP_ERROR();
2163 		return (EINVAL);	/* failed to get queue element */
2164 	}
2165 	if (node == NULL) {
2166 		NG_FREE_ITEM(item);
2167 		TRAP_ERROR();
2168 		return (EINVAL);	/* No address */
2169 	}
2170 	switch(item->el_flags & NGQF_TYPE) {
2171 	case NGQF_DATA:
2172 		/*
2173 		 * DATA MESSAGE
2174 		 * Delivered to a node via a non-optional hook.
2175 		 * Both should be present in the item even though
2176 		 * the node is derivable from the hook.
2177 		 * References are held on both by the item.
2178 		 */
2179 
2180 		/* Protect nodes from sending NULL pointers
2181 		 * to each other
2182 		 */
2183 		if (NGI_M(item) == NULL)
2184 			return (EINVAL);
2185 
2186 		CHECK_DATA_MBUF(NGI_M(item));
2187 		if (hook == NULL) {
2188 			NG_FREE_ITEM(item);
2189 			TRAP_ERROR();
2190 			return(EINVAL);
2191 		}
2192 		if ((NG_HOOK_NOT_VALID(hook))
2193 		|| (NG_NODE_NOT_VALID(NG_HOOK_NODE(hook)))) {
2194 			NG_FREE_ITEM(item);
2195 			return (ENOTCONN);
2196 		}
2197 		if ((hook->hk_flags & HK_QUEUE)) {
2198 			queue = 1;
2199 		}
2200 		break;
2201 	case NGQF_MESG:
2202 		/*
2203 		 * CONTROL MESSAGE
2204 		 * Delivered to a node.
2205 		 * Hook is optional.
2206 		 * References are held by the item on the node and
2207 		 * the hook if it is present.
2208 		 */
2209 		if (hook && (hook->hk_flags & HK_QUEUE)) {
2210 			queue = 1;
2211 		}
2212 		break;
2213 	case NGQF_FN:
2214 		break;
2215 	default:
2216 		NG_FREE_ITEM(item);
2217 		TRAP_ERROR();
2218 		return (EINVAL);
2219 	}
2220 	switch(item->el_flags & NGQF_RW) {
2221 	case NGQF_READER:
2222 		rw = NGQRW_R;
2223 		break;
2224 	case NGQF_WRITER:
2225 		rw = NGQRW_W;
2226 		break;
2227 	default:
2228 		panic("%s: invalid item flags %lx", __func__, item->el_flags);
2229 	}
2230 
2231 	/*
2232 	 * If the node specifies single threading, force writer semantics.
2233 	 * Similarly, the node may say one hook always produces writers.
2234 	 * These are overrides.
2235 	 */
2236 	if ((node->nd_flags & NGF_FORCE_WRITER)
2237 	    || (hook && (hook->hk_flags & HK_FORCE_WRITER)))
2238 			rw = NGQRW_W;
2239 
2240 	if (queue) {
2241 		/* Put it on the queue for that node*/
2242 #ifdef	NETGRAPH_DEBUG
2243 		_ngi_check(item, __FILE__, __LINE__);
2244 #endif
2245 		mtx_lock_spin(&(ngq->q_mtx));
2246 		ng_queue_rw(ngq, item, rw);
2247 		mtx_unlock_spin(&(ngq->q_mtx));
2248 
2249 		if (flags & NG_PROGRESS)
2250 			return (EINPROGRESS);
2251 		else
2252 			return (0);
2253 	}
2254 
2255 	/*
2256 	 * We already decided how we will be queueud or treated.
2257 	 * Try get the appropriate operating permission.
2258 	 */
2259  	if (rw == NGQRW_R)
2260 		item = ng_acquire_read(ngq, item);
2261 	else
2262 		item = ng_acquire_write(ngq, item);
2263 
2264 
2265 	if (item == NULL) {
2266 		if (flags & NG_PROGRESS)
2267 			return (EINPROGRESS);
2268 		else
2269 			return (0);
2270 	}
2271 
2272 #ifdef	NETGRAPH_DEBUG
2273 	_ngi_check(item, __FILE__, __LINE__);
2274 #endif
2275 
2276 	NGI_GET_NODE(item, node); /* zaps stored node */
2277 
2278 	error = ng_apply_item(node, item, rw); /* drops r/w lock when done */
2279 
2280 	/*
2281 	 * If the node goes away when we remove the reference,
2282 	 * whatever we just did caused it.. whatever we do, DO NOT
2283 	 * access the node again!
2284 	 */
2285 	if (NG_NODE_UNREF(node) == 0) {
2286 		return (error);
2287 	}
2288 
2289 	mtx_lock_spin(&(ngq->q_mtx));
2290 	if (NEXT_QUEUED_ITEM_CAN_PROCEED(ngq))
2291 		ng_setisr(ngq->q_node);
2292 	mtx_unlock_spin(&(ngq->q_mtx));
2293 
2294 	return (error);
2295 }
2296 
2297 /*
2298  * We have an item that was possibly queued somewhere.
2299  * It should contain all the information needed
2300  * to run it on the appropriate node/hook.
2301  */
2302 static int
2303 ng_apply_item(node_p node, item_p item, int rw)
2304 {
2305 	hook_p  hook;
2306 	int	error = 0;
2307 	ng_rcvdata_t *rcvdata;
2308 	ng_rcvmsg_t *rcvmsg;
2309 	ng_apply_t *apply = NULL;
2310 	void	*context = NULL;
2311 
2312 	NGI_GET_HOOK(item, hook); /* clears stored hook */
2313 #ifdef	NETGRAPH_DEBUG
2314 	_ngi_check(item, __FILE__, __LINE__);
2315 #endif
2316 
2317 	/*
2318 	 * If the item has an "apply" callback, store it.
2319 	 * Clear item's callback immediately, to avoid an extra call if
2320 	 * the item is reused by the destination node.
2321 	 */
2322 	if (item->apply != NULL) {
2323 		apply = item->apply;
2324 		context = item->context;
2325 		item->apply = NULL;
2326 	}
2327 
2328 	switch (item->el_flags & NGQF_TYPE) {
2329 	case NGQF_DATA:
2330 		/*
2331 		 * Check things are still ok as when we were queued.
2332 		 */
2333 		if ((hook == NULL)
2334 		|| NG_HOOK_NOT_VALID(hook)
2335 		|| NG_NODE_NOT_VALID(node) ) {
2336 			error = EIO;
2337 			NG_FREE_ITEM(item);
2338 			break;
2339 		}
2340 		/*
2341 		 * If no receive method, just silently drop it.
2342 		 * Give preference to the hook over-ride method
2343 		 */
2344 		if ((!(rcvdata = hook->hk_rcvdata))
2345 		&& (!(rcvdata = NG_HOOK_NODE(hook)->nd_type->rcvdata))) {
2346 			error = 0;
2347 			NG_FREE_ITEM(item);
2348 			break;
2349 		}
2350 		error = (*rcvdata)(hook, item);
2351 		break;
2352 	case NGQF_MESG:
2353 		if (hook) {
2354 			if (NG_HOOK_NOT_VALID(hook)) {
2355 				/*
2356 				 * The hook has been zapped then we can't
2357 				 * use it. Immediatly drop its reference.
2358 				 * The message may not need it.
2359 				 */
2360 				NG_HOOK_UNREF(hook);
2361 				hook = NULL;
2362 			}
2363 		}
2364 		/*
2365 		 * Similarly, if the node is a zombie there is
2366 		 * nothing we can do with it, drop everything.
2367 		 */
2368 		if (NG_NODE_NOT_VALID(node)) {
2369 			TRAP_ERROR();
2370 			error = EINVAL;
2371 			NG_FREE_ITEM(item);
2372 		} else {
2373 			/*
2374 			 * Call the appropriate message handler for the object.
2375 			 * It is up to the message handler to free the message.
2376 			 * If it's a generic message, handle it generically,
2377 			 * otherwise call the type's message handler
2378 			 * (if it exists)
2379 			 * XXX (race). Remember that a queued message may
2380 			 * reference a node or hook that has just been
2381 			 * invalidated. It will exist as the queue code
2382 			 * is holding a reference, but..
2383 			 */
2384 
2385 			struct ng_mesg *msg = NGI_MSG(item);
2386 
2387 			/*
2388 			 * check if the generic handler owns it.
2389 			 */
2390 			if ((msg->header.typecookie == NGM_GENERIC_COOKIE)
2391 			&& ((msg->header.flags & NGF_RESP) == 0)) {
2392 				error = ng_generic_msg(node, item, hook);
2393 				break;
2394 			}
2395 			/*
2396 			 * Now see if there is a handler (hook or node specific)
2397 			 * in the target node. If none, silently discard.
2398 			 */
2399 			if (((!hook) || (!(rcvmsg = hook->hk_rcvmsg)))
2400 			&& (!(rcvmsg = node->nd_type->rcvmsg))) {
2401 				TRAP_ERROR();
2402 				error = 0;
2403 				NG_FREE_ITEM(item);
2404 				break;
2405 			}
2406 			error = (*rcvmsg)(node, item, hook);
2407 		}
2408 		break;
2409 	case NGQF_FN:
2410 		/*
2411 		 *  We have to implicitly trust the hook,
2412 		 * as some of these are used for system purposes
2413 		 * where the hook is invalid. In the case of
2414 		 * the shutdown message we allow it to hit
2415 		 * even if the node is invalid.
2416 		 */
2417 		if ((NG_NODE_NOT_VALID(node))
2418 		&& (NGI_FN(item) != &ng_rmnode)) {
2419 			TRAP_ERROR();
2420 			error = EINVAL;
2421 			NG_FREE_ITEM(item);
2422 			break;
2423 		}
2424 		(*NGI_FN(item))(node, hook, NGI_ARG1(item), NGI_ARG2(item));
2425 		NG_FREE_ITEM(item);
2426 		break;
2427 
2428 	}
2429 	/*
2430 	 * We held references on some of the resources
2431 	 * that we took from the item. Now that we have
2432 	 * finished doing everything, drop those references.
2433 	 */
2434 	if (hook) {
2435 		NG_HOOK_UNREF(hook);
2436 	}
2437 
2438  	if (rw == NGQRW_R) {
2439 		ng_leave_read(&node->nd_input_queue);
2440 	} else {
2441 		ng_leave_write(&node->nd_input_queue);
2442 	}
2443 
2444 	/* Apply callback. */
2445 	if (apply != NULL)
2446 		(*apply)(context, error);
2447 
2448 	return (error);
2449 }
2450 
2451 /***********************************************************************
2452  * Implement the 'generic' control messages
2453  ***********************************************************************/
2454 static int
2455 ng_generic_msg(node_p here, item_p item, hook_p lasthook)
2456 {
2457 	int error = 0;
2458 	struct ng_mesg *msg;
2459 	struct ng_mesg *resp = NULL;
2460 
2461 	NGI_GET_MSG(item, msg);
2462 	if (msg->header.typecookie != NGM_GENERIC_COOKIE) {
2463 		TRAP_ERROR();
2464 		error = EINVAL;
2465 		goto out;
2466 	}
2467 	switch (msg->header.cmd) {
2468 	case NGM_SHUTDOWN:
2469 		ng_rmnode(here, NULL, NULL, 0);
2470 		break;
2471 	case NGM_MKPEER:
2472 	    {
2473 		struct ngm_mkpeer *const mkp = (struct ngm_mkpeer *) msg->data;
2474 
2475 		if (msg->header.arglen != sizeof(*mkp)) {
2476 			TRAP_ERROR();
2477 			error = EINVAL;
2478 			break;
2479 		}
2480 		mkp->type[sizeof(mkp->type) - 1] = '\0';
2481 		mkp->ourhook[sizeof(mkp->ourhook) - 1] = '\0';
2482 		mkp->peerhook[sizeof(mkp->peerhook) - 1] = '\0';
2483 		error = ng_mkpeer(here, mkp->ourhook, mkp->peerhook, mkp->type);
2484 		break;
2485 	    }
2486 	case NGM_CONNECT:
2487 	    {
2488 		struct ngm_connect *const con =
2489 			(struct ngm_connect *) msg->data;
2490 		node_p node2;
2491 
2492 		if (msg->header.arglen != sizeof(*con)) {
2493 			TRAP_ERROR();
2494 			error = EINVAL;
2495 			break;
2496 		}
2497 		con->path[sizeof(con->path) - 1] = '\0';
2498 		con->ourhook[sizeof(con->ourhook) - 1] = '\0';
2499 		con->peerhook[sizeof(con->peerhook) - 1] = '\0';
2500 		/* Don't forget we get a reference.. */
2501 		error = ng_path2noderef(here, con->path, &node2, NULL);
2502 		if (error)
2503 			break;
2504 		error = ng_con_nodes(here, con->ourhook, node2, con->peerhook);
2505 		NG_NODE_UNREF(node2);
2506 		break;
2507 	    }
2508 	case NGM_NAME:
2509 	    {
2510 		struct ngm_name *const nam = (struct ngm_name *) msg->data;
2511 
2512 		if (msg->header.arglen != sizeof(*nam)) {
2513 			TRAP_ERROR();
2514 			error = EINVAL;
2515 			break;
2516 		}
2517 		nam->name[sizeof(nam->name) - 1] = '\0';
2518 		error = ng_name_node(here, nam->name);
2519 		break;
2520 	    }
2521 	case NGM_RMHOOK:
2522 	    {
2523 		struct ngm_rmhook *const rmh = (struct ngm_rmhook *) msg->data;
2524 		hook_p hook;
2525 
2526 		if (msg->header.arglen != sizeof(*rmh)) {
2527 			TRAP_ERROR();
2528 			error = EINVAL;
2529 			break;
2530 		}
2531 		rmh->ourhook[sizeof(rmh->ourhook) - 1] = '\0';
2532 		if ((hook = ng_findhook(here, rmh->ourhook)) != NULL)
2533 			ng_destroy_hook(hook);
2534 		break;
2535 	    }
2536 	case NGM_NODEINFO:
2537 	    {
2538 		struct nodeinfo *ni;
2539 
2540 		NG_MKRESPONSE(resp, msg, sizeof(*ni), M_NOWAIT);
2541 		if (resp == NULL) {
2542 			error = ENOMEM;
2543 			break;
2544 		}
2545 
2546 		/* Fill in node info */
2547 		ni = (struct nodeinfo *) resp->data;
2548 		if (NG_NODE_HAS_NAME(here))
2549 			strcpy(ni->name, NG_NODE_NAME(here));
2550 		strcpy(ni->type, here->nd_type->name);
2551 		ni->id = ng_node2ID(here);
2552 		ni->hooks = here->nd_numhooks;
2553 		break;
2554 	    }
2555 	case NGM_LISTHOOKS:
2556 	    {
2557 		const int nhooks = here->nd_numhooks;
2558 		struct hooklist *hl;
2559 		struct nodeinfo *ni;
2560 		hook_p hook;
2561 
2562 		/* Get response struct */
2563 		NG_MKRESPONSE(resp, msg, sizeof(*hl)
2564 		    + (nhooks * sizeof(struct linkinfo)), M_NOWAIT);
2565 		if (resp == NULL) {
2566 			error = ENOMEM;
2567 			break;
2568 		}
2569 		hl = (struct hooklist *) resp->data;
2570 		ni = &hl->nodeinfo;
2571 
2572 		/* Fill in node info */
2573 		if (NG_NODE_HAS_NAME(here))
2574 			strcpy(ni->name, NG_NODE_NAME(here));
2575 		strcpy(ni->type, here->nd_type->name);
2576 		ni->id = ng_node2ID(here);
2577 
2578 		/* Cycle through the linked list of hooks */
2579 		ni->hooks = 0;
2580 		LIST_FOREACH(hook, &here->nd_hooks, hk_hooks) {
2581 			struct linkinfo *const link = &hl->link[ni->hooks];
2582 
2583 			if (ni->hooks >= nhooks) {
2584 				log(LOG_ERR, "%s: number of %s changed\n",
2585 				    __func__, "hooks");
2586 				break;
2587 			}
2588 			if (NG_HOOK_NOT_VALID(hook))
2589 				continue;
2590 			strcpy(link->ourhook, NG_HOOK_NAME(hook));
2591 			strcpy(link->peerhook, NG_PEER_HOOK_NAME(hook));
2592 			if (NG_PEER_NODE_NAME(hook)[0] != '\0')
2593 				strcpy(link->nodeinfo.name,
2594 				    NG_PEER_NODE_NAME(hook));
2595 			strcpy(link->nodeinfo.type,
2596 			   NG_PEER_NODE(hook)->nd_type->name);
2597 			link->nodeinfo.id = ng_node2ID(NG_PEER_NODE(hook));
2598 			link->nodeinfo.hooks = NG_PEER_NODE(hook)->nd_numhooks;
2599 			ni->hooks++;
2600 		}
2601 		break;
2602 	    }
2603 
2604 	case NGM_LISTNAMES:
2605 	case NGM_LISTNODES:
2606 	    {
2607 		const int unnamed = (msg->header.cmd == NGM_LISTNODES);
2608 		struct namelist *nl;
2609 		node_p node;
2610 		int num = 0;
2611 
2612 		mtx_lock(&ng_nodelist_mtx);
2613 		/* Count number of nodes */
2614 		LIST_FOREACH(node, &ng_nodelist, nd_nodes) {
2615 			if (NG_NODE_IS_VALID(node)
2616 			&& (unnamed || NG_NODE_HAS_NAME(node))) {
2617 				num++;
2618 			}
2619 		}
2620 		mtx_unlock(&ng_nodelist_mtx);
2621 
2622 		/* Get response struct */
2623 		NG_MKRESPONSE(resp, msg, sizeof(*nl)
2624 		    + (num * sizeof(struct nodeinfo)), M_NOWAIT);
2625 		if (resp == NULL) {
2626 			error = ENOMEM;
2627 			break;
2628 		}
2629 		nl = (struct namelist *) resp->data;
2630 
2631 		/* Cycle through the linked list of nodes */
2632 		nl->numnames = 0;
2633 		mtx_lock(&ng_nodelist_mtx);
2634 		LIST_FOREACH(node, &ng_nodelist, nd_nodes) {
2635 			struct nodeinfo *const np = &nl->nodeinfo[nl->numnames];
2636 
2637 			if (nl->numnames >= num) {
2638 				log(LOG_ERR, "%s: number of %s changed\n",
2639 				    __func__, "nodes");
2640 				break;
2641 			}
2642 			if (NG_NODE_NOT_VALID(node))
2643 				continue;
2644 			if (!unnamed && (! NG_NODE_HAS_NAME(node)))
2645 				continue;
2646 			if (NG_NODE_HAS_NAME(node))
2647 				strcpy(np->name, NG_NODE_NAME(node));
2648 			strcpy(np->type, node->nd_type->name);
2649 			np->id = ng_node2ID(node);
2650 			np->hooks = node->nd_numhooks;
2651 			nl->numnames++;
2652 		}
2653 		mtx_unlock(&ng_nodelist_mtx);
2654 		break;
2655 	    }
2656 
2657 	case NGM_LISTTYPES:
2658 	    {
2659 		struct typelist *tl;
2660 		struct ng_type *type;
2661 		int num = 0;
2662 
2663 		mtx_lock(&ng_typelist_mtx);
2664 		/* Count number of types */
2665 		LIST_FOREACH(type, &ng_typelist, types) {
2666 			num++;
2667 		}
2668 		mtx_unlock(&ng_typelist_mtx);
2669 
2670 		/* Get response struct */
2671 		NG_MKRESPONSE(resp, msg, sizeof(*tl)
2672 		    + (num * sizeof(struct typeinfo)), M_NOWAIT);
2673 		if (resp == NULL) {
2674 			error = ENOMEM;
2675 			break;
2676 		}
2677 		tl = (struct typelist *) resp->data;
2678 
2679 		/* Cycle through the linked list of types */
2680 		tl->numtypes = 0;
2681 		mtx_lock(&ng_typelist_mtx);
2682 		LIST_FOREACH(type, &ng_typelist, types) {
2683 			struct typeinfo *const tp = &tl->typeinfo[tl->numtypes];
2684 
2685 			if (tl->numtypes >= num) {
2686 				log(LOG_ERR, "%s: number of %s changed\n",
2687 				    __func__, "types");
2688 				break;
2689 			}
2690 			strcpy(tp->type_name, type->name);
2691 			tp->numnodes = type->refs - 1; /* don't count list */
2692 			tl->numtypes++;
2693 		}
2694 		mtx_unlock(&ng_typelist_mtx);
2695 		break;
2696 	    }
2697 
2698 	case NGM_BINARY2ASCII:
2699 	    {
2700 		int bufSize = 20 * 1024;	/* XXX hard coded constant */
2701 		const struct ng_parse_type *argstype;
2702 		const struct ng_cmdlist *c;
2703 		struct ng_mesg *binary, *ascii;
2704 
2705 		/* Data area must contain a valid netgraph message */
2706 		binary = (struct ng_mesg *)msg->data;
2707 		if (msg->header.arglen < sizeof(struct ng_mesg) ||
2708 		    (msg->header.arglen - sizeof(struct ng_mesg) <
2709 		    binary->header.arglen)) {
2710 			TRAP_ERROR();
2711 			error = EINVAL;
2712 			break;
2713 		}
2714 
2715 		/* Get a response message with lots of room */
2716 		NG_MKRESPONSE(resp, msg, sizeof(*ascii) + bufSize, M_NOWAIT);
2717 		if (resp == NULL) {
2718 			error = ENOMEM;
2719 			break;
2720 		}
2721 		ascii = (struct ng_mesg *)resp->data;
2722 
2723 		/* Copy binary message header to response message payload */
2724 		bcopy(binary, ascii, sizeof(*binary));
2725 
2726 		/* Find command by matching typecookie and command number */
2727 		for (c = here->nd_type->cmdlist;
2728 		    c != NULL && c->name != NULL; c++) {
2729 			if (binary->header.typecookie == c->cookie
2730 			    && binary->header.cmd == c->cmd)
2731 				break;
2732 		}
2733 		if (c == NULL || c->name == NULL) {
2734 			for (c = ng_generic_cmds; c->name != NULL; c++) {
2735 				if (binary->header.typecookie == c->cookie
2736 				    && binary->header.cmd == c->cmd)
2737 					break;
2738 			}
2739 			if (c->name == NULL) {
2740 				NG_FREE_MSG(resp);
2741 				error = ENOSYS;
2742 				break;
2743 			}
2744 		}
2745 
2746 		/* Convert command name to ASCII */
2747 		snprintf(ascii->header.cmdstr, sizeof(ascii->header.cmdstr),
2748 		    "%s", c->name);
2749 
2750 		/* Convert command arguments to ASCII */
2751 		argstype = (binary->header.flags & NGF_RESP) ?
2752 		    c->respType : c->mesgType;
2753 		if (argstype == NULL) {
2754 			*ascii->data = '\0';
2755 		} else {
2756 			if ((error = ng_unparse(argstype,
2757 			    (u_char *)binary->data,
2758 			    ascii->data, bufSize)) != 0) {
2759 				NG_FREE_MSG(resp);
2760 				break;
2761 			}
2762 		}
2763 
2764 		/* Return the result as struct ng_mesg plus ASCII string */
2765 		bufSize = strlen(ascii->data) + 1;
2766 		ascii->header.arglen = bufSize;
2767 		resp->header.arglen = sizeof(*ascii) + bufSize;
2768 		break;
2769 	    }
2770 
2771 	case NGM_ASCII2BINARY:
2772 	    {
2773 		int bufSize = 2000;	/* XXX hard coded constant */
2774 		const struct ng_cmdlist *c;
2775 		const struct ng_parse_type *argstype;
2776 		struct ng_mesg *ascii, *binary;
2777 		int off = 0;
2778 
2779 		/* Data area must contain at least a struct ng_mesg + '\0' */
2780 		ascii = (struct ng_mesg *)msg->data;
2781 		if ((msg->header.arglen < sizeof(*ascii) + 1) ||
2782 		    (ascii->header.arglen < 1) ||
2783 		    (msg->header.arglen < sizeof(*ascii) +
2784 		    ascii->header.arglen)) {
2785 			TRAP_ERROR();
2786 			error = EINVAL;
2787 			break;
2788 		}
2789 		ascii->data[ascii->header.arglen - 1] = '\0';
2790 
2791 		/* Get a response message with lots of room */
2792 		NG_MKRESPONSE(resp, msg, sizeof(*binary) + bufSize, M_NOWAIT);
2793 		if (resp == NULL) {
2794 			error = ENOMEM;
2795 			break;
2796 		}
2797 		binary = (struct ng_mesg *)resp->data;
2798 
2799 		/* Copy ASCII message header to response message payload */
2800 		bcopy(ascii, binary, sizeof(*ascii));
2801 
2802 		/* Find command by matching ASCII command string */
2803 		for (c = here->nd_type->cmdlist;
2804 		    c != NULL && c->name != NULL; c++) {
2805 			if (strcmp(ascii->header.cmdstr, c->name) == 0)
2806 				break;
2807 		}
2808 		if (c == NULL || c->name == NULL) {
2809 			for (c = ng_generic_cmds; c->name != NULL; c++) {
2810 				if (strcmp(ascii->header.cmdstr, c->name) == 0)
2811 					break;
2812 			}
2813 			if (c->name == NULL) {
2814 				NG_FREE_MSG(resp);
2815 				error = ENOSYS;
2816 				break;
2817 			}
2818 		}
2819 
2820 		/* Convert command name to binary */
2821 		binary->header.cmd = c->cmd;
2822 		binary->header.typecookie = c->cookie;
2823 
2824 		/* Convert command arguments to binary */
2825 		argstype = (binary->header.flags & NGF_RESP) ?
2826 		    c->respType : c->mesgType;
2827 		if (argstype == NULL) {
2828 			bufSize = 0;
2829 		} else {
2830 			if ((error = ng_parse(argstype, ascii->data,
2831 			    &off, (u_char *)binary->data, &bufSize)) != 0) {
2832 				NG_FREE_MSG(resp);
2833 				break;
2834 			}
2835 		}
2836 
2837 		/* Return the result */
2838 		binary->header.arglen = bufSize;
2839 		resp->header.arglen = sizeof(*binary) + bufSize;
2840 		break;
2841 	    }
2842 
2843 	case NGM_TEXT_CONFIG:
2844 	case NGM_TEXT_STATUS:
2845 		/*
2846 		 * This one is tricky as it passes the command down to the
2847 		 * actual node, even though it is a generic type command.
2848 		 * This means we must assume that the item/msg is already freed
2849 		 * when control passes back to us.
2850 		 */
2851 		if (here->nd_type->rcvmsg != NULL) {
2852 			NGI_MSG(item) = msg; /* put it back as we found it */
2853 			return((*here->nd_type->rcvmsg)(here, item, lasthook));
2854 		}
2855 		/* Fall through if rcvmsg not supported */
2856 	default:
2857 		TRAP_ERROR();
2858 		error = EINVAL;
2859 	}
2860 	/*
2861 	 * Sometimes a generic message may be statically allocated
2862 	 * to avoid problems with allocating when in tight memeory situations.
2863 	 * Don't free it if it is so.
2864 	 * I break them appart here, because erros may cause a free if the item
2865 	 * in which case we'd be doing it twice.
2866 	 * they are kept together above, to simplify freeing.
2867 	 */
2868 out:
2869 	NG_RESPOND_MSG(error, here, item, resp);
2870 	if (msg)
2871 		NG_FREE_MSG(msg);
2872 	return (error);
2873 }
2874 
2875 /************************************************************************
2876 			Queue element get/free routines
2877 ************************************************************************/
2878 
2879 uma_zone_t			ng_qzone;
2880 static int			maxalloc = 512;	/* limit the damage of a leak */
2881 
2882 TUNABLE_INT("net.graph.maxalloc", &maxalloc);
2883 SYSCTL_INT(_net_graph, OID_AUTO, maxalloc, CTLFLAG_RDTUN, &maxalloc,
2884     0, "Maximum number of queue items to allocate");
2885 
2886 #ifdef	NETGRAPH_DEBUG
2887 static TAILQ_HEAD(, ng_item) ng_itemlist = TAILQ_HEAD_INITIALIZER(ng_itemlist);
2888 static int			allocated;	/* number of items malloc'd */
2889 #endif
2890 
2891 /*
2892  * Get a queue entry.
2893  * This is usually called when a packet first enters netgraph.
2894  * By definition, this is usually from an interrupt, or from a user.
2895  * Users are not so important, but try be quick for the times that it's
2896  * an interrupt.
2897  */
2898 static __inline item_p
2899 ng_getqblk(int flags)
2900 {
2901 	item_p item = NULL;
2902 	int wait;
2903 
2904 	wait = (flags & NG_WAITOK) ? M_WAITOK : M_NOWAIT;
2905 
2906 	item = uma_zalloc(ng_qzone, wait | M_ZERO);
2907 
2908 #ifdef	NETGRAPH_DEBUG
2909 	if (item) {
2910 			mtx_lock(&ngq_mtx);
2911 			TAILQ_INSERT_TAIL(&ng_itemlist, item, all);
2912 			allocated++;
2913 			mtx_unlock(&ngq_mtx);
2914 	}
2915 #endif
2916 
2917 	return (item);
2918 }
2919 
2920 /*
2921  * Release a queue entry
2922  */
2923 void
2924 ng_free_item(item_p item)
2925 {
2926 	KASSERT(item->apply == NULL, ("%s: leaking apply callback", __func__));
2927 
2928 	/*
2929 	 * The item may hold resources on it's own. We need to free
2930 	 * these before we can free the item. What they are depends upon
2931 	 * what kind of item it is. it is important that nodes zero
2932 	 * out pointers to resources that they remove from the item
2933 	 * or we release them again here.
2934 	 */
2935 	switch (item->el_flags & NGQF_TYPE) {
2936 	case NGQF_DATA:
2937 		/* If we have an mbuf still attached.. */
2938 		NG_FREE_M(_NGI_M(item));
2939 		break;
2940 	case NGQF_MESG:
2941 		_NGI_RETADDR(item) = 0;
2942 		NG_FREE_MSG(_NGI_MSG(item));
2943 		break;
2944 	case NGQF_FN:
2945 		/* nothing to free really, */
2946 		_NGI_FN(item) = NULL;
2947 		_NGI_ARG1(item) = NULL;
2948 		_NGI_ARG2(item) = 0;
2949 	case NGQF_UNDEF:
2950 		break;
2951 	}
2952 	/* If we still have a node or hook referenced... */
2953 	_NGI_CLR_NODE(item);
2954 	_NGI_CLR_HOOK(item);
2955 
2956 #ifdef	NETGRAPH_DEBUG
2957 	mtx_lock(&ngq_mtx);
2958 	TAILQ_REMOVE(&ng_itemlist, item, all);
2959 	allocated--;
2960 	mtx_unlock(&ngq_mtx);
2961 #endif
2962 	uma_zfree(ng_qzone, item);
2963 }
2964 
2965 /************************************************************************
2966 			Module routines
2967 ************************************************************************/
2968 
2969 /*
2970  * Handle the loading/unloading of a netgraph node type module
2971  */
2972 int
2973 ng_mod_event(module_t mod, int event, void *data)
2974 {
2975 	struct ng_type *const type = data;
2976 	int s, error = 0;
2977 
2978 	switch (event) {
2979 	case MOD_LOAD:
2980 
2981 		/* Register new netgraph node type */
2982 		s = splnet();
2983 		if ((error = ng_newtype(type)) != 0) {
2984 			splx(s);
2985 			break;
2986 		}
2987 
2988 		/* Call type specific code */
2989 		if (type->mod_event != NULL)
2990 			if ((error = (*type->mod_event)(mod, event, data))) {
2991 				mtx_lock(&ng_typelist_mtx);
2992 				type->refs--;	/* undo it */
2993 				LIST_REMOVE(type, types);
2994 				mtx_unlock(&ng_typelist_mtx);
2995 			}
2996 		splx(s);
2997 		break;
2998 
2999 	case MOD_UNLOAD:
3000 		s = splnet();
3001 		if (type->refs > 1) {		/* make sure no nodes exist! */
3002 			error = EBUSY;
3003 		} else {
3004 			if (type->refs == 0) {
3005 				/* failed load, nothing to undo */
3006 				splx(s);
3007 				break;
3008 			}
3009 			if (type->mod_event != NULL) {	/* check with type */
3010 				error = (*type->mod_event)(mod, event, data);
3011 				if (error != 0) {	/* type refuses.. */
3012 					splx(s);
3013 					break;
3014 				}
3015 			}
3016 			mtx_lock(&ng_typelist_mtx);
3017 			LIST_REMOVE(type, types);
3018 			mtx_unlock(&ng_typelist_mtx);
3019 		}
3020 		splx(s);
3021 		break;
3022 
3023 	default:
3024 		if (type->mod_event != NULL)
3025 			error = (*type->mod_event)(mod, event, data);
3026 		else
3027 			error = EOPNOTSUPP;		/* XXX ? */
3028 		break;
3029 	}
3030 	return (error);
3031 }
3032 
3033 /*
3034  * Handle loading and unloading for this code.
3035  * The only thing we need to link into is the NETISR strucure.
3036  */
3037 static int
3038 ngb_mod_event(module_t mod, int event, void *data)
3039 {
3040 	int error = 0;
3041 
3042 	switch (event) {
3043 	case MOD_LOAD:
3044 		/* Initialize everything. */
3045 		mtx_init(&ng_worklist_mtx, "ng_worklist", NULL, MTX_SPIN);
3046 		mtx_init(&ng_typelist_mtx, "netgraph types mutex", NULL,
3047 		    MTX_DEF);
3048 		mtx_init(&ng_nodelist_mtx, "netgraph nodelist mutex", NULL,
3049 		    MTX_DEF);
3050 		mtx_init(&ng_idhash_mtx, "netgraph idhash mutex", NULL,
3051 		    MTX_DEF);
3052 		mtx_init(&ng_topo_mtx, "netgraph topology mutex", NULL,
3053 		    MTX_DEF);
3054 #ifdef	NETGRAPH_DEBUG
3055 		mtx_init(&ngq_mtx, "netgraph item list mutex", NULL,
3056 		    MTX_DEF);
3057 #endif
3058 		ng_qzone = uma_zcreate("NetGraph items", sizeof(struct ng_item),
3059 		    NULL, NULL, NULL, NULL, UMA_ALIGN_CACHE, 0);
3060 		uma_zone_set_max(ng_qzone, maxalloc);
3061 		netisr_register(NETISR_NETGRAPH, (netisr_t *)ngintr, NULL,
3062 		    NETISR_MPSAFE);
3063 		break;
3064 	case MOD_UNLOAD:
3065 		/* You cant unload it because an interface may be using it.  */
3066 		error = EBUSY;
3067 		break;
3068 	default:
3069 		error = EOPNOTSUPP;
3070 		break;
3071 	}
3072 	return (error);
3073 }
3074 
3075 static moduledata_t netgraph_mod = {
3076 	"netgraph",
3077 	ngb_mod_event,
3078 	(NULL)
3079 };
3080 DECLARE_MODULE(netgraph, netgraph_mod, SI_SUB_NETGRAPH, SI_ORDER_MIDDLE);
3081 SYSCTL_NODE(_net, OID_AUTO, graph, CTLFLAG_RW, 0, "netgraph Family");
3082 SYSCTL_INT(_net_graph, OID_AUTO, abi_version, CTLFLAG_RD, 0, NG_ABI_VERSION,"");
3083 SYSCTL_INT(_net_graph, OID_AUTO, msg_version, CTLFLAG_RD, 0, NG_VERSION, "");
3084 
3085 #ifdef	NETGRAPH_DEBUG
3086 void
3087 dumphook (hook_p hook, char *file, int line)
3088 {
3089 	printf("hook: name %s, %d refs, Last touched:\n",
3090 		_NG_HOOK_NAME(hook), hook->hk_refs);
3091 	printf("	Last active @ %s, line %d\n",
3092 		hook->lastfile, hook->lastline);
3093 	if (line) {
3094 		printf(" problem discovered at file %s, line %d\n", file, line);
3095 	}
3096 }
3097 
3098 void
3099 dumpnode(node_p node, char *file, int line)
3100 {
3101 	printf("node: ID [%x]: type '%s', %d hooks, flags 0x%x, %d refs, %s:\n",
3102 		_NG_NODE_ID(node), node->nd_type->name,
3103 		node->nd_numhooks, node->nd_flags,
3104 		node->nd_refs, node->nd_name);
3105 	printf("	Last active @ %s, line %d\n",
3106 		node->lastfile, node->lastline);
3107 	if (line) {
3108 		printf(" problem discovered at file %s, line %d\n", file, line);
3109 	}
3110 }
3111 
3112 void
3113 dumpitem(item_p item, char *file, int line)
3114 {
3115 	printf(" ACTIVE item, last used at %s, line %d",
3116 		item->lastfile, item->lastline);
3117 	switch(item->el_flags & NGQF_TYPE) {
3118 	case NGQF_DATA:
3119 		printf(" - [data]\n");
3120 		break;
3121 	case NGQF_MESG:
3122 		printf(" - retaddr[%d]:\n", _NGI_RETADDR(item));
3123 		break;
3124 	case NGQF_FN:
3125 		printf(" - fn@%p (%p, %p, %p, %d (%x))\n",
3126 			item->body.fn.fn_fn,
3127 			_NGI_NODE(item),
3128 			_NGI_HOOK(item),
3129 			item->body.fn.fn_arg1,
3130 			item->body.fn.fn_arg2,
3131 			item->body.fn.fn_arg2);
3132 		break;
3133 	case NGQF_UNDEF:
3134 		printf(" - UNDEFINED!\n");
3135 	}
3136 	if (line) {
3137 		printf(" problem discovered at file %s, line %d\n", file, line);
3138 		if (_NGI_NODE(item)) {
3139 			printf("node %p ([%x])\n",
3140 				_NGI_NODE(item), ng_node2ID(_NGI_NODE(item)));
3141 		}
3142 	}
3143 }
3144 
3145 static void
3146 ng_dumpitems(void)
3147 {
3148 	item_p item;
3149 	int i = 1;
3150 	TAILQ_FOREACH(item, &ng_itemlist, all) {
3151 		printf("[%d] ", i++);
3152 		dumpitem(item, NULL, 0);
3153 	}
3154 }
3155 
3156 static void
3157 ng_dumpnodes(void)
3158 {
3159 	node_p node;
3160 	int i = 1;
3161 	mtx_lock(&ng_nodelist_mtx);
3162 	SLIST_FOREACH(node, &ng_allnodes, nd_all) {
3163 		printf("[%d] ", i++);
3164 		dumpnode(node, NULL, 0);
3165 	}
3166 	mtx_unlock(&ng_nodelist_mtx);
3167 }
3168 
3169 static void
3170 ng_dumphooks(void)
3171 {
3172 	hook_p hook;
3173 	int i = 1;
3174 	mtx_lock(&ng_nodelist_mtx);
3175 	SLIST_FOREACH(hook, &ng_allhooks, hk_all) {
3176 		printf("[%d] ", i++);
3177 		dumphook(hook, NULL, 0);
3178 	}
3179 	mtx_unlock(&ng_nodelist_mtx);
3180 }
3181 
3182 static int
3183 sysctl_debug_ng_dump_items(SYSCTL_HANDLER_ARGS)
3184 {
3185 	int error;
3186 	int val;
3187 	int i;
3188 
3189 	val = allocated;
3190 	i = 1;
3191 	error = sysctl_handle_int(oidp, &val, sizeof(int), req);
3192 	if (error != 0 || req->newptr == NULL)
3193 		return (error);
3194 	if (val == 42) {
3195 		ng_dumpitems();
3196 		ng_dumpnodes();
3197 		ng_dumphooks();
3198 	}
3199 	return (0);
3200 }
3201 
3202 SYSCTL_PROC(_debug, OID_AUTO, ng_dump_items, CTLTYPE_INT | CTLFLAG_RW,
3203     0, sizeof(int), sysctl_debug_ng_dump_items, "I", "Number of allocated items");
3204 #endif	/* NETGRAPH_DEBUG */
3205 
3206 
3207 /***********************************************************************
3208 * Worklist routines
3209 **********************************************************************/
3210 /* NETISR thread enters here */
3211 /*
3212  * Pick a node off the list of nodes with work,
3213  * try get an item to process off it.
3214  * If there are no more, remove the node from the list.
3215  */
3216 static void
3217 ngintr(void)
3218 {
3219 	item_p item;
3220 	node_p  node = NULL;
3221 
3222 	for (;;) {
3223 		mtx_lock_spin(&ng_worklist_mtx);
3224 		node = TAILQ_FIRST(&ng_worklist);
3225 		if (!node) {
3226 			mtx_unlock_spin(&ng_worklist_mtx);
3227 			break;
3228 		}
3229 		node->nd_flags &= ~NGF_WORKQ;
3230 		TAILQ_REMOVE(&ng_worklist, node, nd_work);
3231 		mtx_unlock_spin(&ng_worklist_mtx);
3232 		/*
3233 		 * We have the node. We also take over the reference
3234 		 * that the list had on it.
3235 		 * Now process as much as you can, until it won't
3236 		 * let you have another item off the queue.
3237 		 * All this time, keep the reference
3238 		 * that lets us be sure that the node still exists.
3239 		 * Let the reference go at the last minute.
3240 		 * ng_dequeue will put us back on the worklist
3241 		 * if there is more too do. This may be of use if there
3242 		 * are Multiple Processors and multiple Net threads in the
3243 		 * future.
3244 		 */
3245 		for (;;) {
3246 			int rw;
3247 
3248 			mtx_lock_spin(&node->nd_input_queue.q_mtx);
3249 			item = ng_dequeue(&node->nd_input_queue, &rw);
3250 			if (item == NULL) {
3251 				mtx_unlock_spin(&node->nd_input_queue.q_mtx);
3252 				break; /* go look for another node */
3253 			} else {
3254 				mtx_unlock_spin(&node->nd_input_queue.q_mtx);
3255 				NGI_GET_NODE(item, node); /* zaps stored node */
3256 				ng_apply_item(node, item, rw);
3257 				NG_NODE_UNREF(node);
3258 			}
3259 		}
3260 		NG_NODE_UNREF(node);
3261 	}
3262 }
3263 
3264 static void
3265 ng_worklist_remove(node_p node)
3266 {
3267 	mtx_lock_spin(&ng_worklist_mtx);
3268 	if (node->nd_flags & NGF_WORKQ) {
3269 		node->nd_flags &= ~NGF_WORKQ;
3270 		TAILQ_REMOVE(&ng_worklist, node, nd_work);
3271 		mtx_unlock_spin(&ng_worklist_mtx);
3272 		NG_NODE_UNREF(node);
3273 	} else {
3274 		mtx_unlock_spin(&ng_worklist_mtx);
3275 	}
3276 }
3277 
3278 /*
3279  * XXX
3280  * It's posible that a debugging NG_NODE_REF may need
3281  * to be outside the mutex zone
3282  */
3283 static void
3284 ng_setisr(node_p node)
3285 {
3286 
3287 	mtx_assert(&node->nd_input_queue.q_mtx, MA_OWNED);
3288 
3289 	if ((node->nd_flags & NGF_WORKQ) == 0) {
3290 		/*
3291 		 * If we are not already on the work queue,
3292 		 * then put us on.
3293 		 */
3294 		node->nd_flags |= NGF_WORKQ;
3295 		mtx_lock_spin(&ng_worklist_mtx);
3296 		TAILQ_INSERT_TAIL(&ng_worklist, node, nd_work);
3297 		mtx_unlock_spin(&ng_worklist_mtx);
3298 		NG_NODE_REF(node); /* XXX fafe in mutex? */
3299 	}
3300 	schednetisr(NETISR_NETGRAPH);
3301 }
3302 
3303 
3304 /***********************************************************************
3305 * Externally useable functions to set up a queue item ready for sending
3306 ***********************************************************************/
3307 
3308 #ifdef	NETGRAPH_DEBUG
3309 #define	ITEM_DEBUG_CHECKS						\
3310 	do {								\
3311 		if (NGI_NODE(item) ) {					\
3312 			printf("item already has node");		\
3313 			kdb_enter("has node");				\
3314 			NGI_CLR_NODE(item);				\
3315 		}							\
3316 		if (NGI_HOOK(item) ) {					\
3317 			printf("item already has hook");		\
3318 			kdb_enter("has hook");				\
3319 			NGI_CLR_HOOK(item);				\
3320 		}							\
3321 	} while (0)
3322 #else
3323 #define ITEM_DEBUG_CHECKS
3324 #endif
3325 
3326 /*
3327  * Put mbuf into the item.
3328  * Hook and node references will be removed when the item is dequeued.
3329  * (or equivalent)
3330  * (XXX) Unsafe because no reference held by peer on remote node.
3331  * remote node might go away in this timescale.
3332  * We know the hooks can't go away because that would require getting
3333  * a writer item on both nodes and we must have at least a  reader
3334  * here to be able to do this.
3335  * Note that the hook loaded is the REMOTE hook.
3336  *
3337  * This is possibly in the critical path for new data.
3338  */
3339 item_p
3340 ng_package_data(struct mbuf *m, int flags)
3341 {
3342 	item_p item;
3343 
3344 	if ((item = ng_getqblk(flags)) == NULL) {
3345 		NG_FREE_M(m);
3346 		return (NULL);
3347 	}
3348 	ITEM_DEBUG_CHECKS;
3349 	item->el_flags = NGQF_DATA | NGQF_READER;
3350 	item->el_next = NULL;
3351 	NGI_M(item) = m;
3352 	return (item);
3353 }
3354 
3355 /*
3356  * Allocate a queue item and put items into it..
3357  * Evaluate the address as this will be needed to queue it and
3358  * to work out what some of the fields should be.
3359  * Hook and node references will be removed when the item is dequeued.
3360  * (or equivalent)
3361  */
3362 item_p
3363 ng_package_msg(struct ng_mesg *msg, int flags)
3364 {
3365 	item_p item;
3366 
3367 	if ((item = ng_getqblk(flags)) == NULL) {
3368 		NG_FREE_MSG(msg);
3369 		return (NULL);
3370 	}
3371 	ITEM_DEBUG_CHECKS;
3372 	/* Messages items count as writers unless explicitly exempted. */
3373 	if (msg->header.cmd & NGM_READONLY)
3374 		item->el_flags = NGQF_MESG | NGQF_READER;
3375 	else
3376 		item->el_flags = NGQF_MESG | NGQF_WRITER;
3377 	item->el_next = NULL;
3378 	/*
3379 	 * Set the current lasthook into the queue item
3380 	 */
3381 	NGI_MSG(item) = msg;
3382 	NGI_RETADDR(item) = 0;
3383 	return (item);
3384 }
3385 
3386 
3387 
3388 #define SET_RETADDR(item, here, retaddr)				\
3389 	do {	/* Data or fn items don't have retaddrs */		\
3390 		if ((item->el_flags & NGQF_TYPE) == NGQF_MESG) {	\
3391 			if (retaddr) {					\
3392 				NGI_RETADDR(item) = retaddr;		\
3393 			} else {					\
3394 				/*					\
3395 				 * The old return address should be ok.	\
3396 				 * If there isn't one, use the address	\
3397 				 * here.				\
3398 				 */					\
3399 				if (NGI_RETADDR(item) == 0) {		\
3400 					NGI_RETADDR(item)		\
3401 						= ng_node2ID(here);	\
3402 				}					\
3403 			}						\
3404 		}							\
3405 	} while (0)
3406 
3407 int
3408 ng_address_hook(node_p here, item_p item, hook_p hook, ng_ID_t retaddr)
3409 {
3410 	hook_p peer;
3411 	node_p peernode;
3412 	ITEM_DEBUG_CHECKS;
3413 	/*
3414 	 * Quick sanity check..
3415 	 * Since a hook holds a reference on it's node, once we know
3416 	 * that the peer is still connected (even if invalid,) we know
3417 	 * that the peer node is present, though maybe invalid.
3418 	 */
3419 	if ((hook == NULL)
3420 	|| NG_HOOK_NOT_VALID(hook)
3421 	|| (NG_HOOK_PEER(hook) == NULL)
3422 	|| NG_HOOK_NOT_VALID(NG_HOOK_PEER(hook))
3423 	|| NG_NODE_NOT_VALID(NG_PEER_NODE(hook))) {
3424 		NG_FREE_ITEM(item);
3425 		TRAP_ERROR();
3426 		return (ENETDOWN);
3427 	}
3428 
3429 	/*
3430 	 * Transfer our interest to the other (peer) end.
3431 	 */
3432 	peer = NG_HOOK_PEER(hook);
3433 	NG_HOOK_REF(peer);
3434 	NGI_SET_HOOK(item, peer);
3435 	peernode = NG_PEER_NODE(hook);
3436 	NG_NODE_REF(peernode);
3437 	NGI_SET_NODE(item, peernode);
3438 	SET_RETADDR(item, here, retaddr);
3439 	return (0);
3440 }
3441 
3442 int
3443 ng_address_path(node_p here, item_p item, char *address, ng_ID_t retaddr)
3444 {
3445 	node_p	dest = NULL;
3446 	hook_p	hook = NULL;
3447 	int	error;
3448 
3449 	ITEM_DEBUG_CHECKS;
3450 	/*
3451 	 * Note that ng_path2noderef increments the reference count
3452 	 * on the node for us if it finds one. So we don't have to.
3453 	 */
3454 	error = ng_path2noderef(here, address, &dest, &hook);
3455 	if (error) {
3456 		NG_FREE_ITEM(item);
3457 		return (error);
3458 	}
3459 	NGI_SET_NODE(item, dest);
3460 	if ( hook) {
3461 		NG_HOOK_REF(hook);	/* don't let it go while on the queue */
3462 		NGI_SET_HOOK(item, hook);
3463 	}
3464 	SET_RETADDR(item, here, retaddr);
3465 	return (0);
3466 }
3467 
3468 int
3469 ng_address_ID(node_p here, item_p item, ng_ID_t ID, ng_ID_t retaddr)
3470 {
3471 	node_p dest;
3472 
3473 	ITEM_DEBUG_CHECKS;
3474 	/*
3475 	 * Find the target node.
3476 	 */
3477 	dest = ng_ID2noderef(ID); /* GETS REFERENCE! */
3478 	if (dest == NULL) {
3479 		NG_FREE_ITEM(item);
3480 		TRAP_ERROR();
3481 		return(EINVAL);
3482 	}
3483 	/* Fill out the contents */
3484 	NGI_SET_NODE(item, dest);
3485 	NGI_CLR_HOOK(item);
3486 	SET_RETADDR(item, here, retaddr);
3487 	return (0);
3488 }
3489 
3490 /*
3491  * special case to send a message to self (e.g. destroy node)
3492  * Possibly indicate an arrival hook too.
3493  * Useful for removing that hook :-)
3494  */
3495 item_p
3496 ng_package_msg_self(node_p here, hook_p hook, struct ng_mesg *msg)
3497 {
3498 	item_p item;
3499 
3500 	/*
3501 	 * Find the target node.
3502 	 * If there is a HOOK argument, then use that in preference
3503 	 * to the address.
3504 	 */
3505 	if ((item = ng_getqblk(NG_NOFLAGS)) == NULL) {
3506 		NG_FREE_MSG(msg);
3507 		return (NULL);
3508 	}
3509 
3510 	/* Fill out the contents */
3511 	item->el_flags = NGQF_MESG | NGQF_WRITER;
3512 	item->el_next = NULL;
3513 	NG_NODE_REF(here);
3514 	NGI_SET_NODE(item, here);
3515 	if (hook) {
3516 		NG_HOOK_REF(hook);
3517 		NGI_SET_HOOK(item, hook);
3518 	}
3519 	NGI_MSG(item) = msg;
3520 	NGI_RETADDR(item) = ng_node2ID(here);
3521 	return (item);
3522 }
3523 
3524 int
3525 ng_send_fn1(node_p node, hook_p hook, ng_item_fn *fn, void * arg1, int arg2,
3526 	int flags)
3527 {
3528 	item_p item;
3529 
3530 	if ((item = ng_getqblk(flags)) == NULL) {
3531 		return (ENOMEM);
3532 	}
3533 	item->el_flags = NGQF_FN | NGQF_WRITER;
3534 	NG_NODE_REF(node); /* and one for the item */
3535 	NGI_SET_NODE(item, node);
3536 	if (hook) {
3537 		NG_HOOK_REF(hook);
3538 		NGI_SET_HOOK(item, hook);
3539 	}
3540 	NGI_FN(item) = fn;
3541 	NGI_ARG1(item) = arg1;
3542 	NGI_ARG2(item) = arg2;
3543 	return(ng_snd_item(item, flags));
3544 }
3545 
3546 /*
3547  * Official timeout routines for Netgraph nodes.
3548  */
3549 static void
3550 ng_callout_trampoline(void *arg)
3551 {
3552 	item_p item = arg;
3553 
3554 	ng_snd_item(item, 0);
3555 }
3556 
3557 
3558 int
3559 ng_callout(struct callout *c, node_p node, hook_p hook, int ticks,
3560     ng_item_fn *fn, void * arg1, int arg2)
3561 {
3562 	item_p item, oitem;
3563 
3564 	if ((item = ng_getqblk(NG_NOFLAGS)) == NULL)
3565 		return (ENOMEM);
3566 
3567 	item->el_flags = NGQF_FN | NGQF_WRITER;
3568 	NG_NODE_REF(node);		/* and one for the item */
3569 	NGI_SET_NODE(item, node);
3570 	if (hook) {
3571 		NG_HOOK_REF(hook);
3572 		NGI_SET_HOOK(item, hook);
3573 	}
3574 	NGI_FN(item) = fn;
3575 	NGI_ARG1(item) = arg1;
3576 	NGI_ARG2(item) = arg2;
3577 	oitem = c->c_arg;
3578 	if (callout_reset(c, ticks, &ng_callout_trampoline, item) == 1 &&
3579 	    oitem != NULL)
3580 		NG_FREE_ITEM(oitem);
3581 	return (0);
3582 }
3583 
3584 /* A special modified version of untimeout() */
3585 int
3586 ng_uncallout(struct callout *c, node_p node)
3587 {
3588 	item_p item;
3589 	int rval;
3590 
3591 	KASSERT(c != NULL, ("ng_uncallout: NULL callout"));
3592 	KASSERT(node != NULL, ("ng_uncallout: NULL node"));
3593 
3594 	rval = callout_stop(c);
3595 	item = c->c_arg;
3596 	/* Do an extra check */
3597 	if ((rval > 0) && (c->c_func == &ng_callout_trampoline) &&
3598 	    (NGI_NODE(item) == node)) {
3599 		/*
3600 		 * We successfully removed it from the queue before it ran
3601 		 * So now we need to unreference everything that was
3602 		 * given extra references. (NG_FREE_ITEM does this).
3603 		 */
3604 		NG_FREE_ITEM(item);
3605 	}
3606 	c->c_arg = NULL;
3607 
3608 	return (rval);
3609 }
3610 
3611 /*
3612  * Set the address, if none given, give the node here.
3613  */
3614 void
3615 ng_replace_retaddr(node_p here, item_p item, ng_ID_t retaddr)
3616 {
3617 	if (retaddr) {
3618 		NGI_RETADDR(item) = retaddr;
3619 	} else {
3620 		/*
3621 		 * The old return address should be ok.
3622 		 * If there isn't one, use the address here.
3623 		 */
3624 		NGI_RETADDR(item) = ng_node2ID(here);
3625 	}
3626 }
3627 
3628 #define TESTING
3629 #ifdef TESTING
3630 /* just test all the macros */
3631 void
3632 ng_macro_test(item_p item);
3633 void
3634 ng_macro_test(item_p item)
3635 {
3636 	node_p node = NULL;
3637 	hook_p hook = NULL;
3638 	struct mbuf *m;
3639 	struct ng_mesg *msg;
3640 	ng_ID_t retaddr;
3641 	int	error;
3642 
3643 	NGI_GET_M(item, m);
3644 	NGI_GET_MSG(item, msg);
3645 	retaddr = NGI_RETADDR(item);
3646 	NG_SEND_DATA(error, hook, m, NULL);
3647 	NG_SEND_DATA_ONLY(error, hook, m);
3648 	NG_FWD_NEW_DATA(error, item, hook, m);
3649 	NG_FWD_ITEM_HOOK(error, item, hook);
3650 	NG_SEND_MSG_HOOK(error, node, msg, hook, retaddr);
3651 	NG_SEND_MSG_ID(error, node, msg, retaddr, retaddr);
3652 	NG_SEND_MSG_PATH(error, node, msg, ".:", retaddr);
3653 	NG_FWD_MSG_HOOK(error, node, item, hook, retaddr);
3654 }
3655 #endif /* TESTING */
3656 
3657