diff options
author | kmacy <kmacy@FreeBSD.org> | 2007-12-17 07:40:34 +0000 |
---|---|---|
committer | kmacy <kmacy@FreeBSD.org> | 2007-12-17 07:40:34 +0000 |
commit | 5d9e84762ff39809f194363b8a97e3fae172b8c1 (patch) | |
tree | 2c1772fe1f6710638d11e67594c7a3c6914651cf | |
parent | 9251d75b0194e548690e01ce21d32516eb8e5c63 (diff) | |
download | FreeBSD-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.c | 4 | ||||
-rw-r--r-- | sys/net/route.h | 9 | ||||
-rw-r--r-- | sys/netinet/if_ether.c | 137 | ||||
-rw-r--r-- | sys/netinet/if_ether.h | 2 |
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 |