net: dsa: tag_ocelot_8021q: calculate TX checksum in software for deferred packets
authorVladimir Oltean <vladimir.oltean@nxp.com>
Wed, 16 Feb 2022 14:30:14 +0000 (16:30 +0200)
committerDavid S. Miller <davem@davemloft.net>
Thu, 17 Feb 2022 14:06:51 +0000 (14:06 +0000)
DSA inherits NETIF_F_CSUM_MASK from master->vlan_features, and the
expectation is that TX checksumming is offloaded and not done in
software.

Normally the DSA master takes care of this, but packets handled by
ocelot_defer_xmit() are a very special exception, because they are
actually injected into the switch through register-based MMIO. So the
DSA master is not involved at all for these packets => no one calculates
the checksum.

This allows PTP over UDP to work using the ocelot-8021q tagging
protocol.

Signed-off-by: Vladimir Oltean <vladimir.oltean@nxp.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
net/dsa/tag_ocelot_8021q.c

index 68982b2789a59267362e035c12bee1650c6a474d..bd6f1d0e5372cbbc57ca49d49d31615c76162ab3 100644 (file)
@@ -32,6 +32,13 @@ static struct sk_buff *ocelot_defer_xmit(struct dsa_port *dp,
        if (!xmit_work_fn || !xmit_worker)
                return NULL;
 
+       /* PTP over IP packets need UDP checksumming. We may have inherited
+        * NETIF_F_HW_CSUM from the DSA master, but these packets are not sent
+        * through the DSA master, so calculate the checksum here.
+        */
+       if (skb->ip_summed == CHECKSUM_PARTIAL && skb_checksum_help(skb))
+               return NULL;
+
        xmit_work = kzalloc(sizeof(*xmit_work), GFP_ATOMIC);
        if (!xmit_work)
                return NULL;