Skip to content

Commit d4b812d

Browse files
edumazetdavem330
authored andcommitted
vlan: mask vlan prio bits
In commit 48cc32d ("vlan: don't deliver frames for unknown vlans to protocols") Florian made sure we set pkt_type to PACKET_OTHERHOST if the vlan id is set and we could find a vlan device for this particular id. But we also have a problem if prio bits are set. Steinar reported an issue on a router receiving IPv6 frames with a vlan tag of 4000 (id 0, prio 2), and tunneled into a sit device, because skb->vlan_tci is set. Forwarded frame is completely corrupted : We can see (8100:4000) being inserted in the middle of IPv6 source address : 16:48:00.780413 IP6 2001:16d8:8100:4000:ee1c:0:9d9:bc87 > 9f94:4d95:2001:67c:29f4::: ICMP6, unknown icmp6 type (0), length 64 0x0000: 0000 0029 8000 c7c3 7103 0001 a0ae e651 0x0010: 0000 0000 ccce 0b00 0000 0000 1011 1213 0x0020: 1415 1617 1819 1a1b 1c1d 1e1f 2021 2223 0x0030: 2425 2627 2829 2a2b 2c2d 2e2f 3031 3233 It seems we are not really ready to properly cope with this right now. We can probably do better in future kernels : vlan_get_ingress_priority() should be a netdev property instead of a per vlan_dev one. For stable kernels, lets clear vlan_tci to fix the bugs. Reported-by: Steinar H. Gunderson <[email protected]> Signed-off-by: Eric Dumazet <[email protected]> Signed-off-by: David S. Miller <[email protected]>
1 parent ece793f commit d4b812d

File tree

3 files changed

+11
-5
lines changed

3 files changed

+11
-5
lines changed

include/linux/if_vlan.h

Lines changed: 1 addition & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -79,9 +79,8 @@ static inline int is_vlan_dev(struct net_device *dev)
7979
}
8080

8181
#define vlan_tx_tag_present(__skb) ((__skb)->vlan_tci & VLAN_TAG_PRESENT)
82-
#define vlan_tx_nonzero_tag_present(__skb) \
83-
(vlan_tx_tag_present(__skb) && ((__skb)->vlan_tci & VLAN_VID_MASK))
8482
#define vlan_tx_tag_get(__skb) ((__skb)->vlan_tci & ~VLAN_TAG_PRESENT)
83+
#define vlan_tx_tag_get_id(__skb) ((__skb)->vlan_tci & VLAN_VID_MASK)
8584

8685
#if defined(CONFIG_VLAN_8021Q) || defined(CONFIG_VLAN_8021Q_MODULE)
8786

net/8021q/vlan_core.c

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -9,7 +9,7 @@ bool vlan_do_receive(struct sk_buff **skbp)
99
{
1010
struct sk_buff *skb = *skbp;
1111
__be16 vlan_proto = skb->vlan_proto;
12-
u16 vlan_id = skb->vlan_tci & VLAN_VID_MASK;
12+
u16 vlan_id = vlan_tx_tag_get_id(skb);
1313
struct net_device *vlan_dev;
1414
struct vlan_pcpu_stats *rx_stats;
1515

net/core/dev.c

Lines changed: 9 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -3580,8 +3580,15 @@ static int __netif_receive_skb_core(struct sk_buff *skb, bool pfmemalloc)
35803580
}
35813581
}
35823582

3583-
if (vlan_tx_nonzero_tag_present(skb))
3584-
skb->pkt_type = PACKET_OTHERHOST;
3583+
if (unlikely(vlan_tx_tag_present(skb))) {
3584+
if (vlan_tx_tag_get_id(skb))
3585+
skb->pkt_type = PACKET_OTHERHOST;
3586+
/* Note: we might in the future use prio bits
3587+
* and set skb->priority like in vlan_do_receive()
3588+
* For the time being, just ignore Priority Code Point
3589+
*/
3590+
skb->vlan_tci = 0;
3591+
}
35853592

35863593
/* deliver only exact match when indicated */
35873594
null_or_dev = deliver_exact ? skb->dev : NULL;

0 commit comments

Comments
 (0)