From 4d5cd9496ea8503c275436c9ac768bcd8d12c1d4 Mon Sep 17 00:00:00 2001 From: Christian Weisgerber Date: Sun, 23 Aug 2015 14:12:06 +0000 Subject: Use simple byte pointer arithmetic and memcpy from/to aligned stack variables to handle the "packed" binary format passed out to userland when querying the prefix/router list. From NetBSD (Martin Husemann). ok mpi@ --- sys/netinet6/nd6.c | 82 ++++++++++++++++++++++++++++-------------------------- 1 file changed, 42 insertions(+), 40 deletions(-) (limited to 'sys/netinet6/nd6.c') diff --git a/sys/netinet6/nd6.c b/sys/netinet6/nd6.c index 31110312523..1f6dca27246 100644 --- a/sys/netinet6/nd6.c +++ b/sys/netinet6/nd6.c @@ -1,4 +1,4 @@ -/* $OpenBSD: nd6.c,v 1.145 2015/08/19 13:27:38 bluhm Exp $ */ +/* $OpenBSD: nd6.c,v 1.146 2015/08/23 14:12:05 naddy Exp $ */ /* $KAME: nd6.c,v 1.280 2002/06/08 19:52:07 itojun Exp $ */ /* @@ -1865,45 +1865,43 @@ fill_prlist(void *oldp, size_t *oldlenp, size_t ol) { int error = 0, s; struct nd_prefix *pr; - struct in6_prefix *p = NULL; - struct in6_prefix *pe = NULL; + char *p = NULL, *ps = NULL; + char *pe = NULL; size_t l; s = splsoftnet(); if (oldp) { - p = (struct in6_prefix *)oldp; - pe = (struct in6_prefix *)((caddr_t)oldp + *oldlenp); + ps = p = (char *)oldp; + pe = (char *)oldp + *oldlenp; } l = 0; LIST_FOREACH(pr, &nd_prefix, ndpr_entry) { u_short advrtrs; - size_t advance; - struct sockaddr_in6 *sin6; - struct sockaddr_in6 *s6; + struct sockaddr_in6 sin6; struct nd_pfxrouter *pfr; + struct in6_prefix pfx; char addr[INET6_ADDRSTRLEN]; - if (oldp && p + 1 <= pe) - { - bzero(p, sizeof(*p)); - sin6 = (struct sockaddr_in6 *)(p + 1); + if (oldp && p + sizeof(struct in6_prefix) <= pe) { + memset(&pfx, 0, sizeof(pfx)); + ps = p; - p->prefix = pr->ndpr_prefix; - if (in6_recoverscope(&p->prefix, - &p->prefix.sin6_addr, pr->ndpr_ifp) != 0) + pfx.prefix = pr->ndpr_prefix; + if (in6_recoverscope(&pfx.prefix, + &pfx.prefix.sin6_addr, pr->ndpr_ifp) != 0) log(LOG_ERR, "scope error in prefix list (%s)\n", - inet_ntop(AF_INET6, &p->prefix.sin6_addr, + inet_ntop(AF_INET6, &pfx.prefix.sin6_addr, addr, sizeof(addr))); - p->raflags = pr->ndpr_raf; - p->prefixlen = pr->ndpr_plen; - p->vltime = pr->ndpr_vltime; - p->pltime = pr->ndpr_pltime; - p->if_index = pr->ndpr_ifp->if_index; + pfx.raflags = pr->ndpr_raf; + pfx.prefixlen = pr->ndpr_plen; + pfx.vltime = pr->ndpr_vltime; + pfx.pltime = pr->ndpr_pltime; + pfx.if_index = pr->ndpr_ifp->if_index; if (pr->ndpr_vltime == ND6_INFINITE_LIFETIME) - p->expire = 0; + pfx.expire = 0; else { time_t maxexpire; @@ -1912,40 +1910,44 @@ fill_prlist(void *oldp, size_t *oldlenp, size_t ol) ((sizeof(maxexpire) * 8) - 1)); if (pr->ndpr_vltime < maxexpire - pr->ndpr_lastupdate) { - p->expire = pr->ndpr_lastupdate + + pfx.expire = pr->ndpr_lastupdate + pr->ndpr_vltime; } else - p->expire = maxexpire; + pfx.expire = maxexpire; } - p->refcnt = pr->ndpr_refcnt; - p->flags = pr->ndpr_stateflags; - p->origin = PR_ORIG_RA; + pfx.refcnt = pr->ndpr_refcnt; + pfx.flags = pr->ndpr_stateflags; + pfx.origin = PR_ORIG_RA; + + p += sizeof(pfx); l += sizeof(pfx); + advrtrs = 0; LIST_FOREACH(pfr, &pr->ndpr_advrtrs, pfr_entry) { - if ((void *)&sin6[advrtrs + 1] > (void *)pe) { + if (p + sizeof(sin6) > pe) { advrtrs++; continue; } - s6 = &sin6[advrtrs]; - s6->sin6_family = AF_INET6; - s6->sin6_len = sizeof(struct sockaddr_in6); - s6->sin6_addr = pfr->router->rtaddr; - in6_recoverscope(s6, &pfr->router->rtaddr, + sin6.sin6_family = AF_INET6; + sin6.sin6_len = sizeof(struct sockaddr_in6); + sin6.sin6_addr = pfr->router->rtaddr; + in6_recoverscope(&sin6, &pfr->router->rtaddr, pfr->router->ifp); advrtrs++; + memcpy(p, &sin6, sizeof(sin6)); + p += sizeof(sin6); + l += sizeof(sin6); } - p->advrtrs = advrtrs; + pfx.advrtrs = advrtrs; + memcpy(ps, &pfx, sizeof(pfx)); } else { + l += sizeof(pfx); advrtrs = 0; - LIST_FOREACH(pfr, &pr->ndpr_advrtrs, pfr_entry) + LIST_FOREACH(pfr, &pr->ndpr_advrtrs, pfr_entry) { advrtrs++; + l += sizeof(sin6); + } } - - advance = sizeof(*p) + sizeof(*sin6) * advrtrs; - l += advance; - if (p) - p = (struct in6_prefix *)((caddr_t)p + advance); } if (oldp) { -- cgit v1.2.3