batman-adv: use vlan_/eth_hdr() instead of skb->data in interface_tx path

Our .ndo_start_xmit handler (batadv_interface_tx()) can rely on having
the skb mac header pointer set correctly since the following commit
present in kernels >= 3.9:

"net: reset mac header in dev_start_xmit()" (6d1ccff627)

Therefore we can safely use eth_hdr() and vlan_eth_hdr() instead of
skb->data now, which spares us some ugly type casts.

At the same time set the mac_header in batadv_dat_snoop_incoming_arp_request()
before sending the skb along the TX path.

Signed-off-by: Linus Lüssing <linus.luessing@web.de>
Signed-off-by: Marek Lindner <mareklindner@neomailbox.ch>
Signed-off-by: Antonio Quartulli <antonio@meshcoding.com>
This commit is contained in:
Linus Lüssing 2014-01-19 22:22:45 +01:00 committed by Antonio Quartulli
parent abae9479ca
commit 927c2ed7e5
5 changed files with 18 additions and 14 deletions

View File

@ -882,7 +882,7 @@ static int batadv_bla_process_claim(struct batadv_priv *bat_priv,
proto = ethhdr->h_proto; proto = ethhdr->h_proto;
headlen = ETH_HLEN; headlen = ETH_HLEN;
if (vid & BATADV_VLAN_HAS_TAG) { if (vid & BATADV_VLAN_HAS_TAG) {
vhdr = (struct vlan_ethhdr *)ethhdr; vhdr = vlan_eth_hdr(skb);
proto = vhdr->h_vlan_encapsulated_proto; proto = vhdr->h_vlan_encapsulated_proto;
headlen += VLAN_HLEN; headlen += VLAN_HLEN;
} }

View File

@ -1027,6 +1027,11 @@ bool batadv_dat_snoop_incoming_arp_request(struct batadv_priv *bat_priv,
if (!skb_new) if (!skb_new)
goto out; goto out;
/* the rest of the TX path assumes that the mac_header offset pointing
* to the inner Ethernet header has been set, therefore reset it now.
*/
skb_reset_mac_header(skb_new);
if (vid & BATADV_VLAN_HAS_TAG) if (vid & BATADV_VLAN_HAS_TAG)
skb_new = vlan_insert_tag(skb_new, htons(ETH_P_8021Q), skb_new = vlan_insert_tag(skb_new, htons(ETH_P_8021Q),
vid & VLAN_VID_MASK); vid & VLAN_VID_MASK);

View File

@ -678,7 +678,7 @@ batadv_gw_dhcp_recipient_get(struct sk_buff *skb, unsigned int *header_len,
if (!pskb_may_pull(skb, *header_len + ETH_HLEN)) if (!pskb_may_pull(skb, *header_len + ETH_HLEN))
return BATADV_DHCP_NO; return BATADV_DHCP_NO;
ethhdr = (struct ethhdr *)skb->data; ethhdr = eth_hdr(skb);
proto = ethhdr->h_proto; proto = ethhdr->h_proto;
*header_len += ETH_HLEN; *header_len += ETH_HLEN;
@ -687,7 +687,7 @@ batadv_gw_dhcp_recipient_get(struct sk_buff *skb, unsigned int *header_len,
if (!pskb_may_pull(skb, *header_len + VLAN_HLEN)) if (!pskb_may_pull(skb, *header_len + VLAN_HLEN))
return BATADV_DHCP_NO; return BATADV_DHCP_NO;
vhdr = (struct vlan_ethhdr *)skb->data; vhdr = vlan_eth_hdr(skb);
proto = vhdr->h_vlan_encapsulated_proto; proto = vhdr->h_vlan_encapsulated_proto;
*header_len += VLAN_HLEN; *header_len += VLAN_HLEN;
} }
@ -726,7 +726,7 @@ batadv_gw_dhcp_recipient_get(struct sk_buff *skb, unsigned int *header_len,
return BATADV_DHCP_NO; return BATADV_DHCP_NO;
/* skb->data might have been reallocated by pskb_may_pull() */ /* skb->data might have been reallocated by pskb_may_pull() */
ethhdr = (struct ethhdr *)skb->data; ethhdr = eth_hdr(skb);
if (ntohs(ethhdr->h_proto) == ETH_P_8021Q) if (ntohs(ethhdr->h_proto) == ETH_P_8021Q)
ethhdr = (struct ethhdr *)(skb->data + VLAN_HLEN); ethhdr = (struct ethhdr *)(skb->data + VLAN_HLEN);

View File

@ -256,7 +256,7 @@ static int batadv_send_skb_unicast(struct batadv_priv *bat_priv,
{ {
struct ethhdr *ethhdr; struct ethhdr *ethhdr;
struct batadv_unicast_packet *unicast_packet; struct batadv_unicast_packet *unicast_packet;
int ret = NET_XMIT_DROP, hdr_size; int ret = NET_XMIT_DROP;
if (!orig_node) if (!orig_node)
goto out; goto out;
@ -265,16 +265,12 @@ static int batadv_send_skb_unicast(struct batadv_priv *bat_priv,
case BATADV_UNICAST: case BATADV_UNICAST:
if (!batadv_send_skb_prepare_unicast(skb, orig_node)) if (!batadv_send_skb_prepare_unicast(skb, orig_node))
goto out; goto out;
hdr_size = sizeof(*unicast_packet);
break; break;
case BATADV_UNICAST_4ADDR: case BATADV_UNICAST_4ADDR:
if (!batadv_send_skb_prepare_unicast_4addr(bat_priv, skb, if (!batadv_send_skb_prepare_unicast_4addr(bat_priv, skb,
orig_node, orig_node,
packet_subtype)) packet_subtype))
goto out; goto out;
hdr_size = sizeof(struct batadv_unicast_4addr_packet);
break; break;
default: default:
/* this function supports UNICAST and UNICAST_4ADDR only. It /* this function supports UNICAST and UNICAST_4ADDR only. It
@ -283,7 +279,10 @@ static int batadv_send_skb_unicast(struct batadv_priv *bat_priv,
goto out; goto out;
} }
ethhdr = (struct ethhdr *)(skb->data + hdr_size); /* skb->data might have been reallocated by
* batadv_send_skb_prepare_unicast{,_4addr}()
*/
ethhdr = eth_hdr(skb);
unicast_packet = (struct batadv_unicast_packet *)skb->data; unicast_packet = (struct batadv_unicast_packet *)skb->data;
/* inform the destination node that we are still missing a correct route /* inform the destination node that we are still missing a correct route

View File

@ -176,11 +176,11 @@ static int batadv_interface_tx(struct sk_buff *skb,
soft_iface->trans_start = jiffies; soft_iface->trans_start = jiffies;
vid = batadv_get_vid(skb, 0); vid = batadv_get_vid(skb, 0);
ethhdr = (struct ethhdr *)skb->data; ethhdr = eth_hdr(skb);
switch (ntohs(ethhdr->h_proto)) { switch (ntohs(ethhdr->h_proto)) {
case ETH_P_8021Q: case ETH_P_8021Q:
vhdr = (struct vlan_ethhdr *)skb->data; vhdr = vlan_eth_hdr(skb);
if (vhdr->h_vlan_encapsulated_proto != ethertype) if (vhdr->h_vlan_encapsulated_proto != ethertype)
break; break;
@ -194,7 +194,7 @@ static int batadv_interface_tx(struct sk_buff *skb,
goto dropped; goto dropped;
/* skb->data might have been reallocated by batadv_bla_tx() */ /* skb->data might have been reallocated by batadv_bla_tx() */
ethhdr = (struct ethhdr *)skb->data; ethhdr = eth_hdr(skb);
/* Register the client MAC in the transtable */ /* Register the client MAC in the transtable */
if (!is_multicast_ether_addr(ethhdr->h_source)) { if (!is_multicast_ether_addr(ethhdr->h_source)) {
@ -230,7 +230,7 @@ static int batadv_interface_tx(struct sk_buff *skb,
/* skb->data may have been modified by /* skb->data may have been modified by
* batadv_gw_dhcp_recipient_get() * batadv_gw_dhcp_recipient_get()
*/ */
ethhdr = (struct ethhdr *)skb->data; ethhdr = eth_hdr(skb);
/* if gw_mode is on, broadcast any non-DHCP message. /* if gw_mode is on, broadcast any non-DHCP message.
* All the DHCP packets are going to be sent as unicast * All the DHCP packets are going to be sent as unicast
*/ */