Source-Changes-HG archive
[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]
[src/trunk]: src/sys/dev/usb Support checksum offloading for mue(4) by Robert...
details: https://anonhg.NetBSD.org/src/rev/6f70a997c535
branches: trunk
changeset: 995185:6f70a997c535
user: rin <rin%NetBSD.org@localhost>
date: Tue Dec 11 08:16:57 2018 +0000
description:
Support checksum offloading for mue(4) by Robert Swindells
and adjusted by myself.
diffstat:
sys/dev/usb/if_mue.c | 72 +++++++++++++++++++++++++++++++++++++++++-------
sys/dev/usb/if_muevar.h | 10 ++++++-
2 files changed, 70 insertions(+), 12 deletions(-)
diffs (168 lines):
diff -r bdcdae6040e2 -r 6f70a997c535 sys/dev/usb/if_mue.c
--- a/sys/dev/usb/if_mue.c Tue Dec 11 06:34:00 2018 +0000
+++ b/sys/dev/usb/if_mue.c Tue Dec 11 08:16:57 2018 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: if_mue.c,v 1.19 2018/12/04 01:35:15 rin Exp $ */
+/* $NetBSD: if_mue.c,v 1.20 2018/12/11 08:16:57 rin Exp $ */
/* $OpenBSD: if_mue.c,v 1.3 2018/08/04 16:42:46 jsg Exp $ */
/*
@@ -20,7 +20,7 @@
/* Driver for Microchip LAN7500/LAN7800 chipsets. */
#include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: if_mue.c,v 1.19 2018/12/04 01:35:15 rin Exp $");
+__KERNEL_RCSID(0, "$NetBSD: if_mue.c,v 1.20 2018/12/11 08:16:57 rin Exp $");
#ifdef _KERNEL_OPT
#include "opt_usb.h"
@@ -1009,7 +1009,12 @@
IFQ_SET_READY(&ifp->if_snd);
- ifp->if_capabilities = IFCAP_TSOv4 | IFCAP_TSOv6;
+ ifp->if_capabilities = IFCAP_TSOv4 | IFCAP_TSOv6 |
+ IFCAP_CSUM_IPv4_Tx | IFCAP_CSUM_IPv4_Rx |
+ IFCAP_CSUM_TCPv4_Tx | IFCAP_CSUM_TCPv4_Rx |
+ IFCAP_CSUM_UDPv4_Tx | IFCAP_CSUM_UDPv4_Rx |
+ IFCAP_CSUM_TCPv6_Tx | IFCAP_CSUM_TCPv6_Rx |
+ IFCAP_CSUM_UDPv6_Tx | IFCAP_CSUM_UDPv6_Rx;
sc->mue_ec.ec_capabilities = ETHERCAP_VLAN_MTU;
@@ -1208,10 +1213,14 @@
usbd_status err;
struct mue_txbuf_hdr hdr;
uint32_t tx_cmd_a, tx_cmd_b;
- int len;
- bool tso;
+ int csum, len;
+ bool tso, ipe, tpe;
- tso = m->m_pkthdr.csum_flags & (M_CSUM_TSOv4 | M_CSUM_TSOv6);
+ csum = m->m_pkthdr.csum_flags;
+ tso = csum & (M_CSUM_TSOv4 | M_CSUM_TSOv6);
+ ipe = csum & M_CSUM_IPv4;
+ tpe = csum & (M_CSUM_TCPv4 | M_CSUM_UDPv4 |
+ M_CSUM_TCPv6 | M_CSUM_UDPv6);
len = m->m_pkthdr.len;
if (__predict_false((!tso && len > MUE_MAX_TX_LEN) ||
@@ -1234,8 +1243,13 @@
tx_cmd_b <<= MUE_TX_CMD_B_MSS_SHIFT;
KASSERT((tx_cmd_b & ~MUE_TX_CMD_B_MSS_MASK) == 0);
mue_tx_offload(sc, m);
- } else
+ } else {
+ if (ipe)
+ tx_cmd_a |= MUE_TX_CMD_A_IPE;
+ if (tpe)
+ tx_cmd_a |= MUE_TX_CMD_A_TPE;
tx_cmd_b = 0;
+ }
hdr.tx_cmd_a = htole32(tx_cmd_a);
hdr.tx_cmd_b = htole32(tx_cmd_b);
@@ -1418,7 +1432,9 @@
uint32_t rx_cmd_a, totlen;
uint16_t pktlen;
int s;
+ int csum;
char *buf = c->mue_buf;
+ bool v6;
if (__predict_false(sc->mue_dying)) {
DPRINTF(sc, "dying\n");
@@ -1457,15 +1473,17 @@
hdrp = (struct mue_rxbuf_hdr *)buf;
rx_cmd_a = le32toh(hdrp->rx_cmd_a);
- if (__predict_false(rx_cmd_a & MUE_RX_CMD_A_RED)) {
+ if (__predict_false(rx_cmd_a & MUE_RX_CMD_A_ERRORS)) {
+ /*
+ * We cannot use MUE_RX_CMD_A_RED bit here;
+ * it is turned on in the cases of L3/L4
+ * checksum errors which we handle below.
+ */
MUE_PRINTF(sc, "rx_cmd_a: 0x%x\n", rx_cmd_a);
ifp->if_ierrors++;
goto done;
}
- /* XXX not yet */
- KASSERT((rx_cmd_a & MUE_RX_CMD_A_ICSM) == 0);
-
pktlen = (uint16_t)(rx_cmd_a & MUE_RX_CMD_A_LEN_MASK);
if (sc->mue_flags & LAN7500)
pktlen -= 2;
@@ -1488,6 +1506,36 @@
m_set_rcvif(m, ifp);
m->m_pkthdr.len = m->m_len = pktlen;
m->m_flags |= M_HASFCS;
+
+ if (__predict_false(rx_cmd_a & MUE_RX_CMD_A_ICSM)) {
+ csum = 0;
+ } else {
+ v6 = rx_cmd_a & MUE_RX_CMD_A_IPV;
+ switch (rx_cmd_a & MUE_RX_CMD_A_PID) {
+ case MUE_RX_CMD_A_PID_TCP:
+ csum = v6 ?
+ M_CSUM_TCPv6 : M_CSUM_IPv4 | M_CSUM_TCPv4;
+ break;
+ case MUE_RX_CMD_A_PID_UDP:
+ csum = v6 ?
+ M_CSUM_UDPv6 : M_CSUM_IPv4 | M_CSUM_UDPv4;
+ break;
+ case MUE_RX_CMD_A_PID_IP:
+ csum = v6 ? 0 : M_CSUM_IPv4;
+ break;
+ default:
+ csum = 0;
+ break;
+ }
+ csum &= ifp->if_csum_flags_rx;
+ if (__predict_false((csum & M_CSUM_IPv4) &&
+ (rx_cmd_a & MUE_RX_CMD_A_ICE)))
+ csum |= M_CSUM_IPv4_BAD;
+ if (__predict_false((csum & ~M_CSUM_IPv4) &&
+ (rx_cmd_a & MUE_RX_CMD_A_TCE)))
+ csum |= M_CSUM_TCP_UDP_BAD;
+ }
+ m->m_pkthdr.csum_flags = csum;
memcpy(mtod(m, char *), buf + sizeof(*hdrp), pktlen);
/* Attention: sizeof(hdr) = 10 */
@@ -1643,6 +1691,8 @@
if ((error = ether_ioctl(ifp, cmd, data)) != ENETRESET)
break;
error = 0;
+ if (cmd == SIOCSIFCAP)
+ mue_sethwcsum(sc);
if (cmd == SIOCADDMULTI || cmd == SIOCDELMULTI)
mue_setmulti(sc);
break;
diff -r bdcdae6040e2 -r 6f70a997c535 sys/dev/usb/if_muevar.h
--- a/sys/dev/usb/if_muevar.h Tue Dec 11 06:34:00 2018 +0000
+++ b/sys/dev/usb/if_muevar.h Tue Dec 11 08:16:57 2018 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: if_muevar.h,v 1.2 2018/08/30 09:00:08 rin Exp $ */
+/* $NetBSD: if_muevar.h,v 1.3 2018/12/11 08:16:57 rin Exp $ */
/* $OpenBSD: if_muereg.h,v 1.1 2018/08/03 01:50:15 kevlo Exp $ */
/*
@@ -45,7 +45,15 @@
uint32_t rx_cmd_a;
#define MUE_RX_CMD_A_LEN_MASK 0x00003fff
#define MUE_RX_CMD_A_ICSM 0x00004000
+#define MUE_RX_CMD_A_ERRORS __BITS(16, 21) /* non-checksum errors */
#define MUE_RX_CMD_A_RED 0x00400000
+#define MUE_RX_CMD_A_PID __BITS(28, 27)
+#define MUE_RX_CMD_A_PID_TCP __SHIFTIN(1, MUE_RX_CMD_A_PID)
+#define MUE_RX_CMD_A_PID_UDP __SHIFTIN(2, MUE_RX_CMD_A_PID)
+#define MUE_RX_CMD_A_PID_IP __SHIFTIN(3, MUE_RX_CMD_A_PID)
+#define MUE_RX_CMD_A_IPV __BIT(29)
+#define MUE_RX_CMD_A_TCE __BIT(30)
+#define MUE_RX_CMD_A_ICE __BIT(31)
uint32_t rx_cmd_b;
uint16_t rx_cmd_c;
Home |
Main Index |
Thread Index |
Old Index