summaryrefslogtreecommitdiff
path: root/sys/netinet6/in6_proto.c
diff options
context:
space:
mode:
Diffstat (limited to 'sys/netinet6/in6_proto.c')
-rw-r--r--sys/netinet6/in6_proto.c387
1 files changed, 387 insertions, 0 deletions
diff --git a/sys/netinet6/in6_proto.c b/sys/netinet6/in6_proto.c
new file mode 100644
index 00000000000..a4222730caa
--- /dev/null
+++ b/sys/netinet6/in6_proto.c
@@ -0,0 +1,387 @@
+/*
+%%% copyright-nrl-95
+This software is 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 this 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/socket.h>
+#include <sys/protosw.h>
+#include <sys/kernel.h>
+#include <sys/domain.h>
+#include <sys/mbuf.h>
+
+#include <net/if.h>
+#include <net/route.h>
+#include <net/radix.h>
+
+#include <netinet/in.h>
+#include <netinet/in_systm.h>
+#include <netinet/ip.h>
+#include <netinet/ip_var.h>
+#include <netinet/in_pcb.h>
+#include <netinet/udp.h>
+#include <netinet/udp_var.h>
+
+#include <netinet/tcp.h>
+#include <netinet/tcp_fsm.h>
+#include <netinet/tcp_seq.h>
+#include <netinet/tcp_timer.h>
+#include <netinet/tcpip.h>
+#if __FreeBSD__ && defined(_NETINET_IN_PCB_H_)
+#undef _NETINET_IN_PCB_H_
+#include <netinet/tcp_var.h>
+#define _NETINET_IN_PCB_H_
+#else /* __FreeBSD__ */
+#include <netinet/tcp_var.h>
+#endif /* __FreeBSD__ */
+#include <netinet/tcpip.h>
+#include <netinet/tcp_debug.h>
+
+#include <netinet6/in6.h>
+#include <netinet6/in6_var.h>
+#include <netinet6/ipv6.h>
+#include <netinet6/ipv6_var.h>
+#include <netinet6/ipv6_icmp.h>
+
+#if defined(IPSEC) || defined(NRL_IPSEC)
+#include <netsec/ipsec.h>
+#endif /* defined(IPSEC) || defined(NRL_IPSEC) */
+
+#if __FreeBSD__
+#include <sys/sysctl.h>
+#endif /* __FreeBSD__ */
+
+extern struct domain inet6domain;
+
+#define CAST (void *)
+
+#if !__FreeBSD__
+struct protosw inet6sw[] = {
+/* normal protocol switch */
+ {
+ 0, &inet6domain, 0, 0, /* NOTE: This 0 is the same as IPPROTO_HOPOPTS,
+ but we specially demux IPPROTO_HOPOPTS
+ in ipv6_input(). */
+ CAST ipv6_hop, CAST ipv6_output, 0, 0, /* Watch for hop-by-hop input! */
+ 0,
+ ipv6_init, 0, ipv6_slowtimo, ipv6_drain, ipv6_sysctl
+ },
+
+ /* ICMPv6 entry */
+
+ {
+ SOCK_RAW, &inet6domain, IPPROTO_ICMPV6, PR_ATOMIC|PR_ADDR,
+ CAST ipv6_icmp_input, CAST ipv6_icmp_output, 0, ripv6_ctloutput,
+ ipv6_icmp_usrreq,
+#if defined(_BSDI_VERSION) && _BSDI_VERSION >= 199802
+ 0, 0, 0, 0, ipv6_icmp_sysctl
+#else /* defined(_BSDI_VERSION) && _BSDI_VERSION >= 199802 */
+ 0, 0, 0, 0, 0
+#endif /* defined(_BSDI_VERSION) && _BSDI_VERSION >= 199802 */
+ },
+
+ /* IPv6-in-IPv6 tunnel entry */
+
+ {
+ SOCK_RAW, &inet6domain, IPPROTO_IPV6, PR_ATOMIC|PR_ADDR,
+ CAST ipv6_input, CAST ripv6_output, 0, ripv6_ctloutput,
+ ripv6_usrreq,
+#if defined(_BSDI_VERSION) && _BSDI_VERSION >= 199802
+ 0, 0, 0, 0, ipv6_sysctl
+#else /* defined(_BSDI_VERSION) && _BSDI_VERSION >= 199802 */
+ 0, 0, 0, 0, 0
+#endif /* defined(_BSDI_VERSION) && _BSDI_VERSION >= 199802 */
+ },
+
+ /* IPv4-in-IPv6 tunnel entry */
+
+ {
+ SOCK_RAW, &inet6domain, IPPROTO_IPV4, PR_ATOMIC|PR_ADDR,
+ CAST ipv4_input, 0, 0, 0,
+ 0,
+#if defined(_BSDI_VERSION) && _BSDI_VERSION >= 199802
+ 0, 0, 0, 0, ip_sysctl
+#else /* defined(_BSDI_VERSION) && _BSDI_VERSION >= 199802 */
+ 0, 0, 0, 0, 0
+#endif /* defined(_BSDI_VERSION) && _BSDI_VERSION >= 199802 */
+ },
+
+ /* Fragment entry */
+
+ {
+ SOCK_RAW, &inet6domain, IPPROTO_FRAGMENT, PR_ATOMIC|PR_ADDR,
+ CAST ipv6_reasm, 0, 0, 0,
+ 0,
+ 0, 0, 0, 0, 0
+ },
+
+
+ /* UDP entry */
+
+ /*
+ * Eventually, that ipv6_ctloutput() will have to be replaced with a
+ * udp_ctloutput(), which knows whether or not to redirect things down to
+ * IP or IPv6 appropriately.
+ */
+
+ {
+ SOCK_DGRAM, &inet6domain, IPPROTO_UDP, PR_ATOMIC|PR_ADDR,
+ CAST udp_input, 0, CAST udp_ctlinput, ipv6_ctloutput,
+ udp_usrreq,
+#if defined(_BSDI_VERSION) && _BSDI_VERSION >= 199802
+ 0, 0, 0, 0, udp_sysctl
+#else /* defined(_BSDI_VERSION) && _BSDI_VERSION >= 199802 */
+ 0, 0, 0, 0, 0
+#endif /* defined(_BSDI_VERSION) && _BSDI_VERSION >= 199802 */
+ },
+
+ /* TCP entry */
+
+ {
+ SOCK_STREAM, &inet6domain, IPPROTO_TCP, PR_CONNREQUIRED|PR_WANTRCVD,
+ CAST tcp_input, 0, CAST tcp_ctlinput, tcp_ctloutput,
+ tcp_usrreq,
+#if defined(_BSDI_VERSION) && _BSDI_VERSION >= 199802
+ 0, 0, 0, 0, tcp_sysctl /* init, fasttimo, etc. in v4 protosw already! */
+#else /* defined(_BSDI_VERSION) && _BSDI_VERSION >= 199802 */
+ 0, 0, 0, 0, 0 /* init, fasttimo, etc. in v4 protosw already! */
+#endif /* defined(_BSDI_VERSION) && _BSDI_VERSION >= 199802 */
+ },
+
+#if defined(IPSEC) || defined(NRL_IPSEC)
+ /* IPv6 & IPv4 Authentication Header */
+ {
+ SOCK_RAW, &inet6domain, IPPROTO_AH, PR_ATOMIC|PR_ADDR,
+ CAST ipsec_ah_input, 0, 0, 0,
+ 0,
+#if defined(_BSDI_VERSION) && _BSDI_VERSION >= 199802
+ 0, 0, 0, 0, ipsec_ah_sysctl
+#else /* defined(_BSDI_VERSION) && _BSDI_VERSION >= 199802 */
+ 0, 0, 0, 0, 0
+#endif /* defined(_BSDI_VERSION) && _BSDI_VERSION >= 199802 */
+ },
+
+#ifdef IPSEC_ESP
+ /* IPv6 & IPv4 Encapsulating Security Payload Header */
+ {
+ SOCK_RAW, &inet6domain, IPPROTO_ESP, PR_ATOMIC|PR_ADDR,
+ CAST ipsec_esp_input, 0, 0, 0,
+ 0,
+#if defined(_BSDI_VERSION) && _BSDI_VERSION >= 199802
+ 0, 0, 0, 0, ipsec_esp_sysctl
+#else /* defined(_BSDI_VERSION) && _BSDI_VERSION >= 199802 */
+ 0, 0, 0, 0, 0
+#endif /* defined(_BSDI_VERSION) && _BSDI_VERSION >= 199802 */
+ },
+#endif /* IPSEC_ESP */
+#endif /* defined(IPSEC) || defined(NRL_IPSEC) */
+
+ /* Unknown header. */
+
+ {
+ SOCK_RAW, &inet6domain, IPPROTO_RAW, PR_ATOMIC|PR_ADDR,
+ CAST ripv6_input, CAST ripv6_output, 0, ripv6_ctloutput,
+ ripv6_usrreq,
+ 0,0,0,0,0
+ },
+
+ /* Raw wildcard */
+ {
+ SOCK_RAW, &inet6domain, 0, PR_ATOMIC|PR_ADDR,
+ CAST ripv6_input, CAST ripv6_output, 0, ripv6_ctloutput,
+ ripv6_usrreq,
+ ripv6_init,0,0,0,0
+ },
+};
+#else /* !__FreeBSD__ */
+extern struct pr_usrreqs nousrreqs;
+struct protosw inet6sw[] = {
+ {
+ 0, &inet6domain, 0, 0, /* NOTE: This 0 is the same as IPPROTO_HOPOPTS,
+ but we specially demux IPPROTO_HOPOPTS
+ in ipv6_input(). */
+ CAST ipv6_hop, CAST ipv6_output, 0, 0, /* Watch for hop-by-hop input! */
+ 0,
+ ipv6_init, 0, ipv6_slowtimo, ipv6_drain,
+ &nousrreqs
+ },
+
+ /* ICMPv6 entry */
+
+ {
+ SOCK_RAW, &inet6domain, IPPROTO_ICMPV6, PR_ATOMIC|PR_ADDR,
+ CAST ipv6_icmp_input, CAST ipv6_icmp_output, 0, CAST ripv6_ctloutput,
+ 0,
+ 0, 0, 0, 0,
+ &ipv6_icmp_usrreqs,
+ },
+
+ /* IPv6-in-IPv6 tunnel entry */
+
+ {
+ SOCK_RAW, &inet6domain, IPPROTO_IPV6, PR_ATOMIC|PR_ADDR,
+ CAST ipv6_input, CAST ripv6_output, 0, ripv6_ctloutput,
+ 0,
+ 0, 0, 0, 0,
+ &ripv6_usrreqs
+ },
+
+ /* IPv4-in-IPv6 tunnel entry */
+
+ {
+ SOCK_RAW, &inet6domain, IPPROTO_IPV4, PR_ATOMIC|PR_ADDR,
+ CAST ipv4_input, 0, 0, 0,
+ 0,
+ 0, 0, 0, 0,
+ &nousrreqs
+ },
+
+ /* Fragment entry */
+
+ {
+ SOCK_RAW, &inet6domain, IPPROTO_FRAGMENT, PR_ATOMIC|PR_ADDR,
+ CAST ipv6_reasm, 0, 0, 0,
+ 0,
+ 0, 0, 0, 0,
+ &nousrreqs
+ },
+
+
+ /* UDP entry */
+
+ /*
+ * Eventually, that ipv6_ctloutput() will have to be replaced with a
+ * udp_ctloutput(), which knows whether or not to redirect things down to
+ * IP or IPv6 appropriately.
+ */
+
+ {
+ SOCK_DGRAM, &inet6domain, IPPROTO_UDP, PR_ATOMIC|PR_ADDR,
+ CAST udp_input, 0, CAST udp_ctlinput, ipv6_ctloutput,
+ 0,
+ udp_init, 0, 0, 0,
+ &udp_usrreqs
+ },
+
+ /* TCP entry */
+
+ {
+ SOCK_STREAM, &inet6domain, IPPROTO_TCP, PR_CONNREQUIRED|PR_WANTRCVD,
+ CAST tcp_input, 0, CAST tcp_ctlinput, tcp_ctloutput,
+ 0,
+ tcp_init, tcp_fasttimo, tcp_slowtimo, tcp_drain,
+ &tcp_usrreqs,
+ },
+
+#if defined(IPSEC) || defined(NRL_IPSEC)
+ /* IPv6 & IPv4 Authentication Header */
+ {
+ SOCK_RAW, &inet6domain, IPPROTO_AH, PR_ATOMIC|PR_ADDR,
+ CAST ipsec_ah_input, 0, 0, 0,
+ 0,
+ 0, 0, 0, 0,
+ &nousrreqs
+ },
+
+#ifdef IPSEC_ESP
+ /* IPv6 & IPv4 Encapsulating Security Payload Header */
+ {
+ SOCK_RAW, &inet6domain, IPPROTO_ESP, PR_ATOMIC|PR_ADDR,
+ CAST ipsec_esp_input, 0, 0, 0,
+ 0,
+ 0, 0, 0, 0,
+ &nousrreqs
+ },
+#endif /* IPSEC_ESP */
+#endif /* defined(IPSEC) || defined(NRL_IPSEC) */
+
+ /* Unknown header. */
+
+ {
+ SOCK_RAW, &inet6domain, IPPROTO_RAW, PR_ATOMIC|PR_ADDR,
+ CAST ripv6_input, CAST ripv6_output, 0, ripv6_ctloutput,
+ 0,
+ 0,0,0,0,
+ &ripv6_usrreqs
+ },
+
+ /* Raw wildcard */
+ {
+ SOCK_RAW, &inet6domain, 0, PR_ATOMIC|PR_ADDR,
+ CAST ripv6_input, CAST ripv6_output, 0, ripv6_ctloutput,
+ 0,
+ ripv6_init,0,0,0,
+ &ripv6_usrreqs,
+ },
+};
+
+#endif /* !__FreeBSD__ */
+
+#if !__FreeBSD__
+struct domain inet6domain =
+{
+ PF_INET6, "IPv6", 0, 0, 0,
+ inet6sw, &inet6sw[sizeof(inet6sw)/sizeof(inet6sw[0])], 0,
+ /*
+ * FreeBSD's IPv4 replaces rn_inithead() with an IPv4-specific function.
+ * Our IPv6 uses the ifa->ifa_rtrequest() function pointer to intercept
+ * rtrequest()s. The consequence of this is that we use the generic
+ * rn_inithead().
+ */
+ rn_inithead, 64, sizeof(struct sockaddr_in6)
+};
+#else /* !__FreeBSD__ */
+struct domain inet6domain =
+{
+ PF_INET6, "IPv6", 0, 0, 0,
+ inet6sw, &inet6sw[sizeof(inet6sw)/sizeof(inet6sw[0])], 0,
+ /*
+ * FreeBSD's IPv4 replaces rn_inithead() with an IPv4-specific function.
+ * Our IPv6 uses the ifa->ifa_rtrequest() function pointer to intercept
+ * rtrequest()s. The consequence of this is that we use the generic
+ * rn_inithead().
+ */
+ rn_inithead, 64, sizeof(struct sockaddr_in6)
+};
+
+DOMAIN_SET(inet6);
+#endif /* !__FreeBSD__ */
+
+/* Eventually, make these go away -- if you want to be a router, twiddle the
+ sysctls before bringing up your interfaces */
+
+#ifndef IPV6FORWARDING
+#ifdef IPV6GATEWAY
+#define IPV6FORWARDING 1
+#else
+#define IPV6FORWARDING 0
+#endif /* IPV6GATEWAY */
+#endif /* IPV6FORWARDING */
+
+#ifndef IPV6RSOLICIT
+#if IPV6FORWARDING
+#define IPV6RSOLICIT 0
+#else /* IPV6FORWARDING */
+#define IPV6RSOLICIT 1
+#endif /* IPV6FORWARDING */
+#endif /* IPV6RSOLICIT */
+
+#ifndef IFQMAXLEN
+#define IFQMAXLEN IFQ_MAXLEN
+#endif
+
+int ipv6forwarding = IPV6FORWARDING;
+int ipv6rsolicit = IPV6RSOLICIT;
+int ipv6_defhoplmt = MAXHOPLIMIT;
+int ipv6qmaxlen = IFQMAXLEN;
+
+#if __FreeBSD__
+SYSCTL_NODE(_net_inet, IPPROTO_IPV6, ipv6, CTLFLAG_RW, 0, "IPV6");
+SYSCTL_NODE(_net_inet, IPPROTO_ICMPV6, icmpv6, CTLFLAG_RW, 0, "ICMPV6");
+#endif /* __FreeBSD__ */