Skip site navigation (1)Skip section navigation (2)
Date:      Mon, 29 Dec 2008 23:31:05 +0000 (UTC)
From:      Kip Macy <kmacy@FreeBSD.org>
To:        src-committers@freebsd.org, svn-src-user@freebsd.org
Subject:   svn commit: r186576 - user/kmacy/HEAD_fast_net/sys/netinet
Message-ID:  <200812292331.mBTNV5gu069760@svn.freebsd.org>

next in thread | raw e-mail | index | archive | help
Author: kmacy
Date: Mon Dec 29 23:31:04 2008
New Revision: 186576
URL: http://svn.freebsd.org/changeset/base/186576

Log:
  install cached llentry in the inpcb and then pass down to ether_output in the struct route

Modified:
  user/kmacy/HEAD_fast_net/sys/netinet/if_ether.c
  user/kmacy/HEAD_fast_net/sys/netinet/in_pcb.c
  user/kmacy/HEAD_fast_net/sys/netinet/in_pcb.h
  user/kmacy/HEAD_fast_net/sys/netinet/ip_output.c

Modified: user/kmacy/HEAD_fast_net/sys/netinet/if_ether.c
==============================================================================
--- user/kmacy/HEAD_fast_net/sys/netinet/if_ether.c	Mon Dec 29 22:36:23 2008	(r186575)
+++ user/kmacy/HEAD_fast_net/sys/netinet/if_ether.c	Mon Dec 29 23:31:04 2008	(r186576)
@@ -80,7 +80,6 @@ __FBSDID("$FreeBSD$");
 
 #define SIN(s) ((struct sockaddr_in *)s)
 #define SDL(s) ((struct sockaddr_dl *)s)
-#define LLTABLE(ifp)	((struct lltable *)(ifp)->if_afdata[AF_INET])
 
 SYSCTL_DECL(_net_link_ether);
 SYSCTL_NODE(_net_link_ether, PF_INET, inet, CTLFLAG_RW, 0, "");

Modified: user/kmacy/HEAD_fast_net/sys/netinet/in_pcb.c
==============================================================================
--- user/kmacy/HEAD_fast_net/sys/netinet/in_pcb.c	Mon Dec 29 22:36:23 2008	(r186575)
+++ user/kmacy/HEAD_fast_net/sys/netinet/in_pcb.c	Mon Dec 29 23:31:04 2008	(r186576)
@@ -62,6 +62,7 @@ __FBSDID("$FreeBSD$");
 
 #include <net/if.h>
 #include <net/if_types.h>
+#include <net/if_llatbl.h>
 #include <net/route.h>
 
 #include <netinet/in.h>
