summaryrefslogtreecommitdiff
path: root/sys/netinet
diff options
context:
space:
mode:
Diffstat (limited to 'sys/netinet')
-rw-r--r--sys/netinet/ip_icmp.c116
1 files changed, 86 insertions, 30 deletions
diff --git a/sys/netinet/ip_icmp.c b/sys/netinet/ip_icmp.c
index 9603862ebd7..ba1071a6a94 100644
--- a/sys/netinet/ip_icmp.c
+++ b/sys/netinet/ip_icmp.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: ip_icmp.c,v 1.15 1999/01/08 01:04:17 deraadt Exp $ */
+/* $OpenBSD: ip_icmp.c,v 1.16 1999/01/08 11:35:09 deraadt Exp $ */
/* $NetBSD: ip_icmp.c,v 1.19 1996/02/13 23:42:22 christos Exp $ */
/*
@@ -36,6 +36,18 @@
* @(#)ip_icmp.c 8.2 (Berkeley) 1/4/94
*/
+/*
+%%% portions-copyright-nrl-95
+Portions of this software are Copyright 1995-1998 by Randall Atkinson,
+Ronald Lee, Daniel McDonald, Bao Phan, and Chris Winters. All Rights
+Reserved. All rights under this copyright have been assigned to the US
+Naval Research Laboratory (NRL). The NRL Copyright Notice and License
+Agreement Version 1.1 (January 17, 1995) applies to these portions of the
+software.
+You should have received a copy of the license with this software. If you
+didn't get a copy, you may request one from <license@ipv6.nrl.navy.mil>.
+*/
+
#include <sys/param.h>
#include <sys/systm.h>
#include <sys/malloc.h>
@@ -259,36 +271,62 @@ icmp_input(m, va_alist)
case ICMP_UNREACH:
switch (code) {
- case ICMP_UNREACH_NET:
- case ICMP_UNREACH_HOST:
- case ICMP_UNREACH_PROTOCOL:
- case ICMP_UNREACH_PORT:
- case ICMP_UNREACH_SRCFAIL:
- code += PRC_UNREACH_NET;
- break;
-
- case ICMP_UNREACH_NEEDFRAG:
- code = PRC_MSGSIZE;
- break;
+ case ICMP_UNREACH_NET:
+ case ICMP_UNREACH_HOST:
+ case ICMP_UNREACH_PROTOCOL:
+ case ICMP_UNREACH_PORT:
+ case ICMP_UNREACH_SRCFAIL:
+ code += PRC_UNREACH_NET;
+ break;
+
+ case ICMP_UNREACH_NEEDFRAG:
+#ifdef INET6
+ if (icp->icmp_nextmtu) {
+ extern int ipv6_trans_mtu
+ __P((struct mbuf **, int, int));
+ struct mbuf *m0 = m;
+
+ /*
+ * Do cool v4-related path MTU, for now,
+ * only v6-in-v4 can handle it.
+ */
+ if (icmplen >= ICMP_V6ADVLENMIN &&
+ icmplen >= ICMP_V6ADVLEN(icp) &&
+ icp->icmp_ip.ip_p == IPPROTO_IPV6) {
+ /*
+ * ipv6_trans_mtu returns 1 if
+ * the mbuf is still intact.
+ */
+ if (ipv6_trans_mtu(&m0,icp->icmp_nextmtu,
+ hlen + ICMP_V6ADVLEN(icp))) {
+ m = m0;
+ goto raw;
+ } else
+ return;
+ }
+ }
+#endif /* INET6 */
+ code = PRC_MSGSIZE;
+ break;
- case ICMP_UNREACH_NET_UNKNOWN:
- case ICMP_UNREACH_NET_PROHIB:
- case ICMP_UNREACH_TOSNET:
- code = PRC_UNREACH_NET;
- break;
-
- case ICMP_UNREACH_HOST_UNKNOWN:
- case ICMP_UNREACH_ISOLATED:
- case ICMP_UNREACH_HOST_PROHIB:
- case ICMP_UNREACH_TOSHOST:
- case ICMP_UNREACH_FILTER_PROHIB:
- case ICMP_UNREACH_HOST_PRECEDENCE:
- case ICMP_UNREACH_PRECEDENCE_CUTOFF:
- code = PRC_UNREACH_HOST;
- break;
-
- default:
- goto badcode;
+ case ICMP_UNREACH_NET_UNKNOWN:
+ case ICMP_UNREACH_NET_PROHIB:
+ case ICMP_UNREACH_TOSNET:
+ code = PRC_UNREACH_NET;
+ break;
+
+ case ICMP_UNREACH_HOST_UNKNOWN:
+ case ICMP_UNREACH_ISOLATED:
+ case ICMP_UNREACH_HOST_PROHIB:
+ case ICMP_UNREACH_TOSHOST:
+ case ICMP_UNREACH_FILTER_PROHIB:
+ case ICMP_UNREACH_HOST_PRECEDENCE:
+ case ICMP_UNREACH_PRECEDENCE_CUTOFF:
+ code = PRC_UNREACH_HOST;
+ break;
+
+ default:
+ goto badcode;
}
goto deliver;
@@ -320,6 +358,24 @@ icmp_input(m, va_alist)
if (IN_MULTICAST(icp->icmp_ip.ip_dst.s_addr))
goto badcode;
NTOHS(icp->icmp_ip.ip_len);
+#ifdef INET6
+ /* Get more contiguous data for a v6 in v4 ICMP message. */
+ if (icp->icmp_ip.ip_p == IPPROTO_IPV6) {
+ if (icmplen < ICMP_V6ADVLENMIN ||
+ icmplen < ICMP_V6ADVLEN(icp)) {
+ icmpstat.icps_badlen++;
+ goto freeit;
+ } else {
+ if (!(m = m_pullup(m, (ip->ip_hl << 2) +
+ ICMP_V6ADVLEN(icp)))) {
+ icmpstat.icps_tooshort++;
+ return;
+ }
+ ip = mtod(m, struct ip *);
+ icp = (struct icmp *)(m->m_data + (ip->ip_hl << 2));
+ }
+ }
+#endif /* INET6 */
#ifdef ICMPPRINTFS
if (icmpprintfs)
printf("deliver to protocol %d\n", icp->icmp_ip.ip_p);