summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorkmacy <kmacy@FreeBSD.org>2007-12-17 07:40:34 +0000
committerkmacy <kmacy@FreeBSD.org>2007-12-17 07:40:34 +0000
commit5d9e84762ff39809f194363b8a97e3fae172b8c1 (patch)
tree2c1772fe1f6710638d11e67594c7a3c6914651cf
parent9251d75b0194e548690e01ce21d32516eb8e5c63 (diff)
downloadFreeBSD-src-5d9e84762ff39809f194363b8a97e3fae172b8c1.zip
FreeBSD-src-5d9e84762ff39809f194363b8a97e3fae172b8c1.tar.gz
widen the routing event interface (arp update, redirect, and eventually pmtu change)
into separate functions revert previous commit's changes to arpresolve and add a new interface arpresolve2 which does arp resolution without an mbuf
-rw-r--r--sys/net/route.c4
-rw-r--r--sys/net/route.h9
-rw-r--r--sys/netinet/if_ether.c137
-rw-r--r--sys/netinet/if_ether.h2
4 files changed, 129 insertions, 23 deletions
diff --git a/sys/net/route.c b/sys/net/route.c
index 543383a..78cf49d 100644
--- a/sys/net/route.c
+++ b/sys/net/route.c
@@ -375,7 +375,7 @@ rtredirect(struct sockaddr *dst,
error = rtrequest1(RTM_ADD, &info, &rt);
if (rt != NULL) {
RT_LOCK(rt);
- EVENTHANDLER_INVOKE(route_event, RTEVENT_REDIRECT_UPDATE, rt0, rt, dst);
+ EVENTHANDLER_INVOKE(route_redirect_event, rt0, rt, dst);
flags = rt->rt_flags;
}
if (rt0)
@@ -397,7 +397,7 @@ rtredirect(struct sockaddr *dst,
*/
rt_setgate(rt, rt_key(rt), gateway);
gwrt = rtalloc1(gateway, 1, 0);
- EVENTHANDLER_INVOKE(route_event, RTEVENT_REDIRECT_UPDATE, rt, gwrt, dst);
+ EVENTHANDLER_INVOKE(route_redirect_event, rt, gwrt, dst);
RTFREE_LOCKED(gwrt);
}
} else
diff --git a/sys/net/route.h b/sys/net/route.h
index 423cccb..8c87f6b 100644
--- a/sys/net/route.h
+++ b/sys/net/route.h
@@ -361,11 +361,10 @@ int rtrequest1(int, struct rt_addrinfo *, struct rtentry **);
int rt_check(struct rtentry **, struct rtentry **, struct sockaddr *);
#include <sys/eventhandler.h>
-#define RTEVENT_ARP_UPDATE 1
-#define RTEVENT_PMTU_UPDATE 2
-#define RTEVENT_REDIRECT_UPDATE 3
-typedef void (*rtevent_fn)(void *, int, struct rtentry *, struct rtentry *, struct sockaddr *);
-EVENTHANDLER_DECLARE(route_event, rtevent_fn);
+typedef void (*rtevent_arp_update_fn)(void *, struct rtentry *, uint8_t *, struct sockaddr *);
+typedef void (*rtevent_redirect_fn)(void *, struct rtentry *, struct rtentry *, struct sockaddr *);
+EVENTHANDLER_DECLARE(route_arp_update_event, rtevent_arp_update_fn);
+EVENTHANDLER_DECLARE(route_redirect_event, rtevent_redirect_fn);
#endif
#endif
diff --git a/sys/netinet/if_ether.c b/sys/netinet/if_ether.c
index 006c966..f6c5c12 100644
--- a/sys/netinet/if_ether.c
+++ b/sys/netinet/if_ether.c
@@ -362,16 +362,13 @@ arpresolve(struct ifnet *ifp, struct rtentry *rt0, struct mbuf *m,
struct sockaddr_dl *sdl;
int error;
- if (m != NULL) {
-
- if (m->m_flags & M_BCAST) { /* broadcast */
- (void)memcpy(desten, ifp->if_broadcastaddr, ifp->if_addrlen);
- return (0);
- }
- if (m->m_flags & M_MCAST && ifp->if_type != IFT_ARCNET) {/* multicast */
- ETHER_MAP_IP_MULTICAST(&SIN(dst)->sin_addr, desten);
- return (0);
- }
+ if (m->m_flags & M_BCAST) { /* broadcast */
+ (void)memcpy(desten, ifp->if_broadcastaddr, ifp->if_addrlen);
+ return (0);
+ }
+ if (m->m_flags & M_MCAST && ifp->if_type != IFT_ARCNET) {/* multicast */
+ ETHER_MAP_IP_MULTICAST(&SIN(dst)->sin_addr, desten);
+ return (0);
}
if (rt0 != NULL) {
@@ -452,10 +449,117 @@ arpresolve(struct ifnet *ifp, struct rtentry *rt0, struct mbuf *m,
* response yet. Replace the held mbuf with this
* latest one.
*/
- if (m != NULL) {
- if (la->la_hold)
- m_freem(la->la_hold);
- la->la_hold = m;
+ if (la->la_hold)
+ m_freem(la->la_hold);
+ la->la_hold = m;
+ KASSERT(rt->rt_expire > 0, ("sending ARP request for static entry"));
+
+ /*
+ * Return EWOULDBLOCK if we have tried less than arp_maxtries. It
+ * will be masked by ether_output(). Return EHOSTDOWN/EHOSTUNREACH
+ * if we have already sent arp_maxtries ARP requests. Retransmit the
+ * ARP request, but not faster than one request per second.
+ */
+ if (la->la_asked < arp_maxtries)
+ error = EWOULDBLOCK; /* First request. */
+ else
+ error = (rt == rt0) ? EHOSTDOWN : EHOSTUNREACH;
+
+ if (la->la_asked == 0 || rt->rt_expire != time_uptime) {
+ struct in_addr sin =
+ SIN(rt->rt_ifa->ifa_addr)->sin_addr;
+
+ rt->rt_expire = time_uptime;
+ callout_reset(&la->la_timer, hz, arptimer, rt);
+ la->la_asked++;
+ RT_UNLOCK(rt);
+
+ arprequest(ifp, &sin, &SIN(dst)->sin_addr,
+ IF_LLADDR(ifp));
+ } else
+ RT_UNLOCK(rt);
+
+ return (error);
+}
+
+
+int
+arpresolve2(struct ifnet *ifp, struct rtentry *rt0, struct sockaddr *dst,
+ u_char *desten)
+{
+ struct llinfo_arp *la = NULL;
+ struct rtentry *rt = NULL;
+ struct sockaddr_dl *sdl;
+ int error;
+
+ if (rt0 != NULL) {
+ error = rt_check(&rt, &rt0, dst);
+ if (error)
+ return (error);
+
+ la = (struct llinfo_arp *)rt->rt_llinfo;
+ if (la == NULL)
+ RT_UNLOCK(rt);
+ }
+ if (la == NULL) {
+ /*
+ * We enter this block in case if rt0 was NULL,
+ * or if rt found by rt_check() didn't have llinfo.
+ */
+ rt = arplookup(SIN(dst)->sin_addr.s_addr, 1, 0);
+ if (rt == NULL) {
+ log(LOG_DEBUG,
+ "arpresolve: can't allocate route for %s\n",
+ inet_ntoa(SIN(dst)->sin_addr));
+ return (EINVAL); /* XXX */
+ }
+ la = (struct llinfo_arp *)rt->rt_llinfo;
+ if (la == NULL) {
+ RT_UNLOCK(rt);
+ log(LOG_DEBUG,
+ "arpresolve: can't allocate llinfo for %s\n",
+ inet_ntoa(SIN(dst)->sin_addr));
+ return (EINVAL); /* XXX */
+ }
+ }
+ sdl = SDL(rt->rt_gateway);
+ /*
+ * Check the address family and length is valid, the address
+ * is resolved; otherwise, try to resolve.
+ */
+ if ((rt->rt_expire == 0 || rt->rt_expire > time_uptime) &&
+ sdl->sdl_family == AF_LINK && sdl->sdl_alen != 0) {
+
+ bcopy(LLADDR(sdl), desten, sdl->sdl_alen);
+
+ /*
+ * If entry has an expiry time and it is approaching,
+ * send an ARP request.
+ */
+ if ((rt->rt_expire != 0) &&
+ (time_uptime + la->la_preempt > rt->rt_expire)) {
+ struct in_addr sin =
+ SIN(rt->rt_ifa->ifa_addr)->sin_addr;
+
+ la->la_preempt--;
+ RT_UNLOCK(rt);
+ arprequest(ifp, &sin, &SIN(dst)->sin_addr,
+ IF_LLADDR(ifp));
+ return (0);
+ }
+
+ RT_UNLOCK(rt);
+ return (0);
+ }
+ /*
+ * If ARP is disabled or static on this interface, stop.
+ * XXX
+ * Probably should not allocate empty llinfo struct if we are
+ * not going to be sending out an arp request.
+ */
+ if (ifp->if_flags & (IFF_NOARP | IFF_STATICARP)) {
+ RT_UNLOCK(rt);
+ return (EINVAL);
}
KASSERT(rt->rt_expire > 0, ("sending ARP request for static entry"));
@@ -585,6 +689,7 @@ in_arpinput(struct mbuf *m)
struct sockaddr_in sin;
sin.sin_len = sizeof(struct sockaddr_in);
sin.sin_family = AF_INET;
+ sin.sin_addr.s_addr = 0;
if (ifp->if_bridge)
bridged = 1;
@@ -673,8 +778,8 @@ match:
rt = arplookup(isaddr.s_addr, itaddr.s_addr == myaddr.s_addr, 0);
if (rt != NULL) {
sin.sin_addr.s_addr = isaddr.s_addr;
- EVENTHANDLER_INVOKE(route_event, RTEVENT_ARP_UPDATE, rt, NULL,
- (struct sockaddr *)&sin);
+ EVENTHANDLER_INVOKE(route_arp_update_event, rt,
+ ar_sha(ah), (struct sockaddr *)&sin);
la = (struct llinfo_arp *)rt->rt_llinfo;
if (la == NULL) {
diff --git a/sys/netinet/if_ether.h b/sys/netinet/if_ether.h
index 14df15f..ae2b0d0 100644
--- a/sys/netinet/if_ether.h
+++ b/sys/netinet/if_ether.h
@@ -111,6 +111,8 @@ extern u_char ether_ipmulticast_max[ETHER_ADDR_LEN];
int arpresolve(struct ifnet *ifp, struct rtentry *rt,
struct mbuf *m, struct sockaddr *dst, u_char *desten);
+int arpresolve2(struct ifnet *ifp, struct rtentry *rt,
+ struct sockaddr *dst, u_char *desten);
void arp_ifinit(struct ifnet *, struct ifaddr *);
void arp_ifinit2(struct ifnet *, struct ifaddr *, u_char *);
#endif
OpenPOWER on IntegriCloud