@@ -494,6 +495,11 @@ void
 in_pcbrtalloc(struct inpcb *inp, in_addr_t faddr, struct route *sro)
 {
 	struct sockaddr_in *sin;
+	struct sockaddr *dst;
+	struct llentry *la;
+	struct rtentry *rt;
+	struct ifnet *ifp;
+	int flags = LLE_EXCLUSIVE;
 
 	INP_WLOCK_ASSERT(inp);
 	bzero(sro, sizeof(*sro));
@@ -516,11 +522,39 @@ in_pcbrtalloc(struct inpcb *inp, in_addr
 #endif		
 	}
 
-	if (sro->ro_rt != NULL) {
-		inp->inp_rt = sro->ro_rt;
-		inp->inp_vflag |= INP_RT_VALID;
+	rt = sro->ro_rt;
+	if (rt == NULL)
+		return;
+	
+	inp->inp_rt = rt;
+	inp->inp_vflag |= INP_RT_VALID;
 
+	if (rt->rt_ifp == NULL)
+		return;
+	
+	ifp = rt->rt_ifp;
+	dst = &sro->ro_dst;
+	if (rt->rt_flags & RTF_GATEWAY)
+		dst = rt->rt_gateway;
+		
+	IF_AFDATA_RLOCK(ifp);	
+	la = lla_lookup(LLTABLE(ifp), flags, dst);
+	IF_AFDATA_RUNLOCK(ifp);
+	if ((la == NULL) && 
+	    (ifp->if_flags & (IFF_NOARP | IFF_STATICARP)) == 0) {
+		flags |= (LLE_CREATE | LLE_EXCLUSIVE);
+		IF_AFDATA_WLOCK(ifp);
+		la = lla_lookup(LLTABLE(ifp), flags, dst);
+		IF_AFDATA_WUNLOCK(ifp);	
 	}
+	if (la == NULL)
+		return;
+	
+	LLE_ADDREF(la);
+	LLE_WUNLOCK(la);
+
+	inp->inp_lle = la;
+	inp->inp_flags |= INP_LLE_VALID;
 }
 
 /*
@@ -905,9 +939,14 @@ in_pcbdisconnect(struct inpcb *inp)
 	INP_WLOCK_ASSERT(inp);
 
 	if (inp->inp_vflag & INP_RT_VALID) {
+		inp->inp_vflag &= ~INP_RT_VALID;
 		RTFREE(inp->inp_rt);
 		inp->inp_rt = NULL;
-		inp->inp_vflag &= ~INP_RT_VALID;
+	}
+	if (inp->inp_flags & INP_LLE_VALID) {
+		inp->inp_flags &= ~INP_LLE_VALID;
+		LLE_FREE(inp->inp_lle);
+		inp->inp_lle = NULL;
 	}
 
 	inp->inp_faddr.s_addr = INADDR_ANY;
@@ -948,11 +987,16 @@ in_pcbfree_internal(struct inpcb *inp)
 	INP_WLOCK_ASSERT(inp);
 
 	if (inp->inp_vflag & INP_RT_VALID) {		
+		inp->inp_vflag &= ~INP_RT_VALID;
 		RTFREE(inp->inp_rt);
 		inp->inp_rt = NULL;
-		inp->inp_vflag &= ~INP_RT_VALID;
 	}
-	
+	if (inp->inp_flags & INP_LLE_VALID) {
+		inp->inp_flags &= ~INP_LLE_VALID;
+		LLE_FREE(inp->inp_lle);
+		inp->inp_lle = NULL;
+	}
+
 #ifdef IPSEC
 	if (inp->inp_sp != NULL)
 		ipsec_delete_pcbpolicy(inp);

Modified: user/kmacy/HEAD_fast_net/sys/netinet/in_pcb.h
==============================================================================
--- user/kmacy/HEAD_fast_net/sys/netinet/in_pcb.h	Mon Dec 29 22:36:23 2008	(r186575)
+++ user/kmacy/HEAD_fast_net/sys/netinet/in_pcb.h	Mon Dec 29 23:31:04 2008	(r186576)
@@ -418,6 +418,7 @@ void 	inp_4tuple_get(struct inpcb *inp, 
 #define	INP_FAITH		0x200	/* accept FAITH'ed connections */
 #define	INP_RECVTTL		0x400	/* receive incoming IP TTL */
 #define	INP_DONTFRAG		0x800	/* don't fragment packet */
+#define	INP_LLE_VALID		0x1000	/* L2 entry is set */
 
 #define IN6P_IPV6_V6ONLY	0x008000 /* restrict AF_INET6 socket for v6 */
 

Modified: user/kmacy/HEAD_fast_net/sys/netinet/ip_output.c
==============================================================================
--- user/kmacy/HEAD_fast_net/sys/netinet/ip_output.c	Mon Dec 29 22:36:23 2008	(r186575)
+++ user/kmacy/HEAD_fast_net/sys/netinet/ip_output.c	Mon Dec 29 23:31:04 2008	(r186576)
@@ -53,6 +53,7 @@ __FBSDID("$FreeBSD$");
 #include <sys/vimage.h>
 
 #include <net/if.h>
+#include <net/if_llatbl.h>
 #include <net/netisr.h>
 #include <net/pfil.h>
 #include <net/route.h>
@@ -121,7 +122,7 @@ ip_output(struct mbuf *m, struct mbuf *o
 	int hlen = sizeof (struct ip);
 	int mtu;
 	int len, error = 0;
-	int neednewroute = 0;
+	int neednewroute = 0, neednewlle = 0;
 	struct sockaddr_in *dst = NULL;	/* keep compiler happy */
 	struct in_ifaddr *ia = NULL;
 	int isbroadcast, sw_csum;
@@ -150,7 +151,13 @@ ip_output(struct mbuf *m, struct mbuf *o
 				ro->ro_rt = inp->inp_rt;
 			} else
 				neednewroute = 1;
-		} 
+		}
+		if ((ro == &iproute) && (inp->inp_flags & INP_LLE_VALID)) {
+			if (inp->inp_lle->la_flags & LLE_VALID) {
+				ro->ro_lle = inp->inp_lle;
+			} else
+				neednewlle = 1;
+		}
 	}
 
 	if (opt) {
@@ -625,19 +632,44 @@ passout:
 done:
 	if (ro == &iproute && ro->ro_rt != NULL) {
 		int wlocked;		
-
+		struct llentry *la;
+		
+		wlocked = INP_WLOCKED(inp);
 		if (inp == NULL || (inp->inp_vflag & INP_RT_VALID) == 0)
-			RTFREE(ro->ro_rt);		
+			RTFREE(ro->ro_rt);
 		else if (neednewroute && ro->ro_rt != inp->inp_rt) {
-			wlocked = INP_WLOCKED(inp);
 			if (!wlocked && INP_TRY_UPGRADE(inp) == 0)
 				return (error);
 			RTFREE(inp->inp_rt);
 			inp->inp_rt = ro->ro_rt;
-			if (!wlocked)
-				INP_DOWNGRADE(inp);
+
+		}
+		if (neednewlle) {
+			if (!wlocked && INP_TRY_UPGRADE(inp) == 0)
+				return (error);
+			IF_AFDATA_RLOCK(ifp);	
+			la = lla_lookup(LLTABLE(ifp), LLE_EXCLUSIVE,
+			    (struct sockaddr *)dst);
+			IF_AFDATA_RUNLOCK(ifp);
+			if ((la == NULL) && 
+			    (ifp->if_flags & (IFF_NOARP | IFF_STATICARP)) == 0) {
+				IF_AFDATA_WLOCK(ifp);
+				la = lla_lookup(LLTABLE(ifp),
+				    (LLE_CREATE | LLE_EXCLUSIVE),
+				    (struct sockaddr *)dst);
+				IF_AFDATA_WUNLOCK(ifp);	
+			}
+			if (la != NULL) {
+				LLE_FREE(inp->inp_lle);
+				LLE_ADDREF(la);
+				LLE_WUNLOCK(la);
+				inp->inp_lle = la;
+			}
 		}
+		if (!wlocked)
+			INP_DOWNGRADE(inp);
 	}
+	
 	return (error);
 bad:
 	m_freem(m);



Want to link to this message? Use this URL: <https://mail-archive.FreeBSD.org/cgi/mid.cgi?200812292331.mBTNV5gu069760>