From: bluhm Date: Thu, 1 Feb 2018 21:06:31 +0000 (+0000) Subject: When processing IPsec AH packets with IP options or IPv6 extension X-Git-Url: http://artulab.com/gitweb/?a=commitdiff_plain;h=df9b9db2c3d93a3b2fa67316cad3e2df6eeb53f5;p=openbsd When processing IPsec AH packets with IP options or IPv6 extension headers, check all lengths before accessing the values. found by Maxime Villard; from NetBSD; with and OK markus@; OK patrick@ --- diff --git a/sys/netinet/ip_ah.c b/sys/netinet/ip_ah.c index 517443ad625..a89ab2571a8 100644 --- a/sys/netinet/ip_ah.c +++ b/sys/netinet/ip_ah.c @@ -1,4 +1,4 @@ -/* $OpenBSD: ip_ah.c,v 1.133 2017/11/08 16:29:20 visa Exp $ */ +/* $OpenBSD: ip_ah.c,v 1.134 2018/02/01 21:06:31 bluhm Exp $ */ /* * The authors of this code are John Ioannidis (ji@tla.org), * Angelos D. Keromytis (kermit@csd.uch.gr) and @@ -201,7 +201,7 @@ ah_massage_headers(struct mbuf **m0, int proto, int skip, int alg, int out) #ifdef INET6 struct ip6_ext *ip6e; struct ip6_hdr ip6; - int ad, alloc, nxt; + int ad, alloc, nxt, noff; #endif /* INET6 */ switch (proto) { @@ -225,7 +225,7 @@ ah_massage_headers(struct mbuf **m0, int proto, int skip, int alg, int out) ip->ip_sum = 0; ip->ip_off = 0; - ptr = mtod(m, unsigned char *) + sizeof(struct ip); + ptr = mtod(m, unsigned char *); /* IPv4 option processing */ for (off = sizeof(struct ip); off < skip;) { @@ -291,10 +291,12 @@ ah_massage_headers(struct mbuf **m0, int proto, int skip, int alg, int out) * what the destination's IP header * will look like. */ - if (out) - bcopy(ptr + off + ptr[off + 1] - + if (out && + ptr[off + 1] >= 2 + sizeof(struct in_addr)) + memcpy(&ip->ip_dst, + ptr + off + ptr[off + 1] - sizeof(struct in_addr), - &(ip->ip_dst), sizeof(struct in_addr)); + sizeof(struct in_addr)); /* FALLTHROUGH */ default: @@ -310,7 +312,7 @@ ah_massage_headers(struct mbuf **m0, int proto, int skip, int alg, int out) /* Zeroize all other options. */ count = ptr[off + 1]; - memcpy(ptr, ipseczeroes, count); + memset(ptr + off, 0, count); off += count; break; } @@ -387,57 +389,46 @@ ah_massage_headers(struct mbuf **m0, int proto, int skip, int alg, int out) nxt = ip6.ip6_nxt & 0xff; /* Next header type. */ for (off = 0; off < skip - sizeof(struct ip6_hdr);) { + if (off + sizeof(struct ip6_ext) > + skip - sizeof(struct ip6_hdr)) + goto error6; + ip6e = (struct ip6_ext *)(ptr + off); + switch (nxt) { case IPPROTO_HOPOPTS: case IPPROTO_DSTOPTS: - ip6e = (struct ip6_ext *) (ptr + off); + noff = off + ((ip6e->ip6e_len + 1) << 3); + + /* Sanity check. */ + if (noff > skip - sizeof(struct ip6_hdr)) + goto error6; /* - * Process the mutable/immutable - * options -- borrows heavily from the - * KAME code. + * Zero out mutable options. */ for (count = off + sizeof(struct ip6_ext); - count < off + ((ip6e->ip6e_len + 1) << 3);) { + count < noff;) { if (ptr[count] == IP6OPT_PAD1) { count++; continue; /* Skip padding. */ } - /* Sanity check. */ - if (count > off + - ((ip6e->ip6e_len + 1) << 3)) { - ahstat_inc(ahs_hdrops); - m_freem(m); - - /* Free, if we allocated. */ - if (alloc) - free(ptr, M_XDATA, 0); - return EINVAL; - } - - ad = ptr[count + 1]; + if (count + 2 > noff) + goto error6; + ad = ptr[count + 1] + 2; + if (count + ad > noff) + goto error6; /* If mutable option, zeroize. */ if (ptr[count] & IP6OPT_MUTABLE) - memcpy(ptr + count, ipseczeroes, - ptr[count + 1]); + memset(ptr + count, 0, ad); count += ad; - - /* Sanity check. */ - if (count > - skip - sizeof(struct ip6_hdr)) { - ahstat_inc(ahs_hdrops); - m_freem(m); - - /* Free, if we allocated. */ - if (alloc) - free(ptr, M_XDATA, 0); - return EINVAL; - } } + if (count != noff) + goto error6; + /* Advance. */ off += ((ip6e->ip6e_len + 1) << 3); nxt = ip6e->ip6e_nxt; @@ -451,7 +442,6 @@ ah_massage_headers(struct mbuf **m0, int proto, int skip, int alg, int out) { struct ip6_rthdr *rh; - ip6e = (struct ip6_ext *) (ptr + off); rh = (struct ip6_rthdr *)(ptr + off); /* * must adjust content to make it look like @@ -496,6 +486,7 @@ ah_massage_headers(struct mbuf **m0, int proto, int skip, int alg, int out) default: DPRINTF(("%s: unexpected IPv6 header type %d\n", __func__, off)); +error6: if (alloc) free(ptr, M_XDATA, 0); ahstat_inc(ahs_hdrops);