Commit 99eed284 authored by Sven Eckelmann's avatar Sven Eckelmann Committed by Greg Kroah-Hartman

Staging: batman-adv: Keep header writable and unshared

my_skb_push provided an easy way to allocate enough headroom in
situation were we don't have enough space left and move the data pointer
to the new position, but we didn't checked wether we are allowed to
write to the new pushed header. This is for example a problem when the
skb was cloned and thus doesn't have a private data part.

my_skb_head_push now replaces my_skb_push by using skb_cow_head to
provide only a large enough, writable header without testing for the
rest of the (maybe shared) data. It will also move the data pointer
using skb_push when skb_cow_head doesn't fail.

This should give us enough flexibility in situation were skbs will be
queued by underlying layers and still doesn't unnecessarily copy the
data in situations when the skb was consumed right away during
dev_queue_xmit.
Reported-by: default avatarMarek Lindner <lindner_marek@yahoo.de>
Signed-off-by: default avatarSven Eckelmann <sven.eckelmann@gmx.de>
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@suse.de>
parent 15cf5523
...@@ -73,7 +73,7 @@ int send_skb_packet(struct sk_buff *skb, ...@@ -73,7 +73,7 @@ int send_skb_packet(struct sk_buff *skb,
} }
/* push to the ethernet header. */ /* push to the ethernet header. */
if (my_skb_push(skb, sizeof(struct ethhdr)) < 0) if (my_skb_head_push(skb, sizeof(struct ethhdr)) < 0)
goto send_skb_err; goto send_skb_err;
skb_reset_mac_header(skb); skb_reset_mac_header(skb);
......
...@@ -31,8 +31,6 @@ ...@@ -31,8 +31,6 @@
static uint32_t bcast_seqno = 1; /* give own bcast messages seq numbers to avoid static uint32_t bcast_seqno = 1; /* give own bcast messages seq numbers to avoid
* broadcast storms */ * broadcast storms */
static int32_t skb_packets;
static int32_t skb_bad_packets;
unsigned char main_if_addr[ETH_ALEN]; unsigned char main_if_addr[ETH_ALEN];
static int bat_get_settings(struct net_device *dev, struct ethtool_cmd *cmd); static int bat_get_settings(struct net_device *dev, struct ethtool_cmd *cmd);
...@@ -59,18 +57,22 @@ void set_main_if_addr(uint8_t *addr) ...@@ -59,18 +57,22 @@ void set_main_if_addr(uint8_t *addr)
memcpy(main_if_addr, addr, ETH_ALEN); memcpy(main_if_addr, addr, ETH_ALEN);
} }
int my_skb_push(struct sk_buff *skb, unsigned int len) int my_skb_head_push(struct sk_buff *skb, unsigned int len)
{ {
int result = 0; int result;
skb_packets++; /**
if (skb_headroom(skb) < len) { * TODO: We must check if we can release all references to non-payload
skb_bad_packets++; * data using skb_header_release in our skbs to allow skb_cow_header to
result = pskb_expand_head(skb, len, 0, GFP_ATOMIC); * work optimally. This means that those skbs are not allowed to read
* or write any data which is before the current position of skb->data
* after that call and thus allow other skbs with the same data buffer
* to write freely in that area.
*/
result = skb_cow_head(skb, len);
if (result < 0) if (result < 0)
return result; return result;
}
skb_push(skb, len); skb_push(skb, len);
return 0; return 0;
...@@ -140,7 +142,7 @@ int interface_tx(struct sk_buff *skb, struct net_device *dev) ...@@ -140,7 +142,7 @@ int interface_tx(struct sk_buff *skb, struct net_device *dev)
/* ethernet packet should be broadcasted */ /* ethernet packet should be broadcasted */
if (is_bcast(ethhdr->h_dest) || is_mcast(ethhdr->h_dest)) { if (is_bcast(ethhdr->h_dest) || is_mcast(ethhdr->h_dest)) {
if (my_skb_push(skb, sizeof(struct bcast_packet)) < 0) if (my_skb_head_push(skb, sizeof(struct bcast_packet)) < 0)
goto dropped; goto dropped;
bcast_packet = (struct bcast_packet *)skb->data; bcast_packet = (struct bcast_packet *)skb->data;
......
...@@ -26,7 +26,7 @@ void set_main_if_addr(uint8_t *addr); ...@@ -26,7 +26,7 @@ void set_main_if_addr(uint8_t *addr);
void interface_setup(struct net_device *dev); void interface_setup(struct net_device *dev);
int interface_tx(struct sk_buff *skb, struct net_device *dev); int interface_tx(struct sk_buff *skb, struct net_device *dev);
void interface_rx(struct sk_buff *skb, int hdr_size); void interface_rx(struct sk_buff *skb, int hdr_size);
int my_skb_push(struct sk_buff *skb, unsigned int len); int my_skb_head_push(struct sk_buff *skb, unsigned int len);
extern unsigned char main_if_addr[]; extern unsigned char main_if_addr[];
......
...@@ -163,8 +163,8 @@ static int unicast_send_frag_skb(struct sk_buff *skb, struct bat_priv *bat_priv, ...@@ -163,8 +163,8 @@ static int unicast_send_frag_skb(struct sk_buff *skb, struct bat_priv *bat_priv,
frag_skb = dev_alloc_skb(data_len - (data_len / 2) + hdr_len); frag_skb = dev_alloc_skb(data_len - (data_len / 2) + hdr_len);
skb_split(skb, frag_skb, data_len / 2); skb_split(skb, frag_skb, data_len / 2);
if (my_skb_push(frag_skb, hdr_len) < 0 || if (my_skb_head_push(frag_skb, hdr_len) < 0 ||
my_skb_push(skb, hdr_len) < 0) my_skb_head_push(skb, hdr_len) < 0)
goto drop_frag; goto drop_frag;
ucast_frag1 = (struct unicast_frag_packet *)skb->data; ucast_frag1 = (struct unicast_frag_packet *)skb->data;
...@@ -240,7 +240,7 @@ int unicast_send_skb(struct sk_buff *skb, struct bat_priv *bat_priv) ...@@ -240,7 +240,7 @@ int unicast_send_skb(struct sk_buff *skb, struct bat_priv *bat_priv)
return unicast_send_frag_skb(skb, bat_priv, batman_if, return unicast_send_frag_skb(skb, bat_priv, batman_if,
dstaddr, orig_node); dstaddr, orig_node);
if (my_skb_push(skb, sizeof(struct unicast_packet)) < 0) if (my_skb_head_push(skb, sizeof(struct unicast_packet)) < 0)
goto dropped; goto dropped;
unicast_packet = (struct unicast_packet *)skb->data; unicast_packet = (struct unicast_packet *)skb->data;
......
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