Date: Thu, 11 Jun 2009 18:56:07 +0200 From: Marko Zec <zec@freebsd.org> To: src-committers@freebsd.org Cc: svn-src-head@freebsd.org, svn-src-all@freebsd.org Subject: Re: svn commit: r194012 - in head: . sys/netgraph sys/sys Message-ID: <200906111856.07618.zec@freebsd.org> In-Reply-To: <200906111650.n5BGonnn053446@svn.freebsd.org> References: <200906111650.n5BGonnn053446@svn.freebsd.org>
next in thread | previous in thread | raw e-mail | index | archive | help
On Thursday 11 June 2009 18:50:49 Marko Zec wrote: > Author: zec > Date: Thu Jun 11 16:50:49 2009 > New Revision: 194012 > URL: http://svn.freebsd.org/changeset/base/194012 > > Log: > Introduce a mechanism for detecting calls from outbound path of the > network stack when reentering the inbound path from netgraph, and > force queueing of mbufs at the outbound netgraph node. ^^^^^^^^ s/outbound/inbound/ -> the framework prevents direct call dispatching at the egress netgraph node, i.e. the one reentering the network stack from below, not on the output path. Sorry for the typo... Marko > > The mechanism relies on two components. First, in netgraph nodes > where outbound path of the network stack calls into netgraph, the > current thread has to be appropriately marked using the new > NG_OUTBOUND_THREAD_REF() macro before proceeding to call further > into the netgraph topology, and unmarked using the > NG_OUTBOUND_THREAD_UNREF() macro before returning to the caller. > Second, netgraph nodes which can potentially reenter the network > stack in the inbound path have to mark their inbound hooks using > NG_HOOK_SET_TO_INBOUND() macro. The netgraph framework will then > detect when there is a danger of a call graph looping back from > outbound to inbound path via netgraph, and defer handing off the > mbufs to the "inbound" node to a worker thread with a clean stack. > > In this first pass only the most obvious netgraph nodes have been > updated to ensure no outbound to inbound calls can occur. Nodes > such as ng_ipfw, ng_gif etc. should be further examined whether a > potential for outbound to inbound call looping exists. > > This commit changes the layout of struct thread, but due to > __FreeBSD_version number shortage a version bump has been omitted > at this time, nevertheless kernel and modules have to be rebuilt. > > Reviewed by: julian, rwatson, bz > Approved by: julian (mentor) > > Modified: > head/UPDATING > head/sys/netgraph/netgraph.h > head/sys/netgraph/ng_base.c > head/sys/netgraph/ng_eiface.c > head/sys/netgraph/ng_ether.c > head/sys/netgraph/ng_iface.c > head/sys/netgraph/ng_ip_input.c > head/sys/sys/proc.h > > Modified: head/UPDATING > =========================================================================== >=== --- head/UPDATING Thu Jun 11 16:48:59 2009 (r194011) > +++ head/UPDATING Thu Jun 11 16:50:49 2009 (r194012) > @@ -22,6 +22,10 @@ NOTE TO PEOPLE WHO THINK THAT FreeBSD 8. > to maximize performance. (To disable malloc debugging, run > ln -s aj /etc/malloc.conf.) > > +20090611: > + The layout of struct thread has changed. Kernel and modules > + need to be rebuilt. > + > 20090608: > The layout of structs ifnet, domain, protosw and vnet_net has > changed. Kernel modules need to be rebuilt. > > Modified: head/sys/netgraph/netgraph.h > =========================================================================== >=== --- head/sys/netgraph/netgraph.h Thu Jun 11 16:48:59 2009 (r194011) +++ > head/sys/netgraph/netgraph.h Thu Jun 11 16:50:49 2009 (r194012) @@ -130,6 > +130,7 @@ struct ng_hook { > #define HK_FORCE_WRITER 0x0004 /* Incoming data queued as a writer */ > #define HK_DEAD 0x0008 /* This is the dead hook.. don't free */ > #define HK_HI_STACK 0x0010 /* Hook has hi stack usage */ > +#define HK_TO_INBOUND 0x0020 /* Hook on ntw. stack inbound path. */ > > /* > * Public Methods for hook > @@ -150,6 +151,8 @@ void ng_unref_hook(hook_p hook); /* don' > #define _NG_HOOK_FORCE_WRITER(hook) \ > do { hook->hk_flags |= HK_FORCE_WRITER; } while (0) > #define _NG_HOOK_FORCE_QUEUE(hook) do { hook->hk_flags |= HK_QUEUE; } > while (0) +#define _NG_HOOK_SET_TO_INBOUND(hook) \ > + do { hook->hk_flags |= HK_TO_INBOUND; } while (0) > #define _NG_HOOK_HI_STACK(hook) do { hook->hk_flags |= HK_HI_STACK; } > while (0) > > /* Some shortcuts */ > @@ -176,8 +179,11 @@ static __inline int _ng_hook_is_valid(ho > static __inline node_p _ng_hook_node(hook_p hook, char * file, int line); > static __inline hook_p _ng_hook_peer(hook_p hook, char * file, int line); > static __inline void _ng_hook_force_writer(hook_p hook, char * file, > - int line); > -static __inline void _ng_hook_force_queue(hook_p hook, char * file, int > line); + int line); > +static __inline void _ng_hook_force_queue(hook_p hook, char * file, > + int line); > +static __inline void _ng_hook_set_to_inbound(hook_p hook, char * file, > + int line); > > static __inline void > _chkhook(hook_p hook, char *file, int line) > @@ -282,6 +288,13 @@ _ng_hook_force_queue(hook_p hook, char * > } > > static __inline void > +_ng_hook_set_to_inbound(hook_p hook, char * file, int line) > +{ > + _chkhook(hook, file, line); > + _NG_HOOK_SET_TO_INBOUND(hook); > +} > + > +static __inline void > _ng_hook_hi_stack(hook_p hook, char * file, int line) > { > _chkhook(hook, file, line); > @@ -302,6 +315,7 @@ _ng_hook_hi_stack(hook_p hook, char * fi > #define NG_HOOK_PEER(hook) _ng_hook_peer(hook, _NN_) > #define NG_HOOK_FORCE_WRITER(hook) _ng_hook_force_writer(hook, _NN_) > #define NG_HOOK_FORCE_QUEUE(hook) _ng_hook_force_queue(hook, _NN_) > +#define NG_HOOK_SET_TO_INBOUND(hook) _ng_hook_set_to_inbound(hook, _NN_) > #define NG_HOOK_HI_STACK(hook) _ng_hook_hi_stack(hook, _NN_) > > #else /* NETGRAPH_DEBUG */ > /*----------------------------------------------*/ @@ -319,6 +333,7 @@ > _ng_hook_hi_stack(hook_p hook, char * fi > #define NG_HOOK_PEER(hook) _NG_HOOK_PEER(hook) > #define NG_HOOK_FORCE_WRITER(hook) _NG_HOOK_FORCE_WRITER(hook) > #define NG_HOOK_FORCE_QUEUE(hook) _NG_HOOK_FORCE_QUEUE(hook) > +#define NG_HOOK_SET_TO_INBOUND(hook) _NG_HOOK_SET_TO_INBOUND(hook) > #define NG_HOOK_HI_STACK(hook) _NG_HOOK_HI_STACK(hook) > > #endif /* NETGRAPH_DEBUG */ > /*----------------------------------------------*/ @@ -1189,6 +1204,20 @@ > typedef void *meta_p; > #define NG_ID_HASH_SIZE 128 /* most systems wont need even this many */ > #define NG_NAME_HASH_SIZE 128 /* most systems wont need even this many */ > > +/* > + * Mark the current thread when called from the outbound path of the > + * network stack, in order to enforce queuing on ng nodes calling into > + * the inbound network stack path. > + */ > +#define NG_OUTBOUND_THREAD_REF() \ > + curthread->td_ng_outbound++ > +#define NG_OUTBOUND_THREAD_UNREF() \ > + do { \ > + curthread->td_ng_outbound--; \ > + KASSERT(curthread->td_ng_outbound >= 0, \ > + ("%s: negative td_ng_outbound", __func__)); \ > + } while (0) > + > /* Virtualization macros */ > #define INIT_VNET_NETGRAPH(vnet) \ > INIT_FROM_VNET(vnet, VNET_MOD_NETGRAPH, \ > > Modified: head/sys/netgraph/ng_base.c > =========================================================================== >=== --- head/sys/netgraph/ng_base.c Thu Jun 11 16:48:59 2009 (r194011) +++ > head/sys/netgraph/ng_base.c Thu Jun 11 16:50:49 2009 (r194012) @@ -2213,11 > +2213,15 @@ ng_snd_item(item_p item, int flags) > } > > /* > - * If sender or receiver requests queued delivery or stack usage > + * If sender or receiver requests queued delivery, or call graph > + * loops back from outbound to inbound path, or stack usage > * level is dangerous - enqueue message. > */ > if ((flags & NG_QUEUE) || (hook && (hook->hk_flags & HK_QUEUE))) { > queue = 1; > + } else if (hook && (hook->hk_flags & HK_TO_INBOUND) && > + curthread->td_ng_outbound) { > + queue = 1; > } else { > queue = 0; > #ifdef GET_STACK_USAGE > > Modified: head/sys/netgraph/ng_eiface.c > =========================================================================== >=== --- head/sys/netgraph/ng_eiface.c Thu Jun 11 16:48:59 2009 (r194011) +++ > head/sys/netgraph/ng_eiface.c Thu Jun 11 16:50:49 2009 (r194012) @@ -261,7 > +261,9 @@ ng_eiface_start2(node_p node, hook_p hoo > * Send packet; if hook is not connected, mbuf will get > * freed. > */ > + NG_OUTBOUND_THREAD_REF(); > NG_SEND_DATA_ONLY(error, priv->ether, m); > + NG_OUTBOUND_THREAD_UNREF(); > > /* Update stats */ > if (error == 0) > @@ -414,6 +416,7 @@ ng_eiface_newhook(node_p node, hook_p ho > return (EISCONN); > priv->ether = hook; > NG_HOOK_SET_PRIVATE(hook, &priv->ether); > + NG_HOOK_SET_TO_INBOUND(hook); > > if_link_state_change(ifp, LINK_STATE_UP); > > > Modified: head/sys/netgraph/ng_ether.c > =========================================================================== >=== --- head/sys/netgraph/ng_ether.c Thu Jun 11 16:48:59 2009 (r194011) +++ > head/sys/netgraph/ng_ether.c Thu Jun 11 16:50:49 2009 (r194012) @@ -282,7 > +282,9 @@ ng_ether_output(struct ifnet *ifp, struc > return (0); > > /* Send it out "upper" hook */ > + NG_OUTBOUND_THREAD_REF(); > NG_SEND_DATA_ONLY(error, priv->upper, *mp); > + NG_OUTBOUND_THREAD_UNREF(); > return (error); > } > > @@ -416,6 +418,7 @@ ng_ether_newhook(node_p node, hook_p hoo > if (strcmp(name, NG_ETHER_HOOK_UPPER) == 0) { > hookptr = &priv->upper; > NG_HOOK_SET_RCVDATA(hook, ng_ether_rcv_upper); > + NG_HOOK_SET_TO_INBOUND(hook); > } else if (strcmp(name, NG_ETHER_HOOK_LOWER) == 0) { > hookptr = &priv->lower; > NG_HOOK_SET_RCVDATA(hook, ng_ether_rcv_lower); > > Modified: head/sys/netgraph/ng_iface.c > =========================================================================== >=== --- head/sys/netgraph/ng_iface.c Thu Jun 11 16:48:59 2009 (r194011) +++ > head/sys/netgraph/ng_iface.c Thu Jun 11 16:50:49 2009 (r194012) @@ -482,9 > +482,10 @@ ng_iface_send(struct ifnet *ifp, struct > /* Copy length before the mbuf gets invalidated. */ > len = m->m_pkthdr.len; > > - /* Send packet. If hook is not connected, > - mbuf will get freed. */ > + /* Send packet. If hook is not connected, mbuf will get freed. */ > + NG_OUTBOUND_THREAD_REF(); > NG_SEND_DATA_ONLY(error, *get_hook_from_iffam(priv, iffam), m); > + NG_OUTBOUND_THREAD_UNREF(); > > /* Update stats. */ > if (error == 0) { > @@ -610,6 +611,7 @@ ng_iface_newhook(node_p node, hook_p hoo > return (EISCONN); > *hookptr = hook; > NG_HOOK_HI_STACK(hook); > + NG_HOOK_SET_TO_INBOUND(hook); > return (0); > } > > > Modified: head/sys/netgraph/ng_ip_input.c > =========================================================================== >=== --- head/sys/netgraph/ng_ip_input.c Thu Jun 11 16:48:59 2009 (r194011) > +++ head/sys/netgraph/ng_ip_input.c Thu Jun 11 16:50:49 2009 (r194012) @@ > -77,6 +77,7 @@ > #include <sys/malloc.h> > #include <sys/mbuf.h> > #include <sys/socket.h> > +#include <sys/proc.h> > #include <net/if.h> > #include <net/if_types.h> > #include <net/if_var.h> > @@ -120,7 +121,10 @@ ngipi_rcvdata(hook_p hook, item_p item) > > NGI_GET_M(item, m); > NG_FREE_ITEM(item); > - netisr_dispatch(NETISR_IP, m); > + if (curthread->td_ng_outbound) > + netisr_queue(NETISR_IP, m); > + else > + netisr_dispatch(NETISR_IP, m); > return 0; > } > > > Modified: head/sys/sys/proc.h > =========================================================================== >=== --- head/sys/sys/proc.h Thu Jun 11 16:48:59 2009 (r194011) > +++ head/sys/sys/proc.h Thu Jun 11 16:50:49 2009 (r194012) > @@ -235,6 +235,7 @@ struct thread { > char td_name[MAXCOMLEN + 1]; /* (*) Thread name. */ > struct file *td_fpop; /* (k) file referencing cdev under op */ > int td_dbgflags; /* (c) Userland debugger flags */ > + int td_ng_outbound; /* (k) Thread entered ng from above. */ > struct osd td_osd; /* (k) Object specific data. */ > #define td_endzero td_base_pri
Want to link to this message? Use this URL: <https://mail-archive.FreeBSD.org/cgi/mid.cgi?200906111856.07618.zec>