diff options
| author | David Lamparter <equinox@opensourcerouting.org> | 2012-09-26 14:52:39 +0200 | 
|---|---|---|
| committer | David Lamparter <equinox@opensourcerouting.org> | 2013-01-16 01:45:57 +0100 | 
| commit | ca3ccd8748434719e4670ce812d1310013fad518 (patch) | |
| tree | 3fae89070206bc47704a456219350c34b5f8ea54 /lib | |
| parent | 8d083b9ec5bb0375ebb6d8b2b05c848febd92cb5 (diff) | |
zebra: fix sockaddr_dl length assumptions (BZ#737)
Quagga makes bad assumptions about sockaddr_dl (on NetBSD, but possibly
on other systems as well).  Particularly, sizeof(struct sockaddr_dl)
returns a size that does not include the full sdl_data field, leading to
not enough data being copied.  This breaks IPv6 RAs in particular, as
a broken mac address from sockaddr_dl will be included in the packets.
From: Matthias-Christian Ott <ott@mirix.org>
Tested-by: Uwe Toenjes <6bone@6bone.informatik.uni-leipzig.de>
[further simplified + more comments]
Signed-off-by: David Lamparter <equinox@opensourcerouting.org>
Diffstat (limited to 'lib')
| -rw-r--r-- | lib/if.h | 8 | ||||
| -rw-r--r-- | lib/zclient.c | 2 | 
2 files changed, 8 insertions, 2 deletions
| @@ -103,7 +103,13 @@ struct interface    /* Hardware address. */  #ifdef HAVE_STRUCT_SOCKADDR_DL -  struct sockaddr_dl sdl; +  union { +    /* note that sdl_storage is never accessed, it only exists to make space. +     * all actual uses refer to sdl - but use sizeof(sdl_storage)!  this fits +     * best with C aliasing rules. */ +    struct sockaddr_dl sdl; +    struct sockaddr_storage sdl_storage; +  };  #else    unsigned short hw_type;    u_char hw_addr[INTERFACE_HWADDR_MAX]; diff --git a/lib/zclient.c b/lib/zclient.c index 61c6f730..d3165962 100644 --- a/lib/zclient.c +++ b/lib/zclient.c @@ -734,7 +734,7 @@ zebra_interface_if_set_value (struct stream *s, struct interface *ifp)    ifp->mtu6 = stream_getl (s);    ifp->bandwidth = stream_getl (s);  #ifdef HAVE_STRUCT_SOCKADDR_DL -  stream_get (&ifp->sdl, s, sizeof (ifp->sdl)); +  stream_get (&ifp->sdl, s, sizeof (ifp->sdl_storage));  #else    ifp->hw_addr_len = stream_getl (s);    if (ifp->hw_addr_len) | 
