Commit c5cbfc87 authored by Sven Eckelmann's avatar Sven Eckelmann Committed by Simon Wunderlich

batman-adv: Reserve needed_*room for fragments

The batadv net_device is trying to propagate the needed_headroom and
needed_tailroom from the lower devices. This is needed to avoid cost
intensive reallocations using pskb_expand_head during the transmission.

But the fragmentation code split the skb's without adding extra room at the
end/beginning of the various fragments. This reduced the performance of
transmissions over complex scenarios (batadv on vxlan on wireguard) because
the lower devices had to perform the reallocations at least once.

Fixes: ee75ed88 ("batman-adv: Fragment and send skbs larger than mtu")
Signed-off-by: default avatarSven Eckelmann <sven@narfation.org>
Signed-off-by: default avatarSimon Wunderlich <sw@simonwunderlich.de>
parent 4ca23e2c
...@@ -391,6 +391,7 @@ bool batadv_frag_skb_fwd(struct sk_buff *skb, ...@@ -391,6 +391,7 @@ bool batadv_frag_skb_fwd(struct sk_buff *skb,
/** /**
* batadv_frag_create() - create a fragment from skb * batadv_frag_create() - create a fragment from skb
* @net_dev: outgoing device for fragment
* @skb: skb to create fragment from * @skb: skb to create fragment from
* @frag_head: header to use in new fragment * @frag_head: header to use in new fragment
* @fragment_size: size of new fragment * @fragment_size: size of new fragment
...@@ -401,22 +402,25 @@ bool batadv_frag_skb_fwd(struct sk_buff *skb, ...@@ -401,22 +402,25 @@ bool batadv_frag_skb_fwd(struct sk_buff *skb,
* *
* Return: the new fragment, NULL on error. * Return: the new fragment, NULL on error.
*/ */
static struct sk_buff *batadv_frag_create(struct sk_buff *skb, static struct sk_buff *batadv_frag_create(struct net_device *net_dev,
struct sk_buff *skb,
struct batadv_frag_packet *frag_head, struct batadv_frag_packet *frag_head,
unsigned int fragment_size) unsigned int fragment_size)
{ {
unsigned int ll_reserved = LL_RESERVED_SPACE(net_dev);
unsigned int tailroom = net_dev->needed_tailroom;
struct sk_buff *skb_fragment; struct sk_buff *skb_fragment;
unsigned int header_size = sizeof(*frag_head); unsigned int header_size = sizeof(*frag_head);
unsigned int mtu = fragment_size + header_size; unsigned int mtu = fragment_size + header_size;
skb_fragment = netdev_alloc_skb(NULL, mtu + ETH_HLEN); skb_fragment = dev_alloc_skb(ll_reserved + mtu + tailroom);
if (!skb_fragment) if (!skb_fragment)
goto err; goto err;
skb_fragment->priority = skb->priority; skb_fragment->priority = skb->priority;
/* Eat the last mtu-bytes of the skb */ /* Eat the last mtu-bytes of the skb */
skb_reserve(skb_fragment, header_size + ETH_HLEN); skb_reserve(skb_fragment, ll_reserved + header_size);
skb_split(skb, skb_fragment, skb->len - fragment_size); skb_split(skb, skb_fragment, skb->len - fragment_size);
/* Add the header */ /* Add the header */
...@@ -439,11 +443,12 @@ int batadv_frag_send_packet(struct sk_buff *skb, ...@@ -439,11 +443,12 @@ int batadv_frag_send_packet(struct sk_buff *skb,
struct batadv_orig_node *orig_node, struct batadv_orig_node *orig_node,
struct batadv_neigh_node *neigh_node) struct batadv_neigh_node *neigh_node)
{ {
struct net_device *net_dev = neigh_node->if_incoming->net_dev;
struct batadv_priv *bat_priv; struct batadv_priv *bat_priv;
struct batadv_hard_iface *primary_if = NULL; struct batadv_hard_iface *primary_if = NULL;
struct batadv_frag_packet frag_header; struct batadv_frag_packet frag_header;
struct sk_buff *skb_fragment; struct sk_buff *skb_fragment;
unsigned int mtu = neigh_node->if_incoming->net_dev->mtu; unsigned int mtu = net_dev->mtu;
unsigned int header_size = sizeof(frag_header); unsigned int header_size = sizeof(frag_header);
unsigned int max_fragment_size, num_fragments; unsigned int max_fragment_size, num_fragments;
int ret; int ret;
...@@ -503,7 +508,7 @@ int batadv_frag_send_packet(struct sk_buff *skb, ...@@ -503,7 +508,7 @@ int batadv_frag_send_packet(struct sk_buff *skb,
goto put_primary_if; goto put_primary_if;
} }
skb_fragment = batadv_frag_create(skb, &frag_header, skb_fragment = batadv_frag_create(net_dev, skb, &frag_header,
max_fragment_size); max_fragment_size);
if (!skb_fragment) { if (!skb_fragment) {
ret = -ENOMEM; ret = -ENOMEM;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment