Commit 36b3dd25 authored by Jukka Rissanen's avatar Jukka Rissanen Committed by Marcel Holtmann

Bluetooth: 6lowpan: Ensure header compression does not corrupt IPv6 header

If skb is going to multiple destinations, then make sure that we
do not overwrite the common IPv6 headers. So before compressing
the IPv6 headers, we copy the skb and that is then sent to 6LoWPAN
Bluetooth devices.

This is a similar patch as what was done for IEEE 802.154 6LoWPAN
in commit f19f4f95 ("ieee802154: 6lowpan: ensure header compression
does not corrupt ipv6 header")
Signed-off-by: default avatarJukka Rissanen <jukka.rissanen@linux.intel.com>
Signed-off-by: default avatarMarcel Holtmann <marcel@holtmann.org>
parent 59790aa2
...@@ -426,38 +426,33 @@ static void convert_dest_bdaddr(struct in6_addr *ip6_daddr, ...@@ -426,38 +426,33 @@ static void convert_dest_bdaddr(struct in6_addr *ip6_daddr,
*addr_type = get_addr_type_from_eui64(addr->b[5]); *addr_type = get_addr_type_from_eui64(addr->b[5]);
} }
static int header_create(struct sk_buff *skb, struct net_device *netdev, static int setup_header(struct sk_buff *skb, struct net_device *netdev,
unsigned short type, const void *_daddr, bdaddr_t *peer_addr, u8 *peer_addr_type)
const void *_saddr, unsigned int len)
{ {
struct ipv6hdr *hdr; struct in6_addr ipv6_daddr;
struct lowpan_dev *dev; struct lowpan_dev *dev;
struct lowpan_peer *peer; struct lowpan_peer *peer;
bdaddr_t addr, *any = BDADDR_ANY; bdaddr_t addr, *any = BDADDR_ANY;
u8 *saddr, *daddr = any->b; u8 *daddr = any->b;
u8 addr_type; int err, status = 0;
if (type != ETH_P_IPV6)
return -EINVAL;
hdr = ipv6_hdr(skb);
dev = lowpan_dev(netdev); dev = lowpan_dev(netdev);
if (ipv6_addr_is_multicast(&hdr->daddr)) { memcpy(&ipv6_daddr, &lowpan_cb(skb)->addr, sizeof(ipv6_daddr));
memcpy(&lowpan_cb(skb)->addr, &hdr->daddr,
sizeof(struct in6_addr)); if (ipv6_addr_is_multicast(&ipv6_daddr)) {
lowpan_cb(skb)->chan = NULL; lowpan_cb(skb)->chan = NULL;
} else { } else {
unsigned long flags; unsigned long flags;
u8 addr_type;
/* Get destination BT device from skb. /* Get destination BT device from skb.
* If there is no such peer then discard the packet. * If there is no such peer then discard the packet.
*/ */
convert_dest_bdaddr(&hdr->daddr, &addr, &addr_type); convert_dest_bdaddr(&ipv6_daddr, &addr, &addr_type);
BT_DBG("dest addr %pMR type %d IP %pI6c", &addr, BT_DBG("dest addr %pMR type %d IP %pI6c", &addr,
addr_type, &hdr->daddr); addr_type, &ipv6_daddr);
read_lock_irqsave(&devices_lock, flags); read_lock_irqsave(&devices_lock, flags);
peer = peer_lookup_ba(dev, &addr, addr_type); peer = peer_lookup_ba(dev, &addr, addr_type);
...@@ -470,7 +465,7 @@ static int header_create(struct sk_buff *skb, struct net_device *netdev, ...@@ -470,7 +465,7 @@ static int header_create(struct sk_buff *skb, struct net_device *netdev,
* the destination address. * the destination address.
*/ */
read_lock_irqsave(&devices_lock, flags); read_lock_irqsave(&devices_lock, flags);
peer = peer_lookup_dst(dev, &hdr->daddr, skb); peer = peer_lookup_dst(dev, &ipv6_daddr, skb);
read_unlock_irqrestore(&devices_lock, flags); read_unlock_irqrestore(&devices_lock, flags);
if (!peer) { if (!peer) {
BT_DBG("no such peer %pMR found", &addr); BT_DBG("no such peer %pMR found", &addr);
...@@ -479,29 +474,56 @@ static int header_create(struct sk_buff *skb, struct net_device *netdev, ...@@ -479,29 +474,56 @@ static int header_create(struct sk_buff *skb, struct net_device *netdev,
} }
daddr = peer->eui64_addr; daddr = peer->eui64_addr;
*peer_addr = addr;
memcpy(&lowpan_cb(skb)->addr, &hdr->daddr, *peer_addr_type = addr_type;
sizeof(struct in6_addr));
lowpan_cb(skb)->chan = peer->chan; lowpan_cb(skb)->chan = peer->chan;
status = 1;
} }
saddr = dev->netdev->dev_addr; lowpan_header_compress(skb, netdev, ETH_P_IPV6, daddr,
dev->netdev->dev_addr, skb->len);
err = dev_hard_header(skb, netdev, ETH_P_IPV6, NULL, NULL, 0);
if (err < 0)
return err;
return status;
}
static int header_create(struct sk_buff *skb, struct net_device *netdev,
unsigned short type, const void *_daddr,
const void *_saddr, unsigned int len)
{
struct ipv6hdr *hdr;
if (type != ETH_P_IPV6)
return -EINVAL;
hdr = ipv6_hdr(skb);
memcpy(&lowpan_cb(skb)->addr, &hdr->daddr, sizeof(struct in6_addr));
return lowpan_header_compress(skb, netdev, type, daddr, saddr, len); return 0;
} }
/* Packet to BT LE device */ /* Packet to BT LE device */
static int send_pkt(struct l2cap_chan *chan, struct sk_buff *skb, static int send_pkt(struct l2cap_chan *chan, struct sk_buff *skb,
struct net_device *netdev) struct net_device *netdev, bool is_mcast)
{ {
struct msghdr msg; struct msghdr msg;
struct kvec iv; struct kvec iv;
int err; int err;
/* Remember the skb so that we can send EAGAIN to the caller if /* Remember the skb so that we can send EAGAIN to the caller if
* we run out of credits. * we run out of credits. This is not done for multicast packets
* because we generate mcast packet in this module and are not
* really able to remember the skb after this packet is sent.
*/ */
chan->data = skb; if (is_mcast)
chan->data = NULL;
else
chan->data = skb;
memset(&msg, 0, sizeof(msg)); memset(&msg, 0, sizeof(msg));
msg.msg_iov = (struct iovec *) &iv; msg.msg_iov = (struct iovec *) &iv;
...@@ -549,7 +571,11 @@ static void send_mcast_pkt(struct sk_buff *skb, struct net_device *netdev) ...@@ -549,7 +571,11 @@ static void send_mcast_pkt(struct sk_buff *skb, struct net_device *netdev)
list_for_each_entry_safe(pentry, ptmp, &dev->peers, list) { list_for_each_entry_safe(pentry, ptmp, &dev->peers, list) {
local_skb = skb_clone(skb, GFP_ATOMIC); local_skb = skb_clone(skb, GFP_ATOMIC);
send_pkt(pentry->chan, local_skb, netdev); BT_DBG("xmit %s to %pMR type %d IP %pI6c chan %p",
netdev->name,
&pentry->chan->dst, pentry->chan->dst_type,
&pentry->peer_addr, pentry->chan);
send_pkt(pentry->chan, local_skb, netdev, true);
kfree_skb(local_skb); kfree_skb(local_skb);
} }
...@@ -561,43 +587,48 @@ static void send_mcast_pkt(struct sk_buff *skb, struct net_device *netdev) ...@@ -561,43 +587,48 @@ static void send_mcast_pkt(struct sk_buff *skb, struct net_device *netdev)
static netdev_tx_t bt_xmit(struct sk_buff *skb, struct net_device *netdev) static netdev_tx_t bt_xmit(struct sk_buff *skb, struct net_device *netdev)
{ {
int err = 0; int err = 0;
struct lowpan_dev *dev;
struct lowpan_peer *peer;
bdaddr_t addr; bdaddr_t addr;
u8 addr_type; u8 addr_type;
if (ipv6_addr_is_multicast(&lowpan_cb(skb)->addr)) { /* We must take a copy of the skb before we modify/replace the ipv6
/* We need to send the packet to every device * header as the header could be used elsewhere
* behind this interface. */
*/ skb = skb_unshare(skb, GFP_ATOMIC);
send_mcast_pkt(skb, netdev); if (!skb)
} else { return NET_XMIT_DROP;
unsigned long flags;
convert_dest_bdaddr(&lowpan_cb(skb)->addr, &addr, &addr_type);
dev = lowpan_dev(netdev);
read_lock_irqsave(&devices_lock, flags);
peer = peer_lookup_ba(dev, &addr, addr_type);
if (!peer)
peer = peer_lookup_dst(dev, &lowpan_cb(skb)->addr, skb);
read_unlock_irqrestore(&devices_lock, flags);
BT_DBG("xmit %s to %pMR type %d IP %pI6c peer %p", /* Return values from setup_header()
netdev->name, &addr, addr_type, * <0 - error, packet is dropped
&lowpan_cb(skb)->addr, peer); * 0 - this is a multicast packet
* 1 - this is unicast packet
*/
err = setup_header(skb, netdev, &addr, &addr_type);
if (err < 0) {
kfree_skb(skb);
return NET_XMIT_DROP;
}
if (peer && peer->chan) if (err) {
err = send_pkt(peer->chan, skb, netdev); if (lowpan_cb(skb)->chan) {
else BT_DBG("xmit %s to %pMR type %d IP %pI6c chan %p",
netdev->name, &addr, addr_type,
&lowpan_cb(skb)->addr, lowpan_cb(skb)->chan);
err = send_pkt(lowpan_cb(skb)->chan, skb, netdev,
false);
} else {
err = -ENOENT; err = -ENOENT;
}
} else {
/* We need to send the packet to every device behind this
* interface.
*/
send_mcast_pkt(skb, netdev);
} }
dev_kfree_skb(skb);
if (err) if (err)
BT_DBG("ERROR: xmit failed (%d)", err); BT_DBG("ERROR: xmit failed (%d)", err);
return (err < 0) ? NET_XMIT_DROP : err; return err < 0 ? NET_XMIT_DROP : err;
} }
static const struct net_device_ops netdev_ops = { static const struct net_device_ops netdev_ops = {
......
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