Source-Changes-HG archive

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]

[src/trunk]: src/sys PR kern/53562



details:   https://anonhg.NetBSD.org/src/rev/62d3bb0f8e71
branches:  trunk
changeset: 995201:62d3bb0f8e71
user:      rin <rin%NetBSD.org@localhost>
date:      Wed Dec 12 01:40:20 2018 +0000

description:
PR kern/53562

Add ether_sw_offload_[tr]x: handle TX/RX offload options in software.
Since this violates separation b/w L2 and L3/L4, new files are added
rather than having the routines in sys/net/if_ethersubr.c.

OK msaitoh thorpej

diffstat:

 sys/net/ether_sw_offload.c       |  312 +++++++++++++++++++++++++++++++++++++++
 sys/net/ether_sw_offload.h       |   55 ++++++
 sys/net/files.net                |    3 +-
 sys/netinet/in_offload.c         |  135 +++++++++-------
 sys/netinet/in_offload.h         |    3 +-
 sys/netinet6/in6_offload.c       |  130 +++++++++------
 sys/netinet6/in6_offload.h       |    3 +-
 sys/rump/net/lib/libnet/Makefile |    3 +-
 8 files changed, 530 insertions(+), 114 deletions(-)

diffs (truncated from 885 to 300 lines):

diff -r 347f9fb412dd -r 62d3bb0f8e71 sys/net/ether_sw_offload.c
--- /dev/null   Thu Jan 01 00:00:00 1970 +0000
+++ b/sys/net/ether_sw_offload.c        Wed Dec 12 01:40:20 2018 +0000
@@ -0,0 +1,312 @@
+/*     $NetBSD: ether_sw_offload.c,v 1.1 2018/12/12 01:40:20 rin Exp $ */
+
+/*
+ * Copyright (c) 2018 The NetBSD Foundation, Inc.
+ * All rights reserved.
+ *
+ * This code is derived from software contributed to The NetBSD Foundation
+ * by Rin Okuyama.
+ *
+ * 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.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. 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 FOUNDATION 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.
+ */
+
+#include <sys/cdefs.h>
+__KERNEL_RCSID(0, "$NetBSD: ether_sw_offload.c,v 1.1 2018/12/12 01:40:20 rin Exp $");
+
+#include <sys/param.h>
+#include <sys/types.h>
+#include <sys/mbuf.h>
+
+#include <net/if.h>
+#include <net/if_ether.h>
+#include <net/ether_sw_offload.h>
+
+#include <netinet/in.h>
+#include <netinet/in_offload.h>
+#include <netinet/ip.h>
+#include <netinet/ip6.h>
+#include <netinet/tcp.h>
+#include <netinet/udp.h>
+
+#include <netinet6/in6.h>
+#include <netinet6/in6_offload.h>
+
+/*
+ * Handle TX offload in software. For TSO, split the packet into
+ * chanks with payloads of size MSS. For chekcsum offload, update
+ * required checksum fields. The results are more than one packet
+ * in general. Return a mbuf chain consists of them.
+ */
+
+struct mbuf *
+ether_sw_offload_tx(struct ifnet *ifp, struct mbuf *m)
+{
+       struct ether_header *ep;
+       int flags, ehlen;
+       bool v4;
+
+       KASSERT(m->m_flags & M_PKTHDR);
+       flags = m->m_pkthdr.csum_flags;
+       if (flags == 0)
+               goto done;
+
+       /* Sanity check */
+       if (!TX_OFFLOAD_SUPPORTED(ifp->if_csum_flags_tx, flags))
+               goto quit;
+
+       KASSERT(m->m_pkthdr.len >= sizeof(*ep));
+       if (m->m_len < sizeof(*ep)) {
+               m = m_pullup(m, sizeof(*ep));
+               if (m == NULL)
+                       return NULL;
+       }
+       ep = mtod(m, struct ether_header *);
+       switch (ntohs(ep->ether_type)) {
+       case ETHERTYPE_IP:
+       case ETHERTYPE_IPV6:
+               ehlen = ETHER_HDR_LEN;
+               break;
+       case ETHERTYPE_VLAN:
+               ehlen = ETHER_HDR_LEN + ETHER_VLAN_ENCAP_LEN;
+               break;
+       default:
+               goto quit;
+       }
+       KASSERT(m->m_pkthdr.len >= ehlen);
+
+       v4 = flags & (M_CSUM_TSOv4 | M_CSUM_IPv4 | M_CSUM_TCPv4 | M_CSUM_UDPv4);
+
+       if (flags & (M_CSUM_TSOv4 | M_CSUM_TSOv6)) {
+               /*
+                * tcp[46]_segment() assume that size of payloads is
+                * a multiple of MSS. Further, tcp6_segment() assumes
+                * no extention headers.
+                *
+                * XXX Do we need some KASSERT's?
+                */
+               if (v4)
+                       return tcp4_segment(m, ehlen);
+               else
+                       return tcp6_segment(m, ehlen);
+       }
+
+       if (v4)
+               in_undefer_cksum(m, ehlen, flags);
+       else
+               in6_undefer_cksum(m, ehlen, flags);
+done:
+       m->m_pkthdr.csum_flags = 0;
+       m->m_nextpkt = NULL;
+       return m;
+quit:
+       m_freem(m);
+       return NULL;
+}
+
+/*
+ * Handle RX offload in software.
+ *
+ * XXX Fragmented packets or packets with IPv6 extension headers
+ * are not currently supported.
+ */
+
+struct mbuf *
+ether_sw_offload_rx(struct ifnet *ifp, struct mbuf *m)
+{
+       struct ether_header *eh;
+       struct ip *ip;
+       struct ip6_hdr *ip6;
+       struct tcphdr *th;
+       struct udphdr *uh;
+       uint16_t sum, osum;
+       uint8_t proto;
+       int flags, enabled, len, ehlen, iphlen, l4offset;
+       bool v4;
+
+       flags = 0;
+
+       enabled = ifp->if_csum_flags_rx;
+       if (!(enabled & (M_CSUM_IPv4 | M_CSUM_TCPv4 | M_CSUM_UDPv4 |
+           M_CSUM_TCPv6 | M_CSUM_UDPv6)))
+               goto done;
+
+       KASSERT(m->m_flags & M_PKTHDR);
+       len = m->m_pkthdr.len;
+
+       KASSERT(len >= sizeof(*eh));
+       if (m->m_len < sizeof(*eh)) {
+               m = m_pullup(m, sizeof(*eh));
+               if (m == NULL)
+                       return NULL;
+       }
+       eh = mtod(m, struct ether_header *);
+       switch (htons(eh->ether_type)) {
+       case ETHERTYPE_IP:
+       case ETHERTYPE_IPV6:
+               ehlen = ETHER_HDR_LEN;
+               break;
+       case ETHERTYPE_VLAN:
+               ehlen = ETHER_HDR_LEN + ETHER_VLAN_ENCAP_LEN;
+               break;
+       default:
+               goto done;
+       }
+
+       KASSERT(len >= ehlen);
+       len = m->m_pkthdr.len - ehlen;
+
+       KASSERT(len >= sizeof(*ip));
+       if (m->m_len < ehlen + sizeof(*ip)) {
+               m = m_pullup(m, ehlen + sizeof(*ip));
+               if (m == NULL)
+                       return NULL;
+       }
+       ip = (void *)(mtod(m, char *) + ehlen);
+       v4 = (ip->ip_v == IPVERSION);
+
+       if (v4) {
+               if (enabled & M_CSUM_IPv4)
+                       flags |= M_CSUM_IPv4;
+
+               iphlen = ip->ip_hl << 2;
+               KASSERT(iphlen >= sizeof(*ip));
+
+               len -= iphlen;
+               KASSERT(len >= 0);
+
+               if (in4_cksum(m, 0, ehlen, iphlen) != 0) {
+                       if (enabled & M_CSUM_IPv4)
+                               flags |= M_CSUM_IPv4_BAD;
+                       /* Broken. Do not check further. */
+                       goto done;
+               }
+
+               /* Check if fragmented. */
+               if (ntohs(ip->ip_off) & ~(IP_DF | IP_RF))
+                       goto done;
+
+               proto = ip->ip_p;
+               switch (proto) {
+               case IPPROTO_TCP:
+                       if (!(enabled & M_CSUM_TCPv4))
+                               goto done;
+                       break;
+               case IPPROTO_UDP:
+                       if (!(enabled & M_CSUM_UDPv4))
+                               goto done;
+                       break;
+               default:
+                       goto done;
+               }
+
+               sum = in_cksum_phdr(ip->ip_src.s_addr, ip->ip_dst.s_addr,
+                   htons((uint16_t)len + proto));
+       } else {
+               KASSERT(len >= sizeof(*ip6));
+               if (m->m_len < ehlen + sizeof(*ip6)) {
+                       m = m_pullup(m, ehlen + sizeof(*ip6));
+                       if (m == NULL)
+                               return NULL;
+               }
+               ip6 = (void *)(mtod(m, char *) + ehlen);
+               KASSERT((ip6->ip6_vfc & IPV6_VERSION_MASK) == IPV6_VERSION);
+
+               iphlen = sizeof(*ip6);
+
+               len -= iphlen;
+
+               proto = ip6->ip6_nxt;
+               switch (proto) {
+               case IPPROTO_TCP:
+                       if (!(enabled & M_CSUM_TCPv6))
+                               goto done;
+                       break;
+               case IPPROTO_UDP:
+                       if (!(enabled & M_CSUM_UDPv6))
+                               goto done;
+                       break;
+               default:
+                       /* XXX Extension headers are not supported. */
+                       goto done;
+               }
+
+               sum = in6_cksum_phdr(&ip6->ip6_src, &ip6->ip6_dst, htonl(len),
+                   htonl(proto));
+       }
+
+       l4offset = ehlen + iphlen;
+       switch (proto) {
+       case IPPROTO_TCP:
+               KASSERT(len >= sizeof(*th));
+               if (m->m_len < l4offset + sizeof(*th)) {
+                       m = m_pullup(m, l4offset + sizeof(*th));
+                       if (m == NULL)
+                               return NULL;
+               }
+               th = (void *)(mtod(m, char *) + l4offset);
+               osum = th->th_sum;
+               th->th_sum = sum;
+               if (v4) {
+                       flags |= M_CSUM_TCPv4;
+                       sum = in4_cksum(m, 0, l4offset, len);
+               } else {
+                       flags |= M_CSUM_TCPv6;
+                       sum = in6_cksum(m, 0, l4offset, len);
+               }
+               if (sum != osum)
+                       flags |= M_CSUM_TCP_UDP_BAD;
+               th->th_sum = osum;
+               break;
+       case IPPROTO_UDP:
+               KASSERT(len >= sizeof(*uh));
+               if (m->m_len < l4offset + sizeof(*uh)) {
+                       m = m_pullup(m, l4offset + sizeof(*uh));
+                       if (m == NULL)
+                               return NULL;
+               }
+               uh = (void *)(mtod(m, char *) + l4offset);
+               osum = uh->uh_sum;
+               if (osum == 0)
+                       break;
+               uh->uh_sum = sum;
+               if (v4) {
+                       flags |= M_CSUM_UDPv4;
+                       sum = in4_cksum(m, 0, l4offset, len);
+               } else {
+                       flags |= M_CSUM_UDPv6;
+                       sum = in6_cksum(m, 0, l4offset, len);



Home | Main Index | Thread Index | Old Index