mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/stable/linux.git
synced 2024-11-01 00:48:50 +00:00
110 lines
3.2 KiB
C
110 lines
3.2 KiB
C
|
/* SPDX-License-Identifier: GPL-2.0-or-later */
|
||
|
|
||
|
#ifndef _NET_GSO_H
|
||
|
#define _NET_GSO_H
|
||
|
|
||
|
#include <linux/skbuff.h>
|
||
|
|
||
|
/* Keeps track of mac header offset relative to skb->head.
|
||
|
* It is useful for TSO of Tunneling protocol. e.g. GRE.
|
||
|
* For non-tunnel skb it points to skb_mac_header() and for
|
||
|
* tunnel skb it points to outer mac header.
|
||
|
* Keeps track of level of encapsulation of network headers.
|
||
|
*/
|
||
|
struct skb_gso_cb {
|
||
|
union {
|
||
|
int mac_offset;
|
||
|
int data_offset;
|
||
|
};
|
||
|
int encap_level;
|
||
|
__wsum csum;
|
||
|
__u16 csum_start;
|
||
|
};
|
||
|
#define SKB_GSO_CB_OFFSET 32
|
||
|
#define SKB_GSO_CB(skb) ((struct skb_gso_cb *)((skb)->cb + SKB_GSO_CB_OFFSET))
|
||
|
|
||
|
static inline int skb_tnl_header_len(const struct sk_buff *inner_skb)
|
||
|
{
|
||
|
return (skb_mac_header(inner_skb) - inner_skb->head) -
|
||
|
SKB_GSO_CB(inner_skb)->mac_offset;
|
||
|
}
|
||
|
|
||
|
static inline int gso_pskb_expand_head(struct sk_buff *skb, int extra)
|
||
|
{
|
||
|
int new_headroom, headroom;
|
||
|
int ret;
|
||
|
|
||
|
headroom = skb_headroom(skb);
|
||
|
ret = pskb_expand_head(skb, extra, 0, GFP_ATOMIC);
|
||
|
if (ret)
|
||
|
return ret;
|
||
|
|
||
|
new_headroom = skb_headroom(skb);
|
||
|
SKB_GSO_CB(skb)->mac_offset += (new_headroom - headroom);
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static inline void gso_reset_checksum(struct sk_buff *skb, __wsum res)
|
||
|
{
|
||
|
/* Do not update partial checksums if remote checksum is enabled. */
|
||
|
if (skb->remcsum_offload)
|
||
|
return;
|
||
|
|
||
|
SKB_GSO_CB(skb)->csum = res;
|
||
|
SKB_GSO_CB(skb)->csum_start = skb_checksum_start(skb) - skb->head;
|
||
|
}
|
||
|
|
||
|
/* Compute the checksum for a gso segment. First compute the checksum value
|
||
|
* from the start of transport header to SKB_GSO_CB(skb)->csum_start, and
|
||
|
* then add in skb->csum (checksum from csum_start to end of packet).
|
||
|
* skb->csum and csum_start are then updated to reflect the checksum of the
|
||
|
* resultant packet starting from the transport header-- the resultant checksum
|
||
|
* is in the res argument (i.e. normally zero or ~ of checksum of a pseudo
|
||
|
* header.
|
||
|
*/
|
||
|
static inline __sum16 gso_make_checksum(struct sk_buff *skb, __wsum res)
|
||
|
{
|
||
|
unsigned char *csum_start = skb_transport_header(skb);
|
||
|
int plen = (skb->head + SKB_GSO_CB(skb)->csum_start) - csum_start;
|
||
|
__wsum partial = SKB_GSO_CB(skb)->csum;
|
||
|
|
||
|
SKB_GSO_CB(skb)->csum = res;
|
||
|
SKB_GSO_CB(skb)->csum_start = csum_start - skb->head;
|
||
|
|
||
|
return csum_fold(csum_partial(csum_start, plen, partial));
|
||
|
}
|
||
|
|
||
|
struct sk_buff *__skb_gso_segment(struct sk_buff *skb,
|
||
|
netdev_features_t features, bool tx_path);
|
||
|
|
||
|
static inline struct sk_buff *skb_gso_segment(struct sk_buff *skb,
|
||
|
netdev_features_t features)
|
||
|
{
|
||
|
return __skb_gso_segment(skb, features, true);
|
||
|
}
|
||
|
|
||
|
struct sk_buff *skb_eth_gso_segment(struct sk_buff *skb,
|
||
|
netdev_features_t features, __be16 type);
|
||
|
|
||
|
struct sk_buff *skb_mac_gso_segment(struct sk_buff *skb,
|
||
|
netdev_features_t features);
|
||
|
|
||
|
bool skb_gso_validate_network_len(const struct sk_buff *skb, unsigned int mtu);
|
||
|
|
||
|
bool skb_gso_validate_mac_len(const struct sk_buff *skb, unsigned int len);
|
||
|
|
||
|
static inline void skb_gso_error_unwind(struct sk_buff *skb, __be16 protocol,
|
||
|
int pulled_hlen, u16 mac_offset,
|
||
|
int mac_len)
|
||
|
{
|
||
|
skb->protocol = protocol;
|
||
|
skb->encapsulation = 1;
|
||
|
skb_push(skb, pulled_hlen);
|
||
|
skb_reset_transport_header(skb);
|
||
|
skb->mac_header = mac_offset;
|
||
|
skb->network_header = skb->mac_header + mac_len;
|
||
|
skb->mac_len = mac_len;
|
||
|
}
|
||
|
|
||
|
#endif /* _NET_GSO_H */
|