/* $NetBSD: ipcomp_input.c,v 1.36 2008/05/05 13:41:30 ad Exp $ */ /* $KAME: ipcomp_input.c,v 1.29 2001/09/04 08:43:19 itojun Exp $ */ /* * Copyright (C) 1999 WIDE Project. * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * 3. Neither the name of the project nor the names of its contributors * may be used to endorse or promote products derived from this software * without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE PROJECT AND CONTRIBUTORS ``AS IS'' AND * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE * ARE DISCLAIMED. IN NO EVENT SHALL THE PROJECT OR CONTRIBUTORS BE LIABLE * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF * SUCH DAMAGE. */ /* * RFC2393 IP payload compression protocol (IPComp). */ #include __KERNEL_RCSID(0, "$NetBSD: ipcomp_input.c,v 1.36 2008/05/05 13:41:30 ad Exp $"); #include "opt_inet.h" #include "opt_ipsec.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #ifdef INET6 #include #include #endif #include #include #include #include #include #include #include /*#define IPLEN_FLIPPED*/ #ifdef INET void ipcomp4_init(void) { ipsec4_init(); } void #if __STDC__ ipcomp4_input(struct mbuf *m, ...) #else ipcomp4_input(m, va_alist) struct mbuf *m; va_dcl #endif { struct mbuf *md; struct ip *ip; struct ipcomp *ipcomp; const struct ipcomp_algorithm *algo; u_int16_t cpi; /* host order */ u_int16_t nxt; size_t hlen; int error; size_t newlen, olen; struct secasvar *sav = NULL; int off, proto; va_list ap; u_int16_t sport = 0; u_int16_t dport = 0; #ifdef IPSEC_NAT_T struct m_tag *tag = NULL; #endif va_start(ap, m); off = va_arg(ap, int); proto = va_arg(ap, int); va_end(ap); if (m->m_pkthdr.len < off + sizeof(struct ipcomp)) { ipseclog((LOG_DEBUG, "IPv4 IPComp input: assumption failed " "(packet too short)\n")); IPSEC_STATINC(IPSEC_STAT_IN_INVAL); goto fail; } #ifdef IPSEC_NAT_T /* find the source port for NAT-T */ if ((tag = m_tag_find(m, PACKET_TAG_IPSEC_NAT_T_PORTS, NULL)) != NULL) { sport = ((u_int16_t *)(tag + 1))[0]; dport = ((u_int16_t *)(tag + 1))[1]; } #endif md = m_pulldown(m, off, sizeof(*ipcomp), NULL); if (!md) { m = NULL; /* already freed */ ipseclog((LOG_DEBUG, "IPv4 IPComp input: assumption failed " "(pulldown failure)\n")); IPSEC_STATINC(IPSEC_STAT_IN_INVAL); goto fail; } ipcomp = mtod(md, struct ipcomp *); ip = mtod(m, struct ip *); nxt = ipcomp->comp_nxt; hlen = ip->ip_hl << 2; cpi = ntohs(ipcomp->comp_cpi); if (cpi >= IPCOMP_CPI_NEGOTIATE_MIN) { sav = key_allocsa(AF_INET, (void *)&ip->ip_src, (void *)&ip->ip_dst, IPPROTO_IPCOMP, htonl(cpi), sport, dport); if (sav != NULL && (sav->state == SADB_SASTATE_MATURE || sav->state == SADB_SASTATE_DYING)) { cpi = sav->alg_enc; /* XXX */ /* other parameters to look at? */ } } algo = ipcomp_algorithm_lookup(cpi); if (!algo) { ipseclog((LOG_WARNING, "IPv4 IPComp input: unknown cpi %u\n", cpi)); IPSEC_STATINC(IPSEC_STAT_IN_NOSA); goto fail; } /* chop ipcomp header */ ipcomp = NULL; md->m_data += sizeof(struct ipcomp); md->m_len -= sizeof(struct ipcomp); m->m_pkthdr.len -= sizeof(struct ipcomp); #ifdef IPLEN_FLIPPED ip->ip_len -= sizeof(struct ipcomp); #else ip->ip_len = htons(ntohs(ip->ip_len) - sizeof(struct ipcomp)); #endif olen = m->m_pkthdr.len; newlen = m->m_pkthdr.len - off; error = (*algo->decompress)(m, m->m_next, &newlen); if (error != 0) { if (error == EINVAL) IPSEC_STATINC(IPSEC_STAT_IN_INVAL); else if (error == ENOBUFS) IPSEC_STATINC(IPSEC_STAT_IN_NOMEM); m = NULL; goto fail; } IPSEC_STATINC(IPSEC_STAT_IN_COMPHIST + cpi); /* * returning decompressed packet onto icmp is meaningless. * mark it decrypted to prevent icmp from attaching original packet. */ m->m_flags |= M_DECRYPTED; m->m_pkthdr.len = off + newlen; ip = mtod(m, struct ip *); { size_t len; #ifdef IPLEN_FLIPPED len = ip->ip_len; #else len = ntohs(ip->ip_len); #endif /* * be careful about underflow. also, do not assign exact value * as ip_len is manipulated differently on *BSDs. */ len += m->m_pkthdr.len; len -= olen; if (len & ~0xffff) { /* packet too big after decompress */ IPSEC_STATINC(IPSEC_STAT_IN_INVAL); goto fail; } #ifdef IPLEN_FLIPPED ip->ip_len = len & 0xffff; #else ip->ip_len = htons(len & 0xffff); #endif ip->ip_p = nxt; } if (sav) { key_sa_recordxfer(sav, m); if (ipsec_addhist(m, IPPROTO_IPCOMP, (u_int32_t)cpi) != 0) { IPSEC_STATINC(IPSEC_STAT_IN_NOMEM); goto fail; } key_freesav(sav); sav = NULL; } if (nxt != IPPROTO_DONE) { if ((inetsw[ip_protox[nxt]].pr_flags & PR_LASTHDR) != 0 && ipsec4_in_reject(m, NULL)) { IPSEC_STATINC(IPSEC_STAT_IN_POLVIO); goto fail; } (*inetsw[ip_protox[nxt]].pr_input)(m, off, nxt); } else m_freem(m); m = NULL; IPSEC_STATINC(IPSEC_STAT_IN_SUCCESS); return; fail: if (sav) key_freesav(sav); if (m) m_freem(m); return; } #endif /* INET */ #ifdef INET6 void ipcomp6_init(void) { ipsec6_init(); } int ipcomp6_input(struct mbuf **mp, int *offp, int proto) { struct mbuf *m, *md; int off; struct ip6_hdr *ip6; struct ipcomp *ipcomp; const struct ipcomp_algorithm *algo; u_int16_t cpi; /* host order */ u_int16_t nxt; int error; size_t newlen; struct secasvar *sav = NULL; u_int8_t *prvnxtp; m = *mp; off = *offp; md = m_pulldown(m, off, sizeof(*ipcomp), NULL); if (!md) { m = NULL; /* already freed */ ipseclog((LOG_DEBUG, "IPv6 IPComp input: assumption failed " "(pulldown failure)\n")); IPSEC6_STATINC(IPSEC_STAT_IN_INVAL); goto fail; } ipcomp = mtod(md, struct ipcomp *); ip6 = mtod(m, struct ip6_hdr *); nxt = ipcomp->comp_nxt; cpi = ntohs(ipcomp->comp_cpi); if (cpi >= IPCOMP_CPI_NEGOTIATE_MIN) { sav = key_allocsa(AF_INET6, (void *)&ip6->ip6_src, (void *)&ip6->ip6_dst, IPPROTO_IPCOMP, htonl(cpi), 0, 0); if (sav != NULL && (sav->state == SADB_SASTATE_MATURE || sav->state == SADB_SASTATE_DYING)) { cpi = sav->alg_enc; /* XXX */ /* other parameters to look at? */ } } algo = ipcomp_algorithm_lookup(cpi); if (!algo) { ipseclog((LOG_WARNING, "IPv6 IPComp input: unknown cpi %u; " "dropping the packet for simplicity\n", cpi)); IPSEC6_STATINC(IPSEC_STAT_IN_NOSA); goto fail; } /* chop ipcomp header */ ipcomp = NULL; md->m_data += sizeof(struct ipcomp); md->m_len -= sizeof(struct ipcomp); m->m_pkthdr.len -= sizeof(struct ipcomp); newlen = m->m_pkthdr.len - off; error = (*algo->decompress)(m, md, &newlen); if (error != 0) { if (error == EINVAL) IPSEC6_STATINC(IPSEC_STAT_IN_INVAL); else if (error == ENOBUFS) IPSEC6_STATINC(IPSEC_STAT_IN_NOMEM); m = NULL; goto fail; } IPSEC6_STATINC(IPSEC_STAT_IN_COMPHIST + cpi); m->m_pkthdr.len = off + newlen; /* * returning decompressed packet onto icmp is meaningless. * mark it decrypted to prevent icmp from attaching original packet. */ m->m_flags |= M_DECRYPTED; /* update next header field */ prvnxtp = ip6_get_prevhdr(m, off); *prvnxtp = nxt; /* * no need to adjust payload length, as all the IPv6 protocols * look at m->m_pkthdr.len */ if (sav) { key_sa_recordxfer(sav, m); if (ipsec_addhist(m, IPPROTO_IPCOMP, (u_int32_t)cpi) != 0) { IPSEC6_STATINC(IPSEC_STAT_IN_NOMEM); goto fail; } key_freesav(sav); sav = NULL; } *offp = off; *mp = m; IPSEC6_STATINC(IPSEC_STAT_IN_SUCCESS); return nxt; fail: if (m) m_freem(m); if (sav) key_freesav(sav); return IPPROTO_DONE; } #endif /* INET6 */