Commit 49b588e7 authored by Andreas Langer's avatar Andreas Langer Committed by Greg Kroah-Hartman

Staging: batman-adv: restructure fragmentation to handle batman unicast packets

The unicast_frag_send_skb() function expected 'raw' packets (without any
batman-adv header) to fragment them. This needs to be changed, so that
this function is able to fragment packets that already traveled inside
the mesh but need to be fragmented now.
Signed-off-by: default avatarAndreas Langer <an.langer@gmx.de>
Signed-off-by: default avatarSven Eckelmann <sven.eckelmann@gmx.de>
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@suse.de>
parent 3f427604
...@@ -152,55 +152,58 @@ void frag_list_free(struct list_head *head) ...@@ -152,55 +152,58 @@ void frag_list_free(struct list_head *head)
return; return;
} }
static int unicast_send_frag_skb(struct sk_buff *skb, struct bat_priv *bat_priv, static int frag_send_skb(struct sk_buff *skb, struct bat_priv *bat_priv,
struct batman_if *batman_if, uint8_t dstaddr[], struct batman_if *batman_if, uint8_t dstaddr[])
struct orig_node *orig_node)
{ {
struct unicast_frag_packet *ucast_frag1, *ucast_frag2; struct unicast_packet tmp_uc, *unicast_packet;
int hdr_len = sizeof(struct unicast_frag_packet);
struct sk_buff *frag_skb; struct sk_buff *frag_skb;
struct unicast_frag_packet *frag1, *frag2;
int uc_hdr_len = sizeof(struct unicast_packet);
int ucf_hdr_len = sizeof(struct unicast_frag_packet);
int data_len = skb->len; int data_len = skb->len;
if (!bat_priv->primary_if) if (!bat_priv->primary_if)
goto dropped; goto dropped;
frag_skb = dev_alloc_skb(data_len - (data_len / 2) + hdr_len); unicast_packet = (struct unicast_packet *) skb->data;
memcpy(&tmp_uc, unicast_packet, uc_hdr_len);
frag_skb = dev_alloc_skb(data_len - (data_len / 2) + ucf_hdr_len);
skb_split(skb, frag_skb, data_len / 2); skb_split(skb, frag_skb, data_len / 2);
if (my_skb_head_push(frag_skb, hdr_len) < 0 || if (my_skb_head_push(skb, ucf_hdr_len - uc_hdr_len) < 0 ||
my_skb_head_push(skb, hdr_len) < 0) my_skb_head_push(frag_skb, ucf_hdr_len) < 0)
goto drop_frag; goto drop_frag;
ucast_frag1 = (struct unicast_frag_packet *)skb->data; frag1 = (struct unicast_frag_packet *)skb->data;
ucast_frag2 = (struct unicast_frag_packet *)frag_skb->data; frag2 = (struct unicast_frag_packet *)frag_skb->data;
ucast_frag1->version = COMPAT_VERSION; memcpy(frag1, &tmp_uc, sizeof(struct unicast_packet));
ucast_frag1->packet_type = BAT_UNICAST_FRAG;
ucast_frag1->ttl = TTL;
memcpy(ucast_frag1->orig,
bat_priv->primary_if->net_dev->dev_addr, ETH_ALEN);
memcpy(ucast_frag1->dest, orig_node->orig, ETH_ALEN);
memcpy(ucast_frag2, ucast_frag1, sizeof(struct unicast_frag_packet)); frag1->ttl--;
frag1->version = COMPAT_VERSION;
frag1->packet_type = BAT_UNICAST_FRAG;
ucast_frag1->flags |= UNI_FRAG_HEAD; memcpy(frag1->orig, bat_priv->primary_if->net_dev->dev_addr, ETH_ALEN);
ucast_frag2->flags &= ~UNI_FRAG_HEAD; memcpy(frag2, frag1, sizeof(struct unicast_frag_packet));
ucast_frag1->seqno = htons((uint16_t)atomic_inc_return( frag1->flags |= UNI_FRAG_HEAD;
&batman_if->frag_seqno)); frag2->flags &= ~UNI_FRAG_HEAD;
ucast_frag2->seqno = htons((uint16_t)atomic_inc_return( frag1->seqno = htons((uint16_t)atomic_inc_return(
&batman_if->frag_seqno)); &batman_if->frag_seqno));
frag2->seqno = htons((uint16_t)atomic_inc_return(
&batman_if->frag_seqno));
send_skb_packet(skb, batman_if, dstaddr); send_skb_packet(skb, batman_if, dstaddr);
send_skb_packet(frag_skb, batman_if, dstaddr); send_skb_packet(frag_skb, batman_if, dstaddr);
return 0; return NET_RX_SUCCESS;
drop_frag: drop_frag:
kfree_skb(frag_skb); kfree_skb(frag_skb);
dropped: dropped:
kfree_skb(skb); kfree_skb(skb);
return 1; return NET_RX_DROP;
} }
int unicast_send_skb(struct sk_buff *skb, struct bat_priv *bat_priv) int unicast_send_skb(struct sk_buff *skb, struct bat_priv *bat_priv)
...@@ -240,11 +243,6 @@ int unicast_send_skb(struct sk_buff *skb, struct bat_priv *bat_priv) ...@@ -240,11 +243,6 @@ int unicast_send_skb(struct sk_buff *skb, struct bat_priv *bat_priv)
if (batman_if->if_status != IF_ACTIVE) if (batman_if->if_status != IF_ACTIVE)
goto dropped; goto dropped;
if (atomic_read(&bat_priv->frag_enabled) &&
data_len + sizeof(struct unicast_packet) > batman_if->net_dev->mtu)
return unicast_send_frag_skb(skb, bat_priv, batman_if,
dstaddr, orig_node);
if (my_skb_head_push(skb, sizeof(struct unicast_packet)) < 0) if (my_skb_head_push(skb, sizeof(struct unicast_packet)) < 0)
goto dropped; goto dropped;
...@@ -258,6 +256,14 @@ int unicast_send_skb(struct sk_buff *skb, struct bat_priv *bat_priv) ...@@ -258,6 +256,14 @@ int unicast_send_skb(struct sk_buff *skb, struct bat_priv *bat_priv)
/* copy the destination for faster routing */ /* copy the destination for faster routing */
memcpy(unicast_packet->dest, orig_node->orig, ETH_ALEN); memcpy(unicast_packet->dest, orig_node->orig, ETH_ALEN);
if (atomic_read(&bat_priv->frag_enabled) &&
data_len + sizeof(struct unicast_packet) >
batman_if->net_dev->mtu) {
/* send frag skb decreases ttl */
unicast_packet->ttl++;
return frag_send_skb(skb, bat_priv, batman_if,
dstaddr);
}
send_skb_packet(skb, batman_if, dstaddr); send_skb_packet(skb, batman_if, dstaddr);
return 0; return 0;
......
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