aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorGordon Tetlow <gordon@FreeBSD.org>2020-12-01 19:38:52 +0000
committerGordon Tetlow <gordon@FreeBSD.org>2020-12-01 19:38:52 +0000
commit1f6a45d684ffaa9d6a2dda37f9f40179478a6c7f (patch)
tree9e63e7f461c48da8065bc9780fe1279fd95a84a4
parenta999e18388407c7f56c2f69a54b8662b9ff28e8a (diff)
downloadsrc-1f6a45d684ffaa9d6a2dda37f9f40179478a6c7f.tar.gz
src-1f6a45d684ffaa9d6a2dda37f9f40179478a6c7f.zip
Fix ICMPv6 use-after-free in error message handling.
Approved by: so Security: FreeBSD-SA-20:31.icmp6 Security: CVE-2020-7469
Notes
Notes: svn path=/releng/11.4/; revision=368255
-rw-r--r--sys/netinet6/icmp6.c13
1 files changed, 4 insertions, 9 deletions
diff --git a/sys/netinet6/icmp6.c b/sys/netinet6/icmp6.c
index 7f36b0de1912..59f37f0ad01c 100644
--- a/sys/netinet6/icmp6.c
+++ b/sys/netinet6/icmp6.c
@@ -903,6 +903,7 @@ icmp6_notify_error(struct mbuf **mp, int off, int icmp6len, int code)
}
#endif
eip6 = (struct ip6_hdr *)(icmp6 + 1);
+ bzero(&icmp6dst, sizeof(icmp6dst));
/* Detect the upper level protocol */
{
@@ -911,7 +912,6 @@ icmp6_notify_error(struct mbuf **mp, int off, int icmp6len, int code)
int eoff = off + sizeof(struct icmp6_hdr) +
sizeof(struct ip6_hdr);
struct ip6ctlparam ip6cp;
- struct in6_addr *finaldst = NULL;
int icmp6type = icmp6->icmp6_type;
struct ip6_frag *fh;
struct ip6_rthdr *rth;
@@ -994,7 +994,7 @@ icmp6_notify_error(struct mbuf **mp, int off, int icmp6len, int code)
/* just ignore a bogus header */
if ((rth0->ip6r0_len % 2) == 0 &&
(hops = rth0->ip6r0_len/2))
- finaldst = (struct in6_addr *)(rth0 + 1) + (hops - 1);
+ icmp6dst.sin6_addr = *((struct in6_addr *)(rth0 + 1) + (hops - 1));
}
eoff += rthlen;
nxt = rth->ip6r_nxt;
@@ -1059,13 +1059,10 @@ icmp6_notify_error(struct mbuf **mp, int off, int icmp6len, int code)
*/
eip6 = (struct ip6_hdr *)(icmp6 + 1);
- bzero(&icmp6dst, sizeof(icmp6dst));
icmp6dst.sin6_len = sizeof(struct sockaddr_in6);
icmp6dst.sin6_family = AF_INET6;
- if (finaldst == NULL)
+ if (IN6_IS_ADDR_UNSPECIFIED(&icmp6dst.sin6_addr))
icmp6dst.sin6_addr = eip6->ip6_dst;
- else
- icmp6dst.sin6_addr = *finaldst;
if (in6_setscope(&icmp6dst.sin6_addr, m->m_pkthdr.rcvif, NULL))
goto freeit;
bzero(&icmp6src, sizeof(icmp6src));
@@ -1077,13 +1074,11 @@ icmp6_notify_error(struct mbuf **mp, int off, int icmp6len, int code)
icmp6src.sin6_flowinfo =
(eip6->ip6_flow & IPV6_FLOWLABEL_MASK);
- if (finaldst == NULL)
- finaldst = &eip6->ip6_dst;
ip6cp.ip6c_m = m;
ip6cp.ip6c_icmp6 = icmp6;
ip6cp.ip6c_ip6 = (struct ip6_hdr *)(icmp6 + 1);
ip6cp.ip6c_off = eoff;
- ip6cp.ip6c_finaldst = finaldst;
+ ip6cp.ip6c_finaldst = &icmp6dst.sin6_addr;
ip6cp.ip6c_src = &icmp6src;
ip6cp.ip6c_nxt = nxt;