diff options
Diffstat (limited to 'sys')
-rw-r--r-- | sys/net/if_enc.c | 269 | ||||
-rw-r--r-- | sys/net/if_enc.h | 7 |
2 files changed, 258 insertions, 18 deletions
diff --git a/sys/net/if_enc.c b/sys/net/if_enc.c index c6f3c95b39b..0e2ec2df115 100644 --- a/sys/net/if_enc.c +++ b/sys/net/if_enc.c @@ -1,4 +1,4 @@ -/* $OpenBSD: if_enc.c,v 1.13 1999/11/02 00:15:56 angelos Exp $ */ +/* $OpenBSD: if_enc.c,v 1.14 1999/12/27 03:06:40 angelos Exp $ */ /* * The authors of this code are John Ioannidis (ji@tla.org), @@ -44,8 +44,7 @@ #include <sys/socket.h> #include <sys/errno.h> #include <sys/ioctl.h> -#include <sys/time.h> -#include <machine/cpu.h> +#include <sys/proc.h> #include <net/if.h> #include <net/if_types.h> @@ -60,9 +59,16 @@ #include <netinet/in.h> #include <netinet/in_systm.h> #include <netinet/in_var.h> +#include <netinet/ip_var.h> #include <netinet/ip.h> #endif +#ifdef INET6 +#include <netinet6/in6.h> +#include <netinet6/ip6.h> +#include <netinet6/ip6_var.h> +#endif /* INET6 */ + #ifdef ISO extern struct ifqueue clnlintrq; #endif @@ -74,6 +80,16 @@ extern struct ifqueue nsintrq; #include "bpfilter.h" #include "enc.h" +#ifdef ENCDEBUG +#define DPRINTF(x) do { if (encdebug) printf x ; } while (0) +#else +#define DPRINTF(x) +#endif + +#ifndef offsetof +#define offsetof(s, e) ((int)&((s *)0)->e) +#endif + struct enc_softc encif[NENC]; void encattach __P((int)); @@ -93,7 +109,8 @@ encattach(int nenc) bzero(encif, sizeof(encif)); - for (i = 0; i < NENC; i++) { + for (i = 0; i < NENC; i++) + { ifp = &encif[i].sc_if; sprintf(ifp->if_xname, "enc%d", i); ifp->if_softc = &encif[i]; @@ -119,7 +136,136 @@ void encstart(ifp) struct ifnet *ifp; { - /* XXX Code needed */ + struct enc_softc *enc = ifp->if_softc; + int s, err = 0, protoflag; + struct mbuf *m, *mp; + struct tdb *tdb; + + /* If the interface is not setup, flush the queue */ + if ((enc->sc_spi == 0) && (enc->sc_sproto == 0) && + ((enc->sc_dst.sa.sa_family == AF_INET) || + (enc->sc_dst.sa.sa_family == AF_INET6))) + { + DPRINTF(("%s: not initialized with SA\n", ifp->if_xname)); + + for (;;) + { + s = splimp(); + IF_DEQUEUE(&ifp->if_snd, m); + splx(s); + if (m == NULL) + return; + else + m_freem(m); + } + + /* Unreachable */ + } + + /* Find what type of processing we need to do */ + tdb = gettdb(enc->sc_spi, &(enc->sc_dst), enc->sc_sproto); + if (tdb == NULL) + { + DPRINTF(("%s: SA non-existant\n", ifp->if_xname)); + + /* Flush the queue */ + for (;;) + { + s = splimp(); + IF_DEQUEUE(&ifp->if_snd, m); + splx(s); + if (m == NULL) + return; + else + m_freem(m); + } + } + + /* See if we need to notify a key mgmt. daemon to setup SAs */ + if (ntohl(enc->sc_spi) == SPI_LOCAL_USE) + { + /* + * XXX Can't do this for now, as there's no way for + * XXX key mgmt. to specify link-layer properties + * XXX (e.g., encrypt everything on this interface) + */ +#ifdef notyet + if (tdb->tdb_satype != SADB_X_SATYPE_BYPASS) + pfkeyv2_acquire(tdb, 0); /* No point checking for errors */ +#endif + + /* Flush the queue */ + for (;;) + { + s = splimp(); + IF_DEQUEUE(&ifp->if_snd, m); + splx(s); + if (m == NULL) + return; + else + m_freem(m); + } + + /* Unreachable */ + } + + /* IPsec-process all packets in the queue */ + for (;;) + { + /* Get a packet from the queue */ + s = splimp(); + IF_DEQUEUE(&ifp->if_snd, m); + splx(s); + + if (m == NULL) /* Empty queue */ + return; + + /* First, we encapsulate in etherip */ + err = etherip_output(m, tdb, &mp, 0, 0); /* Last 2 args not used */ + if ((mp == NULL) || err) + { + /* Just skip this frame */ + if (mp) + m_freem(mp); + continue; + } + else + { + m = mp; + mp = NULL; + } + + protoflag = tdb->tdb_dst.sa.sa_family; + + /* IPsec packet processing -- skip encapsulation */ + err = ipsp_process_packet(m, &mp, tdb, &protoflag, 1); + if ((mp == NULL) || err) + { + if (mp) + m_freem(mp); + continue; + } + else + { + m = mp; + mp = NULL; + } + +#ifdef INET + /* Send the packet on its way, no point checking for errors here */ + if (protoflag == AF_INET) + ip_output(m, NULL, NULL, IP_ENCAPSULATED | IP_RAWOUTPUT, NULL, NULL); +#endif /* INET */ + +#ifdef INET6 + /* Send the packet on its way, no point checking for errors here */ + if (protoflag == AF_INET6) + ip6_output(m, NULL, NULL, IP_ENCAPSULATED | IP_RAWOUTPUT, + NULL, NULL); +#endif /* INET6 */ + + /* XXX Should find a way to avoid bridging-loops, some mbuf flag ? */ + } } /* @@ -141,7 +287,8 @@ register struct rtentry *rt; ifp->if_lastchange = time; m->m_pkthdr.rcvif = ifp; - if (rt && rt->rt_flags & (RTF_REJECT|RTF_BLACKHOLE)) { + if (rt && rt->rt_flags & (RTF_REJECT|RTF_BLACKHOLE)) + { m_freem(m); return (rt->rt_flags & RTF_BLACKHOLE ? 0 : rt->rt_flags & RTF_HOST ? EHOSTUNREACH : ENETUNREACH); @@ -150,13 +297,20 @@ register struct rtentry *rt; ifp->if_opackets++; ifp->if_obytes += m->m_pkthdr.len; - switch (dst->sa_family) { + switch (dst->sa_family) + { #ifdef INET case AF_INET: ifq = &ipintrq; isr = NETISR_IP; break; #endif +#ifdef INET6 + case AF_INET6: + ifq = &ip6intrq; + isr = NETISR_IPV6; + break; +#endif #ifdef NS case AF_NS: ifq = &nsintrq; @@ -175,7 +329,8 @@ register struct rtentry *rt; } s = splimp(); - if (IF_QFULL(ifq)) { + if (IF_QFULL(ifq)) + { IF_DROP(ifq); m_freem(m); splx(s); @@ -210,21 +365,99 @@ register struct ifnet *ifp; u_long cmd; caddr_t data; { - register struct ifaddr *ifa; - register int error = 0; - - switch (cmd) { - case SIOCSIFADDR: - /* - * Everything else is done at a higher level. - */ - ifp->if_flags |= IFF_UP; - ifa = (struct ifaddr *) data; + struct enc_softc *enc = (struct enc_softc *) ifp->if_softc; + struct ifsa *ifsa = (struct ifsa *) data; + struct proc *prc = curproc; /* XXX */ + struct tdb *tdb; + int s, error; + + switch (cmd) + { + case SIOCGENCSA: + ifsa->sa_spi = enc->sc_spi; + ifsa->sa_proto = enc->sc_sproto; + bcopy(&enc->sc_dst, &ifsa->sa_dst, enc->sc_dst.sa.sa_len); + break; + + case SIOCSENCSA: + /* Check for superuser */ + if ((error = suser(prc->p_ucred, &prc->p_acflag)) != 0) + break; + + /* Check for valid TDB */ + s = spltdb(); + + /* Clear interface pointer in pre-existing TDB */ + if (enc->sc_sproto != 0) + { + tdb = gettdb(enc->sc_spi, &enc->sc_dst, enc->sc_sproto); + if (tdb) + tdb->tdb_interface = NULL; + } + + if (ifsa->sa_proto != 0) + { + tdb = gettdb(ifsa->sa_spi, &ifsa->sa_dst, ifsa->sa_proto); + if (tdb == NULL) + { + splx(s); + error = ENOENT; + break; + } + } + else + tdb = NULL; + + /* Clear SA if requested */ + if ((ifsa->sa_spi == 0) && (ifsa->sa_proto == 0)) + { + bzero(&enc->sc_dst, sizeof(union sockaddr_union)); + enc->sc_spi = 0; + enc->sc_sproto = 0; + + splx(s); + break; + } + + if (tdb == NULL) + { + splx(s); + error = ENOENT; + break; + } + +#ifdef INET + if ((ifsa->sa_dst.sa.sa_family == AF_INET) && + (ifsa->sa_dst.sa.sa_len != sizeof(struct sockaddr_in))) + { + splx(s); + error = EINVAL; + break; + } +#endif /* INET */ + +#ifdef INET6 + if ((ifsa->sa_dst.sa.sa_family == AF_INET6) && + (ifsa->sa_dst.sa.sa_len != sizeof(struct sockaddr_in6))) + { + splx(s); + error = EINVAL; + break; + } +#endif /* INET6 */ + + bcopy(&ifsa->sa_dst, &enc->sc_dst, ifsa->sa_dst.sa.sa_len); + enc->sc_spi = ifsa->sa_spi; + enc->sc_sproto = ifsa->sa_proto; + tdb->tdb_interface = (caddr_t) ifp; + + splx(s); break; default: error = EINVAL; break; } + return (error); } diff --git a/sys/net/if_enc.h b/sys/net/if_enc.h index bfae0a16fb4..ea0d41c5ac9 100644 --- a/sys/net/if_enc.h +++ b/sys/net/if_enc.h @@ -44,6 +44,13 @@ struct enc_softc { u_int32_t sc_sproto; }; +struct ifsa { + char sa_ifname[IFNAMSIZ]; /* bridge ifs name */ + u_int32_t sa_spi; + u_int8_t sa_proto; + union sockaddr_union sa_dst; +}; + struct enchdr { u_int32_t af; u_int32_t spi; |