blob: 69b23cf6879e7b20ac9f6a2619623e7b2c609b1a [file] [log] [blame]
Otto Sabart1b23f5e2019-01-06 00:28:59 +01001.. SPDX-License-Identifier: GPL-2.0
2
Otto Sabartb83eb682019-01-06 00:29:28 +01003=================
4Checksum Offloads
5=================
Edward Creee8ae7b02016-02-11 21:03:37 +00006
7
8Introduction
9============
10
Otto Sabart1b23f5e2019-01-06 00:28:59 +010011This document describes a set of techniques in the Linux networking stack to
12take advantage of checksum offload capabilities of various NICs.
Edward Creee8ae7b02016-02-11 21:03:37 +000013
14The following technologies are described:
Otto Sabart1b23f5e2019-01-06 00:28:59 +010015
16* TX Checksum Offload
17* LCO: Local Checksum Offload
18* RCO: Remote Checksum Offload
Edward Creee8ae7b02016-02-11 21:03:37 +000019
20Things that should be documented here but aren't yet:
Otto Sabart1b23f5e2019-01-06 00:28:59 +010021
22* RX Checksum Offload
23* CHECKSUM_UNNECESSARY conversion
Edward Creee8ae7b02016-02-11 21:03:37 +000024
25
26TX Checksum Offload
27===================
28
Otto Sabart1b23f5e2019-01-06 00:28:59 +010029The interface for offloading a transmit checksum to a device is explained in
30detail in comments near the top of include/linux/skbuff.h.
31
Edward Creee8ae7b02016-02-11 21:03:37 +000032In brief, it allows to request the device fill in a single ones-complement
Otto Sabart1b23f5e2019-01-06 00:28:59 +010033checksum defined by the sk_buff fields skb->csum_start and skb->csum_offset.
34The device should compute the 16-bit ones-complement checksum (i.e. the
35'IP-style' checksum) from csum_start to the end of the packet, and fill in the
36result at (csum_start + csum_offset).
37
38Because csum_offset cannot be negative, this ensures that the previous value of
39the checksum field is included in the checksum computation, thus it can be used
40to supply any needed corrections to the checksum (such as the sum of the
41pseudo-header for UDP or TCP).
42
Edward Creee8ae7b02016-02-11 21:03:37 +000043This interface only allows a single checksum to be offloaded. Where
Otto Sabart1b23f5e2019-01-06 00:28:59 +010044encapsulation is used, the packet may have multiple checksum fields in
45different header layers, and the rest will have to be handled by another
46mechanism such as LCO or RCO.
47
Davide Caratti43c26a12017-05-18 15:44:41 +020048CRC32c can also be offloaded using this interface, by means of filling
Otto Sabart1b23f5e2019-01-06 00:28:59 +010049skb->csum_start and skb->csum_offset as described above, and setting
50skb->csum_not_inet: see skbuff.h comment (section 'D') for more details.
51
Edward Creee8ae7b02016-02-11 21:03:37 +000052No offloading of the IP header checksum is performed; it is always done in
Otto Sabart1b23f5e2019-01-06 00:28:59 +010053software. This is OK because when we build the IP header, we obviously have it
54in cache, so summing it isn't expensive. It's also rather short.
55
Edward Creee8ae7b02016-02-11 21:03:37 +000056The requirements for GSO are more complicated, because when segmenting an
Otto Sabart1b23f5e2019-01-06 00:28:59 +010057encapsulated packet both the inner and outer checksums may need to be edited or
58recomputed for each resulting segment. See the skbuff.h comment (section 'E')
59for more details.
Edward Creee8ae7b02016-02-11 21:03:37 +000060
61A driver declares its offload capabilities in netdev->hw_features; see
Mauro Carvalho Chehabea5baca2020-04-30 18:04:03 +020062Documentation/networking/netdev-features.rst for more. Note that a device
Otto Sabart1b23f5e2019-01-06 00:28:59 +010063which only advertises NETIF_F_IP[V6]_CSUM must still obey the csum_start and
64csum_offset given in the SKB; if it tries to deduce these itself in hardware
65(as some NICs do) the driver should check that the values in the SKB match
66those which the hardware will deduce, and if not, fall back to checksumming in
67software instead (with skb_csum_hwoffload_help() or one of the
68skb_checksum_help() / skb_crc32c_csum_help functions, as mentioned in
69include/linux/skbuff.h).
Edward Creee8ae7b02016-02-11 21:03:37 +000070
Otto Sabart1b23f5e2019-01-06 00:28:59 +010071The stack should, for the most part, assume that checksum offload is supported
72by the underlying device. The only place that should check is
73validate_xmit_skb(), and the functions it calls directly or indirectly. That
74function compares the offload features requested by the SKB (which may include
75other offloads besides TX Checksum Offload) and, if they are not supported or
76enabled on the device (determined by netdev->features), performs the
77corresponding offload in software. In the case of TX Checksum Offload, that
78means calling skb_csum_hwoffload_help(skb, features).
Edward Creee8ae7b02016-02-11 21:03:37 +000079
80
81LCO: Local Checksum Offload
82===========================
83
84LCO is a technique for efficiently computing the outer checksum of an
Otto Sabart1b23f5e2019-01-06 00:28:59 +010085encapsulated datagram when the inner checksum is due to be offloaded.
86
87The ones-complement sum of a correctly checksummed TCP or UDP packet is equal
88to the complement of the sum of the pseudo header, because everything else gets
89'cancelled out' by the checksum field. This is because the sum was
90complemented before being written to the checksum field.
91
Edward Creee8ae7b02016-02-11 21:03:37 +000092More generally, this holds in any case where the 'IP-style' ones complement
Otto Sabart1b23f5e2019-01-06 00:28:59 +010093checksum is used, and thus any checksum that TX Checksum Offload supports.
94
Edward Creee8ae7b02016-02-11 21:03:37 +000095That is, if we have set up TX Checksum Offload with a start/offset pair, we
Otto Sabart1b23f5e2019-01-06 00:28:59 +010096know that after the device has filled in that checksum, the ones complement sum
97from csum_start to the end of the packet will be equal to the complement of
98whatever value we put in the checksum field beforehand. This allows us to
99compute the outer checksum without looking at the payload: we simply stop
100summing when we get to csum_start, then add the complement of the 16-bit word
101at (csum_start + csum_offset).
102
Edward Creee8ae7b02016-02-11 21:03:37 +0000103Then, when the true inner checksum is filled in (either by hardware or by
Otto Sabart1b23f5e2019-01-06 00:28:59 +0100104skb_checksum_help()), the outer checksum will become correct by virtue of the
105arithmetic.
Edward Creee8ae7b02016-02-11 21:03:37 +0000106
107LCO is performed by the stack when constructing an outer UDP header for an
Otto Sabart1b23f5e2019-01-06 00:28:59 +0100108encapsulation such as VXLAN or GENEVE, in udp_set_csum(). Similarly for the
109IPv6 equivalents, in udp6_set_csum().
110
Edward Creee8ae7b02016-02-11 21:03:37 +0000111It is also performed when constructing an IPv4 GRE header, in
Otto Sabart1b23f5e2019-01-06 00:28:59 +0100112net/ipv4/ip_gre.c:build_header(). It is *not* currently performed when
113constructing an IPv6 GRE header; the GRE checksum is computed over the whole
114packet in net/ipv6/ip6_gre.c:ip6gre_xmit2(), but it should be possible to use
115LCO here as IPv6 GRE still uses an IP-style checksum.
116
Edward Creee8ae7b02016-02-11 21:03:37 +0000117All of the LCO implementations use a helper function lco_csum(), in
Otto Sabart1b23f5e2019-01-06 00:28:59 +0100118include/linux/skbuff.h.
Edward Creee8ae7b02016-02-11 21:03:37 +0000119
120LCO can safely be used for nested encapsulations; in this case, the outer
Otto Sabart1b23f5e2019-01-06 00:28:59 +0100121encapsulation layer will sum over both its own header and the 'middle' header.
122This does mean that the 'middle' header will get summed multiple times, but
123there doesn't seem to be a way to avoid that without incurring bigger costs
124(e.g. in SKB bloat).
Edward Creee8ae7b02016-02-11 21:03:37 +0000125
126
127RCO: Remote Checksum Offload
128============================
129
Otto Sabart1b23f5e2019-01-06 00:28:59 +0100130RCO is a technique for eliding the inner checksum of an encapsulated datagram,
131allowing the outer checksum to be offloaded. It does, however, involve a
132change to the encapsulation protocols, which the receiver must also support.
133For this reason, it is disabled by default.
134
Edward Creee8ae7b02016-02-11 21:03:37 +0000135RCO is detailed in the following Internet-Drafts:
Otto Sabart1b23f5e2019-01-06 00:28:59 +0100136
137* https://tools.ietf.org/html/draft-herbert-remotecsumoffload-00
138* https://tools.ietf.org/html/draft-herbert-vxlan-rco-00
139
140In Linux, RCO is implemented individually in each encapsulation protocol, and
141most tunnel types have flags controlling its use. For instance, VXLAN has the
142flag VXLAN_F_REMCSUM_TX (per struct vxlan_rdst) to indicate that RCO should be
143used when transmitting to a given remote destination.