diff options
author | Sven Eckelmann <sven@narfation.org> | 2011-01-25 17:02:31 -0500 |
---|---|---|
committer | Sven Eckelmann <sven@narfation.org> | 2011-01-31 08:57:08 -0500 |
commit | ae361ce19fa135035c6b83ac1f07090b72fd4b8f (patch) | |
tree | 00410704b2dd5f0d7310f5cb9796385d7e8b7e91 /net/batman-adv/unicast.c | |
parent | 5c77d8bb8aeb4ec6804b6c32061109ba2ea6988d (diff) |
batman-adv: Calculate correct size for merged packets
The routing algorithm must be able to decide if a fragment can be merged with
the missing part and still be passed to a forwarding interface. The fragments
can only differ by one byte in case that the original payload had an uneven
length. In that situation the sender has to inform all possible receivers that
the tail is one byte longer using the flag UNI_FRAG_LARGETAIL.
The combination of UNI_FRAG_LARGETAIL and UNI_FRAG_HEAD flag makes it possible
to calculate the correct length for even and uneven sized payloads.
The original formula missed to add the unicast header at all and forgot to
remove the fragment header of the second fragment. This made the results highly
unreliable and only useful for machines with large differences between the
configured MTUs.
Reported-by: Russell Senior <russell@personaltelco.net>
Reported-by: Marek Lindner <lindner_marek@yahoo.de>
Signed-off-by: Sven Eckelmann <sven@narfation.org>
Diffstat (limited to 'net/batman-adv/unicast.c')
-rw-r--r-- | net/batman-adv/unicast.c | 8 |
1 files changed, 6 insertions, 2 deletions
diff --git a/net/batman-adv/unicast.c b/net/batman-adv/unicast.c index 811f7fc7932d..fc77079b18bb 100644 --- a/net/batman-adv/unicast.c +++ b/net/batman-adv/unicast.c | |||
@@ -225,6 +225,7 @@ int frag_send_skb(struct sk_buff *skb, struct bat_priv *bat_priv, | |||
225 | int uc_hdr_len = sizeof(struct unicast_packet); | 225 | int uc_hdr_len = sizeof(struct unicast_packet); |
226 | int ucf_hdr_len = sizeof(struct unicast_frag_packet); | 226 | int ucf_hdr_len = sizeof(struct unicast_frag_packet); |
227 | int data_len = skb->len - uc_hdr_len; | 227 | int data_len = skb->len - uc_hdr_len; |
228 | int large_tail = 0; | ||
228 | 229 | ||
229 | if (!bat_priv->primary_if) | 230 | if (!bat_priv->primary_if) |
230 | goto dropped; | 231 | goto dropped; |
@@ -254,8 +255,11 @@ int frag_send_skb(struct sk_buff *skb, struct bat_priv *bat_priv, | |||
254 | memcpy(frag1->orig, bat_priv->primary_if->net_dev->dev_addr, ETH_ALEN); | 255 | memcpy(frag1->orig, bat_priv->primary_if->net_dev->dev_addr, ETH_ALEN); |
255 | memcpy(frag2, frag1, sizeof(struct unicast_frag_packet)); | 256 | memcpy(frag2, frag1, sizeof(struct unicast_frag_packet)); |
256 | 257 | ||
257 | frag1->flags |= UNI_FRAG_HEAD; | 258 | if (data_len & 1) |
258 | frag2->flags &= ~UNI_FRAG_HEAD; | 259 | large_tail = UNI_FRAG_LARGETAIL; |
260 | |||
261 | frag1->flags = UNI_FRAG_HEAD | large_tail; | ||
262 | frag2->flags = large_tail; | ||
259 | 263 | ||
260 | frag1->seqno = htons((uint16_t)atomic_inc_return( | 264 | frag1->seqno = htons((uint16_t)atomic_inc_return( |
261 | &batman_if->frag_seqno)); | 265 | &batman_if->frag_seqno)); |